fanotify: prepare for implicit event flags in mark mask

So far, all flags that can be set in an fanotify mark mask can be set
explicitly by a call to fanotify_mark(2).

Prepare for defining implicit event flags that cannot be set by user with
fanotify_mark(2), similar to how inotify/dnotify implicitly set the
FS_EVENT_ON_CHILD flag.

Implicit event flags cannot be removed by user and mark gets destroyed
when only implicit event flags remain in the mask.

Link: https://lore.kernel.org/r/20200716084230.30611-7-amir73il@gmail.com
Signed-off-by: Amir Goldstein <amir73il@gmail.com>
Signed-off-by: Jan Kara <jack@suse.cz>
This commit is contained in:
Amir Goldstein 2020-07-16 11:42:14 +03:00 committed by Jan Kara
parent 3ef8665366
commit 4ed6814a91

View File

@ -656,12 +656,13 @@ out:
} }
static __u32 fanotify_mark_remove_from_mask(struct fsnotify_mark *fsn_mark, static __u32 fanotify_mark_remove_from_mask(struct fsnotify_mark *fsn_mark,
__u32 mask, __u32 mask, unsigned int flags,
unsigned int flags, __u32 umask, int *destroy)
int *destroy)
{ {
__u32 oldmask = 0; __u32 oldmask = 0;
/* umask bits cannot be removed by user */
mask &= ~umask;
spin_lock(&fsn_mark->lock); spin_lock(&fsn_mark->lock);
if (!(flags & FAN_MARK_IGNORED_MASK)) { if (!(flags & FAN_MARK_IGNORED_MASK)) {
oldmask = fsn_mark->mask; oldmask = fsn_mark->mask;
@ -669,7 +670,13 @@ static __u32 fanotify_mark_remove_from_mask(struct fsnotify_mark *fsn_mark,
} else { } else {
fsn_mark->ignored_mask &= ~mask; fsn_mark->ignored_mask &= ~mask;
} }
*destroy = !(fsn_mark->mask | fsn_mark->ignored_mask); /*
* We need to keep the mark around even if remaining mask cannot
* result in any events (e.g. mask == FAN_ONDIR) to support incremenal
* changes to the mask.
* Destroy mark when only umask bits remain.
*/
*destroy = !((fsn_mark->mask | fsn_mark->ignored_mask) & ~umask);
spin_unlock(&fsn_mark->lock); spin_unlock(&fsn_mark->lock);
return mask & oldmask; return mask & oldmask;
@ -677,7 +684,7 @@ static __u32 fanotify_mark_remove_from_mask(struct fsnotify_mark *fsn_mark,
static int fanotify_remove_mark(struct fsnotify_group *group, static int fanotify_remove_mark(struct fsnotify_group *group,
fsnotify_connp_t *connp, __u32 mask, fsnotify_connp_t *connp, __u32 mask,
unsigned int flags) unsigned int flags, __u32 umask)
{ {
struct fsnotify_mark *fsn_mark = NULL; struct fsnotify_mark *fsn_mark = NULL;
__u32 removed; __u32 removed;
@ -691,7 +698,7 @@ static int fanotify_remove_mark(struct fsnotify_group *group,
} }
removed = fanotify_mark_remove_from_mask(fsn_mark, mask, flags, removed = fanotify_mark_remove_from_mask(fsn_mark, mask, flags,
&destroy_mark); umask, &destroy_mark);
if (removed & fsnotify_conn_mask(fsn_mark->connector)) if (removed & fsnotify_conn_mask(fsn_mark->connector))
fsnotify_recalc_mask(fsn_mark->connector); fsnotify_recalc_mask(fsn_mark->connector);
if (destroy_mark) if (destroy_mark)
@ -707,25 +714,26 @@ static int fanotify_remove_mark(struct fsnotify_group *group,
static int fanotify_remove_vfsmount_mark(struct fsnotify_group *group, static int fanotify_remove_vfsmount_mark(struct fsnotify_group *group,
struct vfsmount *mnt, __u32 mask, struct vfsmount *mnt, __u32 mask,
unsigned int flags) unsigned int flags, __u32 umask)
{ {
return fanotify_remove_mark(group, &real_mount(mnt)->mnt_fsnotify_marks, return fanotify_remove_mark(group, &real_mount(mnt)->mnt_fsnotify_marks,
mask, flags); mask, flags, umask);
} }
static int fanotify_remove_sb_mark(struct fsnotify_group *group, static int fanotify_remove_sb_mark(struct fsnotify_group *group,
struct super_block *sb, __u32 mask, struct super_block *sb, __u32 mask,
unsigned int flags) unsigned int flags, __u32 umask)
{ {
return fanotify_remove_mark(group, &sb->s_fsnotify_marks, mask, flags); return fanotify_remove_mark(group, &sb->s_fsnotify_marks, mask,
flags, umask);
} }
static int fanotify_remove_inode_mark(struct fsnotify_group *group, static int fanotify_remove_inode_mark(struct fsnotify_group *group,
struct inode *inode, __u32 mask, struct inode *inode, __u32 mask,
unsigned int flags) unsigned int flags, __u32 umask)
{ {
return fanotify_remove_mark(group, &inode->i_fsnotify_marks, mask, return fanotify_remove_mark(group, &inode->i_fsnotify_marks, mask,
flags); flags, umask);
} }
static __u32 fanotify_mark_add_to_mask(struct fsnotify_mark *fsn_mark, static __u32 fanotify_mark_add_to_mask(struct fsnotify_mark *fsn_mark,
@ -1175,13 +1183,13 @@ static int do_fanotify_mark(int fanotify_fd, unsigned int flags, __u64 mask,
case FAN_MARK_REMOVE: case FAN_MARK_REMOVE:
if (mark_type == FAN_MARK_MOUNT) if (mark_type == FAN_MARK_MOUNT)
ret = fanotify_remove_vfsmount_mark(group, mnt, mask, ret = fanotify_remove_vfsmount_mark(group, mnt, mask,
flags); flags, 0);
else if (mark_type == FAN_MARK_FILESYSTEM) else if (mark_type == FAN_MARK_FILESYSTEM)
ret = fanotify_remove_sb_mark(group, mnt->mnt_sb, mask, ret = fanotify_remove_sb_mark(group, mnt->mnt_sb, mask,
flags); flags, 0);
else else
ret = fanotify_remove_inode_mark(group, inode, mask, ret = fanotify_remove_inode_mark(group, inode, mask,
flags); flags, 0);
break; break;
default: default:
ret = -EINVAL; ret = -EINVAL;