generic.c

来自「Linux Kernel 2.6.9 for OMAP1710」· C语言 代码 · 共 701 行 · 第 1/2 页

C
701
字号
/* * proc/fs/generic.c --- generic routines for the proc-fs * * This file contains generic proc-fs routines for handling * directories and files. *  * Copyright (C) 1991, 1992 Linus Torvalds. * Copyright (C) 1997 Theodore Ts'o */#include <linux/errno.h>#include <linux/time.h>#include <linux/proc_fs.h>#include <linux/stat.h>#include <linux/module.h>#include <linux/mount.h>#include <linux/smp_lock.h>#include <linux/init.h>#include <linux/idr.h>#include <linux/namei.h>#include <asm/uaccess.h>#include <asm/bitops.h>static ssize_t proc_file_read(struct file *file, char __user *buf,			      size_t nbytes, loff_t *ppos);static ssize_t proc_file_write(struct file *file, const char __user *buffer,			       size_t count, loff_t *ppos);static loff_t proc_file_lseek(struct file *, loff_t, int);int proc_match(int len, const char *name, struct proc_dir_entry *de){	if (de->namelen != len)		return 0;	return !memcmp(name, de->name, len);}static struct file_operations proc_file_operations = {	.llseek		= proc_file_lseek,	.read		= proc_file_read,	.write		= proc_file_write,};/* buffer size is one page but our output routines use some slack for overruns */#define PROC_BLOCK_SIZE	(PAGE_SIZE - 1024)static ssize_tproc_file_read(struct file *file, char __user *buf, size_t nbytes,	       loff_t *ppos){	struct inode * inode = file->f_dentry->d_inode;	char 	*page;	ssize_t	retval=0;	int	eof=0;	ssize_t	n, count;	char	*start;	struct proc_dir_entry * dp;	dp = PDE(inode);	if (!(page = (char*) __get_free_page(GFP_KERNEL)))		return -ENOMEM;	while ((nbytes > 0) && !eof) {		count = min_t(ssize_t, PROC_BLOCK_SIZE, nbytes);		start = NULL;		if (dp->get_info) {			/* Handle old net routines */			n = dp->get_info(page, &start, *ppos, count);			if (n < count)				eof = 1;		} else if (dp->read_proc) {			/*			 * How to be a proc read function			 * ------------------------------			 * Prototype:			 *    int f(char *buffer, char **start, off_t offset,			 *          int count, int *peof, void *dat)			 *			 * Assume that the buffer is "count" bytes in size.			 *			 * If you know you have supplied all the data you			 * have, set *peof.			 *			 * You have three ways to return data:			 * 0) Leave *start = NULL.  (This is the default.)			 *    Put the data of the requested offset at that			 *    offset within the buffer.  Return the number (n)			 *    of bytes there are from the beginning of the			 *    buffer up to the last byte of data.  If the			 *    number of supplied bytes (= n - offset) is 			 *    greater than zero and you didn't signal eof			 *    and the reader is prepared to take more data			 *    you will be called again with the requested			 *    offset advanced by the number of bytes 			 *    absorbed.  This interface is useful for files			 *    no larger than the buffer.			 * 1) Set *start = an unsigned long value less than			 *    the buffer address but greater than zero.			 *    Put the data of the requested offset at the			 *    beginning of the buffer.  Return the number of			 *    bytes of data placed there.  If this number is			 *    greater than zero and you didn't signal eof			 *    and the reader is prepared to take more data			 *    you will be called again with the requested			 *    offset advanced by *start.  This interface is			 *    useful when you have a large file consisting			 *    of a series of blocks which you want to count			 *    and return as wholes.			 *    (Hack by Paul.Russell@rustcorp.com.au)			 * 2) Set *start = an address within the buffer.			 *    Put the data of the requested offset at *start.			 *    Return the number of bytes of data placed there.			 *    If this number is greater than zero and you			 *    didn't signal eof and the reader is prepared to			 *    take more data you will be called again with the			 *    requested offset advanced by the number of bytes			 *    absorbed.			 */			n = dp->read_proc(page, &start, *ppos,					  count, &eof, dp->data);		} else			break;		if (n == 0)   /* end of file */			break;		if (n < 0) {  /* error */			if (retval == 0)				retval = n;			break;		}		if (start == NULL) {			if (n > PAGE_SIZE) {				printk(KERN_ERR				       "proc_file_read: Apparent buffer overflow!\n");				n = PAGE_SIZE;			}			n -= *ppos;			if (n <= 0)				break;			if (n > count)				n = count;			start = page + *ppos;		} else if (start < page) {			if (n > PAGE_SIZE) {				printk(KERN_ERR				       "proc_file_read: Apparent buffer overflow!\n");				n = PAGE_SIZE;			}			if (n > count) {				/*				 * Don't reduce n because doing so might				 * cut off part of a data block.				 */				printk(KERN_WARNING				       "proc_file_read: Read count exceeded\n");			}		} else /* start >= page */ {			unsigned long startoff = (unsigned long)(start - page);			if (n > (PAGE_SIZE - startoff)) {				printk(KERN_ERR				       "proc_file_read: Apparent buffer overflow!\n");				n = PAGE_SIZE - startoff;			}			if (n > count)				n = count;		}		 		n -= copy_to_user(buf, start < page ? page : start, n);		if (n == 0) {			if (retval == 0)				retval = -EFAULT;			break;		}		*ppos += start < page ? (unsigned long)start : n;		nbytes -= n;		buf += n;		retval += n;	}	free_page((unsigned long) page);	return retval;}static ssize_tproc_file_write(struct file *file, const char __user *buffer,		size_t count, loff_t *ppos){	struct inode *inode = file->f_dentry->d_inode;	struct proc_dir_entry * dp;		dp = PDE(inode);	if (!dp->write_proc)		return -EIO;	/* FIXME: does this routine need ppos?  probably... */	return dp->write_proc(file, buffer, count, dp->data);}static loff_tproc_file_lseek(struct file *file, loff_t offset, int orig){    lock_kernel();    switch (orig) {    case 0:	if (offset < 0)	    goto out;	file->f_pos = offset;	unlock_kernel();	return(file->f_pos);    case 1:	if (offset + file->f_pos < 0)	    goto out;	file->f_pos += offset;	unlock_kernel();	return(file->f_pos);    case 2:	goto out;    default:	goto out;    }out:    unlock_kernel();    return -EINVAL;}static int proc_notify_change(struct dentry *dentry, struct iattr *iattr){	struct inode *inode = dentry->d_inode;	struct proc_dir_entry *de = PDE(inode);	int error;	error = inode_change_ok(inode, iattr);	if (error)		goto out;	error = inode_setattr(inode, iattr);	if (error)		goto out;		de->uid = inode->i_uid;	de->gid = inode->i_gid;	de->mode = inode->i_mode;out:	return error;}static struct inode_operations proc_file_inode_operations = {	.setattr	= proc_notify_change,};/* * This function parses a name such as "tty/driver/serial", and * returns the struct proc_dir_entry for "/proc/tty/driver", and * returns "serial" in residual. */static int xlate_proc_name(const char *name,			   struct proc_dir_entry **ret, const char **residual){	const char     		*cp = name, *next;	struct proc_dir_entry	*de;	int			len;	de = &proc_root;	while (1) {		next = strchr(cp, '/');		if (!next)			break;		len = next - cp;		for (de = de->subdir; de ; de = de->next) {			if (proc_match(len, cp, de))				break;		}		if (!de)			return -ENOENT;		cp += len + 1;	}	*residual = cp;	*ret = de;	return 0;}static DEFINE_IDR(proc_inum_idr);static spinlock_t proc_inum_lock = SPIN_LOCK_UNLOCKED; /* protects the above */#define PROC_DYNAMIC_FIRST 0xF0000000UL/* * Return an inode number between PROC_DYNAMIC_FIRST and * 0xffffffff, or zero on failure. */static unsigned int get_inode_number(void){	int i, inum = 0;	int error;retry:	if (idr_pre_get(&proc_inum_idr, GFP_KERNEL) == 0)		return 0;	spin_lock(&proc_inum_lock);	error = idr_get_new(&proc_inum_idr, NULL, &i);	spin_unlock(&proc_inum_lock);	if (error == -EAGAIN)		goto retry;	else if (error)		return 0;	inum = (i & MAX_ID_MASK) + PROC_DYNAMIC_FIRST;	/* inum will never be more than 0xf0ffffff, so no check	 * for overflow.	 */	return inum;}static void release_inode_number(unsigned int inum){	int id = (inum - PROC_DYNAMIC_FIRST) | ~MAX_ID_MASK;	spin_lock(&proc_inum_lock);	idr_remove(&proc_inum_idr, id);	spin_unlock(&proc_inum_lock);}static int proc_follow_link(struct dentry *dentry, struct nameidata *nd){	nd_set_link(nd, PDE(dentry->d_inode)->data);	return 0;}static struct inode_operations proc_link_inode_operations = {	.readlink	= generic_readlink,	.follow_link	= proc_follow_link,};/* * As some entries in /proc are volatile, we want to  * get rid of unused dentries.  This could be made  * smarter: we could keep a "volatile" flag in the  * inode to indicate which ones to keep. */static int proc_delete_dentry(struct dentry * dentry){	return 1;

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?