Merge branch 'fix/misc' into for-linus
[pandora-kernel.git] / fs / notify / inotify / inotify_user.c
index 982a412..ff27a29 100644 (file)
@@ -296,12 +296,15 @@ static int inotify_fasync(int fd, struct file *file, int on)
 static int inotify_release(struct inode *ignored, struct file *file)
 {
        struct fsnotify_group *group = file->private_data;
+       struct user_struct *user = group->inotify_data.user;
 
        fsnotify_clear_marks_by_group(group);
 
        /* free this group, matching get was inotify_init->fsnotify_obtain_group */
        fsnotify_put_group(group);
 
+       atomic_dec(&user->inotify_devs);
+
        return 0;
 }
 
@@ -363,39 +366,17 @@ static int inotify_find_inode(const char __user *dirname, struct path *path, uns
 }
 
 /*
- * When, for whatever reason, inotify is done with a mark (or what used to be a
- * watch) we need to remove that watch from the idr and we need to send IN_IGNORED
- * for the given wd.
- *
- * There is a bit of recursion here.  The loop looks like:
- *     inotify_destroy_mark_entry -> fsnotify_destroy_mark_by_entry ->
- *     inotify_freeing_mark -> inotify_destory_mark_entry -> restart
- * But the loop is broken in 2 places.  fsnotify_destroy_mark_by_entry sets
- * entry->group = NULL before the call to inotify_freeing_mark, so the if (egroup)
- * test below will not call back to fsnotify again.  But even if that test wasn't
- * there this would still be safe since fsnotify_destroy_mark_by_entry() is
- * safe from recursion.
+ * Send IN_IGNORED for this wd, remove this wd from the idr, and drop the
+ * internal reference help on the mark because it is in the idr.
  */
-void inotify_destroy_mark_entry(struct fsnotify_mark_entry *entry, struct fsnotify_group *group)
+void inotify_ignored_and_remove_idr(struct fsnotify_mark_entry *entry,
+                                   struct fsnotify_group *group)
 {
        struct inotify_inode_mark_entry *ientry;
        struct inotify_event_private_data *event_priv;
        struct fsnotify_event_private_data *fsn_event_priv;
-       struct fsnotify_group *egroup;
        struct idr *idr;
 
-       spin_lock(&entry->lock);
-       egroup = entry->group;
-
-       /* if egroup we aren't really done and something might still send events
-        * for this inode, on the callback we'll send the IN_IGNORED */
-       if (egroup) {
-               spin_unlock(&entry->lock);
-               fsnotify_destroy_mark_by_entry(entry);
-               return;
-       }
-       spin_unlock(&entry->lock);
-
        ientry = container_of(entry, struct inotify_inode_mark_entry, fsn_entry);
 
        event_priv = kmem_cache_alloc(event_priv_cachep, GFP_KERNEL);
@@ -699,7 +680,7 @@ SYSCALL_DEFINE2(inotify_rm_watch, int, fd, __s32, wd)
        fsnotify_get_mark(entry);
        spin_unlock(&group->inotify_data.idr_lock);
 
-       inotify_destroy_mark_entry(entry, group);
+       fsnotify_destroy_mark_by_entry(entry);
        fsnotify_put_mark(entry);
 
 out: