Commit 1dce27c5 authored by David Howells's avatar David Howells Committed by H. Peter Anvin

Wrap accesses to the fd_sets in struct fdtable

Wrap accesses to the fd_sets in struct fdtable (for recording open files and
close-on-exec flags) so that we can move away from using fd_sets since we
abuse the fd_set structs by not allocating the full-sized structure under
normal circumstances and by non-core code looking at the internals of the
fd_sets.

The first abuse means that use of FD_ZERO() on these fd_sets is not permitted,
since that cannot be told about their abnormal lengths.

This introduces six wrapper functions for setting, clearing and testing
close-on-exec flags and fd-is-open flags:

	void __set_close_on_exec(int fd, struct fdtable *fdt);
	void __clear_close_on_exec(int fd, struct fdtable *fdt);
	bool close_on_exec(int fd, const struct fdtable *fdt);
	void __set_open_fd(int fd, struct fdtable *fdt);
	void __clear_open_fd(int fd, struct fdtable *fdt);
	bool fd_is_open(int fd, const struct fdtable *fdt);

Note that I've prepended '__' to the names of the set/clear functions because
they require the caller to hold a lock to use them.

Note also that I haven't added wrappers for looking behind the scenes at the
the array.  Possibly that should exist too.
Signed-off-by: default avatarDavid Howells <dhowells@redhat.com>
Link: http://lkml.kernel.org/r/20120216174942.23314.1364.stgit@warthog.procyon.org.ukSigned-off-by: default avatarH. Peter Anvin <hpa@zytor.com>
Cc: Al Viro <viro@zeniv.linux.org.uk>
parent 8b3d1cda
...@@ -113,8 +113,8 @@ the fdtable structure - ...@@ -113,8 +113,8 @@ the fdtable structure -
if (fd >= 0) { if (fd >= 0) {
/* locate_fd() may have expanded fdtable, load the ptr */ /* locate_fd() may have expanded fdtable, load the ptr */
fdt = files_fdtable(files); fdt = files_fdtable(files);
FD_SET(fd, fdt->open_fds); __set_open_fd(fd, fdt);
FD_CLR(fd, fdt->close_on_exec); __clear_close_on_exec(fd, fdt);
spin_unlock(&files->file_lock); spin_unlock(&files->file_lock);
..... .....
......
...@@ -122,7 +122,7 @@ static struct spu_context *coredump_next_context(int *fd) ...@@ -122,7 +122,7 @@ static struct spu_context *coredump_next_context(int *fd)
struct spu_context *ctx = NULL; struct spu_context *ctx = NULL;
for (; *fd < fdt->max_fds; (*fd)++) { for (; *fd < fdt->max_fds; (*fd)++) {
if (!FD_ISSET(*fd, fdt->open_fds)) if (!fd_is_open(*fd, fdt))
continue; continue;
file = fcheck(*fd); file = fcheck(*fd);
......
...@@ -408,11 +408,11 @@ int task_get_unused_fd_flags(struct binder_proc *proc, int flags) ...@@ -408,11 +408,11 @@ int task_get_unused_fd_flags(struct binder_proc *proc, int flags)
goto repeat; goto repeat;
} }
FD_SET(fd, fdt->open_fds); __set_open_fd(fd, fdt);
if (flags & O_CLOEXEC) if (flags & O_CLOEXEC)
FD_SET(fd, fdt->close_on_exec); __set_close_on_exec(fd, fdt);
else else
FD_CLR(fd, fdt->close_on_exec); __clear_close_on_exec(fd, fdt);
files->next_fd = fd + 1; files->next_fd = fd + 1;
#if 1 #if 1
/* Sanity check */ /* Sanity check */
...@@ -453,7 +453,7 @@ static void task_fd_install( ...@@ -453,7 +453,7 @@ static void task_fd_install(
static void __put_unused_fd(struct files_struct *files, unsigned int fd) static void __put_unused_fd(struct files_struct *files, unsigned int fd)
{ {
struct fdtable *fdt = files_fdtable(files); struct fdtable *fdt = files_fdtable(files);
__FD_CLR(fd, fdt->open_fds); __clear_open_fd(fd, fdt);
if (fd < files->next_fd) if (fd < files->next_fd)
files->next_fd = fd; files->next_fd = fd;
} }
...@@ -479,7 +479,7 @@ static long task_close_fd(struct binder_proc *proc, unsigned int fd) ...@@ -479,7 +479,7 @@ static long task_close_fd(struct binder_proc *proc, unsigned int fd)
if (!filp) if (!filp)
goto out_unlock; goto out_unlock;
rcu_assign_pointer(fdt->fd[fd], NULL); rcu_assign_pointer(fdt->fd[fd], NULL);
FD_CLR(fd, fdt->close_on_exec); __clear_close_on_exec(fd, fdt);
__put_unused_fd(files, fd); __put_unused_fd(files, fd);
spin_unlock(&files->file_lock); spin_unlock(&files->file_lock);
retval = filp_close(filp, files); retval = filp_close(filp, files);
......
...@@ -230,7 +230,7 @@ static void autofs_dev_ioctl_fd_install(unsigned int fd, struct file *file) ...@@ -230,7 +230,7 @@ static void autofs_dev_ioctl_fd_install(unsigned int fd, struct file *file)
fdt = files_fdtable(files); fdt = files_fdtable(files);
BUG_ON(fdt->fd[fd] != NULL); BUG_ON(fdt->fd[fd] != NULL);
rcu_assign_pointer(fdt->fd[fd], file); rcu_assign_pointer(fdt->fd[fd], file);
FD_SET(fd, fdt->close_on_exec); __set_close_on_exec(fd, fdt);
spin_unlock(&files->file_lock); spin_unlock(&files->file_lock);
} }
......
...@@ -2078,8 +2078,8 @@ static int umh_pipe_setup(struct subprocess_info *info, struct cred *new) ...@@ -2078,8 +2078,8 @@ static int umh_pipe_setup(struct subprocess_info *info, struct cred *new)
fd_install(0, rp); fd_install(0, rp);
spin_lock(&cf->file_lock); spin_lock(&cf->file_lock);
fdt = files_fdtable(cf); fdt = files_fdtable(cf);
FD_SET(0, fdt->open_fds); __set_open_fd(0, fdt);
FD_CLR(0, fdt->close_on_exec); __clear_close_on_exec(0, fdt);
spin_unlock(&cf->file_lock); spin_unlock(&cf->file_lock);
/* and disallow core files too */ /* and disallow core files too */
......
...@@ -32,20 +32,20 @@ void set_close_on_exec(unsigned int fd, int flag) ...@@ -32,20 +32,20 @@ void set_close_on_exec(unsigned int fd, int flag)
spin_lock(&files->file_lock); spin_lock(&files->file_lock);
fdt = files_fdtable(files); fdt = files_fdtable(files);
if (flag) if (flag)
FD_SET(fd, fdt->close_on_exec); __set_close_on_exec(fd, fdt);
else else
FD_CLR(fd, fdt->close_on_exec); __clear_close_on_exec(fd, fdt);
spin_unlock(&files->file_lock); spin_unlock(&files->file_lock);
} }
static int get_close_on_exec(unsigned int fd) static bool get_close_on_exec(unsigned int fd)
{ {
struct files_struct *files = current->files; struct files_struct *files = current->files;
struct fdtable *fdt; struct fdtable *fdt;
int res; bool res;
rcu_read_lock(); rcu_read_lock();
fdt = files_fdtable(files); fdt = files_fdtable(files);
res = FD_ISSET(fd, fdt->close_on_exec); res = close_on_exec(fd, fdt);
rcu_read_unlock(); rcu_read_unlock();
return res; return res;
} }
...@@ -90,15 +90,15 @@ SYSCALL_DEFINE3(dup3, unsigned int, oldfd, unsigned int, newfd, int, flags) ...@@ -90,15 +90,15 @@ SYSCALL_DEFINE3(dup3, unsigned int, oldfd, unsigned int, newfd, int, flags)
err = -EBUSY; err = -EBUSY;
fdt = files_fdtable(files); fdt = files_fdtable(files);
tofree = fdt->fd[newfd]; tofree = fdt->fd[newfd];
if (!tofree && FD_ISSET(newfd, fdt->open_fds)) if (!tofree && fd_is_open(newfd, fdt))
goto out_unlock; goto out_unlock;
get_file(file); get_file(file);
rcu_assign_pointer(fdt->fd[newfd], file); rcu_assign_pointer(fdt->fd[newfd], file);
FD_SET(newfd, fdt->open_fds); __set_open_fd(newfd, fdt);
if (flags & O_CLOEXEC) if (flags & O_CLOEXEC)
FD_SET(newfd, fdt->close_on_exec); __set_close_on_exec(newfd, fdt);
else else
FD_CLR(newfd, fdt->close_on_exec); __clear_close_on_exec(newfd, fdt);
spin_unlock(&files->file_lock); spin_unlock(&files->file_lock);
if (tofree) if (tofree)
......
...@@ -366,7 +366,7 @@ struct files_struct *dup_fd(struct files_struct *oldf, int *errorp) ...@@ -366,7 +366,7 @@ struct files_struct *dup_fd(struct files_struct *oldf, int *errorp)
* is partway through open(). So make sure that this * is partway through open(). So make sure that this
* fd is available to the new process. * fd is available to the new process.
*/ */
FD_CLR(open_files - i, new_fdt->open_fds); __clear_open_fd(open_files - i, new_fdt);
} }
rcu_assign_pointer(*new_fds++, f); rcu_assign_pointer(*new_fds++, f);
} }
...@@ -460,11 +460,11 @@ int alloc_fd(unsigned start, unsigned flags) ...@@ -460,11 +460,11 @@ int alloc_fd(unsigned start, unsigned flags)
if (start <= files->next_fd) if (start <= files->next_fd)
files->next_fd = fd + 1; files->next_fd = fd + 1;
FD_SET(fd, fdt->open_fds); __set_open_fd(fd, fdt);
if (flags & O_CLOEXEC) if (flags & O_CLOEXEC)
FD_SET(fd, fdt->close_on_exec); __set_close_on_exec(fd, fdt);
else else
FD_CLR(fd, fdt->close_on_exec); __clear_close_on_exec(fd, fdt);
error = fd; error = fd;
#if 1 #if 1
/* Sanity check */ /* Sanity check */
......
...@@ -836,7 +836,7 @@ EXPORT_SYMBOL(dentry_open); ...@@ -836,7 +836,7 @@ EXPORT_SYMBOL(dentry_open);
static void __put_unused_fd(struct files_struct *files, unsigned int fd) static void __put_unused_fd(struct files_struct *files, unsigned int fd)
{ {
struct fdtable *fdt = files_fdtable(files); struct fdtable *fdt = files_fdtable(files);
__FD_CLR(fd, fdt->open_fds); __clear_open_fd(fd, fdt);
if (fd < files->next_fd) if (fd < files->next_fd)
files->next_fd = fd; files->next_fd = fd;
} }
...@@ -1080,7 +1080,7 @@ SYSCALL_DEFINE1(close, unsigned int, fd) ...@@ -1080,7 +1080,7 @@ SYSCALL_DEFINE1(close, unsigned int, fd)
if (!filp) if (!filp)
goto out_unlock; goto out_unlock;
rcu_assign_pointer(fdt->fd[fd], NULL); rcu_assign_pointer(fdt->fd[fd], NULL);
FD_CLR(fd, fdt->close_on_exec); __clear_close_on_exec(fd, fdt);
__put_unused_fd(files, fd); __put_unused_fd(files, fd);
spin_unlock(&files->file_lock); spin_unlock(&files->file_lock);
retval = filp_close(filp, files); retval = filp_close(filp, files);
......
...@@ -1754,7 +1754,7 @@ static int proc_fd_info(struct inode *inode, struct path *path, char *info) ...@@ -1754,7 +1754,7 @@ static int proc_fd_info(struct inode *inode, struct path *path, char *info)
fdt = files_fdtable(files); fdt = files_fdtable(files);
f_flags = file->f_flags & ~O_CLOEXEC; f_flags = file->f_flags & ~O_CLOEXEC;
if (FD_ISSET(fd, fdt->close_on_exec)) if (close_on_exec(fd, fdt))
f_flags |= O_CLOEXEC; f_flags |= O_CLOEXEC;
if (path) { if (path) {
......
...@@ -38,6 +38,36 @@ struct fdtable { ...@@ -38,6 +38,36 @@ struct fdtable {
struct fdtable *next; struct fdtable *next;
}; };
static inline void __set_close_on_exec(int fd, struct fdtable *fdt)
{
FD_SET(fd, fdt->close_on_exec);
}
static inline void __clear_close_on_exec(int fd, struct fdtable *fdt)
{
FD_CLR(fd, fdt->close_on_exec);
}
static inline bool close_on_exec(int fd, const struct fdtable *fdt)
{
return FD_ISSET(fd, fdt->close_on_exec);
}
static inline void __set_open_fd(int fd, struct fdtable *fdt)
{
FD_SET(fd, fdt->open_fds);
}
static inline void __clear_open_fd(int fd, struct fdtable *fdt)
{
FD_CLR(fd, fdt->open_fds);
}
static inline bool fd_is_open(int fd, const struct fdtable *fdt)
{
return FD_ISSET(fd, fdt->open_fds);
}
/* /*
* Open file table structure * Open file table structure
*/ */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment