diff options
| author | Oleg Nesterov <oleg@redhat.com> | 2013-07-26 13:25:43 -0400 |
|---|---|---|
| committer | Greg Kroah-Hartman <gregkh@linuxfoundation.org> | 2013-08-29 12:47:33 -0400 |
| commit | b86d0ba62decb830aed2fa525e7557857d3199f2 (patch) | |
| tree | 8c4a067276d1fd1801ebddf3a202d21945deccf0 /kernel/trace | |
| parent | 70c91fb9a74e7937ef76a542a86c1551d5df4281 (diff) | |
tracing: Change f_start() to take event_mutex and verify i_private != NULL
commit c5a44a1200c6eda2202434f25325e8ad19533fca upstream.
trace_format_open() and trace_format_seq_ops are racy, nothing
protects ftrace_event_call from trace_remove_event_call().
Change f_start() to take event_mutex and verify i_private != NULL,
change f_stop() to drop this lock.
This fixes nothing, but now we can change debugfs_remove("format")
callers to nullify ->i_private and fix the the problem.
Note: the usage of event_mutex is sub-optimal but simple, we can
change this later.
Link: http://lkml.kernel.org/r/20130726172543.GA3622@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.c | 13 |
1 files changed, 9 insertions, 4 deletions
diff --git a/kernel/trace/trace_events.c b/kernel/trace/trace_events.c index 285589354cb7..09782d623c93 100644 --- a/kernel/trace/trace_events.c +++ b/kernel/trace/trace_events.c | |||
| @@ -838,7 +838,7 @@ enum { | |||
| 838 | 838 | ||
| 839 | static void *f_next(struct seq_file *m, void *v, loff_t *pos) | 839 | static void *f_next(struct seq_file *m, void *v, loff_t *pos) |
| 840 | { | 840 | { |
| 841 | struct ftrace_event_call *call = m->private; | 841 | struct ftrace_event_call *call = event_file_data(m->private); |
| 842 | struct ftrace_event_field *field; | 842 | struct ftrace_event_field *field; |
| 843 | struct list_head *common_head = &ftrace_common_fields; | 843 | struct list_head *common_head = &ftrace_common_fields; |
| 844 | struct list_head *head = trace_get_fields(call); | 844 | struct list_head *head = trace_get_fields(call); |
| @@ -882,6 +882,11 @@ static void *f_start(struct seq_file *m, loff_t *pos) | |||
| 882 | loff_t l = 0; | 882 | loff_t l = 0; |
| 883 | void *p; | 883 | void *p; |
| 884 | 884 | ||
| 885 | /* ->stop() is called even if ->start() fails */ | ||
| 886 | mutex_lock(&event_mutex); | ||
| 887 | if (!event_file_data(m->private)) | ||
| 888 | return ERR_PTR(-ENODEV); | ||
| 889 | |||
| 885 | /* Start by showing the header */ | 890 | /* Start by showing the header */ |
| 886 | if (!*pos) | 891 | if (!*pos) |
| 887 | return (void *)FORMAT_HEADER; | 892 | return (void *)FORMAT_HEADER; |
| @@ -896,7 +901,7 @@ static void *f_start(struct seq_file *m, loff_t *pos) | |||
| 896 | 901 | ||
| 897 | static int f_show(struct seq_file *m, void *v) | 902 | static int f_show(struct seq_file *m, void *v) |
| 898 | { | 903 | { |
| 899 | struct ftrace_event_call *call = m->private; | 904 | struct ftrace_event_call *call = event_file_data(m->private); |
| 900 | struct ftrace_event_field *field; | 905 | struct ftrace_event_field *field; |
| 901 | const char *array_descriptor; | 906 | const char *array_descriptor; |
| 902 | 907 | ||
| @@ -947,6 +952,7 @@ static int f_show(struct seq_file *m, void *v) | |||
| 947 | 952 | ||
| 948 | static void f_stop(struct seq_file *m, void *p) | 953 | static void f_stop(struct seq_file *m, void *p) |
| 949 | { | 954 | { |
| 955 | mutex_unlock(&event_mutex); | ||
| 950 | } | 956 | } |
| 951 | 957 | ||
| 952 | static const struct seq_operations trace_format_seq_ops = { | 958 | static const struct seq_operations trace_format_seq_ops = { |
| @@ -958,7 +964,6 @@ static const struct seq_operations trace_format_seq_ops = { | |||
| 958 | 964 | ||
| 959 | static int trace_format_open(struct inode *inode, struct file *file) | 965 | static int trace_format_open(struct inode *inode, struct file *file) |
| 960 | { | 966 | { |
| 961 | struct ftrace_event_call *call = inode->i_private; | ||
| 962 | struct seq_file *m; | 967 | struct seq_file *m; |
| 963 | int ret; | 968 | int ret; |
| 964 | 969 | ||
| @@ -967,7 +972,7 @@ static int trace_format_open(struct inode *inode, struct file *file) | |||
| 967 | return ret; | 972 | return ret; |
| 968 | 973 | ||
| 969 | m = file->private_data; | 974 | m = file->private_data; |
| 970 | m->private = call; | 975 | m->private = file; |
| 971 | 976 | ||
| 972 | return 0; | 977 | return 0; |
| 973 | } | 978 | } |
