Commit 84e1ab4d authored by Eric Paris's avatar Eric Paris

fsnotify: fix ignored mask handling between inode and vfsmount marks

The interesting 2 list lockstep walking didn't quite work out if the inode
marks only had ignores and the vfsmount list requested events.  The code to
shortcut list traversal would not run the inode list since it didn't have real
event requests.  This code forces inode list traversal when a vfsmount mark
matches the event type.  Maybe we could add an i_fsnotify_ignored_mask field
to struct inode to get the shortcut back, but it doesn't seem worth it to grow
struct inode again.

I bet with the recent changes to lock the way we do now it would actually not
be a major perf hit to just drop i_fsnotify_mark_mask altogether.  But that is
for another day.
Signed-off-by: default avatarEric Paris <eparis@redhat.com>
parent 88b2dbdb
...@@ -149,8 +149,8 @@ static int send_to_group(struct inode *to_tell, struct vfsmount *mnt, ...@@ -149,8 +149,8 @@ static int send_to_group(struct inode *to_tell, struct vfsmount *mnt,
struct fsnotify_event **event) struct fsnotify_event **event)
{ {
struct fsnotify_group *group = NULL; struct fsnotify_group *group = NULL;
__u32 inode_test_mask = (mask & ~FS_EVENT_ON_CHILD); __u32 inode_test_mask = 0;
__u32 vfsmount_test_mask = (mask & ~FS_EVENT_ON_CHILD); __u32 vfsmount_test_mask = 0;
if (unlikely(!inode_mark && !vfsmount_mark)) { if (unlikely(!inode_mark && !vfsmount_mark)) {
BUG(); BUG();
...@@ -170,12 +170,14 @@ static int send_to_group(struct inode *to_tell, struct vfsmount *mnt, ...@@ -170,12 +170,14 @@ static int send_to_group(struct inode *to_tell, struct vfsmount *mnt,
/* does the inode mark tell us to do something? */ /* does the inode mark tell us to do something? */
if (inode_mark) { if (inode_mark) {
group = inode_mark->group; group = inode_mark->group;
inode_test_mask = (mask & ~FS_EVENT_ON_CHILD);
inode_test_mask &= inode_mark->mask; inode_test_mask &= inode_mark->mask;
inode_test_mask &= ~inode_mark->ignored_mask; inode_test_mask &= ~inode_mark->ignored_mask;
} }
/* does the vfsmount_mark tell us to do something? */ /* does the vfsmount_mark tell us to do something? */
if (vfsmount_mark) { if (vfsmount_mark) {
vfsmount_test_mask = (mask & ~FS_EVENT_ON_CHILD);
group = vfsmount_mark->group; group = vfsmount_mark->group;
vfsmount_test_mask &= vfsmount_mark->mask; vfsmount_test_mask &= vfsmount_mark->mask;
vfsmount_test_mask &= ~vfsmount_mark->ignored_mask; vfsmount_test_mask &= ~vfsmount_mark->ignored_mask;
...@@ -183,9 +185,12 @@ static int send_to_group(struct inode *to_tell, struct vfsmount *mnt, ...@@ -183,9 +185,12 @@ static int send_to_group(struct inode *to_tell, struct vfsmount *mnt,
vfsmount_test_mask &= ~inode_mark->ignored_mask; vfsmount_test_mask &= ~inode_mark->ignored_mask;
} }
pr_debug("%s: group=%p to_tell=%p mnt=%p mark=%p mask=%x data=%p" pr_debug("%s: group=%p to_tell=%p mnt=%p mask=%x inode_mark=%p"
" data_is=%d cookie=%d event=%p\n", __func__, group, to_tell, " inode_test_mask=%x vfsmount_mark=%p vfsmount_test_mask=%x"
mnt, inode_mark, mask, data, data_is, cookie, *event); " data=%p data_is=%d cookie=%d event=%p\n",
__func__, group, to_tell, mnt, mask, inode_mark,
inode_test_mask, vfsmount_mark, vfsmount_test_mask, data,
data_is, cookie, *event);
if (!inode_test_mask && !vfsmount_test_mask) if (!inode_test_mask && !vfsmount_test_mask)
return 0; return 0;
...@@ -214,7 +219,7 @@ static int send_to_group(struct inode *to_tell, struct vfsmount *mnt, ...@@ -214,7 +219,7 @@ static int send_to_group(struct inode *to_tell, struct vfsmount *mnt,
int fsnotify(struct inode *to_tell, __u32 mask, void *data, int data_is, int fsnotify(struct inode *to_tell, __u32 mask, void *data, int data_is,
const unsigned char *file_name, u32 cookie) const unsigned char *file_name, u32 cookie)
{ {
struct hlist_node *inode_node, *vfsmount_node; struct hlist_node *inode_node = NULL, *vfsmount_node = NULL;
struct fsnotify_mark *inode_mark = NULL, *vfsmount_mark = NULL; struct fsnotify_mark *inode_mark = NULL, *vfsmount_mark = NULL;
struct fsnotify_group *inode_group, *vfsmount_group; struct fsnotify_group *inode_group, *vfsmount_group;
struct fsnotify_event *event = NULL; struct fsnotify_event *event = NULL;
...@@ -245,19 +250,13 @@ int fsnotify(struct inode *to_tell, __u32 mask, void *data, int data_is, ...@@ -245,19 +250,13 @@ int fsnotify(struct inode *to_tell, __u32 mask, void *data, int data_is,
(test_mask & to_tell->i_fsnotify_mask)) (test_mask & to_tell->i_fsnotify_mask))
inode_node = srcu_dereference(to_tell->i_fsnotify_marks.first, inode_node = srcu_dereference(to_tell->i_fsnotify_marks.first,
&fsnotify_mark_srcu); &fsnotify_mark_srcu);
else
inode_node = NULL;
if (mnt) { if (mnt && ((mask & FS_MODIFY) ||
if ((mask & FS_MODIFY) || (test_mask & mnt->mnt_fsnotify_mask))) {
(test_mask & mnt->mnt_fsnotify_mask))
vfsmount_node = srcu_dereference(mnt->mnt_fsnotify_marks.first, vfsmount_node = srcu_dereference(mnt->mnt_fsnotify_marks.first,
&fsnotify_mark_srcu); &fsnotify_mark_srcu);
else inode_node = srcu_dereference(to_tell->i_fsnotify_marks.first,
vfsmount_node = NULL; &fsnotify_mark_srcu);
} else {
mnt = NULL;
vfsmount_node = NULL;
} }
while (inode_node || vfsmount_node) { while (inode_node || vfsmount_node) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment