aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/trace/trace_events_trigger.c
diff options
context:
space:
mode:
authorTom Zanussi <tom.zanussi@linux.intel.com>2013-10-24 09:59:24 -0400
committerSteven Rostedt <rostedt@goodmis.org>2013-12-20 18:40:22 -0500
commit85f2b08268c014e290b600ba49fa85530600eaa1 (patch)
tree911b2e24ac56434fc04e8ecafe6251536b6f06a0 /kernel/trace/trace_events_trigger.c
parent319e2e3f63c348a9b66db4667efa73178e18b17d (diff)
tracing: Add basic event trigger framework
Add a 'trigger' file for each trace event, enabling 'trace event triggers' to be set for trace events. 'trace event triggers' are patterned after the existing 'ftrace function triggers' implementation except that triggers are written to per-event 'trigger' files instead of to a single file such as the 'set_ftrace_filter' used for ftrace function triggers. The implementation is meant to be entirely separate from ftrace function triggers, in order to keep the respective implementations relatively simple and to allow them to diverge. The event trigger functionality is built on top of SOFT_DISABLE functionality. It adds a TRIGGER_MODE bit to the ftrace_event_file flags which is checked when any trace event fires. Triggers set for a particular event need to be checked regardless of whether that event is actually enabled or not - getting an event to fire even if it's not enabled is what's already implemented by SOFT_DISABLE mode, so trigger mode directly reuses that. Event trigger essentially inherit the soft disable logic in __ftrace_event_enable_disable() while adding a bit of logic and trigger reference counting via tm_ref on top of that in a new trace_event_trigger_enable_disable() function. Because the base __ftrace_event_enable_disable() code now needs to be invoked from outside trace_events.c, a wrapper is also added for those usages. The triggers for an event are actually invoked via a new function, event_triggers_call(), and code is also added to invoke them for ftrace_raw_event calls as well as syscall events. The main part of the patch creates a new trace_events_trigger.c file to contain the trace event triggers implementation. The standard open, read, and release file operations are implemented here. The open() implementation sets up for the various open modes of the 'trigger' file. It creates and attaches the trigger iterator and sets up the command parser. If opened for reading set up the trigger seq_ops. The read() implementation parses the event trigger written to the 'trigger' file, looks up the trigger command, and passes it along to that event_command's func() implementation for command-specific processing. The release() implementation does whatever cleanup is needed to release the 'trigger' file, like releasing the parser and trigger iterator, etc. A couple of functions for event command registration and unregistration are added, along with a list to add them to and a mutex to protect them, as well as an (initially empty) registration function to add the set of commands that will be added by future commits, and call to it from the trace event initialization code. also added are a couple trigger-specific data structures needed for these implementations such as a trigger iterator and a struct for trigger-specific data. A couple structs consisting mostly of function meant to be implemented in command-specific ways, event_command and event_trigger_ops, are used by the generic event trigger command implementations. They're being put into trace.h alongside the other trace_event data structures and functions, in the expectation that they'll be needed in several trace_event-related files such as trace_events_trigger.c and trace_events.c. The event_command.func() function is meant to be called by the trigger parsing code in order to add a trigger instance to the corresponding event. It essentially coordinates adding a live trigger instance to the event, and arming the triggering the event. Every event_command func() implementation essentially does the same thing for any command: - choose ops - use the value of param to choose either a number or count version of event_trigger_ops specific to the command - do the register or unregister of those ops - associate a filter, if specified, with the triggering event The reg() and unreg() ops allow command-specific implementations for event_trigger_op registration and unregistration, and the get_trigger_ops() op allows command-specific event_trigger_ops selection to be parameterized. When a trigger instance is added, the reg() op essentially adds that trigger to the triggering event and arms it, while unreg() does the opposite. The set_filter() function is used to associate a filter with the trigger - if the command doesn't specify a set_filter() implementation, the command will ignore filters. Each command has an associated trigger_type, which serves double duty, both as a unique identifier for the command as well as a value that can be used for setting a trigger mode bit during trigger invocation. The signature of func() adds a pointer to the event_command struct, used to invoke those functions, along with a command_data param that can be passed to the reg/unreg functions. This allows func() implementations to use command-specific blobs and supports code re-use. The event_trigger_ops.func() command corrsponds to the trigger 'probe' function that gets called when the triggering event is actually invoked. The other functions are used to list the trigger when needed, along with a couple mundane book-keeping functions. This also moves event_file_data() into trace.h so it can be used outside of trace_events.c. Link: http://lkml.kernel.org/r/316d95061accdee070aac8e5750afba0192fa5b9.1382622043.git.tom.zanussi@linux.intel.com Signed-off-by: Tom Zanussi <tom.zanussi@linux.intel.com> Idea-by: Steve Rostedt <rostedt@goodmis.org> Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
Diffstat (limited to 'kernel/trace/trace_events_trigger.c')
-rw-r--r--kernel/trace/trace_events_trigger.c278
1 files changed, 278 insertions, 0 deletions
diff --git a/kernel/trace/trace_events_trigger.c b/kernel/trace/trace_events_trigger.c
new file mode 100644
index 000000000000..60a6a6d66dc0
--- /dev/null
+++ b/kernel/trace/trace_events_trigger.c
@@ -0,0 +1,278 @@
1/*
2 * trace_events_trigger - trace event triggers
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; either version 2 of the License, or
7 * (at your option) any later version.
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
17 *
18 * Copyright (C) 2013 Tom Zanussi <tom.zanussi@linux.intel.com>
19 */
20
21#include <linux/module.h>
22#include <linux/ctype.h>
23#include <linux/mutex.h>
24#include <linux/slab.h>
25
26#include "trace.h"
27
28static LIST_HEAD(trigger_commands);
29static DEFINE_MUTEX(trigger_cmd_mutex);
30
31/**
32 * event_triggers_call - Call triggers associated with a trace event
33 * @file: The ftrace_event_file associated with the event
34 *
35 * For each trigger associated with an event, invoke the trigger
36 * function registered with the associated trigger command.
37 *
38 * Called from tracepoint handlers (with rcu_read_lock_sched() held).
39 *
40 * Return: an enum event_trigger_type value containing a set bit for
41 * any trigger that should be deferred, ETT_NONE if nothing to defer.
42 */
43void event_triggers_call(struct ftrace_event_file *file)
44{
45 struct event_trigger_data *data;
46
47 if (list_empty(&file->triggers))
48 return;
49
50 list_for_each_entry_rcu(data, &file->triggers, list)
51 data->ops->func(data);
52}
53EXPORT_SYMBOL_GPL(event_triggers_call);
54
55static void *trigger_next(struct seq_file *m, void *t, loff_t *pos)
56{
57 struct ftrace_event_file *event_file = event_file_data(m->private);
58
59 return seq_list_next(t, &event_file->triggers, pos);
60}
61
62static void *trigger_start(struct seq_file *m, loff_t *pos)
63{
64 struct ftrace_event_file *event_file;
65
66 /* ->stop() is called even if ->start() fails */
67 mutex_lock(&event_mutex);
68 event_file = event_file_data(m->private);
69 if (unlikely(!event_file))
70 return ERR_PTR(-ENODEV);
71
72 return seq_list_start(&event_file->triggers, *pos);
73}
74
75static void trigger_stop(struct seq_file *m, void *t)
76{
77 mutex_unlock(&event_mutex);
78}
79
80static int trigger_show(struct seq_file *m, void *v)
81{
82 struct event_trigger_data *data;
83
84 data = list_entry(v, struct event_trigger_data, list);
85 data->ops->print(m, data->ops, data);
86
87 return 0;
88}
89
90static const struct seq_operations event_triggers_seq_ops = {
91 .start = trigger_start,
92 .next = trigger_next,
93 .stop = trigger_stop,
94 .show = trigger_show,
95};
96
97static int event_trigger_regex_open(struct inode *inode, struct file *file)
98{
99 int ret = 0;
100
101 mutex_lock(&event_mutex);
102
103 if (unlikely(!event_file_data(file))) {
104 mutex_unlock(&event_mutex);
105 return -ENODEV;
106 }
107
108 if (file->f_mode & FMODE_READ) {
109 ret = seq_open(file, &event_triggers_seq_ops);
110 if (!ret) {
111 struct seq_file *m = file->private_data;
112 m->private = file;
113 }
114 }
115
116 mutex_unlock(&event_mutex);
117
118 return ret;
119}
120
121static int trigger_process_regex(struct ftrace_event_file *file, char *buff)
122{
123 char *command, *next = buff;
124 struct event_command *p;
125 int ret = -EINVAL;
126
127 command = strsep(&next, ": \t");
128 command = (command[0] != '!') ? command : command + 1;
129
130 mutex_lock(&trigger_cmd_mutex);
131 list_for_each_entry(p, &trigger_commands, list) {
132 if (strcmp(p->name, command) == 0) {
133 ret = p->func(p, file, buff, command, next);
134 goto out_unlock;
135 }
136 }
137 out_unlock:
138 mutex_unlock(&trigger_cmd_mutex);
139
140 return ret;
141}
142
143static ssize_t event_trigger_regex_write(struct file *file,
144 const char __user *ubuf,
145 size_t cnt, loff_t *ppos)
146{
147 struct ftrace_event_file *event_file;
148 ssize_t ret;
149 char *buf;
150
151 if (!cnt)
152 return 0;
153
154 if (cnt >= PAGE_SIZE)
155 return -EINVAL;
156
157 buf = (char *)__get_free_page(GFP_TEMPORARY);
158 if (!buf)
159 return -ENOMEM;
160
161 if (copy_from_user(buf, ubuf, cnt)) {
162 free_page((unsigned long)buf);
163 return -EFAULT;
164 }
165 buf[cnt] = '\0';
166 strim(buf);
167
168 mutex_lock(&event_mutex);
169 event_file = event_file_data(file);
170 if (unlikely(!event_file)) {
171 mutex_unlock(&event_mutex);
172 free_page((unsigned long)buf);
173 return -ENODEV;
174 }
175 ret = trigger_process_regex(event_file, buf);
176 mutex_unlock(&event_mutex);
177
178 free_page((unsigned long)buf);
179 if (ret < 0)
180 goto out;
181
182 *ppos += cnt;
183 ret = cnt;
184 out:
185 return ret;
186}
187
188static int event_trigger_regex_release(struct inode *inode, struct file *file)
189{
190 mutex_lock(&event_mutex);
191
192 if (file->f_mode & FMODE_READ)
193 seq_release(inode, file);
194
195 mutex_unlock(&event_mutex);
196
197 return 0;
198}
199
200static ssize_t
201event_trigger_write(struct file *filp, const char __user *ubuf,
202 size_t cnt, loff_t *ppos)
203{
204 return event_trigger_regex_write(filp, ubuf, cnt, ppos);
205}
206
207static int
208event_trigger_open(struct inode *inode, struct file *filp)
209{
210 return event_trigger_regex_open(inode, filp);
211}
212
213static int
214event_trigger_release(struct inode *inode, struct file *file)
215{
216 return event_trigger_regex_release(inode, file);
217}
218
219const struct file_operations event_trigger_fops = {
220 .open = event_trigger_open,
221 .read = seq_read,
222 .write = event_trigger_write,
223 .llseek = ftrace_filter_lseek,
224 .release = event_trigger_release,
225};
226
227static int trace_event_trigger_enable_disable(struct ftrace_event_file *file,
228 int trigger_enable)
229{
230 int ret = 0;
231
232 if (trigger_enable) {
233 if (atomic_inc_return(&file->tm_ref) > 1)
234 return ret;
235 set_bit(FTRACE_EVENT_FL_TRIGGER_MODE_BIT, &file->flags);
236 ret = trace_event_enable_disable(file, 1, 1);
237 } else {
238 if (atomic_dec_return(&file->tm_ref) > 0)
239 return ret;
240 clear_bit(FTRACE_EVENT_FL_TRIGGER_MODE_BIT, &file->flags);
241 ret = trace_event_enable_disable(file, 0, 1);
242 }
243
244 return ret;
245}
246
247/**
248 * clear_event_triggers - Clear all triggers associated with a trace array
249 * @tr: The trace array to clear
250 *
251 * For each trigger, the triggering event has its tm_ref decremented
252 * via trace_event_trigger_enable_disable(), and any associated event
253 * (in the case of enable/disable_event triggers) will have its sm_ref
254 * decremented via free()->trace_event_enable_disable(). That
255 * combination effectively reverses the soft-mode/trigger state added
256 * by trigger registration.
257 *
258 * Must be called with event_mutex held.
259 */
260void
261clear_event_triggers(struct trace_array *tr)
262{
263 struct ftrace_event_file *file;
264
265 list_for_each_entry(file, &tr->events, list) {
266 struct event_trigger_data *data;
267 list_for_each_entry_rcu(data, &file->triggers, list) {
268 trace_event_trigger_enable_disable(file, 0);
269 if (data->ops->free)
270 data->ops->free(data->ops, data);
271 }
272 }
273}
274
275__init int register_trigger_cmds(void)
276{
277 return 0;
278}