diff options
author | Dipankar Sarma <dipankar@in.ibm.com> | 2005-09-09 16:04:13 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@g5.osdl.org> | 2005-09-09 16:57:55 -0400 |
commit | ab2af1f5005069321c5d130f09cce577b03f43ef (patch) | |
tree | 73a70ba486f522cd9eeeef376ede2b5a1c1b473b /fs/open.c | |
parent | 6e72ad2c581de121cc7e772469e2a8f6b1fd4379 (diff) |
[PATCH] files: files struct with RCU
Patch to eliminate struct files_struct.file_lock spinlock on the reader side
and use rcu refcounting rcuref_xxx api for the f_count refcounter. The
updates to the fdtable are done by allocating a new fdtable structure and
setting files->fdt to point to the new structure. The fdtable structure is
protected by RCU thereby allowing lock-free lookup. For fd arrays/sets that
are vmalloced, we use keventd to free them since RCU callbacks can't sleep. A
global list of fdtable to be freed is not scalable, so we use a per-cpu list.
If keventd is already handling the current cpu's work, we use a timer to defer
queueing of that work.
Since the last publication, this patch has been re-written to avoid using
explicit memory barriers and use rcu_assign_pointer(), rcu_dereference()
premitives instead. This required that the fd information is kept in a
separate structure (fdtable) and updated atomically.
Signed-off-by: Dipankar Sarma <dipankar@in.ibm.com>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'fs/open.c')
-rw-r--r-- | fs/open.c | 8 |
1 files changed, 4 insertions, 4 deletions
@@ -24,6 +24,7 @@ | |||
24 | #include <linux/personality.h> | 24 | #include <linux/personality.h> |
25 | #include <linux/pagemap.h> | 25 | #include <linux/pagemap.h> |
26 | #include <linux/syscalls.h> | 26 | #include <linux/syscalls.h> |
27 | #include <linux/rcupdate.h> | ||
27 | 28 | ||
28 | #include <asm/unistd.h> | 29 | #include <asm/unistd.h> |
29 | 30 | ||
@@ -930,9 +931,8 @@ void fastcall fd_install(unsigned int fd, struct file * file) | |||
930 | struct fdtable *fdt; | 931 | struct fdtable *fdt; |
931 | spin_lock(&files->file_lock); | 932 | spin_lock(&files->file_lock); |
932 | fdt = files_fdtable(files); | 933 | fdt = files_fdtable(files); |
933 | if (unlikely(fdt->fd[fd] != NULL)) | 934 | BUG_ON(fdt->fd[fd] != NULL); |
934 | BUG(); | 935 | rcu_assign_pointer(fdt->fd[fd], file); |
935 | fdt->fd[fd] = file; | ||
936 | spin_unlock(&files->file_lock); | 936 | spin_unlock(&files->file_lock); |
937 | } | 937 | } |
938 | 938 | ||
@@ -1024,7 +1024,7 @@ asmlinkage long sys_close(unsigned int fd) | |||
1024 | filp = fdt->fd[fd]; | 1024 | filp = fdt->fd[fd]; |
1025 | if (!filp) | 1025 | if (!filp) |
1026 | goto out_unlock; | 1026 | goto out_unlock; |
1027 | fdt->fd[fd] = NULL; | 1027 | rcu_assign_pointer(fdt->fd[fd], NULL); |
1028 | FD_CLR(fd, fdt->close_on_exec); | 1028 | FD_CLR(fd, fdt->close_on_exec); |
1029 | __put_unused_fd(files, fd); | 1029 | __put_unused_fd(files, fd); |
1030 | spin_unlock(&files->file_lock); | 1030 | spin_unlock(&files->file_lock); |