flags, umask);
 }
 
+static void fanotify_mark_add_ignored_mask(struct fsnotify_mark *fsn_mark,
+                                          __u32 mask, unsigned int flags,
+                                          __u32 *removed)
+{
+       fsn_mark->ignored_mask |= mask;
+
+       /*
+        * Setting FAN_MARK_IGNORED_SURV_MODIFY for the first time may lead to
+        * the removal of the FS_MODIFY bit in calculated mask if it was set
+        * because of an ignored mask that is now going to survive FS_MODIFY.
+        */
+       if ((flags & FAN_MARK_IGNORED_SURV_MODIFY) &&
+           !(fsn_mark->flags & FSNOTIFY_MARK_FLAG_IGNORED_SURV_MODIFY)) {
+               fsn_mark->flags |= FSNOTIFY_MARK_FLAG_IGNORED_SURV_MODIFY;
+               if (!(fsn_mark->mask & FS_MODIFY))
+                       *removed = FS_MODIFY;
+       }
+}
+
 static __u32 fanotify_mark_add_to_mask(struct fsnotify_mark *fsn_mark,
-                                      __u32 mask, unsigned int flags)
+                                      __u32 mask, unsigned int flags,
+                                      __u32 *removed)
 {
        __u32 oldmask, newmask;
 
        if (!(flags & FAN_MARK_IGNORED_MASK)) {
                fsn_mark->mask |= mask;
        } else {
-               fsn_mark->ignored_mask |= mask;
-               if (flags & FAN_MARK_IGNORED_SURV_MODIFY)
-                       fsn_mark->flags |= FSNOTIFY_MARK_FLAG_IGNORED_SURV_MODIFY;
+               fanotify_mark_add_ignored_mask(fsn_mark, mask, flags, removed);
        }
        newmask = fsnotify_calc_mask(fsn_mark);
        spin_unlock(&fsn_mark->lock);
                             __kernel_fsid_t *fsid)
 {
        struct fsnotify_mark *fsn_mark;
-       __u32 added;
+       __u32 added, removed = 0;
        int ret = 0;
 
        mutex_lock(&group->mark_mutex);
                        goto out;
        }
 
-       added = fanotify_mark_add_to_mask(fsn_mark, mask, flags);
-       if (added & ~fsnotify_conn_mask(fsn_mark->connector))
+       added = fanotify_mark_add_to_mask(fsn_mark, mask, flags, &removed);
+       if (removed || (added & ~fsnotify_conn_mask(fsn_mark->connector)))
                fsnotify_recalc_mask(fsn_mark->connector);
 
 out:
 
 
 
        /*
-        * if this is a modify event we may need to clear the ignored masks
-        * otherwise return if none of the marks care about this type of event.
+        * If this is a modify event we may need to clear some ignored masks.
+        * In that case, the object with ignored masks will have the FS_MODIFY
+        * event in its mask.
+        * Otherwise, return if none of the marks care about this type of event.
         */
        test_mask = (mask & ALL_FSNOTIFY_EVENTS);
-       if (!(mask & FS_MODIFY) && !(test_mask & marks_mask))
+       if (!(test_mask & marks_mask))
                return 0;
 
        iter_info.srcu_idx = srcu_read_lock(&fsnotify_mark_srcu);