cregit-Linux how code gets into the kernel

Release 4.18 fs/pstore/inode.c

Directory: fs/pstore
/*
 * Persistent Storage - ramfs parts.
 *
 * Copyright (C) 2010 Intel Corporation <tony.luck@intel.com>
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License version 2 as
 *  published by the Free Software Foundation.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with this program; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */

#include <linux/module.h>
#include <linux/fs.h>
#include <linux/fsnotify.h>
#include <linux/pagemap.h>
#include <linux/highmem.h>
#include <linux/time.h>
#include <linux/init.h>
#include <linux/list.h>
#include <linux/string.h>
#include <linux/mount.h>
#include <linux/seq_file.h>
#include <linux/ramfs.h>
#include <linux/parser.h>
#include <linux/sched.h>
#include <linux/magic.h>
#include <linux/pstore.h>
#include <linux/slab.h>
#include <linux/spinlock.h>
#include <linux/uaccess.h>

#include "internal.h"


#define	PSTORE_NAMELEN	64

static DEFINE_SPINLOCK(allpstore_lock);
static LIST_HEAD(allpstore);


struct pstore_private {
	
struct list_head list;
	
struct pstore_record *record;
	
size_t total_size;
};


struct pstore_ftrace_seq_data {
	
const void *ptr;
	
size_t off;
	
size_t size;
};


#define REC_SIZE sizeof(struct pstore_ftrace_record)


static void free_pstore_private(struct pstore_private *private) { if (!private) return; if (private->record) { kfree(private->record->buf); kfree(private->record); } kfree(private); }

Contributors

PersonTokensPropCommitsCommitProp
Kees Cook46100.00%2100.00%
Total46100.00%2100.00%


static void *pstore_ftrace_seq_start(struct seq_file *s, loff_t *pos) { struct pstore_private *ps = s->private; struct pstore_ftrace_seq_data *data; data = kzalloc(sizeof(*data), GFP_KERNEL); if (!data) return NULL; data->off = ps->total_size % REC_SIZE; data->off += *pos * REC_SIZE; if (data->off + REC_SIZE > ps->total_size) { kfree(data); return NULL; } return data; }

Contributors

PersonTokensPropCommitsCommitProp
Anton Vorontsov9397.89%150.00%
Kees Cook22.11%150.00%
Total95100.00%2100.00%


static void pstore_ftrace_seq_stop(struct seq_file *s, void *v) { kfree(v); }

Contributors

PersonTokensPropCommitsCommitProp
Anton Vorontsov20100.00%1100.00%
Total20100.00%1100.00%


static void *pstore_ftrace_seq_next(struct seq_file *s, void *v, loff_t *pos) { struct pstore_private *ps = s->private; struct pstore_ftrace_seq_data *data = v; data->off += REC_SIZE; if (data->off + REC_SIZE > ps->total_size) return NULL; (*pos)++; return data; }

Contributors

PersonTokensPropCommitsCommitProp
Anton Vorontsov6598.48%150.00%
Kees Cook11.52%150.00%
Total66100.00%2100.00%


