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 | |
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>
-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 | } |