diff options
| -rw-r--r-- | fs/dnotify.c | 11 |
1 files changed, 11 insertions, 0 deletions
diff --git a/fs/dnotify.c b/fs/dnotify.c index 28d01ed66de0..eaecc4cfe540 100644 --- a/fs/dnotify.c +++ b/fs/dnotify.c | |||
| @@ -20,6 +20,7 @@ | |||
| 20 | #include <linux/init.h> | 20 | #include <linux/init.h> |
| 21 | #include <linux/spinlock.h> | 21 | #include <linux/spinlock.h> |
| 22 | #include <linux/slab.h> | 22 | #include <linux/slab.h> |
| 23 | #include <linux/file.h> | ||
| 23 | 24 | ||
| 24 | int dir_notify_enable __read_mostly = 1; | 25 | int dir_notify_enable __read_mostly = 1; |
| 25 | 26 | ||
| @@ -66,6 +67,7 @@ int fcntl_dirnotify(int fd, struct file *filp, unsigned long arg) | |||
| 66 | struct dnotify_struct **prev; | 67 | struct dnotify_struct **prev; |
| 67 | struct inode *inode; | 68 | struct inode *inode; |
| 68 | fl_owner_t id = current->files; | 69 | fl_owner_t id = current->files; |
| 70 | struct file *f; | ||
| 69 | int error = 0; | 71 | int error = 0; |
| 70 | 72 | ||
| 71 | if ((arg & ~DN_MULTISHOT) == 0) { | 73 | if ((arg & ~DN_MULTISHOT) == 0) { |
| @@ -92,6 +94,15 @@ int fcntl_dirnotify(int fd, struct file *filp, unsigned long arg) | |||
| 92 | prev = &odn->dn_next; | 94 | prev = &odn->dn_next; |
| 93 | } | 95 | } |
| 94 | 96 | ||
| 97 | rcu_read_lock(); | ||
| 98 | f = fcheck(fd); | ||
| 99 | rcu_read_unlock(); | ||
| 100 | /* we'd lost the race with close(), sod off silently */ | ||
| 101 | /* note that inode->i_lock prevents reordering problems | ||
| 102 | * between accesses to descriptor table and ->i_dnotify */ | ||
| 103 | if (f != filp) | ||
| 104 | goto out_free; | ||
| 105 | |||
| 95 | error = __f_setown(filp, task_pid(current), PIDTYPE_PID, 0); | 106 | error = __f_setown(filp, task_pid(current), PIDTYPE_PID, 0); |
| 96 | if (error) | 107 | if (error) |
| 97 | goto out_free; | 108 | goto out_free; |
