diff options
author | Jan Kara <jack@suse.cz> | 2014-08-06 19:03:28 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2014-08-06 21:01:12 -0400 |
commit | 5838d4442bd5971687b72221736222637e03140d (patch) | |
tree | e639967a7baf1b1c6a0d1b5e0babc8083c7faad4 /fs/notify/fanotify/fanotify.c | |
parent | 8ba8fa917093510cdcb4ec8ff8b9603e1b525658 (diff) |
fanotify: fix double free of pending permission events
Commit 85816794240b ("fanotify: Fix use after free for permission
events") introduced a double free issue for permission events which are
pending in group's notification queue while group is being destroyed.
These events are freed from fanotify_handle_event() but they are not
removed from groups notification queue and thus they get freed again
from fsnotify_flush_notify().
Fix the problem by removing permission events from notification queue
before freeing them if we skip processing access response. Also expand
comments in fanotify_release() to explain group shutdown in detail.
Fixes: 85816794240b9659e66e4d9b0df7c6e814e5f603
Signed-off-by: Jan Kara <jack@suse.cz>
Reported-by: Douglas Leeder <douglas.leeder@sophos.com>
Tested-by: Douglas Leeder <douglas.leeder@sophos.com>
Reported-by: Heinrich Schuchard <xypron.glpk@gmx.de>
Cc: <stable@vger.kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'fs/notify/fanotify/fanotify.c')
-rw-r--r-- | fs/notify/fanotify/fanotify.c | 9 |
1 files changed, 8 insertions, 1 deletions
diff --git a/fs/notify/fanotify/fanotify.c b/fs/notify/fanotify/fanotify.c index fdeb36b70c65..30d3addfad75 100644 --- a/fs/notify/fanotify/fanotify.c +++ b/fs/notify/fanotify/fanotify.c | |||
@@ -70,8 +70,15 @@ static int fanotify_get_response(struct fsnotify_group *group, | |||
70 | wait_event(group->fanotify_data.access_waitq, event->response || | 70 | wait_event(group->fanotify_data.access_waitq, event->response || |
71 | atomic_read(&group->fanotify_data.bypass_perm)); | 71 | atomic_read(&group->fanotify_data.bypass_perm)); |
72 | 72 | ||
73 | if (!event->response) /* bypass_perm set */ | 73 | if (!event->response) { /* bypass_perm set */ |
74 | /* | ||
75 | * Event was canceled because group is being destroyed. Remove | ||
76 | * it from group's event list because we are responsible for | ||
77 | * freeing the permission event. | ||
78 | */ | ||
79 | fsnotify_remove_event(group, &event->fae.fse); | ||
74 | return 0; | 80 | return 0; |
81 | } | ||
75 | 82 | ||
76 | /* userspace responded, convert to something usable */ | 83 | /* userspace responded, convert to something usable */ |
77 | switch (event->response) { | 84 | switch (event->response) { |