diff options
| author | Steven Rostedt (Red Hat) <rostedt@goodmis.org> | 2013-12-21 21:55:17 -0500 |
|---|---|---|
| committer | Steven Rostedt <rostedt@goodmis.org> | 2014-01-02 16:17:22 -0500 |
| commit | d8a30f20347a60a796a5221e07711c0d30d42dc3 (patch) | |
| tree | eed4dc37ddf15b981194a573701522605b58e69f /kernel/trace | |
| parent | 098c879e1f2d6ee7afbfe959f6b04070065cec90 (diff) | |
tracing: Fix rcu handling of event_trigger_data filter field
The filter field of the event_trigger_data structure is protected under
RCU sched locks. It was not annotated as such, and after doing so,
sparse pointed out several locations that required fix ups.
Reported-by: kbuild test robot <fengguang.wu@intel.com>
Tested-by: Tom Zanussi <tom.zanussi@linux.intel.com>
Acked-by: Tom Zanussi <tom.zanussi@linux.intel.com>
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
Diffstat (limited to 'kernel/trace')
| -rw-r--r-- | kernel/trace/trace.h | 2 | ||||
| -rw-r--r-- | kernel/trace/trace_events_trigger.c | 8 |
2 files changed, 6 insertions, 4 deletions
diff --git a/kernel/trace/trace.h b/kernel/trace/trace.h index 0380cab0af9b..02b592f2d4b7 100644 --- a/kernel/trace/trace.h +++ b/kernel/trace/trace.h | |||
| @@ -1057,7 +1057,7 @@ struct event_trigger_data { | |||
| 1057 | int ref; | 1057 | int ref; |
| 1058 | struct event_trigger_ops *ops; | 1058 | struct event_trigger_ops *ops; |
| 1059 | struct event_command *cmd_ops; | 1059 | struct event_command *cmd_ops; |
| 1060 | struct event_filter *filter; | 1060 | struct event_filter __rcu *filter; |
| 1061 | char *filter_str; | 1061 | char *filter_str; |
| 1062 | void *private_data; | 1062 | void *private_data; |
| 1063 | struct list_head list; | 1063 | struct list_head list; |
diff --git a/kernel/trace/trace_events_trigger.c b/kernel/trace/trace_events_trigger.c index 12ac8a5d4d95..f6dd115b958d 100644 --- a/kernel/trace/trace_events_trigger.c +++ b/kernel/trace/trace_events_trigger.c | |||
| @@ -67,6 +67,7 @@ event_triggers_call(struct ftrace_event_file *file, void *rec) | |||
| 67 | { | 67 | { |
| 68 | struct event_trigger_data *data; | 68 | struct event_trigger_data *data; |
| 69 | enum event_trigger_type tt = ETT_NONE; | 69 | enum event_trigger_type tt = ETT_NONE; |
| 70 | struct event_filter *filter; | ||
| 70 | 71 | ||
| 71 | if (list_empty(&file->triggers)) | 72 | if (list_empty(&file->triggers)) |
| 72 | return tt; | 73 | return tt; |
| @@ -76,7 +77,8 @@ event_triggers_call(struct ftrace_event_file *file, void *rec) | |||
| 76 | data->ops->func(data); | 77 | data->ops->func(data); |
| 77 | continue; | 78 | continue; |
| 78 | } | 79 | } |
| 79 | if (data->filter && !filter_match_preds(data->filter, rec)) | 80 | filter = rcu_dereference(data->filter); |
| 81 | if (filter && !filter_match_preds(filter, rec)) | ||
| 80 | continue; | 82 | continue; |
| 81 | if (data->cmd_ops->post_trigger) { | 83 | if (data->cmd_ops->post_trigger) { |
| 82 | tt |= data->cmd_ops->trigger_type; | 84 | tt |= data->cmd_ops->trigger_type; |
| @@ -703,7 +705,7 @@ static int set_trigger_filter(char *filter_str, | |||
| 703 | if (ret) | 705 | if (ret) |
| 704 | goto out; | 706 | goto out; |
| 705 | assign: | 707 | assign: |
| 706 | tmp = data->filter; | 708 | tmp = rcu_access_pointer(data->filter); |
| 707 | 709 | ||
| 708 | rcu_assign_pointer(data->filter, filter); | 710 | rcu_assign_pointer(data->filter, filter); |
| 709 | 711 | ||
| @@ -719,7 +721,7 @@ static int set_trigger_filter(char *filter_str, | |||
| 719 | if (filter_str) { | 721 | if (filter_str) { |
| 720 | data->filter_str = kstrdup(filter_str, GFP_KERNEL); | 722 | data->filter_str = kstrdup(filter_str, GFP_KERNEL); |
| 721 | if (!data->filter_str) { | 723 | if (!data->filter_str) { |
| 722 | free_event_filter(data->filter); | 724 | free_event_filter(rcu_access_pointer(data->filter)); |
| 723 | data->filter = NULL; | 725 | data->filter = NULL; |
| 724 | ret = -ENOMEM; | 726 | ret = -ENOMEM; |
| 725 | } | 727 | } |
