diff options
Diffstat (limited to 'fs/proc/fd.c')
| -rw-r--r-- | fs/proc/fd.c | 114 | 
1 files changed, 48 insertions, 66 deletions
| diff --git a/fs/proc/fd.c b/fs/proc/fd.c index d7a4a28ef630..75f2890abbd8 100644 --- a/fs/proc/fd.c +++ b/fs/proc/fd.c @@ -167,11 +167,10 @@ static int proc_fd_link(struct dentry *dentry, struct path *path)  	return ret;  } -static struct dentry * +static int  proc_fd_instantiate(struct inode *dir, struct dentry *dentry,  		    struct task_struct *task, const void *ptr)  { -	struct dentry *error = ERR_PTR(-ENOENT);  	unsigned fd = (unsigned long)ptr;  	struct proc_inode *ei;  	struct inode *inode; @@ -194,9 +193,9 @@ proc_fd_instantiate(struct inode *dir, struct dentry *dentry,  	/* Close the race of the process dying before we return the dentry */  	if (tid_fd_revalidate(dentry, 0)) -		error = NULL; +		return 0;   out: -	return error; +	return -ENOENT;  }  static struct dentry *proc_lookupfd_common(struct inode *dir, @@ -204,7 +203,7 @@ static struct dentry *proc_lookupfd_common(struct inode *dir,  					   instantiate_t instantiate)  {  	struct task_struct *task = get_proc_task(dir); -	struct dentry *result = ERR_PTR(-ENOENT); +	int result = -ENOENT;  	unsigned fd = name_to_int(dentry);  	if (!task) @@ -216,77 +215,61 @@ static struct dentry *proc_lookupfd_common(struct inode *dir,  out:  	put_task_struct(task);  out_no_task: -	return result; +	return ERR_PTR(result);  } -static int proc_readfd_common(struct file * filp, void * dirent, -			      filldir_t filldir, instantiate_t instantiate) +static int proc_readfd_common(struct file *file, struct dir_context *ctx, +			      instantiate_t instantiate)  { -	struct dentry *dentry = filp->f_path.dentry; -	struct inode *inode = dentry->d_inode; -	struct task_struct *p = get_proc_task(inode); +	struct task_struct *p = get_proc_task(file_inode(file));  	struct files_struct *files; -	unsigned int fd, ino; -	int retval; +	unsigned int fd; -	retval = -ENOENT;  	if (!p) -		goto out_no_task; -	retval = 0; - -	fd = filp->f_pos; -	switch (fd) { -		case 0: -			if (filldir(dirent, ".", 1, 0, inode->i_ino, DT_DIR) < 0) -				goto out; -			filp->f_pos++; -		case 1: -			ino = parent_ino(dentry); -			if (filldir(dirent, "..", 2, 1, ino, DT_DIR) < 0) -				goto out; -			filp->f_pos++; -		default: -			files = get_files_struct(p); -			if (!files) -				goto out; -			rcu_read_lock(); -			for (fd = filp->f_pos - 2; -			     fd < files_fdtable(files)->max_fds; -			     fd++, filp->f_pos++) { -				char name[PROC_NUMBUF]; -				int len; -				int rv; - -				if (!fcheck_files(files, fd)) -					continue; -				rcu_read_unlock(); +		return -ENOENT; -				len = snprintf(name, sizeof(name), "%d", fd); -				rv = proc_fill_cache(filp, dirent, filldir, -						     name, len, instantiate, p, -						     (void *)(unsigned long)fd); -				if (rv < 0) -					goto out_fd_loop; -				rcu_read_lock(); -			} -			rcu_read_unlock(); -out_fd_loop: -			put_files_struct(files); +	if (!dir_emit_dots(file, ctx)) +		goto out; +	if (!dir_emit_dots(file, ctx)) +		goto out; +	files = get_files_struct(p); +	if (!files) +		goto out; + +	rcu_read_lock(); +	for (fd = ctx->pos - 2; +	     fd < files_fdtable(files)->max_fds; +	     fd++, ctx->pos++) { +		char name[PROC_NUMBUF]; +		int len; + +		if (!fcheck_files(files, fd)) +			continue; +		rcu_read_unlock(); + +		len = snprintf(name, sizeof(name), "%d", fd); +		if (!proc_fill_cache(file, ctx, +				     name, len, instantiate, p, +				     (void *)(unsigned long)fd)) +			goto out_fd_loop; +		rcu_read_lock();  	} +	rcu_read_unlock(); +out_fd_loop: +	put_files_struct(files);  out:  	put_task_struct(p); -out_no_task: -	return retval; +	return 0;  } -static int proc_readfd(struct file *filp, void *dirent, filldir_t filldir) +static int proc_readfd(struct file *file, struct dir_context *ctx)  { -	return proc_readfd_common(filp, dirent, filldir, proc_fd_instantiate); +	return proc_readfd_common(file, ctx, proc_fd_instantiate);  }  const struct file_operations proc_fd_operations = {  	.read		= generic_read_dir, -	.readdir	= proc_readfd, +	.iterate	= proc_readfd,  	.llseek		= default_llseek,  }; @@ -316,11 +299,10 @@ const struct inode_operations proc_fd_inode_operations = {  	.setattr	= proc_setattr,  }; -static struct dentry * +static int  proc_fdinfo_instantiate(struct inode *dir, struct dentry *dentry,  			struct task_struct *task, const void *ptr)  { -	struct dentry *error = ERR_PTR(-ENOENT);  	unsigned fd = (unsigned long)ptr;  	struct proc_inode *ei;  	struct inode *inode; @@ -340,9 +322,9 @@ proc_fdinfo_instantiate(struct inode *dir, struct dentry *dentry,  	/* Close the race of the process dying before we return the dentry */  	if (tid_fd_revalidate(dentry, 0)) -		error = NULL; +		return 0;   out: -	return error; +	return -ENOENT;  }  static struct dentry * @@ -351,9 +333,9 @@ proc_lookupfdinfo(struct inode *dir, struct dentry *dentry, unsigned int flags)  	return proc_lookupfd_common(dir, dentry, proc_fdinfo_instantiate);  } -static int proc_readfdinfo(struct file *filp, void *dirent, filldir_t filldir) +static int proc_readfdinfo(struct file *file, struct dir_context *ctx)  { -	return proc_readfd_common(filp, dirent, filldir, +	return proc_readfd_common(file, ctx,  				  proc_fdinfo_instantiate);  } @@ -364,6 +346,6 @@ const struct inode_operations proc_fdinfo_inode_operations = {  const struct file_operations proc_fdinfo_operations = {  	.read		= generic_read_dir, -	.readdir	= proc_readfdinfo, +	.iterate	= proc_readfdinfo,  	.llseek		= default_llseek,  }; | 