static int pstore_ftrace_seq_show(struct seq_file *s, void *v) { struct pstore_private *ps = s->private; struct pstore_ftrace_seq_data *data = v; struct pstore_ftrace_record *rec; rec = (struct pstore_ftrace_record *)(ps->record->buf + data->off); seq_printf(s, "CPU:%d ts:%llu %08lx %08lx %pf <- %pF\n", pstore_ftrace_decode_cpu(rec), pstore_ftrace_read_timestamp(rec), rec->ip, rec->parent_ip, (void *)rec->ip, (void *)rec->parent_ip); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Anton Vorontsov8484.85%125.00%
Kees Cook99.09%250.00%
Joel Fernandes66.06%125.00%
Total99100.00%4100.00%

static const struct seq_operations pstore_ftrace_seq_ops = { .start = pstore_ftrace_seq_start, .next = pstore_ftrace_seq_next, .stop = pstore_ftrace_seq_stop, .show = pstore_ftrace_seq_show, };
static ssize_t pstore_file_read(struct file *file, char __user *userbuf, size_t count, loff_t *ppos) { struct seq_file *sf = file->private_data; struct pstore_private *ps = sf->private; if (ps->record->type == PSTORE_TYPE_FTRACE) return seq_read(file, userbuf, count, ppos); return simple_read_from_buffer(userbuf, count, ppos, ps->record->buf, ps->total_size); }

Contributors

PersonTokensPropCommitsCommitProp
Tony Luck4655.42%125.00%
Anton Vorontsov3137.35%125.00%
Kees Cook67.23%250.00%
Total83100.00%4100.00%


static int pstore_file_open(struct inode *inode, struct file *file) { struct pstore_private *ps = inode->i_private; struct seq_file *sf; int err; const struct seq_operations *sops = NULL; if (ps->record->type == PSTORE_TYPE_FTRACE) sops = &pstore_ftrace_seq_ops; err = seq_open(file, sops); if (err < 0) return err; sf = file->private_data; sf->private = ps; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Anton Vorontsov8797.75%150.00%
Kees Cook22.25%150.00%
Total89100.00%2100.00%


static loff_t pstore_file_llseek(struct file *file, loff_t off, int whence) { struct seq_file *sf = file->private_data; if (sf->op) return seq_lseek(file, off, whence); return default_llseek(file, off, whence); }

Contributors

PersonTokensPropCommitsCommitProp
Anton Vorontsov4994.23%150.00%
Andrew Morton35.77%150.00%
Total52100.00%2100.00%

static const struct file_operations pstore_file_operations = { .open = pstore_file_open, .read = pstore_file_read, .llseek = pstore_file_llseek, .release = seq_release, }; /* * When a file is unlinked from our file system we call the * platform driver to erase the record from persistent store. */
static int pstore_unlink(struct inode *dir, struct dentry *dentry) { struct pstore_private *p = d_inode(dentry)->i_private; struct pstore_record *record = p->record; if (!record->psi->erase) return -EPERM; mutex_lock(&record->psi->read_mutex); record->psi->erase(record); mutex_unlock(&record->psi->read_mutex); return simple_unlink(dir, dentry); }

Contributors

PersonTokensPropCommitsCommitProp
Tony Luck3843.68%114.29%
Kees Cook2629.89%342.86%
Namhyung Kim1820.69%114.29%
David Howells33.45%114.29%
Matthew Garrett22.30%114.29%
Total87100.00%7100.00%


static void pstore_evict_inode(struct inode *inode) { struct pstore_private *p = inode->i_private; unsigned long flags; clear_inode(inode); if (p) { spin_lock_irqsave(&allpstore_lock, flags); list_del(&p->list); spin_unlock_irqrestore(&allpstore_lock, flags); free_pstore_private(p); } }

Contributors

PersonTokensPropCommitsCommitProp
Tony Luck6296.88%250.00%
Jan Kara11.56%125.00%
Kees Cook11.56%125.00%
Total64100.00%4100.00%

static const struct inode_operations pstore_dir_inode_operations = { .lookup = simple_lookup, .unlink = pstore_unlink, };
static struct inode *pstore_get_inode(struct super_block *sb) { struct inode *inode = new_inode(sb); if (inode) { inode->i_ino = get_next_ino(); inode->i_atime = inode->i_mtime = inode->i_ctime = current_time(inode); } return inode; }

Contributors

PersonTokensPropCommitsCommitProp
Tony Luck5292.86%150.00%
Deepa Dinamani47.14%150.00%
Total56100.00%2100.00%

enum { Opt_kmsg_bytes, Opt_err }; static const match_table_t tokens = { {Opt_kmsg_bytes, "kmsg_bytes=%u"}, {Opt_err, NULL} };
static void parse_options(char *options) { char *p; substring_t args[MAX_OPT_ARGS]; int option; if (!options) return; while ((p = strsep(&options, ",")) != NULL) { int token; if (!*p) continue; token = match_token(p, tokens, args); switch (token) { case Opt_kmsg_bytes: if (!match_int(&args[0], &option)) pstore_set_kmsg_bytes(option); break; } } }

Contributors

PersonTokensPropCommitsCommitProp
Tony Luck98100.00%1100.00%
Total98100.00%1100.00%

/* * Display the mount options in /proc/mounts. */
static int pstore_show_options(struct seq_file *m, struct dentry *root) { if (kmsg_bytes != PSTORE_DEFAULT_KMSG_BYTES) seq_printf(m, ",kmsg_bytes=%lu", kmsg_bytes); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
David Howells34100.00%1100.00%
Total34100.00%1100.00%


static int pstore_remount(struct super_block *sb, int *flags, char *data) { sync_filesystem(sb); parse_options(data); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tony Luck2784.38%150.00%
Theodore Y. Ts'o515.62%150.00%
Total32100.00%2100.00%

static const struct super_operations pstore_ops = { .statfs = simple_statfs, .drop_inode = generic_delete_inode, .evict_inode = pstore_evict_inode, .remount_fs = pstore_remount, .show_options = pstore_show_options, }; static struct super_block *pstore_sb;
bool pstore_is_mounted(void) { return pstore_sb != NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Tony Luck1191.67%266.67%
Geliang Tang18.33%133.33%
Total12100.00%3100.00%

/* * Make a regular file in the root directory of our file system. * Load it up with "size" bytes of data from "buf". * Set the mtime & ctime to the date that this record was originally stored. */
int pstore_mkfile(struct dentry *root, struct pstore_record *record) { struct dentry *dentry; struct inode *inode; int rc = 0; char name[PSTORE_NAMELEN]; struct pstore_private *private, *pos; unsigned long flags; size_t size = record->size + record->ecc_notice_size; WARN_ON(!inode_is_locked(d_inode(root))); spin_lock_irqsave(&allpstore_lock, flags); list_for_each_entry(pos, &allpstore, list) { if (pos->record->type == record->type && pos->record->id == record->id && pos->record->psi == record->psi) { rc = -EEXIST; break; } } spin_unlock_irqrestore(&allpstore_lock, flags); if (rc) return rc; rc = -ENOMEM; inode = pstore_get_inode(root->d_sb); if (!inode) goto fail; inode->i_mode = S_IFREG | 0444; inode->i_fop = &pstore_file_operations; private = kzalloc(sizeof(*private), GFP_KERNEL); if (!private) goto fail_alloc; private->record = record; switch (record->type) { case PSTORE_TYPE_DMESG: scnprintf(name, sizeof(name), "dmesg-%s-%llu%s", record->psi->name, record->id, record->compressed ? ".enc.z" : ""); break; case PSTORE_TYPE_CONSOLE: scnprintf(name, sizeof(name), "console-%s-%llu", record->psi->name, record->id); break; case PSTORE_TYPE_FTRACE: scnprintf(name, sizeof(name), "ftrace-%s-%llu", record->psi->name, record->id); break; case PSTORE_TYPE_MCE: scnprintf(name, sizeof(name), "mce-%s-%llu", record->psi->name, record->id); break; case PSTORE_TYPE_PPC_RTAS: scnprintf(name, sizeof(name), "rtas-%s-%llu", record->psi->name, record->id); break; case PSTORE_TYPE_PPC_OF: scnprintf(name, sizeof(name), "powerpc-ofw-%s-%llu", record->psi->name, record->id); break; case PSTORE_TYPE_PPC_COMMON: scnprintf(name, sizeof(name), "powerpc-common-%s-%llu", record->psi->name, record->id); break; case PSTORE_TYPE_PMSG: scnprintf(name, sizeof(name), "pmsg-%s-%llu", record->psi->name, record->id); break; case PSTORE_TYPE_PPC_OPAL: scnprintf(name, sizeof(name), "powerpc-opal-%s-%llu", record->psi->name, record->id); break; case PSTORE_TYPE_UNKNOWN: scnprintf(name, sizeof(name), "unknown-%s-%llu", record->psi->name, record->id); break; default: scnprintf(name, sizeof(name), "type%d-%s-%llu", record->type, record->psi->name, record->id); break; } dentry = d_alloc_name(root, name); if (!dentry) goto fail_private; inode->i_size = private->total_size = size; inode->i_private = private; if (record->time.tv_sec) inode->i_mtime = inode->i_ctime = record->time; d_add(dentry, inode); spin_lock_irqsave(&allpstore_lock, flags); list_add(&private->list, &allpstore); spin_unlock_irqrestore(&allpstore_lock, flags); return 0; fail_private: free_pstore_private(private); fail_alloc: iput(inode); fail: return rc; }

Contributors

PersonTokensPropCommitsCommitProp
Tony Luck29647.67%315.00%
Kees Cook15925.60%525.00%
Mark Salyzyn7211.59%210.00%
Aruna Balakrishnaiah426.76%420.00%
Anton Vorontsov203.22%210.00%
Al Viro152.42%15.00%
Hari Bathini121.93%15.00%
Valdis Kletnieks40.64%15.00%
Dan Carpenter10.16%15.00%
Total621100.00%20100.00%

/* * Read all the records from the persistent store. Create * files in our filesystem. Don't warn about -EEXIST errors * when we are re-scanning the backing store looking to add new * error records. */
void pstore_get_records(int quiet) { struct pstore_info *psi = psinfo; struct dentry *root; if (!psi || !pstore_sb) return; root = pstore_sb->s_root; inode_lock(d_inode(root)); pstore_get_backend_records(psi, root, quiet); inode_unlock(d_inode(root)); }

Contributors

PersonTokensPropCommitsCommitProp
Kees Cook60100.00%1100.00%
Total60100.00%1100.00%


static int pstore_fill_super(struct super_block *sb, void *data, int silent) { struct inode *inode; pstore_sb = sb; sb->s_maxbytes = MAX_LFS_FILESIZE; sb->s_blocksize = PAGE_SIZE; sb->s_blocksize_bits = PAGE_SHIFT; sb->s_magic = PSTOREFS_MAGIC; sb->s_op = &pstore_ops; sb->s_time_gran = 1; parse_options(data); inode = pstore_get_inode(sb); if (inode) { inode->i_mode = S_IFDIR | 0750; inode->i_op = &pstore_dir_inode_operations; inode->i_fop = &simple_dir_operations; inc_nlink(inode); } sb->s_root = d_make_root(inode); if (!sb->s_root) return -ENOMEM; pstore_get_records(0); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tony Luck10475.91%333.33%
Al Viro2921.17%333.33%
Kirill A. Shutemov21.46%111.11%
Anton Vorontsov10.73%111.11%
Kees Cook10.73%111.11%
Total137100.00%9100.00%


static struct dentry *pstore_mount(struct file_system_type *fs_type, int flags, const char *dev_name, void *data) { return mount_single(fs_type, flags, data, pstore_fill_super); }

Contributors

PersonTokensPropCommitsCommitProp
Tony Luck37100.00%2100.00%
Total37100.00%2100.00%


static void pstore_kill_sb(struct super_block *sb) { kill_litter_super(sb); pstore_sb = NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Tony Luck20100.00%1100.00%
Total20100.00%1100.00%

static struct file_system_type pstore_fs_type = { .owner = THIS_MODULE, .name = "pstore", .mount = pstore_mount, .kill_sb = pstore_kill_sb, };
static int __init init_pstore_fs(void) { int err; pstore_choose_compression(); /* Create a convenient mount point for people to access pstore */ err = sysfs_create_mount_point(fs_kobj, "pstore"); if (err) goto out; err = register_filesystem(&pstore_fs_type); if (err < 0) sysfs_remove_mount_point(fs_kobj, "pstore"); out: return err; }

Contributors

PersonTokensPropCommitsCommitProp
Josh Boyer3153.45%125.00%
Tony Luck1525.86%125.00%
Eric W. Biedermann915.52%125.00%
Kees Cook35.17%125.00%
Total58100.00%4100.00%

module_init(init_pstore_fs)
static void __exit exit_pstore_fs(void) { unregister_filesystem(&pstore_fs_type); sysfs_remove_mount_point(fs_kobj, "pstore"); }

Contributors

PersonTokensPropCommitsCommitProp
Geliang Tang22100.00%1100.00%
Total22100.00%1100.00%

module_exit(exit_pstore_fs) MODULE_AUTHOR("Tony Luck <tony.luck@intel.com>"); MODULE_LICENSE("GPL");

Overall Contributors

PersonTokensPropCommitsCommitProp
Tony Luck103347.17%511.63%
Anton Vorontsov50823.20%36.98%
Kees Cook32114.66%920.93%
Mark Salyzyn723.29%24.65%
Al Viro442.01%36.98%
Aruna Balakrishnaiah421.92%49.30%
David Howells391.78%24.65%
Geliang Tang321.46%24.65%
Josh Boyer311.42%12.33%
Namhyung Kim180.82%12.33%
Hari Bathini120.55%12.33%
Eric W. Biedermann90.41%12.33%
Joel Fernandes60.27%12.33%
Theodore Y. Ts'o50.23%12.33%
Deepa Dinamani40.18%12.33%
Valdis Kletnieks40.18%12.33%
Matthew Garrett30.14%12.33%
Andrew Morton30.14%12.33%
Kirill A. Shutemov20.09%12.33%
Jan Kara10.05%12.33%
Dan Carpenter10.05%12.33%
Total2190100.00%43100.00%
Directory: fs/pstore
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.