aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/trace
diff options
context:
space:
mode:
authorOleg Nesterov <oleg@redhat.com>2013-07-28 14:35:27 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2013-08-29 12:47:34 -0400
commit012dc156d6af49e02a30fcba7d688e251608d97c (patch)
tree9df8015705569ced239b24ae56872eabc2b15054 /kernel/trace
parentc6febdf258313a120acd1298dd1dd41442131a37 (diff)
tracing: Change remove_event_file_dir() to clear "d_subdirs"->i_private
commit bf682c3159c4d298d1126a56793ed3f5e80395f7 upstream. Change remove_event_file_dir() to clear ->i_private for every file we are going to remove. We need to check file->dir != NULL because event_create_dir() can fail. debugfs_remove_recursive(NULL) is fine but the patch moves it under the same check anyway for readability. spin_lock(d_lock) and "d_inode != NULL" check are not needed afaics, but I do not understand this code enough. tracing_open_generic_file() and tracing_release_generic_file() can go away, ftrace_enable_fops and ftrace_event_filter_fops() use tracing_open_generic() but only to check tracing_disabled. This fixes all races with event_remove() or instance_delete(). f_op->read/write/whatever can never use the freed file/call, all event/* files were changed to check and use ->i_private under event_mutex. Note: this doesn't not fix other problems, event_remove() can destroy the active ftrace_event_call, we need more changes but those changes are completely orthogonal. Link: http://lkml.kernel.org/r/20130728183527.GB16723@redhat.com Reviewed-by: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com> Signed-off-by: Oleg Nesterov <oleg@redhat.com> Signed-off-by: Steven Rostedt <rostedt@goodmis.org> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'kernel/trace')
-rw-r--r--kernel/trace/trace_events.c47
1 files changed, 15 insertions, 32 deletions
diff --git a/kernel/trace/trace_events.c b/kernel/trace/trace_events.c
index 422f99dfe699..0bff8aaf581b 100644
--- a/kernel/trace/trace_events.c
+++ b/kernel/trace/trace_events.c
@@ -426,42 +426,26 @@ static void *event_file_data(struct file *filp)
426 426
427static void remove_event_file_dir(struct ftrace_event_file *file) 427static void remove_event_file_dir(struct ftrace_event_file *file)
428{ 428{
429 struct dentry *dir = file->dir;
430 struct dentry *child;
431
432 if (dir) {
433 spin_lock(&dir->d_lock); /* probably unneeded */
434 list_for_each_entry(child, &dir->d_subdirs, d_u.d_child) {
435 if (child->d_inode) /* probably unneeded */
436 child->d_inode->i_private = NULL;
437 }
438 spin_unlock(&dir->d_lock);
439
440 debugfs_remove_recursive(dir);
441 }
442
429 list_del(&file->list); 443 list_del(&file->list);
430 debugfs_remove_recursive(file->dir);
431 remove_subsystem(file->system); 444 remove_subsystem(file->system);
432 kmem_cache_free(file_cachep, file); 445 kmem_cache_free(file_cachep, file);
433} 446}
434 447
435/* 448/*
436 * Open and update trace_array ref count.
437 * Must have the current trace_array passed to it.
438 */
439static int tracing_open_generic_file(struct inode *inode, struct file *filp)
440{
441 struct ftrace_event_file *file = inode->i_private;
442 struct trace_array *tr = file->tr;
443 int ret;
444
445 if (trace_array_get(tr) < 0)
446 return -ENODEV;
447
448 ret = tracing_open_generic(inode, filp);
449 if (ret < 0)
450 trace_array_put(tr);
451 return ret;
452}
453
454static int tracing_release_generic_file(struct inode *inode, struct file *filp)
455{
456 struct ftrace_event_file *file = inode->i_private;
457 struct trace_array *tr = file->tr;
458
459 trace_array_put(tr);
460
461 return 0;
462}
463
464/*
465 * __ftrace_set_clr_event(NULL, NULL, NULL, set) will set/unset all events. 449 * __ftrace_set_clr_event(NULL, NULL, NULL, set) will set/unset all events.
466 */ 450 */
467static int 451static int
@@ -1303,10 +1287,9 @@ static const struct file_operations ftrace_set_event_fops = {
1303}; 1287};
1304 1288
1305static const struct file_operations ftrace_enable_fops = { 1289static const struct file_operations ftrace_enable_fops = {
1306 .open = tracing_open_generic_file, 1290 .open = tracing_open_generic,
1307 .read = event_enable_read, 1291 .read = event_enable_read,
1308 .write = event_enable_write, 1292 .write = event_enable_write,
1309 .release = tracing_release_generic_file,
1310 .llseek = default_llseek, 1293 .llseek = default_llseek,
1311}; 1294};
1312 1295