Blame view

kernel/trace/trace_events_hist.c 142 KB
bcea3f96e   Steven Rostedt (VMware)   tracing: Add SPDX...
1
  // SPDX-License-Identifier: GPL-2.0
7ef224d1d   Tom Zanussi   tracing: Add 'his...
2
3
4
  /*
   * trace_events_hist - trace event hist triggers
   *
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5
6
7
8
9
   * Copyright (C) 2015 Tom Zanussi <tom.zanussi@linux.intel.com>
   */
  
  #include <linux/module.h>
  #include <linux/kallsyms.h>
17911ff38   Steven Rostedt (VMware)   tracing: Add lock...
10
  #include <linux/security.h>
7ef224d1d   Tom Zanussi   tracing: Add 'his...
11
12
13
  #include <linux/mutex.h>
  #include <linux/slab.h>
  #include <linux/stacktrace.h>
b2d091031   Ingo Molnar   sched/headers: Pr...
14
  #include <linux/rculist.h>
4b147936f   Tom Zanussi   tracing: Add supp...
15
  #include <linux/tracefs.h>
7ef224d1d   Tom Zanussi   tracing: Add 'his...
16

ac6815462   Zhengjun Xing   tracing: Add "gfp...
17
18
19
  /* for gfp flag names */
  #include <linux/trace_events.h>
  #include <trace/events/mmflags.h>
7ef224d1d   Tom Zanussi   tracing: Add 'his...
20
  #include "tracing_map.h"
726721a51   Tom Zanussi   tracing: Move syn...
21
  #include "trace_synth.h"
4b147936f   Tom Zanussi   tracing: Add supp...
22

d566c5e9d   Tom Zanussi   tracing: Use trac...
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
  #define ERRORS								\
  	C(NONE,			"No error"),				\
  	C(DUPLICATE_VAR,	"Variable already defined"),		\
  	C(VAR_NOT_UNIQUE,	"Variable name not unique, need to use fully qualified name (subsys.event.var) for variable"), \
  	C(TOO_MANY_VARS,	"Too many variables defined"),		\
  	C(MALFORMED_ASSIGNMENT,	"Malformed assignment"),		\
  	C(NAMED_MISMATCH,	"Named hist trigger doesn't match existing named trigger (includes variables)"), \
  	C(TRIGGER_EEXIST,	"Hist trigger already exists"),		\
  	C(TRIGGER_ENOENT_CLEAR,	"Can't clear or continue a nonexistent hist trigger"), \
  	C(SET_CLOCK_FAIL,	"Couldn't set trace_clock"),		\
  	C(BAD_FIELD_MODIFIER,	"Invalid field modifier"),		\
  	C(TOO_MANY_SUBEXPR,	"Too many subexpressions (3 max)"),	\
  	C(TIMESTAMP_MISMATCH,	"Timestamp units in expression don't match"), \
  	C(TOO_MANY_FIELD_VARS,	"Too many field variables defined"),	\
  	C(EVENT_FILE_NOT_FOUND,	"Event file not found"),		\
  	C(HIST_NOT_FOUND,	"Matching event histogram not found"),	\
  	C(HIST_CREATE_FAIL,	"Couldn't create histogram for field"),	\
  	C(SYNTH_VAR_NOT_FOUND,	"Couldn't find synthetic variable"),	\
  	C(SYNTH_EVENT_NOT_FOUND,"Couldn't find synthetic event"),	\
  	C(SYNTH_TYPE_MISMATCH,	"Param type doesn't match synthetic event field type"), \
  	C(SYNTH_COUNT_MISMATCH,	"Param count doesn't match synthetic event field count"), \
  	C(FIELD_VAR_PARSE_FAIL,	"Couldn't parse field variable"),	\
  	C(VAR_CREATE_FIND_FAIL,	"Couldn't create or find variable"),	\
  	C(ONX_NOT_VAR,		"For onmax(x) or onchange(x), x must be a variable"), \
  	C(ONX_VAR_NOT_FOUND,	"Couldn't find onmax or onchange variable"), \
  	C(ONX_VAR_CREATE_FAIL,	"Couldn't create onmax or onchange variable"), \
  	C(FIELD_VAR_CREATE_FAIL,"Couldn't create field variable"),	\
  	C(TOO_MANY_PARAMS,	"Too many action params"),		\
  	C(PARAM_NOT_FOUND,	"Couldn't find param"),			\
  	C(INVALID_PARAM,	"Invalid action param"),		\
  	C(ACTION_NOT_FOUND,	"No action found"),			\
  	C(NO_SAVE_PARAMS,	"No params found for save()"),		\
  	C(TOO_MANY_SAVE_ACTIONS,"Can't have more than one save() action per hist"), \
  	C(ACTION_MISMATCH,	"Handler doesn't support action"),	\
  	C(NO_CLOSING_PAREN,	"No closing paren found"),		\
  	C(SUBSYS_NOT_FOUND,	"Missing subsystem"),			\
  	C(INVALID_SUBSYS_EVENT,	"Invalid subsystem or event name"),	\
c8d94a187   Tom Zanussi   tracing: Check ke...
60
  	C(INVALID_REF_KEY,	"Using variable references in keys not supported"), \
d566c5e9d   Tom Zanussi   tracing: Use trac...
61
  	C(VAR_NOT_FOUND,	"Couldn't find variable"),		\
4de26c8c9   Tom Zanussi   tracing: Add hist...
62
63
64
65
66
67
  	C(FIELD_NOT_FOUND,	"Couldn't find field"),			\
  	C(EMPTY_ASSIGNMENT,	"Empty assignment"),			\
  	C(INVALID_SORT_MODIFIER,"Invalid sort modifier"),		\
  	C(EMPTY_SORT_FIELD,	"Empty sort field"),			\
  	C(TOO_MANY_SORT_FIELDS,	"Too many sort fields (Max = 2)"),	\
  	C(INVALID_SORT_FIELD,	"Sort field must be a key or a val"),
d566c5e9d   Tom Zanussi   tracing: Use trac...
68
69
70
71
72
73
74
75
76
77
  
  #undef C
  #define C(a, b)		HIST_ERR_##a
  
  enum { ERRORS };
  
  #undef C
  #define C(a, b)		b
  
  static const char *err_text[] = { ERRORS };
7ef224d1d   Tom Zanussi   tracing: Add 'his...
78
  struct hist_field;
df35d93bb   Tom Zanussi   tracing: Pass tra...
79
80
81
82
  typedef u64 (*hist_field_fn_t) (struct hist_field *field,
  				struct tracing_map_elt *elt,
  				struct ring_buffer_event *rbe,
  				void *event);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
83

5819eaddf   Tom Zanussi   tracing: Reimplem...
84
  #define HIST_FIELD_OPERANDS_MAX	2
30350d65a   Tom Zanussi   tracing: Add vari...
85
  #define HIST_FIELDS_MAX		(TRACING_MAP_FIELDS_MAX + TRACING_MAP_VARS_MAX)
0212e2aa3   Tom Zanussi   tracing: Add hist...
86
  #define HIST_ACTIONS_MAX	8
30350d65a   Tom Zanussi   tracing: Add vari...
87

100719dce   Tom Zanussi   tracing: Add simp...
88
89
90
91
92
93
  enum field_op_id {
  	FIELD_OP_NONE,
  	FIELD_OP_PLUS,
  	FIELD_OP_MINUS,
  	FIELD_OP_UNARY_MINUS,
  };
05ddb25cb   Tom Zanussi   tracing: Add hist...
94
95
96
97
98
99
100
101
102
103
  /*
   * A hist_var (histogram variable) contains variable information for
   * hist_fields having the HIST_FIELD_FL_VAR or HIST_FIELD_FL_VAR_REF
   * flag set.  A hist_var has a variable name e.g. ts0, and is
   * associated with a given histogram trigger, as specified by
   * hist_data.  The hist_var idx is the unique index assigned to the
   * variable by the hist trigger's tracing_map.  The idx is what is
   * used to set a variable's value and, by a variable reference, to
   * retrieve it.
   */
30350d65a   Tom Zanussi   tracing: Add vari...
104
105
106
107
108
  struct hist_var {
  	char				*name;
  	struct hist_trigger_data	*hist_data;
  	unsigned int			idx;
  };
5819eaddf   Tom Zanussi   tracing: Reimplem...
109

7ef224d1d   Tom Zanussi   tracing: Add 'his...
110
111
112
113
  struct hist_field {
  	struct ftrace_event_field	*field;
  	unsigned long			flags;
  	hist_field_fn_t			fn;
8bcebc77e   Steven Rostedt (VMware)   tracing: Fix hist...
114
  	unsigned int			ref;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
115
  	unsigned int			size;
76a3b0c8a   Tom Zanussi   tracing: Add hist...
116
  	unsigned int			offset;
5819eaddf   Tom Zanussi   tracing: Reimplem...
117
  	unsigned int                    is_signed;
19a9facd0   Tom Zanussi   tracing: Add hist...
118
  	const char			*type;
5819eaddf   Tom Zanussi   tracing: Reimplem...
119
  	struct hist_field		*operands[HIST_FIELD_OPERANDS_MAX];
b559d003a   Tom Zanussi   tracing: Add hist...
120
  	struct hist_trigger_data	*hist_data;
05ddb25cb   Tom Zanussi   tracing: Add hist...
121
122
123
124
  
  	/*
  	 * Variable fields contain variable-specific info in var.
  	 */
30350d65a   Tom Zanussi   tracing: Add vari...
125
  	struct hist_var			var;
100719dce   Tom Zanussi   tracing: Add simp...
126
  	enum field_op_id		operator;
067fe038e   Tom Zanussi   tracing: Add vari...
127
128
  	char				*system;
  	char				*event_name;
05ddb25cb   Tom Zanussi   tracing: Add hist...
129
130
131
132
133
  
  	/*
  	 * The name field is used for EXPR and VAR_REF fields.  VAR
  	 * fields contain the variable name in var.name.
  	 */
100719dce   Tom Zanussi   tracing: Add simp...
134
  	char				*name;
05ddb25cb   Tom Zanussi   tracing: Add hist...
135
136
137
138
139
140
141
142
143
  
  	/*
  	 * When a histogram trigger is hit, if it has any references
  	 * to variables, the values of those variables are collected
  	 * into a var_ref_vals array by resolve_var_refs().  The
  	 * current value of each variable is read from the tracing_map
  	 * using the hist field's hist_var.idx and entered into the
  	 * var_ref_idx entry i.e. var_ref_vals[var_ref_idx].
  	 */
067fe038e   Tom Zanussi   tracing: Add vari...
144
145
  	unsigned int			var_ref_idx;
  	bool                            read_once;
63a1e5de3   Tom Zanussi   tracing: Save nor...
146
147
  
  	unsigned int			var_str_idx;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
148
  };
df35d93bb   Tom Zanussi   tracing: Pass tra...
149
150
151
152
  static u64 hist_field_none(struct hist_field *field,
  			   struct tracing_map_elt *elt,
  			   struct ring_buffer_event *rbe,
  			   void *event)
69a0200c2   Tom Zanussi   tracing: Add hist...
153
154
155
  {
  	return 0;
  }
df35d93bb   Tom Zanussi   tracing: Pass tra...
156
157
158
159
  static u64 hist_field_counter(struct hist_field *field,
  			      struct tracing_map_elt *elt,
  			      struct ring_buffer_event *rbe,
  			      void *event)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
160
161
162
  {
  	return 1;
  }
df35d93bb   Tom Zanussi   tracing: Pass tra...
163
164
165
166
  static u64 hist_field_string(struct hist_field *hist_field,
  			     struct tracing_map_elt *elt,
  			     struct ring_buffer_event *rbe,
  			     void *event)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
167
168
169
170
171
  {
  	char *addr = (char *)(event + hist_field->field->offset);
  
  	return (u64)(unsigned long)addr;
  }
df35d93bb   Tom Zanussi   tracing: Pass tra...
172
173
174
175
  static u64 hist_field_dynstring(struct hist_field *hist_field,
  				struct tracing_map_elt *elt,
  				struct ring_buffer_event *rbe,
  				void *event)
79e577cbc   Namhyung Kim   tracing: Support ...
176
177
178
179
180
181
182
  {
  	u32 str_item = *(u32 *)(event + hist_field->field->offset);
  	int str_loc = str_item & 0xffff;
  	char *addr = (char *)(event + str_loc);
  
  	return (u64)(unsigned long)addr;
  }
df35d93bb   Tom Zanussi   tracing: Pass tra...
183
184
185
186
  static u64 hist_field_pstring(struct hist_field *hist_field,
  			      struct tracing_map_elt *elt,
  			      struct ring_buffer_event *rbe,
  			      void *event)
79e577cbc   Namhyung Kim   tracing: Support ...
187
188
189
190
191
  {
  	char **addr = (char **)(event + hist_field->field->offset);
  
  	return (u64)(unsigned long)*addr;
  }
df35d93bb   Tom Zanussi   tracing: Pass tra...
192
193
194
195
  static u64 hist_field_log2(struct hist_field *hist_field,
  			   struct tracing_map_elt *elt,
  			   struct ring_buffer_event *rbe,
  			   void *event)
4b94f5b7b   Namhyung Kim   tracing: Add hist...
196
  {
5819eaddf   Tom Zanussi   tracing: Reimplem...
197
  	struct hist_field *operand = hist_field->operands[0];
df35d93bb   Tom Zanussi   tracing: Pass tra...
198
  	u64 val = operand->fn(operand, elt, rbe, event);
4b94f5b7b   Namhyung Kim   tracing: Add hist...
199
200
201
  
  	return (u64) ilog2(roundup_pow_of_two(val));
  }
df35d93bb   Tom Zanussi   tracing: Pass tra...
202
203
204
205
  static u64 hist_field_plus(struct hist_field *hist_field,
  			   struct tracing_map_elt *elt,
  			   struct ring_buffer_event *rbe,
  			   void *event)
100719dce   Tom Zanussi   tracing: Add simp...
206
207
208
  {
  	struct hist_field *operand1 = hist_field->operands[0];
  	struct hist_field *operand2 = hist_field->operands[1];
df35d93bb   Tom Zanussi   tracing: Pass tra...
209
210
  	u64 val1 = operand1->fn(operand1, elt, rbe, event);
  	u64 val2 = operand2->fn(operand2, elt, rbe, event);
100719dce   Tom Zanussi   tracing: Add simp...
211
212
213
  
  	return val1 + val2;
  }
df35d93bb   Tom Zanussi   tracing: Pass tra...
214
215
216
217
  static u64 hist_field_minus(struct hist_field *hist_field,
  			    struct tracing_map_elt *elt,
  			    struct ring_buffer_event *rbe,
  			    void *event)
100719dce   Tom Zanussi   tracing: Add simp...
218
219
220
  {
  	struct hist_field *operand1 = hist_field->operands[0];
  	struct hist_field *operand2 = hist_field->operands[1];
df35d93bb   Tom Zanussi   tracing: Pass tra...
221
222
  	u64 val1 = operand1->fn(operand1, elt, rbe, event);
  	u64 val2 = operand2->fn(operand2, elt, rbe, event);
100719dce   Tom Zanussi   tracing: Add simp...
223
224
225
  
  	return val1 - val2;
  }
df35d93bb   Tom Zanussi   tracing: Pass tra...
226
227
228
229
  static u64 hist_field_unary_minus(struct hist_field *hist_field,
  				  struct tracing_map_elt *elt,
  				  struct ring_buffer_event *rbe,
  				  void *event)
100719dce   Tom Zanussi   tracing: Add simp...
230
231
  {
  	struct hist_field *operand = hist_field->operands[0];
df35d93bb   Tom Zanussi   tracing: Pass tra...
232
  	s64 sval = (s64)operand->fn(operand, elt, rbe, event);
100719dce   Tom Zanussi   tracing: Add simp...
233
234
235
236
  	u64 val = (u64)-sval;
  
  	return val;
  }
7ef224d1d   Tom Zanussi   tracing: Add 'his...
237
  #define DEFINE_HIST_FIELD_FN(type)					\
fbd302cbe   Tom Zanussi   tracing: Add ring...
238
  	static u64 hist_field_##type(struct hist_field *hist_field,	\
df35d93bb   Tom Zanussi   tracing: Pass tra...
239
240
241
  				     struct tracing_map_elt *elt,	\
  				     struct ring_buffer_event *rbe,	\
  				     void *event)			\
7ef224d1d   Tom Zanussi   tracing: Add 'his...
242
243
244
  {									\
  	type *addr = (type *)(event + hist_field->field->offset);	\
  									\
79e577cbc   Namhyung Kim   tracing: Support ...
245
  	return (u64)(unsigned long)*addr;				\
7ef224d1d   Tom Zanussi   tracing: Add 'his...
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
  }
  
  DEFINE_HIST_FIELD_FN(s64);
  DEFINE_HIST_FIELD_FN(u64);
  DEFINE_HIST_FIELD_FN(s32);
  DEFINE_HIST_FIELD_FN(u32);
  DEFINE_HIST_FIELD_FN(s16);
  DEFINE_HIST_FIELD_FN(u16);
  DEFINE_HIST_FIELD_FN(s8);
  DEFINE_HIST_FIELD_FN(u8);
  
  #define for_each_hist_field(i, hist_data)	\
  	for ((i) = 0; (i) < (hist_data)->n_fields; (i)++)
  
  #define for_each_hist_val_field(i, hist_data)	\
  	for ((i) = 0; (i) < (hist_data)->n_vals; (i)++)
  
  #define for_each_hist_key_field(i, hist_data)	\
  	for ((i) = (hist_data)->n_vals; (i) < (hist_data)->n_fields; (i)++)
69a0200c2   Tom Zanussi   tracing: Add hist...
265
266
267
  #define HIST_STACKTRACE_DEPTH	16
  #define HIST_STACKTRACE_SIZE	(HIST_STACKTRACE_DEPTH * sizeof(unsigned long))
  #define HIST_STACKTRACE_SKIP	5
7ef224d1d   Tom Zanussi   tracing: Add 'his...
268
  #define HITCOUNT_IDX		0
69a0200c2   Tom Zanussi   tracing: Add hist...
269
  #define HIST_KEY_SIZE_MAX	(MAX_FILTER_STR_VAL + HIST_STACKTRACE_SIZE)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
270
271
  
  enum hist_field_flags {
0d7a8325b   Tom Zanussi   tracing: Clean up...
272
273
274
275
276
277
278
279
280
281
  	HIST_FIELD_FL_HITCOUNT		= 1 << 0,
  	HIST_FIELD_FL_KEY		= 1 << 1,
  	HIST_FIELD_FL_STRING		= 1 << 2,
  	HIST_FIELD_FL_HEX		= 1 << 3,
  	HIST_FIELD_FL_SYM		= 1 << 4,
  	HIST_FIELD_FL_SYM_OFFSET	= 1 << 5,
  	HIST_FIELD_FL_EXECNAME		= 1 << 6,
  	HIST_FIELD_FL_SYSCALL		= 1 << 7,
  	HIST_FIELD_FL_STACKTRACE	= 1 << 8,
  	HIST_FIELD_FL_LOG2		= 1 << 9,
ad42febe5   Tom Zanussi   tracing: Add hist...
282
  	HIST_FIELD_FL_TIMESTAMP		= 1 << 10,
860f9f6b0   Tom Zanussi   tracing: Add usec...
283
  	HIST_FIELD_FL_TIMESTAMP_USECS	= 1 << 11,
30350d65a   Tom Zanussi   tracing: Add vari...
284
  	HIST_FIELD_FL_VAR		= 1 << 12,
100719dce   Tom Zanussi   tracing: Add simp...
285
  	HIST_FIELD_FL_EXPR		= 1 << 13,
067fe038e   Tom Zanussi   tracing: Add vari...
286
  	HIST_FIELD_FL_VAR_REF		= 1 << 14,
8b7622bf9   Tom Zanussi   tracing: Add cpu ...
287
  	HIST_FIELD_FL_CPU		= 1 << 15,
7e8b88a30   Tom Zanussi   tracing: Add hist...
288
  	HIST_FIELD_FL_ALIAS		= 1 << 16,
30350d65a   Tom Zanussi   tracing: Add vari...
289
290
291
292
293
294
  };
  
  struct var_defs {
  	unsigned int	n_vars;
  	char		*name[TRACING_MAP_VARS_MAX];
  	char		*expr[TRACING_MAP_VARS_MAX];
7ef224d1d   Tom Zanussi   tracing: Add 'his...
295
296
297
298
  };
  
  struct hist_trigger_attrs {
  	char		*keys_str;
f2606835d   Tom Zanussi   tracing: Add hist...
299
  	char		*vals_str;
e62347d24   Tom Zanussi   tracing: Add hist...
300
  	char		*sort_key_str;
5463bfda3   Tom Zanussi   tracing: Add supp...
301
  	char		*name;
a4072fe85   Tom Zanussi   tracing: Add a cl...
302
  	char		*clock;
83e99914c   Tom Zanussi   tracing: Add hist...
303
304
  	bool		pause;
  	bool		cont;
e86ae9baa   Tom Zanussi   tracing: Add hist...
305
  	bool		clear;
860f9f6b0   Tom Zanussi   tracing: Add usec...
306
  	bool		ts_in_usecs;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
307
  	unsigned int	map_bits;
30350d65a   Tom Zanussi   tracing: Add vari...
308
309
310
  
  	char		*assignment_str[TRACING_MAP_VARS_MAX];
  	unsigned int	n_assignments;
0212e2aa3   Tom Zanussi   tracing: Add hist...
311
312
  	char		*action_str[HIST_ACTIONS_MAX];
  	unsigned int	n_actions;
30350d65a   Tom Zanussi   tracing: Add vari...
313
  	struct var_defs	var_defs;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
314
  };
02205a675   Tom Zanussi   tracing: Add supp...
315
316
317
318
319
320
321
322
323
  struct field_var {
  	struct hist_field	*var;
  	struct hist_field	*val;
  };
  
  struct field_var_hist {
  	struct hist_trigger_data	*hist_data;
  	char				*cmd;
  };
7ef224d1d   Tom Zanussi   tracing: Add 'his...
324
  struct hist_trigger_data {
30350d65a   Tom Zanussi   tracing: Add vari...
325
  	struct hist_field               *fields[HIST_FIELDS_MAX];
7ef224d1d   Tom Zanussi   tracing: Add 'his...
326
327
328
  	unsigned int			n_vals;
  	unsigned int			n_keys;
  	unsigned int			n_fields;
30350d65a   Tom Zanussi   tracing: Add vari...
329
  	unsigned int			n_vars;
63a1e5de3   Tom Zanussi   tracing: Save nor...
330
  	unsigned int			n_var_str;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
331
332
333
334
335
336
  	unsigned int			key_size;
  	struct tracing_map_sort_key	sort_keys[TRACING_MAP_SORT_KEYS_MAX];
  	unsigned int			n_sort_keys;
  	struct trace_event_file		*event_file;
  	struct hist_trigger_attrs	*attrs;
  	struct tracing_map		*map;
ad42febe5   Tom Zanussi   tracing: Add hist...
337
  	bool				enable_timestamps;
30350d65a   Tom Zanussi   tracing: Add vari...
338
  	bool				remove;
067fe038e   Tom Zanussi   tracing: Add vari...
339
340
  	struct hist_field               *var_refs[TRACING_MAP_VARS_MAX];
  	unsigned int			n_var_refs;
0212e2aa3   Tom Zanussi   tracing: Add hist...
341
342
343
  
  	struct action_data		*actions[HIST_ACTIONS_MAX];
  	unsigned int			n_actions;
02205a675   Tom Zanussi   tracing: Add supp...
344
345
346
347
348
349
  
  	struct field_var		*field_vars[SYNTH_FIELDS_MAX];
  	unsigned int			n_field_vars;
  	unsigned int			n_field_var_str;
  	struct field_var_hist		*field_var_hists[SYNTH_FIELDS_MAX];
  	unsigned int			n_field_var_hists;
50450603e   Tom Zanussi   tracing: Add 'onm...
350

7d18a10c3   Tom Zanussi   tracing: Refactor...
351
352
353
  	struct field_var		*save_vars[SYNTH_FIELDS_MAX];
  	unsigned int			n_save_vars;
  	unsigned int			n_save_var_str;
0212e2aa3   Tom Zanussi   tracing: Add hist...
354
355
356
357
358
359
  };
  
  struct action_data;
  
  typedef void (*action_fn_t) (struct hist_trigger_data *hist_data,
  			     struct tracing_map_elt *elt, void *rec,
7d18a10c3   Tom Zanussi   tracing: Refactor...
360
  			     struct ring_buffer_event *rbe, void *key,
0212e2aa3   Tom Zanussi   tracing: Add hist...
361
  			     struct action_data *data, u64 *var_ref_vals);
466f4528f   Tom Zanussi   tracing: Generali...
362
  typedef bool (*check_track_val_fn_t) (u64 track_val, u64 var_val);
7d18a10c3   Tom Zanussi   tracing: Refactor...
363
364
365
  enum handler_id {
  	HANDLER_ONMATCH = 1,
  	HANDLER_ONMAX,
dff81f559   Tom Zanussi   tracing: Add hist...
366
  	HANDLER_ONCHANGE,
7d18a10c3   Tom Zanussi   tracing: Refactor...
367
368
369
370
371
  };
  
  enum action_id {
  	ACTION_SAVE = 1,
  	ACTION_TRACE,
a3785b7ec   Tom Zanussi   tracing: Add hist...
372
  	ACTION_SNAPSHOT,
7d18a10c3   Tom Zanussi   tracing: Refactor...
373
  };
0212e2aa3   Tom Zanussi   tracing: Add hist...
374
  struct action_data {
7d18a10c3   Tom Zanussi   tracing: Refactor...
375
376
377
  	enum handler_id		handler;
  	enum action_id		action;
  	char			*action_name;
0212e2aa3   Tom Zanussi   tracing: Add hist...
378
  	action_fn_t		fn;
7d18a10c3   Tom Zanussi   tracing: Refactor...
379

c282a386a   Tom Zanussi   tracing: Add 'onm...
380
381
  	unsigned int		n_params;
  	char			*params[SYNTH_FIELDS_MAX];
c3e49506a   Tom Zanussi   tracing: Split up...
382
383
384
385
  	/*
  	 * When a histogram trigger is hit, the values of any
  	 * references to variables, including variables being passed
  	 * as parameters to synthetic events, are collected into a
d380dcde9   Tom Zanussi   tracing: Fix now ...
386
387
388
389
  	 * var_ref_vals array.  This var_ref_idx array is an array of
  	 * indices into the var_ref_vals array, one for each synthetic
  	 * event param, and is passed to the synthetic event
  	 * invocation.
c3e49506a   Tom Zanussi   tracing: Split up...
390
  	 */
d380dcde9   Tom Zanussi   tracing: Fix now ...
391
  	unsigned int		var_ref_idx[TRACING_MAP_VARS_MAX];
c3e49506a   Tom Zanussi   tracing: Split up...
392
  	struct synth_event	*synth_event;
e91eefd73   Tom Zanussi   tracing: Add alte...
393
394
  	bool			use_trace_keyword;
  	char			*synth_event_name;
c3e49506a   Tom Zanussi   tracing: Split up...
395

c282a386a   Tom Zanussi   tracing: Add 'onm...
396
397
  	union {
  		struct {
c3e49506a   Tom Zanussi   tracing: Split up...
398
  			char			*event;
726721a51   Tom Zanussi   tracing: Move syn...
399
400
  			char			*event_system;
  		} match_data;
8dcc53ad9   Tom Zanussi   tracing: Add synt...
401

726721a51   Tom Zanussi   tracing: Move syn...
402
403
404
405
406
407
408
409
410
411
  		struct {
  			/*
  			 * var_str contains the $-unstripped variable
  			 * name referenced by var_ref, and used when
  			 * printing the action.  Because var_ref
  			 * creation is deferred to create_actions(),
  			 * we need a per-action way to save it until
  			 * then, thus var_str.
  			 */
  			char			*var_str;
8dcc53ad9   Tom Zanussi   tracing: Add synt...
412

726721a51   Tom Zanussi   tracing: Move syn...
413
414
415
416
417
  			/*
  			 * var_ref refers to the variable being
  			 * tracked e.g onmax($var).
  			 */
  			struct hist_field	*var_ref;
249d7b2ef   Tom Zanussi   tracing: Consolid...
418

726721a51   Tom Zanussi   tracing: Move syn...
419
420
421
422
423
424
  			/*
  			 * track_var contains the 'invisible' tracking
  			 * variable created to keep the current
  			 * e.g. max value.
  			 */
  			struct hist_field	*track_var;
249d7b2ef   Tom Zanussi   tracing: Consolid...
425

726721a51   Tom Zanussi   tracing: Move syn...
426
427
428
429
430
  			check_track_val_fn_t	check_val;
  			action_fn_t		save_data;
  		} track_data;
  	};
  };
8dcc53ad9   Tom Zanussi   tracing: Add synt...
431

726721a51   Tom Zanussi   tracing: Move syn...
432
433
434
  struct track_data {
  	u64				track_val;
  	bool				updated;
249d7b2ef   Tom Zanussi   tracing: Consolid...
435

726721a51   Tom Zanussi   tracing: Move syn...
436
437
438
  	unsigned int			key_len;
  	void				*key;
  	struct tracing_map_elt		elt;
249d7b2ef   Tom Zanussi   tracing: Consolid...
439

726721a51   Tom Zanussi   tracing: Move syn...
440
441
442
  	struct action_data		*action_data;
  	struct hist_trigger_data	*hist_data;
  };
8dcc53ad9   Tom Zanussi   tracing: Add synt...
443

726721a51   Tom Zanussi   tracing: Move syn...
444
445
446
447
448
  struct hist_elt_data {
  	char *comm;
  	u64 *var_ref_vals;
  	char *field_var_str[SYNTH_FIELDS_MAX];
  };
8dcc53ad9   Tom Zanussi   tracing: Add synt...
449

726721a51   Tom Zanussi   tracing: Move syn...
450
451
452
453
  struct snapshot_context {
  	struct tracing_map_elt	*elt;
  	void			*key;
  };
1d9d4c901   Tom Zanussi   tracing: Make syn...
454

726721a51   Tom Zanussi   tracing: Move syn...
455
456
457
  static void track_data_free(struct track_data *track_data)
  {
  	struct hist_elt_data *elt_data;
1d9d4c901   Tom Zanussi   tracing: Make syn...
458

726721a51   Tom Zanussi   tracing: Move syn...
459
460
  	if (!track_data)
  		return;
1d9d4c901   Tom Zanussi   tracing: Make syn...
461

726721a51   Tom Zanussi   tracing: Move syn...
462
463
464
465
466
467
  	kfree(track_data->key);
  
  	elt_data = track_data->elt.private_data;
  	if (elt_data) {
  		kfree(elt_data->comm);
  		kfree(elt_data);
1d9d4c901   Tom Zanussi   tracing: Make syn...
468
  	}
8dcc53ad9   Tom Zanussi   tracing: Add synt...
469

726721a51   Tom Zanussi   tracing: Move syn...
470
  	kfree(track_data);
8dcc53ad9   Tom Zanussi   tracing: Add synt...
471
  }
8dcc53ad9   Tom Zanussi   tracing: Add synt...
472

726721a51   Tom Zanussi   tracing: Move syn...
473
474
475
  static struct track_data *track_data_alloc(unsigned int key_len,
  					   struct action_data *action_data,
  					   struct hist_trigger_data *hist_data)
8dcc53ad9   Tom Zanussi   tracing: Add synt...
476
  {
726721a51   Tom Zanussi   tracing: Move syn...
477
478
  	struct track_data *data = kzalloc(sizeof(*data), GFP_KERNEL);
  	struct hist_elt_data *elt_data;
8dcc53ad9   Tom Zanussi   tracing: Add synt...
479

726721a51   Tom Zanussi   tracing: Move syn...
480
481
  	if (!data)
  		return ERR_PTR(-ENOMEM);
8dcc53ad9   Tom Zanussi   tracing: Add synt...
482

726721a51   Tom Zanussi   tracing: Move syn...
483
484
485
486
487
  	data->key = kzalloc(key_len, GFP_KERNEL);
  	if (!data->key) {
  		track_data_free(data);
  		return ERR_PTR(-ENOMEM);
  	}
8dcc53ad9   Tom Zanussi   tracing: Add synt...
488

726721a51   Tom Zanussi   tracing: Move syn...
489
490
491
  	data->key_len = key_len;
  	data->action_data = action_data;
  	data->hist_data = hist_data;
8dcc53ad9   Tom Zanussi   tracing: Add synt...
492

726721a51   Tom Zanussi   tracing: Move syn...
493
494
495
496
497
  	elt_data = kzalloc(sizeof(*elt_data), GFP_KERNEL);
  	if (!elt_data) {
  		track_data_free(data);
  		return ERR_PTR(-ENOMEM);
  	}
4b147936f   Tom Zanussi   tracing: Add supp...
498

726721a51   Tom Zanussi   tracing: Move syn...
499
  	data->elt.private_data = elt_data;
4b147936f   Tom Zanussi   tracing: Add supp...
500

726721a51   Tom Zanussi   tracing: Move syn...
501
502
503
504
  	elt_data->comm = kzalloc(TASK_COMM_LEN, GFP_KERNEL);
  	if (!elt_data->comm) {
  		track_data_free(data);
  		return ERR_PTR(-ENOMEM);
7bbab38d0   Masami Hiramatsu   tracing: Use dyn_...
505
  	}
7bbab38d0   Masami Hiramatsu   tracing: Use dyn_...
506

726721a51   Tom Zanussi   tracing: Move syn...
507
508
  	return data;
  }
7bbab38d0   Masami Hiramatsu   tracing: Use dyn_...
509

726721a51   Tom Zanussi   tracing: Move syn...
510
511
  static char last_cmd[MAX_FILTER_STR_VAL];
  static char last_cmd_loc[MAX_FILTER_STR_VAL];
7bbab38d0   Masami Hiramatsu   tracing: Use dyn_...
512

726721a51   Tom Zanussi   tracing: Move syn...
513
514
515
  static int errpos(char *str)
  {
  	return err_pos(last_cmd, str);
4b147936f   Tom Zanussi   tracing: Add supp...
516
  }
726721a51   Tom Zanussi   tracing: Move syn...
517
  static void last_cmd_set(struct trace_event_file *file, char *str)
4b147936f   Tom Zanussi   tracing: Add supp...
518
  {
726721a51   Tom Zanussi   tracing: Move syn...
519
520
  	const char *system = NULL, *name = NULL;
  	struct trace_event_call *call;
4b147936f   Tom Zanussi   tracing: Add supp...
521

726721a51   Tom Zanussi   tracing: Move syn...
522
523
  	if (!str)
  		return;
4b147936f   Tom Zanussi   tracing: Add supp...
524

726721a51   Tom Zanussi   tracing: Move syn...
525
526
  	strcpy(last_cmd, "hist:");
  	strncat(last_cmd, str, MAX_FILTER_STR_VAL - 1 - sizeof("hist:"));
4b147936f   Tom Zanussi   tracing: Add supp...
527

726721a51   Tom Zanussi   tracing: Move syn...
528
529
530
531
532
533
534
535
  	if (file) {
  		call = file->event_call;
  		system = call->class->system;
  		if (system) {
  			name = trace_event_name(call);
  			if (!name)
  				system = NULL;
  		}
4b147936f   Tom Zanussi   tracing: Add supp...
536
  	}
726721a51   Tom Zanussi   tracing: Move syn...
537
538
  	if (system)
  		snprintf(last_cmd_loc, MAX_FILTER_STR_VAL, "hist:%s:%s", system, name);
4b147936f   Tom Zanussi   tracing: Add supp...
539
  }
726721a51   Tom Zanussi   tracing: Move syn...
540
  static void hist_err(struct trace_array *tr, u8 err_type, u8 err_pos)
7bbab38d0   Masami Hiramatsu   tracing: Use dyn_...
541
  {
726721a51   Tom Zanussi   tracing: Move syn...
542
543
  	tracing_log_err(tr, last_cmd_loc, last_cmd, err_text,
  			err_type, err_pos);
7bbab38d0   Masami Hiramatsu   tracing: Use dyn_...
544
  }
726721a51   Tom Zanussi   tracing: Move syn...
545
  static void hist_err_clear(void)
7bbab38d0   Masami Hiramatsu   tracing: Use dyn_...
546
  {
726721a51   Tom Zanussi   tracing: Move syn...
547
548
  	last_cmd[0] = '\0';
  	last_cmd_loc[0] = '\0';
7bbab38d0   Masami Hiramatsu   tracing: Use dyn_...
549
  }
726721a51   Tom Zanussi   tracing: Move syn...
550
551
  typedef void (*synth_probe_func_t) (void *__data, u64 *var_ref_vals,
  				    unsigned int *var_ref_idx);
4b147936f   Tom Zanussi   tracing: Add supp...
552

726721a51   Tom Zanussi   tracing: Move syn...
553
554
  static inline void trace_synth(struct synth_event *event, u64 *var_ref_vals,
  			       unsigned int *var_ref_idx)
4b147936f   Tom Zanussi   tracing: Add supp...
555
  {
726721a51   Tom Zanussi   tracing: Move syn...
556
  	struct tracepoint *tp = event->tp;
4b147936f   Tom Zanussi   tracing: Add supp...
557

726721a51   Tom Zanussi   tracing: Move syn...
558
559
560
561
  	if (unlikely(atomic_read(&tp->key.enabled) > 0)) {
  		struct tracepoint_func *probe_func_ptr;
  		synth_probe_func_t probe_func;
  		void *__data;
17911ff38   Steven Rostedt (VMware)   tracing: Add lock...
562

726721a51   Tom Zanussi   tracing: Move syn...
563
564
  		if (!(cpu_online(raw_smp_processor_id())))
  			return;
4b147936f   Tom Zanussi   tracing: Add supp...
565

726721a51   Tom Zanussi   tracing: Move syn...
566
567
568
569
570
571
572
573
574
  		probe_func_ptr = rcu_dereference_sched((tp)->funcs);
  		if (probe_func_ptr) {
  			do {
  				probe_func = probe_func_ptr->func;
  				__data = probe_func_ptr->data;
  				probe_func(__data, var_ref_vals, var_ref_idx);
  			} while ((++probe_func_ptr)->func);
  		}
  	}
4b147936f   Tom Zanussi   tracing: Add supp...
575
  }
726721a51   Tom Zanussi   tracing: Move syn...
576
577
578
579
  static void action_trace(struct hist_trigger_data *hist_data,
  			 struct tracing_map_elt *elt, void *rec,
  			 struct ring_buffer_event *rbe, void *key,
  			 struct action_data *data, u64 *var_ref_vals)
4b147936f   Tom Zanussi   tracing: Add supp...
580
  {
726721a51   Tom Zanussi   tracing: Move syn...
581
582
583
  	struct synth_event *event = data->synth_event;
  
  	trace_synth(event, var_ref_vals, data->var_ref_idx);
4b147936f   Tom Zanussi   tracing: Add supp...
584
  }
726721a51   Tom Zanussi   tracing: Move syn...
585
586
587
  struct hist_var_data {
  	struct list_head list;
  	struct hist_trigger_data *hist_data;
4b147936f   Tom Zanussi   tracing: Add supp...
588
  };
df35d93bb   Tom Zanussi   tracing: Pass tra...
589
590
591
592
  static u64 hist_field_timestamp(struct hist_field *hist_field,
  				struct tracing_map_elt *elt,
  				struct ring_buffer_event *rbe,
  				void *event)
860f9f6b0   Tom Zanussi   tracing: Add usec...
593
594
595
596
597
598
599
600
601
602
603
  {
  	struct hist_trigger_data *hist_data = hist_field->hist_data;
  	struct trace_array *tr = hist_data->event_file->tr;
  
  	u64 ts = ring_buffer_event_time_stamp(rbe);
  
  	if (hist_data->attrs->ts_in_usecs && trace_clock_in_ns(tr))
  		ts = ns2usecs(ts);
  
  	return ts;
  }
8b7622bf9   Tom Zanussi   tracing: Add cpu ...
604
605
606
607
608
609
610
611
612
  static u64 hist_field_cpu(struct hist_field *hist_field,
  			  struct tracing_map_elt *elt,
  			  struct ring_buffer_event *rbe,
  			  void *event)
  {
  	int cpu = smp_processor_id();
  
  	return cpu;
  }
de40f033d   Tom Zanussi   tracing: Remove o...
613
614
615
616
617
618
619
620
621
622
623
  /**
   * check_field_for_var_ref - Check if a VAR_REF field references a variable
   * @hist_field: The VAR_REF field to check
   * @var_data: The hist trigger that owns the variable
   * @var_idx: The trigger variable identifier
   *
   * Check the given VAR_REF field to see whether or not it references
   * the given variable associated with the given trigger.
   *
   * Return: The VAR_REF field if it does reference the variable, NULL if not
   */
067fe038e   Tom Zanussi   tracing: Add vari...
624
625
626
627
628
  static struct hist_field *
  check_field_for_var_ref(struct hist_field *hist_field,
  			struct hist_trigger_data *var_data,
  			unsigned int var_idx)
  {
e4f6d2450   Tom Zanussi   tracing: Use var_...
629
  	WARN_ON(!(hist_field && hist_field->flags & HIST_FIELD_FL_VAR_REF));
067fe038e   Tom Zanussi   tracing: Add vari...
630

e4f6d2450   Tom Zanussi   tracing: Use var_...
631
632
633
  	if (hist_field && hist_field->var.idx == var_idx &&
  	    hist_field->var.hist_data == var_data)
  		return hist_field;
067fe038e   Tom Zanussi   tracing: Add vari...
634

e4f6d2450   Tom Zanussi   tracing: Use var_...
635
  	return NULL;
067fe038e   Tom Zanussi   tracing: Add vari...
636
  }
de40f033d   Tom Zanussi   tracing: Remove o...
637
638
639
640
641
642
643
644
645
646
647
648
  /**
   * find_var_ref - Check if a trigger has a reference to a trigger variable
   * @hist_data: The hist trigger that might have a reference to the variable
   * @var_data: The hist trigger that owns the variable
   * @var_idx: The trigger variable identifier
   *
   * Check the list of var_refs[] on the first hist trigger to see
   * whether any of them are references to the variable on the second
   * trigger.
   *
   * Return: The VAR_REF field referencing the variable if so, NULL if not
   */
067fe038e   Tom Zanussi   tracing: Add vari...
649
650
651
652
  static struct hist_field *find_var_ref(struct hist_trigger_data *hist_data,
  				       struct hist_trigger_data *var_data,
  				       unsigned int var_idx)
  {
e4f6d2450   Tom Zanussi   tracing: Use var_...
653
  	struct hist_field *hist_field;
067fe038e   Tom Zanussi   tracing: Add vari...
654
  	unsigned int i;
e4f6d2450   Tom Zanussi   tracing: Use var_...
655
656
657
658
  	for (i = 0; i < hist_data->n_var_refs; i++) {
  		hist_field = hist_data->var_refs[i];
  		if (check_field_for_var_ref(hist_field, var_data, var_idx))
  			return hist_field;
067fe038e   Tom Zanussi   tracing: Add vari...
659
  	}
e4f6d2450   Tom Zanussi   tracing: Use var_...
660
  	return NULL;
067fe038e   Tom Zanussi   tracing: Add vari...
661
  }
de40f033d   Tom Zanussi   tracing: Remove o...
662
663
664
665
666
667
668
669
670
671
672
673
674
675
  /**
   * find_any_var_ref - Check if there is a reference to a given trigger variable
   * @hist_data: The hist trigger
   * @var_idx: The trigger variable identifier
   *
   * Check to see whether the given variable is currently referenced by
   * any other trigger.
   *
   * The trigger the variable is defined on is explicitly excluded - the
   * assumption being that a self-reference doesn't prevent a trigger
   * from being removed.
   *
   * Return: The VAR_REF field referencing the variable if so, NULL if not
   */
067fe038e   Tom Zanussi   tracing: Add vari...
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
  static struct hist_field *find_any_var_ref(struct hist_trigger_data *hist_data,
  					   unsigned int var_idx)
  {
  	struct trace_array *tr = hist_data->event_file->tr;
  	struct hist_field *found = NULL;
  	struct hist_var_data *var_data;
  
  	list_for_each_entry(var_data, &tr->hist_vars, list) {
  		if (var_data->hist_data == hist_data)
  			continue;
  		found = find_var_ref(var_data->hist_data, hist_data, var_idx);
  		if (found)
  			break;
  	}
  
  	return found;
  }
de40f033d   Tom Zanussi   tracing: Remove o...
693
694
695
696
697
698
699
700
701
702
703
704
705
  /**
   * check_var_refs - Check if there is a reference to any of trigger's variables
   * @hist_data: The hist trigger
   *
   * A trigger can define one or more variables.  If any one of them is
   * currently referenced by any other trigger, this function will
   * determine that.
  
   * Typically used to determine whether or not a trigger can be removed
   * - if there are any references to a trigger's variables, it cannot.
   *
   * Return: True if there is a reference to any of trigger's variables
   */
067fe038e   Tom Zanussi   tracing: Add vari...
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
  static bool check_var_refs(struct hist_trigger_data *hist_data)
  {
  	struct hist_field *field;
  	bool found = false;
  	int i;
  
  	for_each_hist_field(i, hist_data) {
  		field = hist_data->fields[i];
  		if (field && field->flags & HIST_FIELD_FL_VAR) {
  			if (find_any_var_ref(hist_data, field->var.idx)) {
  				found = true;
  				break;
  			}
  		}
  	}
  
  	return found;
  }
  
  static struct hist_var_data *find_hist_vars(struct hist_trigger_data *hist_data)
  {
  	struct trace_array *tr = hist_data->event_file->tr;
  	struct hist_var_data *var_data, *found = NULL;
  
  	list_for_each_entry(var_data, &tr->hist_vars, list) {
  		if (var_data->hist_data == hist_data) {
  			found = var_data;
  			break;
  		}
  	}
  
  	return found;
  }
  
  static bool field_has_hist_vars(struct hist_field *hist_field,
  				unsigned int level)
  {
  	int i;
  
  	if (level > 3)
  		return false;
  
  	if (!hist_field)
  		return false;
  
  	if (hist_field->flags & HIST_FIELD_FL_VAR ||
  	    hist_field->flags & HIST_FIELD_FL_VAR_REF)
  		return true;
  
  	for (i = 0; i < HIST_FIELD_OPERANDS_MAX; i++) {
  		struct hist_field *operand;
  
  		operand = hist_field->operands[i];
  		if (field_has_hist_vars(operand, level + 1))
  			return true;
  	}
  
  	return false;
  }
  
  static bool has_hist_vars(struct hist_trigger_data *hist_data)
  {
  	struct hist_field *hist_field;
  	int i;
  
  	for_each_hist_field(i, hist_data) {
  		hist_field = hist_data->fields[i];
  		if (field_has_hist_vars(hist_field, 0))
  			return true;
  	}
  
  	return false;
  }
  
  static int save_hist_vars(struct hist_trigger_data *hist_data)
  {
  	struct trace_array *tr = hist_data->event_file->tr;
  	struct hist_var_data *var_data;
  
  	var_data = find_hist_vars(hist_data);
  	if (var_data)
  		return 0;
8530dec63   Steven Rostedt (VMware)   tracing: Add trac...
788
  	if (tracing_check_open_get_tr(tr))
067fe038e   Tom Zanussi   tracing: Add vari...
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
  		return -ENODEV;
  
  	var_data = kzalloc(sizeof(*var_data), GFP_KERNEL);
  	if (!var_data) {
  		trace_array_put(tr);
  		return -ENOMEM;
  	}
  
  	var_data->hist_data = hist_data;
  	list_add(&var_data->list, &tr->hist_vars);
  
  	return 0;
  }
  
  static void remove_hist_vars(struct hist_trigger_data *hist_data)
  {
  	struct trace_array *tr = hist_data->event_file->tr;
  	struct hist_var_data *var_data;
  
  	var_data = find_hist_vars(hist_data);
  	if (!var_data)
  		return;
  
  	if (WARN_ON(check_var_refs(hist_data)))
  		return;
  
  	list_del(&var_data->list);
  
  	kfree(var_data);
  
  	trace_array_put(tr);
  }
30350d65a   Tom Zanussi   tracing: Add vari...
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
  static struct hist_field *find_var_field(struct hist_trigger_data *hist_data,
  					 const char *var_name)
  {
  	struct hist_field *hist_field, *found = NULL;
  	int i;
  
  	for_each_hist_field(i, hist_data) {
  		hist_field = hist_data->fields[i];
  		if (hist_field && hist_field->flags & HIST_FIELD_FL_VAR &&
  		    strcmp(hist_field->var.name, var_name) == 0) {
  			found = hist_field;
  			break;
  		}
  	}
  
  	return found;
  }
  
  static struct hist_field *find_var(struct hist_trigger_data *hist_data,
  				   struct trace_event_file *file,
  				   const char *var_name)
  {
  	struct hist_trigger_data *test_data;
  	struct event_trigger_data *test;
  	struct hist_field *hist_field;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
846
  	lockdep_assert_held(&event_mutex);
30350d65a   Tom Zanussi   tracing: Add vari...
847
848
849
  	hist_field = find_var_field(hist_data, var_name);
  	if (hist_field)
  		return hist_field;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
850
  	list_for_each_entry(test, &file->triggers, list) {
30350d65a   Tom Zanussi   tracing: Add vari...
851
852
853
854
855
856
857
858
859
860
  		if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
  			test_data = test->private_data;
  			hist_field = find_var_field(test_data, var_name);
  			if (hist_field)
  				return hist_field;
  		}
  	}
  
  	return NULL;
  }
067fe038e   Tom Zanussi   tracing: Add vari...
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
  static struct trace_event_file *find_var_file(struct trace_array *tr,
  					      char *system,
  					      char *event_name,
  					      char *var_name)
  {
  	struct hist_trigger_data *var_hist_data;
  	struct hist_var_data *var_data;
  	struct trace_event_file *file, *found = NULL;
  
  	if (system)
  		return find_event_file(tr, system, event_name);
  
  	list_for_each_entry(var_data, &tr->hist_vars, list) {
  		var_hist_data = var_data->hist_data;
  		file = var_hist_data->event_file;
  		if (file == found)
  			continue;
  
  		if (find_var_field(var_hist_data, var_name)) {
f404da6e1   Tom Zanussi   tracing: Add 'las...
880
  			if (found) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
881
  				hist_err(tr, HIST_ERR_VAR_NOT_UNIQUE, errpos(var_name));
067fe038e   Tom Zanussi   tracing: Add vari...
882
  				return NULL;
f404da6e1   Tom Zanussi   tracing: Add 'las...
883
  			}
067fe038e   Tom Zanussi   tracing: Add vari...
884
885
886
887
888
889
890
891
892
893
894
895
896
897
  
  			found = file;
  		}
  	}
  
  	return found;
  }
  
  static struct hist_field *find_file_var(struct trace_event_file *file,
  					const char *var_name)
  {
  	struct hist_trigger_data *test_data;
  	struct event_trigger_data *test;
  	struct hist_field *hist_field;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
898
899
900
  	lockdep_assert_held(&event_mutex);
  
  	list_for_each_entry(test, &file->triggers, list) {
067fe038e   Tom Zanussi   tracing: Add vari...
901
902
903
904
905
906
907
908
909
910
  		if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
  			test_data = test->private_data;
  			hist_field = find_var_field(test_data, var_name);
  			if (hist_field)
  				return hist_field;
  		}
  	}
  
  	return NULL;
  }
c282a386a   Tom Zanussi   tracing: Add 'onm...
911
912
913
914
915
916
917
918
919
920
  static struct hist_field *
  find_match_var(struct hist_trigger_data *hist_data, char *var_name)
  {
  	struct trace_array *tr = hist_data->event_file->tr;
  	struct hist_field *hist_field, *found = NULL;
  	struct trace_event_file *file;
  	unsigned int i;
  
  	for (i = 0; i < hist_data->n_actions; i++) {
  		struct action_data *data = hist_data->actions[i];
7d18a10c3   Tom Zanussi   tracing: Refactor...
921
  		if (data->handler == HANDLER_ONMATCH) {
c3e49506a   Tom Zanussi   tracing: Split up...
922
923
  			char *system = data->match_data.event_system;
  			char *event_name = data->match_data.event;
c282a386a   Tom Zanussi   tracing: Add 'onm...
924
925
926
927
928
929
930
  
  			file = find_var_file(tr, system, event_name, var_name);
  			if (!file)
  				continue;
  			hist_field = find_file_var(file, var_name);
  			if (hist_field) {
  				if (found) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
931
932
  					hist_err(tr, HIST_ERR_VAR_NOT_UNIQUE,
  						 errpos(var_name));
c282a386a   Tom Zanussi   tracing: Add 'onm...
933
934
935
936
937
938
939
940
941
  					return ERR_PTR(-EINVAL);
  				}
  
  				found = hist_field;
  			}
  		}
  	}
  	return found;
  }
067fe038e   Tom Zanussi   tracing: Add vari...
942
943
944
945
946
947
948
949
  static struct hist_field *find_event_var(struct hist_trigger_data *hist_data,
  					 char *system,
  					 char *event_name,
  					 char *var_name)
  {
  	struct trace_array *tr = hist_data->event_file->tr;
  	struct hist_field *hist_field = NULL;
  	struct trace_event_file *file;
c282a386a   Tom Zanussi   tracing: Add 'onm...
950
951
952
953
954
955
956
  	if (!system || !event_name) {
  		hist_field = find_match_var(hist_data, var_name);
  		if (IS_ERR(hist_field))
  			return NULL;
  		if (hist_field)
  			return hist_field;
  	}
067fe038e   Tom Zanussi   tracing: Add vari...
957
958
959
960
961
962
963
964
  	file = find_var_file(tr, system, event_name, var_name);
  	if (!file)
  		return NULL;
  
  	hist_field = find_file_var(file, var_name);
  
  	return hist_field;
  }
067fe038e   Tom Zanussi   tracing: Add vari...
965
966
967
968
969
970
971
  static u64 hist_field_var_ref(struct hist_field *hist_field,
  			      struct tracing_map_elt *elt,
  			      struct ring_buffer_event *rbe,
  			      void *event)
  {
  	struct hist_elt_data *elt_data;
  	u64 var_val = 0;
55267c88c   Tom Zanussi   tracing: Prevent ...
972
973
  	if (WARN_ON_ONCE(!elt))
  		return var_val;
067fe038e   Tom Zanussi   tracing: Add vari...
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
  	elt_data = elt->private_data;
  	var_val = elt_data->var_ref_vals[hist_field->var_ref_idx];
  
  	return var_val;
  }
  
  static bool resolve_var_refs(struct hist_trigger_data *hist_data, void *key,
  			     u64 *var_ref_vals, bool self)
  {
  	struct hist_trigger_data *var_data;
  	struct tracing_map_elt *var_elt;
  	struct hist_field *hist_field;
  	unsigned int i, var_idx;
  	bool resolved = true;
  	u64 var_val = 0;
  
  	for (i = 0; i < hist_data->n_var_refs; i++) {
  		hist_field = hist_data->var_refs[i];
  		var_idx = hist_field->var.idx;
  		var_data = hist_field->var.hist_data;
  
  		if (var_data == NULL) {
  			resolved = false;
  			break;
  		}
  
  		if ((self && var_data != hist_data) ||
  		    (!self && var_data == hist_data))
  			continue;
  
  		var_elt = tracing_map_lookup(var_data->map, key);
  		if (!var_elt) {
  			resolved = false;
  			break;
  		}
  
  		if (!tracing_map_var_set(var_elt, var_idx)) {
  			resolved = false;
  			break;
  		}
  
  		if (self || !hist_field->read_once)
  			var_val = tracing_map_read_var(var_elt, var_idx);
  		else
  			var_val = tracing_map_read_var_once(var_elt, var_idx);
  
  		var_ref_vals[i] = var_val;
  	}
  
  	return resolved;
  }
85013256c   Tom Zanussi   tracing: Add hist...
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
  static const char *hist_field_name(struct hist_field *field,
  				   unsigned int level)
  {
  	const char *field_name = "";
  
  	if (level > 1)
  		return field_name;
  
  	if (field->field)
  		field_name = field->field->name;
7e8b88a30   Tom Zanussi   tracing: Add hist...
1035
1036
  	else if (field->flags & HIST_FIELD_FL_LOG2 ||
  		 field->flags & HIST_FIELD_FL_ALIAS)
5819eaddf   Tom Zanussi   tracing: Reimplem...
1037
  		field_name = hist_field_name(field->operands[0], ++level);
8b7622bf9   Tom Zanussi   tracing: Add cpu ...
1038
1039
  	else if (field->flags & HIST_FIELD_FL_CPU)
  		field_name = "cpu";
067fe038e   Tom Zanussi   tracing: Add vari...
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
  	else if (field->flags & HIST_FIELD_FL_EXPR ||
  		 field->flags & HIST_FIELD_FL_VAR_REF) {
  		if (field->system) {
  			static char full_name[MAX_FILTER_STR_VAL];
  
  			strcat(full_name, field->system);
  			strcat(full_name, ".");
  			strcat(full_name, field->event_name);
  			strcat(full_name, ".");
  			strcat(full_name, field->name);
  			field_name = full_name;
  		} else
  			field_name = field->name;
0ae7961e7   Tom Zanussi   tracing: Fix disp...
1053
1054
  	} else if (field->flags & HIST_FIELD_FL_TIMESTAMP)
  		field_name = "common_timestamp";
85013256c   Tom Zanussi   tracing: Add hist...
1055
1056
1057
1058
1059
1060
  
  	if (field_name == NULL)
  		field_name = "";
  
  	return field_name;
  }
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
  static hist_field_fn_t select_value_fn(int field_size, int field_is_signed)
  {
  	hist_field_fn_t fn = NULL;
  
  	switch (field_size) {
  	case 8:
  		if (field_is_signed)
  			fn = hist_field_s64;
  		else
  			fn = hist_field_u64;
  		break;
  	case 4:
  		if (field_is_signed)
  			fn = hist_field_s32;
  		else
  			fn = hist_field_u32;
  		break;
  	case 2:
  		if (field_is_signed)
  			fn = hist_field_s16;
  		else
  			fn = hist_field_u16;
  		break;
  	case 1:
  		if (field_is_signed)
  			fn = hist_field_s8;
  		else
  			fn = hist_field_u8;
  		break;
  	}
  
  	return fn;
  }
  
  static int parse_map_size(char *str)
  {
  	unsigned long size, map_bits;
  	int ret;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
  	ret = kstrtoul(str, 0, &size);
  	if (ret)
  		goto out;
  
  	map_bits = ilog2(roundup_pow_of_two(size));
  	if (map_bits < TRACING_MAP_BITS_MIN ||
  	    map_bits > TRACING_MAP_BITS_MAX)
  		ret = -EINVAL;
  	else
  		ret = map_bits;
   out:
  	return ret;
  }
  
  static void destroy_hist_trigger_attrs(struct hist_trigger_attrs *attrs)
  {
30350d65a   Tom Zanussi   tracing: Add vari...
1115
  	unsigned int i;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1116
1117
  	if (!attrs)
  		return;
30350d65a   Tom Zanussi   tracing: Add vari...
1118
1119
  	for (i = 0; i < attrs->n_assignments; i++)
  		kfree(attrs->assignment_str[i]);
0212e2aa3   Tom Zanussi   tracing: Add hist...
1120
1121
  	for (i = 0; i < attrs->n_actions; i++)
  		kfree(attrs->action_str[i]);
5463bfda3   Tom Zanussi   tracing: Add supp...
1122
  	kfree(attrs->name);
e62347d24   Tom Zanussi   tracing: Add hist...
1123
  	kfree(attrs->sort_key_str);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1124
  	kfree(attrs->keys_str);
f2606835d   Tom Zanussi   tracing: Add hist...
1125
  	kfree(attrs->vals_str);
a4072fe85   Tom Zanussi   tracing: Add a cl...
1126
  	kfree(attrs->clock);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1127
1128
  	kfree(attrs);
  }
0212e2aa3   Tom Zanussi   tracing: Add hist...
1129
1130
  static int parse_action(char *str, struct hist_trigger_attrs *attrs)
  {
c282a386a   Tom Zanussi   tracing: Add 'onm...
1131
  	int ret = -EINVAL;
0212e2aa3   Tom Zanussi   tracing: Add hist...
1132
1133
1134
  
  	if (attrs->n_actions >= HIST_ACTIONS_MAX)
  		return ret;
754481e69   Steven Rostedt (VMware)   tracing: Use str_...
1135
  	if ((str_has_prefix(str, "onmatch(")) ||
dff81f559   Tom Zanussi   tracing: Add hist...
1136
1137
  	    (str_has_prefix(str, "onmax(")) ||
  	    (str_has_prefix(str, "onchange("))) {
c282a386a   Tom Zanussi   tracing: Add 'onm...
1138
1139
1140
1141
1142
1143
1144
1145
  		attrs->action_str[attrs->n_actions] = kstrdup(str, GFP_KERNEL);
  		if (!attrs->action_str[attrs->n_actions]) {
  			ret = -ENOMEM;
  			return ret;
  		}
  		attrs->n_actions++;
  		ret = 0;
  	}
0212e2aa3   Tom Zanussi   tracing: Add hist...
1146
1147
  	return ret;
  }
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
1148
1149
  static int parse_assignment(struct trace_array *tr,
  			    char *str, struct hist_trigger_attrs *attrs)
9b1ae035c   Tom Zanussi   tracing: Break ou...
1150
  {
b527b638f   Tom Zanussi   tracing: Simplify...
1151
  	int len, ret = 0;
9b1ae035c   Tom Zanussi   tracing: Break ou...
1152

b527b638f   Tom Zanussi   tracing: Simplify...
1153
1154
1155
  	if ((len = str_has_prefix(str, "key=")) ||
  	    (len = str_has_prefix(str, "keys="))) {
  		attrs->keys_str = kstrdup(str + len, GFP_KERNEL);
9b1ae035c   Tom Zanussi   tracing: Break ou...
1156
1157
1158
1159
  		if (!attrs->keys_str) {
  			ret = -ENOMEM;
  			goto out;
  		}
b527b638f   Tom Zanussi   tracing: Simplify...
1160
1161
1162
1163
  	} else if ((len = str_has_prefix(str, "val=")) ||
  		   (len = str_has_prefix(str, "vals=")) ||
  		   (len = str_has_prefix(str, "values="))) {
  		attrs->vals_str = kstrdup(str + len, GFP_KERNEL);
9b1ae035c   Tom Zanussi   tracing: Break ou...
1164
1165
1166
1167
  		if (!attrs->vals_str) {
  			ret = -ENOMEM;
  			goto out;
  		}
b527b638f   Tom Zanussi   tracing: Simplify...
1168
1169
  	} else if ((len = str_has_prefix(str, "sort="))) {
  		attrs->sort_key_str = kstrdup(str + len, GFP_KERNEL);
9b1ae035c   Tom Zanussi   tracing: Break ou...
1170
1171
1172
1173
  		if (!attrs->sort_key_str) {
  			ret = -ENOMEM;
  			goto out;
  		}
754481e69   Steven Rostedt (VMware)   tracing: Use str_...
1174
  	} else if (str_has_prefix(str, "name=")) {
9b1ae035c   Tom Zanussi   tracing: Break ou...
1175
1176
1177
1178
1179
  		attrs->name = kstrdup(str, GFP_KERNEL);
  		if (!attrs->name) {
  			ret = -ENOMEM;
  			goto out;
  		}
b527b638f   Tom Zanussi   tracing: Simplify...
1180
1181
  	} else if ((len = str_has_prefix(str, "clock="))) {
  		str += len;
a4072fe85   Tom Zanussi   tracing: Add a cl...
1182
1183
1184
1185
1186
1187
1188
  
  		str = strstrip(str);
  		attrs->clock = kstrdup(str, GFP_KERNEL);
  		if (!attrs->clock) {
  			ret = -ENOMEM;
  			goto out;
  		}
b527b638f   Tom Zanussi   tracing: Simplify...
1189
1190
  	} else if ((len = str_has_prefix(str, "size="))) {
  		int map_bits = parse_map_size(str + len);
9b1ae035c   Tom Zanussi   tracing: Break ou...
1191
1192
1193
1194
1195
1196
  
  		if (map_bits < 0) {
  			ret = map_bits;
  			goto out;
  		}
  		attrs->map_bits = map_bits;
30350d65a   Tom Zanussi   tracing: Add vari...
1197
1198
1199
1200
  	} else {
  		char *assignment;
  
  		if (attrs->n_assignments == TRACING_MAP_VARS_MAX) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
1201
  			hist_err(tr, HIST_ERR_TOO_MANY_VARS, errpos(str));
30350d65a   Tom Zanussi   tracing: Add vari...
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
  			ret = -EINVAL;
  			goto out;
  		}
  
  		assignment = kstrdup(str, GFP_KERNEL);
  		if (!assignment) {
  			ret = -ENOMEM;
  			goto out;
  		}
  
  		attrs->assignment_str[attrs->n_assignments++] = assignment;
  	}
9b1ae035c   Tom Zanussi   tracing: Break ou...
1214
1215
1216
   out:
  	return ret;
  }
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
1217
1218
  static struct hist_trigger_attrs *
  parse_hist_trigger_attrs(struct trace_array *tr, char *trigger_str)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
  {
  	struct hist_trigger_attrs *attrs;
  	int ret = 0;
  
  	attrs = kzalloc(sizeof(*attrs), GFP_KERNEL);
  	if (!attrs)
  		return ERR_PTR(-ENOMEM);
  
  	while (trigger_str) {
  		char *str = strsep(&trigger_str, ":");
b527b638f   Tom Zanussi   tracing: Simplify...
1229
  		char *rhs;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1230

b527b638f   Tom Zanussi   tracing: Simplify...
1231
1232
1233
1234
  		rhs = strchr(str, '=');
  		if (rhs) {
  			if (!strlen(++rhs)) {
  				ret = -EINVAL;
4de26c8c9   Tom Zanussi   tracing: Add hist...
1235
  				hist_err(tr, HIST_ERR_EMPTY_ASSIGNMENT, errpos(str));
b527b638f   Tom Zanussi   tracing: Simplify...
1236
1237
  				goto free;
  			}
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
1238
  			ret = parse_assignment(tr, str, attrs);
9b1ae035c   Tom Zanussi   tracing: Break ou...
1239
1240
1241
  			if (ret)
  				goto free;
  		} else if (strcmp(str, "pause") == 0)
83e99914c   Tom Zanussi   tracing: Add hist...
1242
1243
1244
1245
  			attrs->pause = true;
  		else if ((strcmp(str, "cont") == 0) ||
  			 (strcmp(str, "continue") == 0))
  			attrs->cont = true;
e86ae9baa   Tom Zanussi   tracing: Add hist...
1246
1247
  		else if (strcmp(str, "clear") == 0)
  			attrs->clear = true;
9b1ae035c   Tom Zanussi   tracing: Break ou...
1248
  		else {
0212e2aa3   Tom Zanussi   tracing: Add hist...
1249
1250
1251
  			ret = parse_action(str, attrs);
  			if (ret)
  				goto free;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1252
1253
1254
1255
1256
1257
1258
  		}
  	}
  
  	if (!attrs->keys_str) {
  		ret = -EINVAL;
  		goto free;
  	}
a4072fe85   Tom Zanussi   tracing: Add a cl...
1259
1260
1261
1262
1263
1264
1265
  	if (!attrs->clock) {
  		attrs->clock = kstrdup("global", GFP_KERNEL);
  		if (!attrs->clock) {
  			ret = -ENOMEM;
  			goto free;
  		}
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1266
1267
1268
1269
1270
1271
  	return attrs;
   free:
  	destroy_hist_trigger_attrs(attrs);
  
  	return ERR_PTR(ret);
  }
6b4827ad0   Tom Zanussi   tracing: Add hist...
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
  static inline void save_comm(char *comm, struct task_struct *task)
  {
  	if (!task->pid) {
  		strcpy(comm, "<idle>");
  		return;
  	}
  
  	if (WARN_ON_ONCE(task->pid < 0)) {
  		strcpy(comm, "<XXX>");
  		return;
  	}
27242c62b   Tom Zanussi   tracing: Use strn...
1283
  	strncpy(comm, task->comm, TASK_COMM_LEN);
6b4827ad0   Tom Zanussi   tracing: Add hist...
1284
  }
af6a29bca   Tom Zanussi   tracing: Generali...
1285
1286
  static void hist_elt_data_free(struct hist_elt_data *elt_data)
  {
02205a675   Tom Zanussi   tracing: Add supp...
1287
1288
1289
1290
  	unsigned int i;
  
  	for (i = 0; i < SYNTH_FIELDS_MAX; i++)
  		kfree(elt_data->field_var_str[i]);
af6a29bca   Tom Zanussi   tracing: Generali...
1291
1292
1293
1294
1295
  	kfree(elt_data->comm);
  	kfree(elt_data);
  }
  
  static void hist_trigger_elt_data_free(struct tracing_map_elt *elt)
6b4827ad0   Tom Zanussi   tracing: Add hist...
1296
  {
af6a29bca   Tom Zanussi   tracing: Generali...
1297
1298
1299
  	struct hist_elt_data *elt_data = elt->private_data;
  
  	hist_elt_data_free(elt_data);
6b4827ad0   Tom Zanussi   tracing: Add hist...
1300
  }
af6a29bca   Tom Zanussi   tracing: Generali...
1301
  static int hist_trigger_elt_data_alloc(struct tracing_map_elt *elt)
6b4827ad0   Tom Zanussi   tracing: Add hist...
1302
1303
  {
  	struct hist_trigger_data *hist_data = elt->map->private_data;
af6a29bca   Tom Zanussi   tracing: Generali...
1304
1305
  	unsigned int size = TASK_COMM_LEN;
  	struct hist_elt_data *elt_data;
6b4827ad0   Tom Zanussi   tracing: Add hist...
1306
  	struct hist_field *key_field;
02205a675   Tom Zanussi   tracing: Add supp...
1307
  	unsigned int i, n_str;
6b4827ad0   Tom Zanussi   tracing: Add hist...
1308

af6a29bca   Tom Zanussi   tracing: Generali...
1309
1310
1311
  	elt_data = kzalloc(sizeof(*elt_data), GFP_KERNEL);
  	if (!elt_data)
  		return -ENOMEM;
6b4827ad0   Tom Zanussi   tracing: Add hist...
1312
1313
1314
1315
  	for_each_hist_key_field(i, hist_data) {
  		key_field = hist_data->fields[i];
  
  		if (key_field->flags & HIST_FIELD_FL_EXECNAME) {
af6a29bca   Tom Zanussi   tracing: Generali...
1316
1317
1318
  			elt_data->comm = kzalloc(size, GFP_KERNEL);
  			if (!elt_data->comm) {
  				kfree(elt_data);
6b4827ad0   Tom Zanussi   tracing: Add hist...
1319
  				return -ENOMEM;
af6a29bca   Tom Zanussi   tracing: Generali...
1320
  			}
6b4827ad0   Tom Zanussi   tracing: Add hist...
1321
1322
1323
  			break;
  		}
  	}
63a1e5de3   Tom Zanussi   tracing: Save nor...
1324
1325
1326
1327
1328
1329
  	n_str = hist_data->n_field_var_str + hist_data->n_save_var_str +
  		hist_data->n_var_str;
  	if (n_str > SYNTH_FIELDS_MAX) {
  		hist_elt_data_free(elt_data);
  		return -EINVAL;
  	}
02205a675   Tom Zanussi   tracing: Add supp...
1330

4a4a56b4e   Tom Zanussi   tracing: Change S...
1331
  	BUILD_BUG_ON(STR_VAR_LEN_MAX & (sizeof(u64) - 1));
02205a675   Tom Zanussi   tracing: Add supp...
1332
1333
1334
1335
1336
1337
1338
1339
1340
  	size = STR_VAR_LEN_MAX;
  
  	for (i = 0; i < n_str; i++) {
  		elt_data->field_var_str[i] = kzalloc(size, GFP_KERNEL);
  		if (!elt_data->field_var_str[i]) {
  			hist_elt_data_free(elt_data);
  			return -ENOMEM;
  		}
  	}
af6a29bca   Tom Zanussi   tracing: Generali...
1341
  	elt->private_data = elt_data;
6b4827ad0   Tom Zanussi   tracing: Add hist...
1342
1343
  	return 0;
  }
af6a29bca   Tom Zanussi   tracing: Generali...
1344
  static void hist_trigger_elt_data_init(struct tracing_map_elt *elt)
6b4827ad0   Tom Zanussi   tracing: Add hist...
1345
  {
af6a29bca   Tom Zanussi   tracing: Generali...
1346
  	struct hist_elt_data *elt_data = elt->private_data;
6b4827ad0   Tom Zanussi   tracing: Add hist...
1347

af6a29bca   Tom Zanussi   tracing: Generali...
1348
1349
  	if (elt_data->comm)
  		save_comm(elt_data->comm, current);
6b4827ad0   Tom Zanussi   tracing: Add hist...
1350
  }
af6a29bca   Tom Zanussi   tracing: Generali...
1351
1352
1353
1354
  static const struct tracing_map_ops hist_trigger_elt_data_ops = {
  	.elt_alloc	= hist_trigger_elt_data_alloc,
  	.elt_free	= hist_trigger_elt_data_free,
  	.elt_init	= hist_trigger_elt_data_init,
6b4827ad0   Tom Zanussi   tracing: Add hist...
1355
  };
2ece94fbd   Tom Zanussi   tracing: Move get...
1356
1357
1358
1359
1360
1361
1362
1363
1364
1365
1366
1367
1368
1369
1370
1371
1372
1373
1374
1375
1376
  static const char *get_hist_field_flags(struct hist_field *hist_field)
  {
  	const char *flags_str = NULL;
  
  	if (hist_field->flags & HIST_FIELD_FL_HEX)
  		flags_str = "hex";
  	else if (hist_field->flags & HIST_FIELD_FL_SYM)
  		flags_str = "sym";
  	else if (hist_field->flags & HIST_FIELD_FL_SYM_OFFSET)
  		flags_str = "sym-offset";
  	else if (hist_field->flags & HIST_FIELD_FL_EXECNAME)
  		flags_str = "execname";
  	else if (hist_field->flags & HIST_FIELD_FL_SYSCALL)
  		flags_str = "syscall";
  	else if (hist_field->flags & HIST_FIELD_FL_LOG2)
  		flags_str = "log2";
  	else if (hist_field->flags & HIST_FIELD_FL_TIMESTAMP_USECS)
  		flags_str = "usecs";
  
  	return flags_str;
  }
100719dce   Tom Zanussi   tracing: Add simp...
1377
1378
  static void expr_field_str(struct hist_field *field, char *expr)
  {
067fe038e   Tom Zanussi   tracing: Add vari...
1379
1380
  	if (field->flags & HIST_FIELD_FL_VAR_REF)
  		strcat(expr, "$");
100719dce   Tom Zanussi   tracing: Add simp...
1381
  	strcat(expr, hist_field_name(field, 0));
76690945f   Tom Zanussi   tracing: Don't ad...
1382
  	if (field->flags && !(field->flags & HIST_FIELD_FL_VAR_REF)) {
100719dce   Tom Zanussi   tracing: Add simp...
1383
1384
1385
1386
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
1404
1405
1406
1407
1408
1409
1410
1411
1412
1413
1414
1415
1416
1417
1418
1419
1420
1421
1422
1423
1424
1425
1426
1427
1428
1429
1430
1431
1432
1433
1434
1435
1436
1437
1438
1439
1440
1441
1442
1443
1444
1445
1446
1447
1448
1449
1450
1451
1452
1453
1454
1455
1456
1457
1458
1459
1460
1461
1462
1463
1464
1465
1466
1467
1468
  		const char *flags_str = get_hist_field_flags(field);
  
  		if (flags_str) {
  			strcat(expr, ".");
  			strcat(expr, flags_str);
  		}
  	}
  }
  
  static char *expr_str(struct hist_field *field, unsigned int level)
  {
  	char *expr;
  
  	if (level > 1)
  		return NULL;
  
  	expr = kzalloc(MAX_FILTER_STR_VAL, GFP_KERNEL);
  	if (!expr)
  		return NULL;
  
  	if (!field->operands[0]) {
  		expr_field_str(field, expr);
  		return expr;
  	}
  
  	if (field->operator == FIELD_OP_UNARY_MINUS) {
  		char *subexpr;
  
  		strcat(expr, "-(");
  		subexpr = expr_str(field->operands[0], ++level);
  		if (!subexpr) {
  			kfree(expr);
  			return NULL;
  		}
  		strcat(expr, subexpr);
  		strcat(expr, ")");
  
  		kfree(subexpr);
  
  		return expr;
  	}
  
  	expr_field_str(field->operands[0], expr);
  
  	switch (field->operator) {
  	case FIELD_OP_MINUS:
  		strcat(expr, "-");
  		break;
  	case FIELD_OP_PLUS:
  		strcat(expr, "+");
  		break;
  	default:
  		kfree(expr);
  		return NULL;
  	}
  
  	expr_field_str(field->operands[1], expr);
  
  	return expr;
  }
  
  static int contains_operator(char *str)
  {
  	enum field_op_id field_op = FIELD_OP_NONE;
  	char *op;
  
  	op = strpbrk(str, "+-");
  	if (!op)
  		return FIELD_OP_NONE;
  
  	switch (*op) {
  	case '-':
  		if (*str == '-')
  			field_op = FIELD_OP_UNARY_MINUS;
  		else
  			field_op = FIELD_OP_MINUS;
  		break;
  	case '+':
  		field_op = FIELD_OP_PLUS;
  		break;
  	default:
  		break;
  	}
  
  	return field_op;
  }
8bcebc77e   Steven Rostedt (VMware)   tracing: Fix hist...
1469
1470
1471
1472
  static void get_hist_field(struct hist_field *hist_field)
  {
  	hist_field->ref++;
  }
656fe2ba8   Tom Zanussi   tracing: Use hist...
1473
1474
  static void __destroy_hist_field(struct hist_field *hist_field)
  {
8bcebc77e   Steven Rostedt (VMware)   tracing: Fix hist...
1475
1476
  	if (--hist_field->ref > 1)
  		return;
656fe2ba8   Tom Zanussi   tracing: Use hist...
1477
1478
1479
  	kfree(hist_field->var.name);
  	kfree(hist_field->name);
  	kfree(hist_field->type);
9da73974e   Vamshi K Sthambamkadi   tracing: Fix memo...
1480
1481
  	kfree(hist_field->system);
  	kfree(hist_field->event_name);
656fe2ba8   Tom Zanussi   tracing: Use hist...
1482
1483
  	kfree(hist_field);
  }
5819eaddf   Tom Zanussi   tracing: Reimplem...
1484
1485
  static void destroy_hist_field(struct hist_field *hist_field,
  			       unsigned int level)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1486
  {
5819eaddf   Tom Zanussi   tracing: Reimplem...
1487
  	unsigned int i;
100719dce   Tom Zanussi   tracing: Add simp...
1488
  	if (level > 3)
5819eaddf   Tom Zanussi   tracing: Reimplem...
1489
1490
1491
1492
  		return;
  
  	if (!hist_field)
  		return;
656fe2ba8   Tom Zanussi   tracing: Use hist...
1493
1494
  	if (hist_field->flags & HIST_FIELD_FL_VAR_REF)
  		return; /* var refs will be destroyed separately */
5819eaddf   Tom Zanussi   tracing: Reimplem...
1495
1496
  	for (i = 0; i < HIST_FIELD_OPERANDS_MAX; i++)
  		destroy_hist_field(hist_field->operands[i], level + 1);
656fe2ba8   Tom Zanussi   tracing: Use hist...
1497
  	__destroy_hist_field(hist_field);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1498
  }
b559d003a   Tom Zanussi   tracing: Add hist...
1499
1500
  static struct hist_field *create_hist_field(struct hist_trigger_data *hist_data,
  					    struct ftrace_event_field *field,
30350d65a   Tom Zanussi   tracing: Add vari...
1501
1502
  					    unsigned long flags,
  					    char *var_name)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1503
1504
1505
1506
1507
1508
1509
1510
1511
  {
  	struct hist_field *hist_field;
  
  	if (field && is_function_field(field))
  		return NULL;
  
  	hist_field = kzalloc(sizeof(struct hist_field), GFP_KERNEL);
  	if (!hist_field)
  		return NULL;
8bcebc77e   Steven Rostedt (VMware)   tracing: Fix hist...
1512
  	hist_field->ref = 1;
b559d003a   Tom Zanussi   tracing: Add hist...
1513
  	hist_field->hist_data = hist_data;
7e8b88a30   Tom Zanussi   tracing: Add hist...
1514
  	if (flags & HIST_FIELD_FL_EXPR || flags & HIST_FIELD_FL_ALIAS)
100719dce   Tom Zanussi   tracing: Add simp...
1515
  		goto out; /* caller will populate */
067fe038e   Tom Zanussi   tracing: Add vari...
1516
1517
1518
1519
  	if (flags & HIST_FIELD_FL_VAR_REF) {
  		hist_field->fn = hist_field_var_ref;
  		goto out;
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1520
1521
  	if (flags & HIST_FIELD_FL_HITCOUNT) {
  		hist_field->fn = hist_field_counter;
19a9facd0   Tom Zanussi   tracing: Add hist...
1522
1523
1524
1525
  		hist_field->size = sizeof(u64);
  		hist_field->type = kstrdup("u64", GFP_KERNEL);
  		if (!hist_field->type)
  			goto free;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1526
1527
  		goto out;
  	}
69a0200c2   Tom Zanussi   tracing: Add hist...
1528
1529
1530
1531
  	if (flags & HIST_FIELD_FL_STACKTRACE) {
  		hist_field->fn = hist_field_none;
  		goto out;
  	}
4b94f5b7b   Namhyung Kim   tracing: Add hist...
1532
  	if (flags & HIST_FIELD_FL_LOG2) {
5819eaddf   Tom Zanussi   tracing: Reimplem...
1533
  		unsigned long fl = flags & ~HIST_FIELD_FL_LOG2;
4b94f5b7b   Namhyung Kim   tracing: Add hist...
1534
  		hist_field->fn = hist_field_log2;
30350d65a   Tom Zanussi   tracing: Add vari...
1535
  		hist_field->operands[0] = create_hist_field(hist_data, field, fl, NULL);
5819eaddf   Tom Zanussi   tracing: Reimplem...
1536
  		hist_field->size = hist_field->operands[0]->size;
19a9facd0   Tom Zanussi   tracing: Add hist...
1537
1538
1539
  		hist_field->type = kstrdup(hist_field->operands[0]->type, GFP_KERNEL);
  		if (!hist_field->type)
  			goto free;
4b94f5b7b   Namhyung Kim   tracing: Add hist...
1540
1541
  		goto out;
  	}
ad42febe5   Tom Zanussi   tracing: Add hist...
1542
1543
1544
  	if (flags & HIST_FIELD_FL_TIMESTAMP) {
  		hist_field->fn = hist_field_timestamp;
  		hist_field->size = sizeof(u64);
19a9facd0   Tom Zanussi   tracing: Add hist...
1545
1546
1547
  		hist_field->type = kstrdup("u64", GFP_KERNEL);
  		if (!hist_field->type)
  			goto free;
ad42febe5   Tom Zanussi   tracing: Add hist...
1548
1549
  		goto out;
  	}
8b7622bf9   Tom Zanussi   tracing: Add cpu ...
1550
1551
1552
1553
1554
1555
1556
1557
  	if (flags & HIST_FIELD_FL_CPU) {
  		hist_field->fn = hist_field_cpu;
  		hist_field->size = sizeof(int);
  		hist_field->type = kstrdup("unsigned int", GFP_KERNEL);
  		if (!hist_field->type)
  			goto free;
  		goto out;
  	}
432480c58   Tom Zanussi   tracing: Add chec...
1558
1559
  	if (WARN_ON_ONCE(!field))
  		goto out;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1560
1561
  	if (is_string_field(field)) {
  		flags |= HIST_FIELD_FL_STRING;
79e577cbc   Namhyung Kim   tracing: Support ...
1562

19a9facd0   Tom Zanussi   tracing: Add hist...
1563
1564
1565
1566
  		hist_field->size = MAX_FILTER_STR_VAL;
  		hist_field->type = kstrdup(field->type, GFP_KERNEL);
  		if (!hist_field->type)
  			goto free;
79e577cbc   Namhyung Kim   tracing: Support ...
1567
1568
1569
1570
1571
1572
  		if (field->filter_type == FILTER_STATIC_STRING)
  			hist_field->fn = hist_field_string;
  		else if (field->filter_type == FILTER_DYN_STRING)
  			hist_field->fn = hist_field_dynstring;
  		else
  			hist_field->fn = hist_field_pstring;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1573
  	} else {
19a9facd0   Tom Zanussi   tracing: Add hist...
1574
1575
1576
1577
1578
  		hist_field->size = field->size;
  		hist_field->is_signed = field->is_signed;
  		hist_field->type = kstrdup(field->type, GFP_KERNEL);
  		if (!hist_field->type)
  			goto free;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1579
1580
1581
  		hist_field->fn = select_value_fn(field->size,
  						 field->is_signed);
  		if (!hist_field->fn) {
5819eaddf   Tom Zanussi   tracing: Reimplem...
1582
  			destroy_hist_field(hist_field, 0);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1583
1584
1585
1586
1587
1588
  			return NULL;
  		}
  	}
   out:
  	hist_field->field = field;
  	hist_field->flags = flags;
30350d65a   Tom Zanussi   tracing: Add vari...
1589
1590
1591
1592
1593
  	if (var_name) {
  		hist_field->var.name = kstrdup(var_name, GFP_KERNEL);
  		if (!hist_field->var.name)
  			goto free;
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1594
  	return hist_field;
30350d65a   Tom Zanussi   tracing: Add vari...
1595
1596
1597
   free:
  	destroy_hist_field(hist_field, 0);
  	return NULL;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1598
1599
1600
1601
1602
  }
  
  static void destroy_hist_fields(struct hist_trigger_data *hist_data)
  {
  	unsigned int i;
30350d65a   Tom Zanussi   tracing: Add vari...
1603
  	for (i = 0; i < HIST_FIELDS_MAX; i++) {
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1604
  		if (hist_data->fields[i]) {
5819eaddf   Tom Zanussi   tracing: Reimplem...
1605
  			destroy_hist_field(hist_data->fields[i], 0);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1606
1607
1608
  			hist_data->fields[i] = NULL;
  		}
  	}
656fe2ba8   Tom Zanussi   tracing: Use hist...
1609
1610
1611
1612
1613
1614
  
  	for (i = 0; i < hist_data->n_var_refs; i++) {
  		WARN_ON(!(hist_data->var_refs[i]->flags & HIST_FIELD_FL_VAR_REF));
  		__destroy_hist_field(hist_data->var_refs[i]);
  		hist_data->var_refs[i] = NULL;
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
1615
  }
067fe038e   Tom Zanussi   tracing: Add vari...
1616
1617
1618
1619
1620
1621
1622
1623
1624
1625
1626
1627
1628
1629
1630
1631
1632
1633
1634
1635
1636
1637
1638
1639
1640
1641
  static int init_var_ref(struct hist_field *ref_field,
  			struct hist_field *var_field,
  			char *system, char *event_name)
  {
  	int err = 0;
  
  	ref_field->var.idx = var_field->var.idx;
  	ref_field->var.hist_data = var_field->hist_data;
  	ref_field->size = var_field->size;
  	ref_field->is_signed = var_field->is_signed;
  	ref_field->flags |= var_field->flags &
  		(HIST_FIELD_FL_TIMESTAMP | HIST_FIELD_FL_TIMESTAMP_USECS);
  
  	if (system) {
  		ref_field->system = kstrdup(system, GFP_KERNEL);
  		if (!ref_field->system)
  			return -ENOMEM;
  	}
  
  	if (event_name) {
  		ref_field->event_name = kstrdup(event_name, GFP_KERNEL);
  		if (!ref_field->event_name) {
  			err = -ENOMEM;
  			goto free;
  		}
  	}
7e8b88a30   Tom Zanussi   tracing: Add hist...
1642
1643
1644
1645
1646
1647
1648
1649
1650
1651
1652
1653
  	if (var_field->var.name) {
  		ref_field->name = kstrdup(var_field->var.name, GFP_KERNEL);
  		if (!ref_field->name) {
  			err = -ENOMEM;
  			goto free;
  		}
  	} else if (var_field->name) {
  		ref_field->name = kstrdup(var_field->name, GFP_KERNEL);
  		if (!ref_field->name) {
  			err = -ENOMEM;
  			goto free;
  		}
067fe038e   Tom Zanussi   tracing: Add vari...
1654
1655
1656
1657
1658
1659
1660
1661
1662
1663
1664
1665
1666
1667
1668
1669
  	}
  
  	ref_field->type = kstrdup(var_field->type, GFP_KERNEL);
  	if (!ref_field->type) {
  		err = -ENOMEM;
  		goto free;
  	}
   out:
  	return err;
   free:
  	kfree(ref_field->system);
  	kfree(ref_field->event_name);
  	kfree(ref_field->name);
  
  	goto out;
  }
d380dcde9   Tom Zanussi   tracing: Fix now ...
1670
1671
1672
1673
1674
1675
1676
1677
1678
1679
1680
1681
1682
1683
1684
  static int find_var_ref_idx(struct hist_trigger_data *hist_data,
  			    struct hist_field *var_field)
  {
  	struct hist_field *ref_field;
  	int i;
  
  	for (i = 0; i < hist_data->n_var_refs; i++) {
  		ref_field = hist_data->var_refs[i];
  		if (ref_field->var.idx == var_field->var.idx &&
  		    ref_field->var.hist_data == var_field->hist_data)
  			return i;
  	}
  
  	return -ENOENT;
  }
de40f033d   Tom Zanussi   tracing: Remove o...
1685
1686
1687
1688
1689
1690
1691
1692
1693
1694
1695
1696
1697
1698
1699
1700
1701
  /**
   * create_var_ref - Create a variable reference and attach it to trigger
   * @hist_data: The trigger that will be referencing the variable
   * @var_field: The VAR field to create a reference to
   * @system: The optional system string
   * @event_name: The optional event_name string
   *
   * Given a variable hist_field, create a VAR_REF hist_field that
   * represents a reference to it.
   *
   * This function also adds the reference to the trigger that
   * now references the variable.
   *
   * Return: The VAR_REF field if successful, NULL if not
   */
  static struct hist_field *create_var_ref(struct hist_trigger_data *hist_data,
  					 struct hist_field *var_field,
067fe038e   Tom Zanussi   tracing: Add vari...
1702
1703
1704
1705
  					 char *system, char *event_name)
  {
  	unsigned long flags = HIST_FIELD_FL_VAR_REF;
  	struct hist_field *ref_field;
8bcebc77e   Steven Rostedt (VMware)   tracing: Fix hist...
1706
1707
1708
1709
1710
1711
1712
1713
1714
1715
1716
  	int i;
  
  	/* Check if the variable already exists */
  	for (i = 0; i < hist_data->n_var_refs; i++) {
  		ref_field = hist_data->var_refs[i];
  		if (ref_field->var.idx == var_field->var.idx &&
  		    ref_field->var.hist_data == var_field->hist_data) {
  			get_hist_field(ref_field);
  			return ref_field;
  		}
  	}
067fe038e   Tom Zanussi   tracing: Add vari...
1717
1718
1719
1720
1721
1722
1723
  
  	ref_field = create_hist_field(var_field->hist_data, NULL, flags, NULL);
  	if (ref_field) {
  		if (init_var_ref(ref_field, var_field, system, event_name)) {
  			destroy_hist_field(ref_field, 0);
  			return NULL;
  		}
de40f033d   Tom Zanussi   tracing: Remove o...
1724
1725
1726
  
  		hist_data->var_refs[hist_data->n_var_refs] = ref_field;
  		ref_field->var_ref_idx = hist_data->n_var_refs++;
067fe038e   Tom Zanussi   tracing: Add vari...
1727
1728
1729
1730
1731
1732
1733
1734
1735
1736
1737
1738
1739
1740
1741
1742
1743
1744
1745
1746
1747
1748
1749
1750
1751
1752
1753
1754
1755
1756
1757
1758
1759
1760
1761
1762
1763
1764
1765
1766
1767
1768
1769
1770
1771
1772
1773
1774
1775
1776
1777
1778
1779
1780
1781
1782
1783
1784
1785
1786
1787
1788
1789
1790
1791
  	}
  
  	return ref_field;
  }
  
  static bool is_var_ref(char *var_name)
  {
  	if (!var_name || strlen(var_name) < 2 || var_name[0] != '$')
  		return false;
  
  	return true;
  }
  
  static char *field_name_from_var(struct hist_trigger_data *hist_data,
  				 char *var_name)
  {
  	char *name, *field;
  	unsigned int i;
  
  	for (i = 0; i < hist_data->attrs->var_defs.n_vars; i++) {
  		name = hist_data->attrs->var_defs.name[i];
  
  		if (strcmp(var_name, name) == 0) {
  			field = hist_data->attrs->var_defs.expr[i];
  			if (contains_operator(field) || is_var_ref(field))
  				continue;
  			return field;
  		}
  	}
  
  	return NULL;
  }
  
  static char *local_field_var_ref(struct hist_trigger_data *hist_data,
  				 char *system, char *event_name,
  				 char *var_name)
  {
  	struct trace_event_call *call;
  
  	if (system && event_name) {
  		call = hist_data->event_file->event_call;
  
  		if (strcmp(system, call->class->system) != 0)
  			return NULL;
  
  		if (strcmp(event_name, trace_event_name(call)) != 0)
  			return NULL;
  	}
  
  	if (!!system != !!event_name)
  		return NULL;
  
  	if (!is_var_ref(var_name))
  		return NULL;
  
  	var_name++;
  
  	return field_name_from_var(hist_data, var_name);
  }
  
  static struct hist_field *parse_var_ref(struct hist_trigger_data *hist_data,
  					char *system, char *event_name,
  					char *var_name)
  {
  	struct hist_field *var_field = NULL, *ref_field = NULL;
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
1792
  	struct trace_array *tr = hist_data->event_file->tr;
067fe038e   Tom Zanussi   tracing: Add vari...
1793
1794
1795
1796
1797
1798
1799
1800
  
  	if (!is_var_ref(var_name))
  		return NULL;
  
  	var_name++;
  
  	var_field = find_event_var(hist_data, system, event_name, var_name);
  	if (var_field)
de40f033d   Tom Zanussi   tracing: Remove o...
1801
1802
  		ref_field = create_var_ref(hist_data, var_field,
  					   system, event_name);
067fe038e   Tom Zanussi   tracing: Add vari...
1803

f404da6e1   Tom Zanussi   tracing: Add 'las...
1804
  	if (!ref_field)
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
1805
  		hist_err(tr, HIST_ERR_VAR_NOT_FOUND, errpos(var_name));
f404da6e1   Tom Zanussi   tracing: Add 'las...
1806

067fe038e   Tom Zanussi   tracing: Add vari...
1807
1808
  	return ref_field;
  }
100719dce   Tom Zanussi   tracing: Add simp...
1809
1810
1811
1812
1813
1814
  static struct ftrace_event_field *
  parse_field(struct hist_trigger_data *hist_data, struct trace_event_file *file,
  	    char *field_str, unsigned long *flags)
  {
  	struct ftrace_event_field *field = NULL;
  	char *field_name, *modifier, *str;
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
1815
  	struct trace_array *tr = file->tr;
100719dce   Tom Zanussi   tracing: Add simp...
1816
1817
1818
1819
1820
1821
1822
1823
1824
1825
1826
1827
1828
1829
1830
1831
1832
1833
1834
1835
1836
1837
1838
  
  	modifier = str = kstrdup(field_str, GFP_KERNEL);
  	if (!modifier)
  		return ERR_PTR(-ENOMEM);
  
  	field_name = strsep(&modifier, ".");
  	if (modifier) {
  		if (strcmp(modifier, "hex") == 0)
  			*flags |= HIST_FIELD_FL_HEX;
  		else if (strcmp(modifier, "sym") == 0)
  			*flags |= HIST_FIELD_FL_SYM;
  		else if (strcmp(modifier, "sym-offset") == 0)
  			*flags |= HIST_FIELD_FL_SYM_OFFSET;
  		else if ((strcmp(modifier, "execname") == 0) &&
  			 (strcmp(field_name, "common_pid") == 0))
  			*flags |= HIST_FIELD_FL_EXECNAME;
  		else if (strcmp(modifier, "syscall") == 0)
  			*flags |= HIST_FIELD_FL_SYSCALL;
  		else if (strcmp(modifier, "log2") == 0)
  			*flags |= HIST_FIELD_FL_LOG2;
  		else if (strcmp(modifier, "usecs") == 0)
  			*flags |= HIST_FIELD_FL_TIMESTAMP_USECS;
  		else {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
1839
  			hist_err(tr, HIST_ERR_BAD_FIELD_MODIFIER, errpos(modifier));
100719dce   Tom Zanussi   tracing: Add simp...
1840
1841
1842
1843
1844
1845
1846
1847
1848
1849
  			field = ERR_PTR(-EINVAL);
  			goto out;
  		}
  	}
  
  	if (strcmp(field_name, "common_timestamp") == 0) {
  		*flags |= HIST_FIELD_FL_TIMESTAMP;
  		hist_data->enable_timestamps = true;
  		if (*flags & HIST_FIELD_FL_TIMESTAMP_USECS)
  			hist_data->attrs->ts_in_usecs = true;
8b7622bf9   Tom Zanussi   tracing: Add cpu ...
1850
1851
1852
  	} else if (strcmp(field_name, "cpu") == 0)
  		*flags |= HIST_FIELD_FL_CPU;
  	else {
100719dce   Tom Zanussi   tracing: Add simp...
1853
1854
  		field = trace_find_event_field(file->event_call, field_name);
  		if (!field || !field->size) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
1855
  			hist_err(tr, HIST_ERR_FIELD_NOT_FOUND, errpos(field_name));
100719dce   Tom Zanussi   tracing: Add simp...
1856
1857
1858
1859
1860
1861
1862
1863
1864
  			field = ERR_PTR(-EINVAL);
  			goto out;
  		}
  	}
   out:
  	kfree(str);
  
  	return field;
  }
7e8b88a30   Tom Zanussi   tracing: Add hist...
1865
1866
1867
1868
1869
1870
1871
1872
1873
1874
1875
1876
1877
1878
1879
1880
1881
1882
  static struct hist_field *create_alias(struct hist_trigger_data *hist_data,
  				       struct hist_field *var_ref,
  				       char *var_name)
  {
  	struct hist_field *alias = NULL;
  	unsigned long flags = HIST_FIELD_FL_ALIAS | HIST_FIELD_FL_VAR;
  
  	alias = create_hist_field(hist_data, NULL, flags, var_name);
  	if (!alias)
  		return NULL;
  
  	alias->fn = var_ref->fn;
  	alias->operands[0] = var_ref;
  
  	if (init_var_ref(alias, var_ref, var_ref->system, var_ref->event_name)) {
  		destroy_hist_field(alias, 0);
  		return NULL;
  	}
17f8607a1   Tom Zanussi   tracing: Make sur...
1883
  	alias->var_ref_idx = var_ref->var_ref_idx;
7e8b88a30   Tom Zanussi   tracing: Add hist...
1884
1885
  	return alias;
  }
100719dce   Tom Zanussi   tracing: Add simp...
1886
1887
1888
1889
  static struct hist_field *parse_atom(struct hist_trigger_data *hist_data,
  				     struct trace_event_file *file, char *str,
  				     unsigned long *flags, char *var_name)
  {
067fe038e   Tom Zanussi   tracing: Add vari...
1890
  	char *s, *ref_system = NULL, *ref_event = NULL, *ref_var = str;
100719dce   Tom Zanussi   tracing: Add simp...
1891
1892
1893
  	struct ftrace_event_field *field = NULL;
  	struct hist_field *hist_field = NULL;
  	int ret = 0;
067fe038e   Tom Zanussi   tracing: Add vari...
1894
1895
1896
1897
1898
1899
1900
1901
1902
1903
1904
1905
1906
1907
1908
1909
1910
1911
1912
1913
  	s = strchr(str, '.');
  	if (s) {
  		s = strchr(++s, '.');
  		if (s) {
  			ref_system = strsep(&str, ".");
  			if (!str) {
  				ret = -EINVAL;
  				goto out;
  			}
  			ref_event = strsep(&str, ".");
  			if (!str) {
  				ret = -EINVAL;
  				goto out;
  			}
  			ref_var = str;
  		}
  	}
  
  	s = local_field_var_ref(hist_data, ref_system, ref_event, ref_var);
  	if (!s) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
1914
1915
  		hist_field = parse_var_ref(hist_data, ref_system,
  					   ref_event, ref_var);
067fe038e   Tom Zanussi   tracing: Add vari...
1916
  		if (hist_field) {
7e8b88a30   Tom Zanussi   tracing: Add hist...
1917
1918
1919
1920
1921
1922
1923
  			if (var_name) {
  				hist_field = create_alias(hist_data, hist_field, var_name);
  				if (!hist_field) {
  					ret = -ENOMEM;
  					goto out;
  				}
  			}
067fe038e   Tom Zanussi   tracing: Add vari...
1924
1925
1926
1927
  			return hist_field;
  		}
  	} else
  		str = s;
100719dce   Tom Zanussi   tracing: Add simp...
1928
1929
1930
1931
1932
1933
1934
1935
1936
1937
1938
1939
1940
1941
1942
1943
1944
1945
1946
1947
1948
1949
1950
1951
1952
1953
1954
1955
1956
1957
1958
1959
1960
1961
1962
  	field = parse_field(hist_data, file, str, flags);
  	if (IS_ERR(field)) {
  		ret = PTR_ERR(field);
  		goto out;
  	}
  
  	hist_field = create_hist_field(hist_data, field, *flags, var_name);
  	if (!hist_field) {
  		ret = -ENOMEM;
  		goto out;
  	}
  
  	return hist_field;
   out:
  	return ERR_PTR(ret);
  }
  
  static struct hist_field *parse_expr(struct hist_trigger_data *hist_data,
  				     struct trace_event_file *file,
  				     char *str, unsigned long flags,
  				     char *var_name, unsigned int level);
  
  static struct hist_field *parse_unary(struct hist_trigger_data *hist_data,
  				      struct trace_event_file *file,
  				      char *str, unsigned long flags,
  				      char *var_name, unsigned int level)
  {
  	struct hist_field *operand1, *expr = NULL;
  	unsigned long operand_flags;
  	int ret = 0;
  	char *s;
  
  	/* we support only -(xxx) i.e. explicit parens required */
  
  	if (level > 3) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
1963
  		hist_err(file->tr, HIST_ERR_TOO_MANY_SUBEXPR, errpos(str));
100719dce   Tom Zanussi   tracing: Add simp...
1964
1965
1966
1967
1968
1969
1970
1971
1972
1973
1974
1975
1976
1977
1978
1979
1980
1981
1982
1983
1984
1985
1986
1987
1988
1989
1990
1991
1992
1993
1994
1995
1996
1997
1998
1999
2000
2001
2002
2003
2004
2005
  		ret = -EINVAL;
  		goto free;
  	}
  
  	str++; /* skip leading '-' */
  
  	s = strchr(str, '(');
  	if (s)
  		str++;
  	else {
  		ret = -EINVAL;
  		goto free;
  	}
  
  	s = strrchr(str, ')');
  	if (s)
  		*s = '\0';
  	else {
  		ret = -EINVAL; /* no closing ')' */
  		goto free;
  	}
  
  	flags |= HIST_FIELD_FL_EXPR;
  	expr = create_hist_field(hist_data, NULL, flags, var_name);
  	if (!expr) {
  		ret = -ENOMEM;
  		goto free;
  	}
  
  	operand_flags = 0;
  	operand1 = parse_expr(hist_data, file, str, operand_flags, NULL, ++level);
  	if (IS_ERR(operand1)) {
  		ret = PTR_ERR(operand1);
  		goto free;
  	}
  
  	expr->flags |= operand1->flags &
  		(HIST_FIELD_FL_TIMESTAMP | HIST_FIELD_FL_TIMESTAMP_USECS);
  	expr->fn = hist_field_unary_minus;
  	expr->operands[0] = operand1;
  	expr->operator = FIELD_OP_UNARY_MINUS;
  	expr->name = expr_str(expr, 0);
19a9facd0   Tom Zanussi   tracing: Add hist...
2006
2007
2008
2009
2010
  	expr->type = kstrdup(operand1->type, GFP_KERNEL);
  	if (!expr->type) {
  		ret = -ENOMEM;
  		goto free;
  	}
100719dce   Tom Zanussi   tracing: Add simp...
2011
2012
2013
2014
2015
2016
  
  	return expr;
   free:
  	destroy_hist_field(expr, 0);
  	return ERR_PTR(ret);
  }
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2017
2018
  static int check_expr_operands(struct trace_array *tr,
  			       struct hist_field *operand1,
100719dce   Tom Zanussi   tracing: Add simp...
2019
2020
2021
2022
  			       struct hist_field *operand2)
  {
  	unsigned long operand1_flags = operand1->flags;
  	unsigned long operand2_flags = operand2->flags;
7e8b88a30   Tom Zanussi   tracing: Add hist...
2023
2024
2025
2026
2027
2028
2029
2030
2031
2032
2033
2034
2035
2036
2037
2038
2039
2040
2041
  	if ((operand1_flags & HIST_FIELD_FL_VAR_REF) ||
  	    (operand1_flags & HIST_FIELD_FL_ALIAS)) {
  		struct hist_field *var;
  
  		var = find_var_field(operand1->var.hist_data, operand1->name);
  		if (!var)
  			return -EINVAL;
  		operand1_flags = var->flags;
  	}
  
  	if ((operand2_flags & HIST_FIELD_FL_VAR_REF) ||
  	    (operand2_flags & HIST_FIELD_FL_ALIAS)) {
  		struct hist_field *var;
  
  		var = find_var_field(operand2->var.hist_data, operand2->name);
  		if (!var)
  			return -EINVAL;
  		operand2_flags = var->flags;
  	}
100719dce   Tom Zanussi   tracing: Add simp...
2042
  	if ((operand1_flags & HIST_FIELD_FL_TIMESTAMP_USECS) !=
f404da6e1   Tom Zanussi   tracing: Add 'las...
2043
  	    (operand2_flags & HIST_FIELD_FL_TIMESTAMP_USECS)) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2044
  		hist_err(tr, HIST_ERR_TIMESTAMP_MISMATCH, 0);
100719dce   Tom Zanussi   tracing: Add simp...
2045
  		return -EINVAL;
f404da6e1   Tom Zanussi   tracing: Add 'las...
2046
  	}
100719dce   Tom Zanussi   tracing: Add simp...
2047
2048
2049
2050
2051
2052
2053
2054
2055
2056
2057
2058
2059
  
  	return 0;
  }
  
  static struct hist_field *parse_expr(struct hist_trigger_data *hist_data,
  				     struct trace_event_file *file,
  				     char *str, unsigned long flags,
  				     char *var_name, unsigned int level)
  {
  	struct hist_field *operand1 = NULL, *operand2 = NULL, *expr = NULL;
  	unsigned long operand_flags;
  	int field_op, ret = -EINVAL;
  	char *sep, *operand1_str;
f404da6e1   Tom Zanussi   tracing: Add 'las...
2060
  	if (level > 3) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2061
  		hist_err(file->tr, HIST_ERR_TOO_MANY_SUBEXPR, errpos(str));
100719dce   Tom Zanussi   tracing: Add simp...
2062
  		return ERR_PTR(-EINVAL);
f404da6e1   Tom Zanussi   tracing: Add 'las...
2063
  	}
100719dce   Tom Zanussi   tracing: Add simp...
2064
2065
2066
2067
2068
2069
2070
2071
2072
2073
2074
2075
2076
2077
2078
2079
2080
2081
2082
2083
2084
2085
2086
2087
2088
2089
2090
2091
2092
2093
2094
2095
2096
2097
2098
2099
2100
2101
2102
2103
2104
  
  	field_op = contains_operator(str);
  
  	if (field_op == FIELD_OP_NONE)
  		return parse_atom(hist_data, file, str, &flags, var_name);
  
  	if (field_op == FIELD_OP_UNARY_MINUS)
  		return parse_unary(hist_data, file, str, flags, var_name, ++level);
  
  	switch (field_op) {
  	case FIELD_OP_MINUS:
  		sep = "-";
  		break;
  	case FIELD_OP_PLUS:
  		sep = "+";
  		break;
  	default:
  		goto free;
  	}
  
  	operand1_str = strsep(&str, sep);
  	if (!operand1_str || !str)
  		goto free;
  
  	operand_flags = 0;
  	operand1 = parse_atom(hist_data, file, operand1_str,
  			      &operand_flags, NULL);
  	if (IS_ERR(operand1)) {
  		ret = PTR_ERR(operand1);
  		operand1 = NULL;
  		goto free;
  	}
  
  	/* rest of string could be another expression e.g. b+c in a+b+c */
  	operand_flags = 0;
  	operand2 = parse_expr(hist_data, file, str, operand_flags, NULL, ++level);
  	if (IS_ERR(operand2)) {
  		ret = PTR_ERR(operand2);
  		operand2 = NULL;
  		goto free;
  	}
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2105
  	ret = check_expr_operands(file->tr, operand1, operand2);
100719dce   Tom Zanussi   tracing: Add simp...
2106
2107
2108
2109
2110
2111
2112
2113
2114
2115
2116
2117
2118
  	if (ret)
  		goto free;
  
  	flags |= HIST_FIELD_FL_EXPR;
  
  	flags |= operand1->flags &
  		(HIST_FIELD_FL_TIMESTAMP | HIST_FIELD_FL_TIMESTAMP_USECS);
  
  	expr = create_hist_field(hist_data, NULL, flags, var_name);
  	if (!expr) {
  		ret = -ENOMEM;
  		goto free;
  	}
067fe038e   Tom Zanussi   tracing: Add vari...
2119
2120
  	operand1->read_once = true;
  	operand2->read_once = true;
100719dce   Tom Zanussi   tracing: Add simp...
2121
2122
2123
2124
  	expr->operands[0] = operand1;
  	expr->operands[1] = operand2;
  	expr->operator = field_op;
  	expr->name = expr_str(expr, 0);
19a9facd0   Tom Zanussi   tracing: Add hist...
2125
2126
2127
2128
2129
  	expr->type = kstrdup(operand1->type, GFP_KERNEL);
  	if (!expr->type) {
  		ret = -ENOMEM;
  		goto free;
  	}
100719dce   Tom Zanussi   tracing: Add simp...
2130
2131
2132
2133
2134
2135
2136
2137
2138
  
  	switch (field_op) {
  	case FIELD_OP_MINUS:
  		expr->fn = hist_field_minus;
  		break;
  	case FIELD_OP_PLUS:
  		expr->fn = hist_field_plus;
  		break;
  	default:
5e4cf2bf6   Dan Carpenter   tracing: Fix a po...
2139
  		ret = -EINVAL;
100719dce   Tom Zanussi   tracing: Add simp...
2140
2141
2142
2143
2144
2145
2146
2147
2148
2149
2150
  		goto free;
  	}
  
  	return expr;
   free:
  	destroy_hist_field(operand1, 0);
  	destroy_hist_field(operand2, 0);
  	destroy_hist_field(expr, 0);
  
  	return ERR_PTR(ret);
  }
02205a675   Tom Zanussi   tracing: Add supp...
2151
2152
2153
2154
  static char *find_trigger_filter(struct hist_trigger_data *hist_data,
  				 struct trace_event_file *file)
  {
  	struct event_trigger_data *test;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
2155
2156
2157
  	lockdep_assert_held(&event_mutex);
  
  	list_for_each_entry(test, &file->triggers, list) {
02205a675   Tom Zanussi   tracing: Add supp...
2158
2159
2160
2161
2162
2163
2164
2165
2166
2167
2168
2169
2170
2171
2172
2173
2174
2175
2176
2177
2178
2179
2180
2181
2182
2183
2184
2185
2186
2187
2188
2189
2190
2191
2192
2193
2194
2195
2196
2197
2198
2199
2200
2201
2202
2203
2204
2205
2206
  		if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
  			if (test->private_data == hist_data)
  				return test->filter_str;
  		}
  	}
  
  	return NULL;
  }
  
  static struct event_command trigger_hist_cmd;
  static int event_hist_trigger_func(struct event_command *cmd_ops,
  				   struct trace_event_file *file,
  				   char *glob, char *cmd, char *param);
  
  static bool compatible_keys(struct hist_trigger_data *target_hist_data,
  			    struct hist_trigger_data *hist_data,
  			    unsigned int n_keys)
  {
  	struct hist_field *target_hist_field, *hist_field;
  	unsigned int n, i, j;
  
  	if (hist_data->n_fields - hist_data->n_vals != n_keys)
  		return false;
  
  	i = hist_data->n_vals;
  	j = target_hist_data->n_vals;
  
  	for (n = 0; n < n_keys; n++) {
  		hist_field = hist_data->fields[i + n];
  		target_hist_field = target_hist_data->fields[j + n];
  
  		if (strcmp(hist_field->type, target_hist_field->type) != 0)
  			return false;
  		if (hist_field->size != target_hist_field->size)
  			return false;
  		if (hist_field->is_signed != target_hist_field->is_signed)
  			return false;
  	}
  
  	return true;
  }
  
  static struct hist_trigger_data *
  find_compatible_hist(struct hist_trigger_data *target_hist_data,
  		     struct trace_event_file *file)
  {
  	struct hist_trigger_data *hist_data;
  	struct event_trigger_data *test;
  	unsigned int n_keys;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
2207
  	lockdep_assert_held(&event_mutex);
02205a675   Tom Zanussi   tracing: Add supp...
2208
  	n_keys = target_hist_data->n_fields - target_hist_data->n_vals;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
2209
  	list_for_each_entry(test, &file->triggers, list) {
02205a675   Tom Zanussi   tracing: Add supp...
2210
2211
2212
2213
2214
2215
2216
2217
2218
2219
2220
2221
2222
2223
2224
  		if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
  			hist_data = test->private_data;
  
  			if (compatible_keys(target_hist_data, hist_data, n_keys))
  				return hist_data;
  		}
  	}
  
  	return NULL;
  }
  
  static struct trace_event_file *event_file(struct trace_array *tr,
  					   char *system, char *event_name)
  {
  	struct trace_event_file *file;
3be4c1e52   Steven Rostedt (VMware)   tracing: Allow hi...
2225
  	file = __find_event_file(tr, system, event_name);
02205a675   Tom Zanussi   tracing: Add supp...
2226
2227
2228
2229
2230
2231
2232
2233
2234
2235
2236
2237
2238
2239
2240
2241
2242
2243
2244
2245
2246
2247
2248
2249
2250
2251
2252
2253
2254
2255
2256
2257
2258
2259
2260
2261
2262
2263
2264
2265
2266
2267
2268
2269
2270
2271
2272
2273
2274
2275
2276
2277
  	if (!file)
  		return ERR_PTR(-EINVAL);
  
  	return file;
  }
  
  static struct hist_field *
  find_synthetic_field_var(struct hist_trigger_data *target_hist_data,
  			 char *system, char *event_name, char *field_name)
  {
  	struct hist_field *event_var;
  	char *synthetic_name;
  
  	synthetic_name = kzalloc(MAX_FILTER_STR_VAL, GFP_KERNEL);
  	if (!synthetic_name)
  		return ERR_PTR(-ENOMEM);
  
  	strcpy(synthetic_name, "synthetic_");
  	strcat(synthetic_name, field_name);
  
  	event_var = find_event_var(target_hist_data, system, event_name, synthetic_name);
  
  	kfree(synthetic_name);
  
  	return event_var;
  }
  
  /**
   * create_field_var_hist - Automatically create a histogram and var for a field
   * @target_hist_data: The target hist trigger
   * @subsys_name: Optional subsystem name
   * @event_name: Optional event name
   * @field_name: The name of the field (and the resulting variable)
   *
   * Hist trigger actions fetch data from variables, not directly from
   * events.  However, for convenience, users are allowed to directly
   * specify an event field in an action, which will be automatically
   * converted into a variable on their behalf.
  
   * If a user specifies a field on an event that isn't the event the
   * histogram currently being defined (the target event histogram), the
   * only way that can be accomplished is if a new hist trigger is
   * created and the field variable defined on that.
   *
   * This function creates a new histogram compatible with the target
   * event (meaning a histogram with the same key as the target
   * histogram), and creates a variable for the specified field, but
   * with 'synthetic_' prepended to the variable name in order to avoid
   * collision with normal field variables.
   *
   * Return: The variable created for the field.
   */
c282a386a   Tom Zanussi   tracing: Add 'onm...
2278
  static struct hist_field *
02205a675   Tom Zanussi   tracing: Add supp...
2279
2280
2281
2282
2283
2284
2285
2286
2287
2288
2289
2290
2291
  create_field_var_hist(struct hist_trigger_data *target_hist_data,
  		      char *subsys_name, char *event_name, char *field_name)
  {
  	struct trace_array *tr = target_hist_data->event_file->tr;
  	struct hist_field *event_var = ERR_PTR(-EINVAL);
  	struct hist_trigger_data *hist_data;
  	unsigned int i, n, first = true;
  	struct field_var_hist *var_hist;
  	struct trace_event_file *file;
  	struct hist_field *key_field;
  	char *saved_filter;
  	char *cmd;
  	int ret;
f404da6e1   Tom Zanussi   tracing: Add 'las...
2292
  	if (target_hist_data->n_field_var_hists >= SYNTH_FIELDS_MAX) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2293
  		hist_err(tr, HIST_ERR_TOO_MANY_FIELD_VARS, errpos(field_name));
02205a675   Tom Zanussi   tracing: Add supp...
2294
  		return ERR_PTR(-EINVAL);
f404da6e1   Tom Zanussi   tracing: Add 'las...
2295
  	}
02205a675   Tom Zanussi   tracing: Add supp...
2296
2297
2298
2299
  
  	file = event_file(tr, subsys_name, event_name);
  
  	if (IS_ERR(file)) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2300
  		hist_err(tr, HIST_ERR_EVENT_FILE_NOT_FOUND, errpos(field_name));
02205a675   Tom Zanussi   tracing: Add supp...
2301
2302
2303
2304
2305
2306
2307
2308
2309
2310
2311
  		ret = PTR_ERR(file);
  		return ERR_PTR(ret);
  	}
  
  	/*
  	 * Look for a histogram compatible with target.  We'll use the
  	 * found histogram specification to create a new matching
  	 * histogram with our variable on it.  target_hist_data is not
  	 * yet a registered histogram so we can't use that.
  	 */
  	hist_data = find_compatible_hist(target_hist_data, file);
f404da6e1   Tom Zanussi   tracing: Add 'las...
2312
  	if (!hist_data) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2313
  		hist_err(tr, HIST_ERR_HIST_NOT_FOUND, errpos(field_name));
02205a675   Tom Zanussi   tracing: Add supp...
2314
  		return ERR_PTR(-EINVAL);
f404da6e1   Tom Zanussi   tracing: Add 'las...
2315
  	}
02205a675   Tom Zanussi   tracing: Add supp...
2316
2317
2318
2319
2320
2321
2322
2323
2324
2325
2326
2327
2328
2329
2330
2331
2332
2333
2334
2335
2336
2337
2338
2339
2340
2341
2342
2343
2344
2345
2346
2347
2348
2349
2350
2351
2352
2353
2354
2355
2356
2357
2358
2359
2360
2361
2362
2363
2364
2365
2366
2367
2368
2369
2370
2371
2372
2373
  
  	/* See if a synthetic field variable has already been created */
  	event_var = find_synthetic_field_var(target_hist_data, subsys_name,
  					     event_name, field_name);
  	if (!IS_ERR_OR_NULL(event_var))
  		return event_var;
  
  	var_hist = kzalloc(sizeof(*var_hist), GFP_KERNEL);
  	if (!var_hist)
  		return ERR_PTR(-ENOMEM);
  
  	cmd = kzalloc(MAX_FILTER_STR_VAL, GFP_KERNEL);
  	if (!cmd) {
  		kfree(var_hist);
  		return ERR_PTR(-ENOMEM);
  	}
  
  	/* Use the same keys as the compatible histogram */
  	strcat(cmd, "keys=");
  
  	for_each_hist_key_field(i, hist_data) {
  		key_field = hist_data->fields[i];
  		if (!first)
  			strcat(cmd, ",");
  		strcat(cmd, key_field->field->name);
  		first = false;
  	}
  
  	/* Create the synthetic field variable specification */
  	strcat(cmd, ":synthetic_");
  	strcat(cmd, field_name);
  	strcat(cmd, "=");
  	strcat(cmd, field_name);
  
  	/* Use the same filter as the compatible histogram */
  	saved_filter = find_trigger_filter(hist_data, file);
  	if (saved_filter) {
  		strcat(cmd, " if ");
  		strcat(cmd, saved_filter);
  	}
  
  	var_hist->cmd = kstrdup(cmd, GFP_KERNEL);
  	if (!var_hist->cmd) {
  		kfree(cmd);
  		kfree(var_hist);
  		return ERR_PTR(-ENOMEM);
  	}
  
  	/* Save the compatible histogram information */
  	var_hist->hist_data = hist_data;
  
  	/* Create the new histogram with our variable */
  	ret = event_hist_trigger_func(&trigger_hist_cmd, file,
  				      "", "hist", cmd);
  	if (ret) {
  		kfree(cmd);
  		kfree(var_hist->cmd);
  		kfree(var_hist);
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2374
  		hist_err(tr, HIST_ERR_HIST_CREATE_FAIL, errpos(field_name));
02205a675   Tom Zanussi   tracing: Add supp...
2375
2376
2377
2378
2379
2380
2381
2382
2383
2384
2385
  		return ERR_PTR(ret);
  	}
  
  	kfree(cmd);
  
  	/* If we can't find the variable, something went wrong */
  	event_var = find_synthetic_field_var(target_hist_data, subsys_name,
  					     event_name, field_name);
  	if (IS_ERR_OR_NULL(event_var)) {
  		kfree(var_hist->cmd);
  		kfree(var_hist);
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2386
  		hist_err(tr, HIST_ERR_SYNTH_VAR_NOT_FOUND, errpos(field_name));
02205a675   Tom Zanussi   tracing: Add supp...
2387
2388
2389
2390
2391
2392
2393
2394
2395
  		return ERR_PTR(-EINVAL);
  	}
  
  	n = target_hist_data->n_field_var_hists;
  	target_hist_data->field_var_hists[n] = var_hist;
  	target_hist_data->n_field_var_hists++;
  
  	return event_var;
  }
c282a386a   Tom Zanussi   tracing: Add 'onm...
2396
  static struct hist_field *
02205a675   Tom Zanussi   tracing: Add supp...
2397
2398
2399
2400
2401
2402
2403
2404
2405
2406
2407
2408
2409
2410
2411
2412
2413
2414
2415
2416
2417
2418
2419
2420
2421
2422
2423
2424
2425
2426
2427
2428
2429
2430
2431
2432
2433
2434
2435
2436
2437
2438
2439
2440
2441
2442
2443
2444
  find_target_event_var(struct hist_trigger_data *hist_data,
  		      char *subsys_name, char *event_name, char *var_name)
  {
  	struct trace_event_file *file = hist_data->event_file;
  	struct hist_field *hist_field = NULL;
  
  	if (subsys_name) {
  		struct trace_event_call *call;
  
  		if (!event_name)
  			return NULL;
  
  		call = file->event_call;
  
  		if (strcmp(subsys_name, call->class->system) != 0)
  			return NULL;
  
  		if (strcmp(event_name, trace_event_name(call)) != 0)
  			return NULL;
  	}
  
  	hist_field = find_var_field(hist_data, var_name);
  
  	return hist_field;
  }
  
  static inline void __update_field_vars(struct tracing_map_elt *elt,
  				       struct ring_buffer_event *rbe,
  				       void *rec,
  				       struct field_var **field_vars,
  				       unsigned int n_field_vars,
  				       unsigned int field_var_str_start)
  {
  	struct hist_elt_data *elt_data = elt->private_data;
  	unsigned int i, j, var_idx;
  	u64 var_val;
  
  	for (i = 0, j = field_var_str_start; i < n_field_vars; i++) {
  		struct field_var *field_var = field_vars[i];
  		struct hist_field *var = field_var->var;
  		struct hist_field *val = field_var->val;
  
  		var_val = val->fn(val, elt, rbe, rec);
  		var_idx = var->var.idx;
  
  		if (val->flags & HIST_FIELD_FL_STRING) {
  			char *str = elt_data->field_var_str[j++];
  			char *val_str = (char *)(uintptr_t)var_val;
ad452870c   Tom Zanussi   tracing: Make sur...
2445
  			strscpy(str, val_str, STR_VAR_LEN_MAX);
02205a675   Tom Zanussi   tracing: Add supp...
2446
2447
2448
2449
2450
2451
2452
2453
2454
2455
2456
2457
2458
2459
  			var_val = (u64)(uintptr_t)str;
  		}
  		tracing_map_set_var(elt, var_idx, var_val);
  	}
  }
  
  static void update_field_vars(struct hist_trigger_data *hist_data,
  			      struct tracing_map_elt *elt,
  			      struct ring_buffer_event *rbe,
  			      void *rec)
  {
  	__update_field_vars(elt, rbe, rec, hist_data->field_vars,
  			    hist_data->n_field_vars, 0);
  }
466f4528f   Tom Zanussi   tracing: Generali...
2460
2461
2462
2463
  static void save_track_data_vars(struct hist_trigger_data *hist_data,
  				 struct tracing_map_elt *elt, void *rec,
  				 struct ring_buffer_event *rbe, void *key,
  				 struct action_data *data, u64 *var_ref_vals)
50450603e   Tom Zanussi   tracing: Add 'onm...
2464
  {
7d18a10c3   Tom Zanussi   tracing: Refactor...
2465
2466
  	__update_field_vars(elt, rbe, rec, hist_data->save_vars,
  			    hist_data->n_save_vars, hist_data->n_field_var_str);
50450603e   Tom Zanussi   tracing: Add 'onm...
2467
  }
02205a675   Tom Zanussi   tracing: Add supp...
2468
2469
2470
2471
2472
2473
2474
2475
2476
2477
2478
2479
2480
2481
2482
2483
2484
2485
2486
2487
2488
2489
2490
2491
  static struct hist_field *create_var(struct hist_trigger_data *hist_data,
  				     struct trace_event_file *file,
  				     char *name, int size, const char *type)
  {
  	struct hist_field *var;
  	int idx;
  
  	if (find_var(hist_data, file, name) && !hist_data->remove) {
  		var = ERR_PTR(-EINVAL);
  		goto out;
  	}
  
  	var = kzalloc(sizeof(struct hist_field), GFP_KERNEL);
  	if (!var) {
  		var = ERR_PTR(-ENOMEM);
  		goto out;
  	}
  
  	idx = tracing_map_add_var(hist_data->map);
  	if (idx < 0) {
  		kfree(var);
  		var = ERR_PTR(-EINVAL);
  		goto out;
  	}
9da73974e   Vamshi K Sthambamkadi   tracing: Fix memo...
2492
  	var->ref = 1;
02205a675   Tom Zanussi   tracing: Add supp...
2493
2494
2495
2496
2497
2498
2499
2500
2501
2502
2503
2504
2505
2506
2507
2508
2509
2510
2511
2512
2513
2514
  	var->flags = HIST_FIELD_FL_VAR;
  	var->var.idx = idx;
  	var->var.hist_data = var->hist_data = hist_data;
  	var->size = size;
  	var->var.name = kstrdup(name, GFP_KERNEL);
  	var->type = kstrdup(type, GFP_KERNEL);
  	if (!var->var.name || !var->type) {
  		kfree(var->var.name);
  		kfree(var->type);
  		kfree(var);
  		var = ERR_PTR(-ENOMEM);
  	}
   out:
  	return var;
  }
  
  static struct field_var *create_field_var(struct hist_trigger_data *hist_data,
  					  struct trace_event_file *file,
  					  char *field_name)
  {
  	struct hist_field *val = NULL, *var = NULL;
  	unsigned long flags = HIST_FIELD_FL_VAR;
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2515
  	struct trace_array *tr = file->tr;
02205a675   Tom Zanussi   tracing: Add supp...
2516
2517
2518
2519
  	struct field_var *field_var;
  	int ret = 0;
  
  	if (hist_data->n_field_vars >= SYNTH_FIELDS_MAX) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2520
  		hist_err(tr, HIST_ERR_TOO_MANY_FIELD_VARS, errpos(field_name));
02205a675   Tom Zanussi   tracing: Add supp...
2521
2522
2523
2524
2525
2526
  		ret = -EINVAL;
  		goto err;
  	}
  
  	val = parse_atom(hist_data, file, field_name, &flags, NULL);
  	if (IS_ERR(val)) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2527
  		hist_err(tr, HIST_ERR_FIELD_VAR_PARSE_FAIL, errpos(field_name));
02205a675   Tom Zanussi   tracing: Add supp...
2528
2529
2530
2531
2532
2533
  		ret = PTR_ERR(val);
  		goto err;
  	}
  
  	var = create_var(hist_data, file, field_name, val->size, val->type);
  	if (IS_ERR(var)) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2534
  		hist_err(tr, HIST_ERR_VAR_CREATE_FIND_FAIL, errpos(field_name));
02205a675   Tom Zanussi   tracing: Add supp...
2535
2536
2537
2538
2539
2540
2541
2542
2543
2544
2545
2546
2547
2548
2549
2550
2551
2552
2553
2554
2555
2556
2557
2558
2559
2560
2561
2562
2563
2564
2565
2566
2567
2568
2569
2570
2571
2572
2573
2574
2575
2576
  		kfree(val);
  		ret = PTR_ERR(var);
  		goto err;
  	}
  
  	field_var = kzalloc(sizeof(struct field_var), GFP_KERNEL);
  	if (!field_var) {
  		kfree(val);
  		kfree(var);
  		ret =  -ENOMEM;
  		goto err;
  	}
  
  	field_var->var = var;
  	field_var->val = val;
   out:
  	return field_var;
   err:
  	field_var = ERR_PTR(ret);
  	goto out;
  }
  
  /**
   * create_target_field_var - Automatically create a variable for a field
   * @target_hist_data: The target hist trigger
   * @subsys_name: Optional subsystem name
   * @event_name: Optional event name
   * @var_name: The name of the field (and the resulting variable)
   *
   * Hist trigger actions fetch data from variables, not directly from
   * events.  However, for convenience, users are allowed to directly
   * specify an event field in an action, which will be automatically
   * converted into a variable on their behalf.
  
   * This function creates a field variable with the name var_name on
   * the hist trigger currently being defined on the target event.  If
   * subsys_name and event_name are specified, this function simply
   * verifies that they do in fact match the target event subsystem and
   * event name.
   *
   * Return: The variable created for the field.
   */
c282a386a   Tom Zanussi   tracing: Add 'onm...
2577
  static struct field_var *
02205a675   Tom Zanussi   tracing: Add supp...
2578
2579
2580
2581
2582
2583
2584
2585
2586
2587
2588
2589
2590
2591
2592
2593
2594
2595
2596
2597
2598
2599
  create_target_field_var(struct hist_trigger_data *target_hist_data,
  			char *subsys_name, char *event_name, char *var_name)
  {
  	struct trace_event_file *file = target_hist_data->event_file;
  
  	if (subsys_name) {
  		struct trace_event_call *call;
  
  		if (!event_name)
  			return NULL;
  
  		call = file->event_call;
  
  		if (strcmp(subsys_name, call->class->system) != 0)
  			return NULL;
  
  		if (strcmp(event_name, trace_event_name(call)) != 0)
  			return NULL;
  	}
  
  	return create_field_var(target_hist_data, file, var_name);
  }
466f4528f   Tom Zanussi   tracing: Generali...
2600
2601
2602
2603
2604
2605
2606
  static bool check_track_val_max(u64 track_val, u64 var_val)
  {
  	if (var_val <= track_val)
  		return false;
  
  	return true;
  }
dff81f559   Tom Zanussi   tracing: Add hist...
2607
2608
2609
2610
2611
2612
2613
  static bool check_track_val_changed(u64 track_val, u64 var_val)
  {
  	if (var_val == track_val)
  		return false;
  
  	return true;
  }
466f4528f   Tom Zanussi   tracing: Generali...
2614
2615
2616
  static u64 get_track_val(struct hist_trigger_data *hist_data,
  			 struct tracing_map_elt *elt,
  			 struct action_data *data)
50450603e   Tom Zanussi   tracing: Add 'onm...
2617
  {
466f4528f   Tom Zanussi   tracing: Generali...
2618
2619
2620
2621
  	unsigned int track_var_idx = data->track_data.track_var->var.idx;
  	u64 track_val;
  
  	track_val = tracing_map_read_var(elt, track_var_idx);
50450603e   Tom Zanussi   tracing: Add 'onm...
2622

466f4528f   Tom Zanussi   tracing: Generali...
2623
2624
2625
2626
2627
2628
2629
2630
2631
2632
2633
2634
2635
2636
2637
2638
2639
2640
2641
2642
2643
2644
2645
2646
2647
2648
2649
2650
2651
2652
2653
2654
2655
  	return track_val;
  }
  
  static void save_track_val(struct hist_trigger_data *hist_data,
  			   struct tracing_map_elt *elt,
  			   struct action_data *data, u64 var_val)
  {
  	unsigned int track_var_idx = data->track_data.track_var->var.idx;
  
  	tracing_map_set_var(elt, track_var_idx, var_val);
  }
  
  static void save_track_data(struct hist_trigger_data *hist_data,
  			    struct tracing_map_elt *elt, void *rec,
  			    struct ring_buffer_event *rbe, void *key,
  			    struct action_data *data, u64 *var_ref_vals)
  {
  	if (data->track_data.save_data)
  		data->track_data.save_data(hist_data, elt, rec, rbe, key, data, var_ref_vals);
  }
  
  static bool check_track_val(struct tracing_map_elt *elt,
  			    struct action_data *data,
  			    u64 var_val)
  {
  	struct hist_trigger_data *hist_data;
  	u64 track_val;
  
  	hist_data = data->track_data.track_var->hist_data;
  	track_val = get_track_val(hist_data, elt, data);
  
  	return data->track_data.check_val(track_val, var_val);
  }
a3785b7ec   Tom Zanussi   tracing: Add hist...
2656
2657
2658
2659
2660
2661
2662
  #ifdef CONFIG_TRACER_SNAPSHOT
  static bool cond_snapshot_update(struct trace_array *tr, void *cond_data)
  {
  	/* called with tr->max_lock held */
  	struct track_data *track_data = tr->cond_snapshot->cond_data;
  	struct hist_elt_data *elt_data, *track_elt_data;
  	struct snapshot_context *context = cond_data;
9b2ca371b   Tom Zanussi   tracing: Add a ch...
2663
  	struct action_data *action;
a3785b7ec   Tom Zanussi   tracing: Add hist...
2664
2665
2666
2667
  	u64 track_val;
  
  	if (!track_data)
  		return false;
9b2ca371b   Tom Zanussi   tracing: Add a ch...
2668
  	action = track_data->action_data;
a3785b7ec   Tom Zanussi   tracing: Add hist...
2669
2670
  	track_val = get_track_val(track_data->hist_data, context->elt,
  				  track_data->action_data);
9b2ca371b   Tom Zanussi   tracing: Add a ch...
2671
2672
  	if (!action->track_data.check_val(track_data->track_val, track_val))
  		return false;
a3785b7ec   Tom Zanussi   tracing: Add hist...
2673
2674
2675
2676
2677
2678
  	track_data->track_val = track_val;
  	memcpy(track_data->key, context->key, track_data->key_len);
  
  	elt_data = context->elt->private_data;
  	track_elt_data = track_data->elt.private_data;
  	if (elt_data->comm)
27242c62b   Tom Zanussi   tracing: Use strn...
2679
  		strncpy(track_elt_data->comm, elt_data->comm, TASK_COMM_LEN);
a3785b7ec   Tom Zanussi   tracing: Add hist...
2680
2681
2682
2683
2684
2685
2686
2687
2688
2689
2690
2691
2692
2693
2694
2695
2696
2697
2698
2699
2700
2701
2702
2703
2704
2705
2706
2707
2708
2709
2710
2711
2712
2713
2714
2715
2716
2717
2718
2719
2720
2721
2722
2723
2724
2725
2726
2727
2728
2729
2730
2731
2732
2733
2734
2735
2736
2737
2738
2739
2740
2741
2742
2743
2744
2745
2746
2747
2748
2749
2750
2751
2752
2753
2754
2755
2756
2757
2758
2759
2760
2761
2762
2763
2764
2765
  
  	track_data->updated = true;
  
  	return true;
  }
  
  static void save_track_data_snapshot(struct hist_trigger_data *hist_data,
  				     struct tracing_map_elt *elt, void *rec,
  				     struct ring_buffer_event *rbe, void *key,
  				     struct action_data *data,
  				     u64 *var_ref_vals)
  {
  	struct trace_event_file *file = hist_data->event_file;
  	struct snapshot_context context;
  
  	context.elt = elt;
  	context.key = key;
  
  	tracing_snapshot_cond(file->tr, &context);
  }
  
  static void hist_trigger_print_key(struct seq_file *m,
  				   struct hist_trigger_data *hist_data,
  				   void *key,
  				   struct tracing_map_elt *elt);
  
  static struct action_data *snapshot_action(struct hist_trigger_data *hist_data)
  {
  	unsigned int i;
  
  	if (!hist_data->n_actions)
  		return NULL;
  
  	for (i = 0; i < hist_data->n_actions; i++) {
  		struct action_data *data = hist_data->actions[i];
  
  		if (data->action == ACTION_SNAPSHOT)
  			return data;
  	}
  
  	return NULL;
  }
  
  static void track_data_snapshot_print(struct seq_file *m,
  				      struct hist_trigger_data *hist_data)
  {
  	struct trace_event_file *file = hist_data->event_file;
  	struct track_data *track_data;
  	struct action_data *action;
  
  	track_data = tracing_cond_snapshot_data(file->tr);
  	if (!track_data)
  		return;
  
  	if (!track_data->updated)
  		return;
  
  	action = snapshot_action(hist_data);
  	if (!action)
  		return;
  
  	seq_puts(m, "
  Snapshot taken (see tracing/snapshot).  Details:
  ");
  	seq_printf(m, "\ttriggering value { %s(%s) }: %10llu",
  		   action->handler == HANDLER_ONMAX ? "onmax" : "onchange",
  		   action->track_data.var_str, track_data->track_val);
  
  	seq_puts(m, "\ttriggered by event with key: ");
  	hist_trigger_print_key(m, hist_data, track_data->key, &track_data->elt);
  	seq_putc(m, '
  ');
  }
  #else
  static bool cond_snapshot_update(struct trace_array *tr, void *cond_data)
  {
  	return false;
  }
  static void save_track_data_snapshot(struct hist_trigger_data *hist_data,
  				     struct tracing_map_elt *elt, void *rec,
  				     struct ring_buffer_event *rbe, void *key,
  				     struct action_data *data,
  				     u64 *var_ref_vals) {}
  static void track_data_snapshot_print(struct seq_file *m,
  				      struct hist_trigger_data *hist_data) {}
  #endif /* CONFIG_TRACER_SNAPSHOT */
466f4528f   Tom Zanussi   tracing: Generali...
2766
2767
2768
2769
2770
2771
2772
2773
2774
2775
2776
  static void track_data_print(struct seq_file *m,
  			     struct hist_trigger_data *hist_data,
  			     struct tracing_map_elt *elt,
  			     struct action_data *data)
  {
  	u64 track_val = get_track_val(hist_data, elt, data);
  	unsigned int i, save_var_idx;
  
  	if (data->handler == HANDLER_ONMAX)
  		seq_printf(m, "
  \tmax: %10llu", track_val);
dff81f559   Tom Zanussi   tracing: Add hist...
2777
2778
2779
  	else if (data->handler == HANDLER_ONCHANGE)
  		seq_printf(m, "
  \tchanged: %10llu", track_val);
50450603e   Tom Zanussi   tracing: Add 'onm...
2780

a3785b7ec   Tom Zanussi   tracing: Add hist...
2781
2782
  	if (data->action == ACTION_SNAPSHOT)
  		return;
7d18a10c3   Tom Zanussi   tracing: Refactor...
2783
2784
2785
  	for (i = 0; i < hist_data->n_save_vars; i++) {
  		struct hist_field *save_val = hist_data->save_vars[i]->val;
  		struct hist_field *save_var = hist_data->save_vars[i]->var;
50450603e   Tom Zanussi   tracing: Add 'onm...
2786
2787
2788
2789
2790
2791
2792
2793
2794
2795
2796
2797
2798
  		u64 val;
  
  		save_var_idx = save_var->var.idx;
  
  		val = tracing_map_read_var(elt, save_var_idx);
  
  		if (save_val->flags & HIST_FIELD_FL_STRING) {
  			seq_printf(m, "  %s: %-32s", save_var->var.name,
  				   (char *)(uintptr_t)(val));
  		} else
  			seq_printf(m, "  %s: %10llu", save_var->var.name, val);
  	}
  }
466f4528f   Tom Zanussi   tracing: Generali...
2799
2800
2801
2802
  static void ontrack_action(struct hist_trigger_data *hist_data,
  			   struct tracing_map_elt *elt, void *rec,
  			   struct ring_buffer_event *rbe, void *key,
  			   struct action_data *data, u64 *var_ref_vals)
50450603e   Tom Zanussi   tracing: Add 'onm...
2803
  {
466f4528f   Tom Zanussi   tracing: Generali...
2804
  	u64 var_val = var_ref_vals[data->track_data.var_ref->var_ref_idx];
50450603e   Tom Zanussi   tracing: Add 'onm...
2805

466f4528f   Tom Zanussi   tracing: Generali...
2806
2807
2808
2809
  	if (check_track_val(elt, data, var_val)) {
  		save_track_val(hist_data, elt, data, var_val);
  		save_track_data(hist_data, elt, rec, rbe, key, data, var_ref_vals);
  	}
50450603e   Tom Zanussi   tracing: Add 'onm...
2810
  }
c3e49506a   Tom Zanussi   tracing: Split up...
2811
  static void action_data_destroy(struct action_data *data)
50450603e   Tom Zanussi   tracing: Add 'onm...
2812
2813
  {
  	unsigned int i;
c3e49506a   Tom Zanussi   tracing: Split up...
2814
  	lockdep_assert_held(&event_mutex);
50450603e   Tom Zanussi   tracing: Add 'onm...
2815

7d18a10c3   Tom Zanussi   tracing: Refactor...
2816
  	kfree(data->action_name);
50450603e   Tom Zanussi   tracing: Add 'onm...
2817
2818
2819
  
  	for (i = 0; i < data->n_params; i++)
  		kfree(data->params[i]);
c3e49506a   Tom Zanussi   tracing: Split up...
2820
2821
  	if (data->synth_event)
  		data->synth_event->ref--;
e91eefd73   Tom Zanussi   tracing: Add alte...
2822
  	kfree(data->synth_event_name);
50450603e   Tom Zanussi   tracing: Add 'onm...
2823
2824
  	kfree(data);
  }
466f4528f   Tom Zanussi   tracing: Generali...
2825
2826
  static void track_data_destroy(struct hist_trigger_data *hist_data,
  			       struct action_data *data)
c3e49506a   Tom Zanussi   tracing: Split up...
2827
  {
a3785b7ec   Tom Zanussi   tracing: Add hist...
2828
  	struct trace_event_file *file = hist_data->event_file;
466f4528f   Tom Zanussi   tracing: Generali...
2829
  	destroy_hist_field(data->track_data.track_var, 0);
c3e49506a   Tom Zanussi   tracing: Split up...
2830

a3785b7ec   Tom Zanussi   tracing: Add hist...
2831
2832
2833
2834
2835
2836
2837
2838
2839
  	if (data->action == ACTION_SNAPSHOT) {
  		struct track_data *track_data;
  
  		track_data = tracing_cond_snapshot_data(file->tr);
  		if (track_data && track_data->hist_data == hist_data) {
  			tracing_snapshot_cond_disable(file->tr);
  			track_data_free(track_data);
  		}
  	}
466f4528f   Tom Zanussi   tracing: Generali...
2840
  	kfree(data->track_data.var_str);
c3e49506a   Tom Zanussi   tracing: Split up...
2841
2842
2843
  
  	action_data_destroy(data);
  }
7d18a10c3   Tom Zanussi   tracing: Refactor...
2844
2845
  static int action_create(struct hist_trigger_data *hist_data,
  			 struct action_data *data);
466f4528f   Tom Zanussi   tracing: Generali...
2846
2847
  static int track_data_create(struct hist_trigger_data *hist_data,
  			     struct action_data *data)
50450603e   Tom Zanussi   tracing: Add 'onm...
2848
  {
466f4528f   Tom Zanussi   tracing: Generali...
2849
  	struct hist_field *var_field, *ref_field, *track_var = NULL;
50450603e   Tom Zanussi   tracing: Add 'onm...
2850
  	struct trace_event_file *file = hist_data->event_file;
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2851
  	struct trace_array *tr = file->tr;
466f4528f   Tom Zanussi   tracing: Generali...
2852
  	char *track_data_var_str;
50450603e   Tom Zanussi   tracing: Add 'onm...
2853
  	int ret = 0;
466f4528f   Tom Zanussi   tracing: Generali...
2854
2855
  	track_data_var_str = data->track_data.var_str;
  	if (track_data_var_str[0] != '$') {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2856
  		hist_err(tr, HIST_ERR_ONX_NOT_VAR, errpos(track_data_var_str));
50450603e   Tom Zanussi   tracing: Add 'onm...
2857
  		return -EINVAL;
f404da6e1   Tom Zanussi   tracing: Add 'las...
2858
  	}
466f4528f   Tom Zanussi   tracing: Generali...
2859
  	track_data_var_str++;
50450603e   Tom Zanussi   tracing: Add 'onm...
2860

466f4528f   Tom Zanussi   tracing: Generali...
2861
  	var_field = find_target_event_var(hist_data, NULL, NULL, track_data_var_str);
f404da6e1   Tom Zanussi   tracing: Add 'las...
2862
  	if (!var_field) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2863
  		hist_err(tr, HIST_ERR_ONX_VAR_NOT_FOUND, errpos(track_data_var_str));
50450603e   Tom Zanussi   tracing: Add 'onm...
2864
  		return -EINVAL;
f404da6e1   Tom Zanussi   tracing: Add 'las...
2865
  	}
50450603e   Tom Zanussi   tracing: Add 'onm...
2866

de40f033d   Tom Zanussi   tracing: Remove o...
2867
  	ref_field = create_var_ref(hist_data, var_field, NULL, NULL);
50450603e   Tom Zanussi   tracing: Add 'onm...
2868
2869
  	if (!ref_field)
  		return -ENOMEM;
466f4528f   Tom Zanussi   tracing: Generali...
2870
  	data->track_data.var_ref = ref_field;
50450603e   Tom Zanussi   tracing: Add 'onm...
2871

466f4528f   Tom Zanussi   tracing: Generali...
2872
2873
2874
  	if (data->handler == HANDLER_ONMAX)
  		track_var = create_var(hist_data, file, "__max", sizeof(u64), "u64");
  	if (IS_ERR(track_var)) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2875
  		hist_err(tr, HIST_ERR_ONX_VAR_CREATE_FAIL, 0);
466f4528f   Tom Zanussi   tracing: Generali...
2876
  		ret = PTR_ERR(track_var);
50450603e   Tom Zanussi   tracing: Add 'onm...
2877
2878
  		goto out;
  	}
dff81f559   Tom Zanussi   tracing: Add hist...
2879
2880
2881
2882
  
  	if (data->handler == HANDLER_ONCHANGE)
  		track_var = create_var(hist_data, file, "__change", sizeof(u64), "u64");
  	if (IS_ERR(track_var)) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2883
  		hist_err(tr, HIST_ERR_ONX_VAR_CREATE_FAIL, 0);
dff81f559   Tom Zanussi   tracing: Add hist...
2884
2885
2886
  		ret = PTR_ERR(track_var);
  		goto out;
  	}
466f4528f   Tom Zanussi   tracing: Generali...
2887
  	data->track_data.track_var = track_var;
50450603e   Tom Zanussi   tracing: Add 'onm...
2888

7d18a10c3   Tom Zanussi   tracing: Refactor...
2889
  	ret = action_create(hist_data, data);
50450603e   Tom Zanussi   tracing: Add 'onm...
2890
2891
2892
   out:
  	return ret;
  }
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2893
2894
  static int parse_action_params(struct trace_array *tr, char *params,
  			       struct action_data *data)
50450603e   Tom Zanussi   tracing: Add 'onm...
2895
2896
  {
  	char *param, *saved_param;
e91eefd73   Tom Zanussi   tracing: Add alte...
2897
  	bool first_param = true;
50450603e   Tom Zanussi   tracing: Add 'onm...
2898
2899
2900
  	int ret = 0;
  
  	while (params) {
7d18a10c3   Tom Zanussi   tracing: Refactor...
2901
  		if (data->n_params >= SYNTH_FIELDS_MAX) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2902
  			hist_err(tr, HIST_ERR_TOO_MANY_PARAMS, 0);
50450603e   Tom Zanussi   tracing: Add 'onm...
2903
  			goto out;
7d18a10c3   Tom Zanussi   tracing: Refactor...
2904
  		}
50450603e   Tom Zanussi   tracing: Add 'onm...
2905
2906
2907
  
  		param = strsep(&params, ",");
  		if (!param) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2908
  			hist_err(tr, HIST_ERR_PARAM_NOT_FOUND, 0);
50450603e   Tom Zanussi   tracing: Add 'onm...
2909
2910
2911
2912
2913
2914
  			ret = -EINVAL;
  			goto out;
  		}
  
  		param = strstrip(param);
  		if (strlen(param) < 2) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2915
  			hist_err(tr, HIST_ERR_INVALID_PARAM, errpos(param));
50450603e   Tom Zanussi   tracing: Add 'onm...
2916
2917
2918
2919
2920
2921
2922
2923
2924
  			ret = -EINVAL;
  			goto out;
  		}
  
  		saved_param = kstrdup(param, GFP_KERNEL);
  		if (!saved_param) {
  			ret = -ENOMEM;
  			goto out;
  		}
e91eefd73   Tom Zanussi   tracing: Add alte...
2925
2926
2927
2928
2929
2930
  		if (first_param && data->use_trace_keyword) {
  			data->synth_event_name = saved_param;
  			first_param = false;
  			continue;
  		}
  		first_param = false;
50450603e   Tom Zanussi   tracing: Add 'onm...
2931
2932
2933
2934
2935
  		data->params[data->n_params++] = saved_param;
  	}
   out:
  	return ret;
  }
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2936
  static int action_parse(struct trace_array *tr, char *str, struct action_data *data,
7d18a10c3   Tom Zanussi   tracing: Refactor...
2937
2938
2939
2940
2941
2942
2943
  			enum handler_id handler)
  {
  	char *action_name;
  	int ret = 0;
  
  	strsep(&str, ".");
  	if (!str) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2944
  		hist_err(tr, HIST_ERR_ACTION_NOT_FOUND, 0);
7d18a10c3   Tom Zanussi   tracing: Refactor...
2945
2946
2947
2948
2949
2950
  		ret = -EINVAL;
  		goto out;
  	}
  
  	action_name = strsep(&str, "(");
  	if (!action_name || !str) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2951
  		hist_err(tr, HIST_ERR_ACTION_NOT_FOUND, 0);
7d18a10c3   Tom Zanussi   tracing: Refactor...
2952
2953
2954
2955
2956
2957
2958
2959
  		ret = -EINVAL;
  		goto out;
  	}
  
  	if (str_has_prefix(action_name, "save")) {
  		char *params = strsep(&str, ")");
  
  		if (!params) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2960
  			hist_err(tr, HIST_ERR_NO_SAVE_PARAMS, 0);
7d18a10c3   Tom Zanussi   tracing: Refactor...
2961
2962
2963
  			ret = -EINVAL;
  			goto out;
  		}
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2964
  		ret = parse_action_params(tr, params, data);
7d18a10c3   Tom Zanussi   tracing: Refactor...
2965
2966
2967
2968
  		if (ret)
  			goto out;
  
  		if (handler == HANDLER_ONMAX)
466f4528f   Tom Zanussi   tracing: Generali...
2969
  			data->track_data.check_val = check_track_val_max;
dff81f559   Tom Zanussi   tracing: Add hist...
2970
2971
  		else if (handler == HANDLER_ONCHANGE)
  			data->track_data.check_val = check_track_val_changed;
466f4528f   Tom Zanussi   tracing: Generali...
2972
  		else {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2973
  			hist_err(tr, HIST_ERR_ACTION_MISMATCH, errpos(action_name));
466f4528f   Tom Zanussi   tracing: Generali...
2974
2975
2976
  			ret = -EINVAL;
  			goto out;
  		}
7d18a10c3   Tom Zanussi   tracing: Refactor...
2977

466f4528f   Tom Zanussi   tracing: Generali...
2978
2979
  		data->track_data.save_data = save_track_data_vars;
  		data->fn = ontrack_action;
7d18a10c3   Tom Zanussi   tracing: Refactor...
2980
  		data->action = ACTION_SAVE;
a3785b7ec   Tom Zanussi   tracing: Add hist...
2981
2982
2983
2984
  	} else if (str_has_prefix(action_name, "snapshot")) {
  		char *params = strsep(&str, ")");
  
  		if (!str) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2985
  			hist_err(tr, HIST_ERR_NO_CLOSING_PAREN, errpos(params));
a3785b7ec   Tom Zanussi   tracing: Add hist...
2986
2987
2988
2989
2990
2991
  			ret = -EINVAL;
  			goto out;
  		}
  
  		if (handler == HANDLER_ONMAX)
  			data->track_data.check_val = check_track_val_max;
dff81f559   Tom Zanussi   tracing: Add hist...
2992
2993
  		else if (handler == HANDLER_ONCHANGE)
  			data->track_data.check_val = check_track_val_changed;
a3785b7ec   Tom Zanussi   tracing: Add hist...
2994
  		else {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
2995
  			hist_err(tr, HIST_ERR_ACTION_MISMATCH, errpos(action_name));
a3785b7ec   Tom Zanussi   tracing: Add hist...
2996
2997
2998
2999
3000
3001
3002
  			ret = -EINVAL;
  			goto out;
  		}
  
  		data->track_data.save_data = save_track_data_snapshot;
  		data->fn = ontrack_action;
  		data->action = ACTION_SNAPSHOT;
7d18a10c3   Tom Zanussi   tracing: Refactor...
3003
3004
  	} else {
  		char *params = strsep(&str, ")");
e91eefd73   Tom Zanussi   tracing: Add alte...
3005
3006
  		if (str_has_prefix(action_name, "trace"))
  			data->use_trace_keyword = true;
7d18a10c3   Tom Zanussi   tracing: Refactor...
3007
  		if (params) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3008
  			ret = parse_action_params(tr, params, data);
7d18a10c3   Tom Zanussi   tracing: Refactor...
3009
3010
3011
  			if (ret)
  				goto out;
  		}
466f4528f   Tom Zanussi   tracing: Generali...
3012
3013
  		if (handler == HANDLER_ONMAX)
  			data->track_data.check_val = check_track_val_max;
dff81f559   Tom Zanussi   tracing: Add hist...
3014
3015
  		else if (handler == HANDLER_ONCHANGE)
  			data->track_data.check_val = check_track_val_changed;
466f4528f   Tom Zanussi   tracing: Generali...
3016
3017
3018
3019
3020
3021
  
  		if (handler != HANDLER_ONMATCH) {
  			data->track_data.save_data = action_trace;
  			data->fn = ontrack_action;
  		} else
  			data->fn = action_trace;
7d18a10c3   Tom Zanussi   tracing: Refactor...
3022
3023
3024
3025
3026
3027
3028
3029
3030
3031
3032
3033
3034
  		data->action = ACTION_TRACE;
  	}
  
  	data->action_name = kstrdup(action_name, GFP_KERNEL);
  	if (!data->action_name) {
  		ret = -ENOMEM;
  		goto out;
  	}
  
  	data->handler = handler;
   out:
  	return ret;
  }
466f4528f   Tom Zanussi   tracing: Generali...
3035
3036
  static struct action_data *track_data_parse(struct hist_trigger_data *hist_data,
  					    char *str, enum handler_id handler)
50450603e   Tom Zanussi   tracing: Add 'onm...
3037
  {
50450603e   Tom Zanussi   tracing: Add 'onm...
3038
3039
  	struct action_data *data;
  	int ret = -EINVAL;
466f4528f   Tom Zanussi   tracing: Generali...
3040
  	char *var_str;
50450603e   Tom Zanussi   tracing: Add 'onm...
3041
3042
3043
3044
  
  	data = kzalloc(sizeof(*data), GFP_KERNEL);
  	if (!data)
  		return ERR_PTR(-ENOMEM);
466f4528f   Tom Zanussi   tracing: Generali...
3045
3046
  	var_str = strsep(&str, ")");
  	if (!var_str || !str) {
50450603e   Tom Zanussi   tracing: Add 'onm...
3047
3048
3049
  		ret = -EINVAL;
  		goto free;
  	}
466f4528f   Tom Zanussi   tracing: Generali...
3050
3051
  	data->track_data.var_str = kstrdup(var_str, GFP_KERNEL);
  	if (!data->track_data.var_str) {
50450603e   Tom Zanussi   tracing: Add 'onm...
3052
3053
3054
  		ret = -ENOMEM;
  		goto free;
  	}
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3055
  	ret = action_parse(hist_data->event_file->tr, str, data, handler);
7d18a10c3   Tom Zanussi   tracing: Refactor...
3056
  	if (ret)
50450603e   Tom Zanussi   tracing: Add 'onm...
3057
  		goto free;
50450603e   Tom Zanussi   tracing: Add 'onm...
3058
3059
3060
   out:
  	return data;
   free:
466f4528f   Tom Zanussi   tracing: Generali...
3061
  	track_data_destroy(hist_data, data);
50450603e   Tom Zanussi   tracing: Add 'onm...
3062
3063
3064
  	data = ERR_PTR(ret);
  	goto out;
  }
c282a386a   Tom Zanussi   tracing: Add 'onm...
3065
3066
  static void onmatch_destroy(struct action_data *data)
  {
c3e49506a   Tom Zanussi   tracing: Split up...
3067
3068
  	kfree(data->match_data.event);
  	kfree(data->match_data.event_system);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3069

c3e49506a   Tom Zanussi   tracing: Split up...
3070
  	action_data_destroy(data);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3071
  }
02205a675   Tom Zanussi   tracing: Add supp...
3072
3073
3074
3075
3076
3077
3078
3079
3080
3081
3082
3083
3084
3085
3086
3087
3088
  static void destroy_field_var(struct field_var *field_var)
  {
  	if (!field_var)
  		return;
  
  	destroy_hist_field(field_var->var, 0);
  	destroy_hist_field(field_var->val, 0);
  
  	kfree(field_var);
  }
  
  static void destroy_field_vars(struct hist_trigger_data *hist_data)
  {
  	unsigned int i;
  
  	for (i = 0; i < hist_data->n_field_vars; i++)
  		destroy_field_var(hist_data->field_vars[i]);
9da73974e   Vamshi K Sthambamkadi   tracing: Fix memo...
3089
3090
3091
  
  	for (i = 0; i < hist_data->n_save_vars; i++)
  		destroy_field_var(hist_data->save_vars[i]);
02205a675   Tom Zanussi   tracing: Add supp...
3092
  }
c282a386a   Tom Zanussi   tracing: Add 'onm...
3093
3094
  static void save_field_var(struct hist_trigger_data *hist_data,
  			   struct field_var *field_var)
02205a675   Tom Zanussi   tracing: Add supp...
3095
3096
3097
3098
3099
3100
  {
  	hist_data->field_vars[hist_data->n_field_vars++] = field_var;
  
  	if (field_var->val->flags & HIST_FIELD_FL_STRING)
  		hist_data->n_field_var_str++;
  }
c282a386a   Tom Zanussi   tracing: Add 'onm...
3101

c282a386a   Tom Zanussi   tracing: Add 'onm...
3102
3103
3104
3105
3106
3107
3108
3109
3110
3111
  static int check_synth_field(struct synth_event *event,
  			     struct hist_field *hist_field,
  			     unsigned int field_pos)
  {
  	struct synth_field *field;
  
  	if (field_pos >= event->n_fields)
  		return -EINVAL;
  
  	field = event->fields[field_pos];
bd82631d7   Tom Zanussi   tracing: Add supp...
3112
3113
3114
3115
3116
3117
3118
3119
  	/*
  	 * A dynamic string synth field can accept static or
  	 * dynamic. A static string synth field can only accept a
  	 * same-sized static string, which is checked for later.
  	 */
  	if (strstr(hist_field->type, "char[") && field->is_string
  	    && field->is_dynamic)
  		return 0;
b05e89ae7   Masami Hiramatsu   tracing: Accept d...
3120
3121
3122
3123
3124
  	if (strcmp(field->type, hist_field->type) != 0) {
  		if (field->size != hist_field->size ||
  		    field->is_signed != hist_field->is_signed)
  			return -EINVAL;
  	}
c282a386a   Tom Zanussi   tracing: Add 'onm...
3125
3126
3127
  
  	return 0;
  }
c282a386a   Tom Zanussi   tracing: Add 'onm...
3128
  static struct hist_field *
7d18a10c3   Tom Zanussi   tracing: Refactor...
3129
3130
3131
  trace_action_find_var(struct hist_trigger_data *hist_data,
  		      struct action_data *data,
  		      char *system, char *event, char *var)
c282a386a   Tom Zanussi   tracing: Add 'onm...
3132
  {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3133
  	struct trace_array *tr = hist_data->event_file->tr;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3134
3135
3136
3137
3138
3139
  	struct hist_field *hist_field;
  
  	var++; /* skip '$' */
  
  	hist_field = find_target_event_var(hist_data, system, event, var);
  	if (!hist_field) {
7d18a10c3   Tom Zanussi   tracing: Refactor...
3140
  		if (!system && data->handler == HANDLER_ONMATCH) {
c3e49506a   Tom Zanussi   tracing: Split up...
3141
3142
  			system = data->match_data.event_system;
  			event = data->match_data.event;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3143
3144
3145
3146
  		}
  
  		hist_field = find_event_var(hist_data, system, event, var);
  	}
f404da6e1   Tom Zanussi   tracing: Add 'las...
3147
  	if (!hist_field)
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3148
  		hist_err(tr, HIST_ERR_PARAM_NOT_FOUND, errpos(var));
f404da6e1   Tom Zanussi   tracing: Add 'las...
3149

c282a386a   Tom Zanussi   tracing: Add 'onm...
3150
3151
3152
3153
  	return hist_field;
  }
  
  static struct hist_field *
7d18a10c3   Tom Zanussi   tracing: Refactor...
3154
3155
3156
  trace_action_create_field_var(struct hist_trigger_data *hist_data,
  			      struct action_data *data, char *system,
  			      char *event, char *var)
c282a386a   Tom Zanussi   tracing: Add 'onm...
3157
3158
3159
3160
3161
3162
3163
3164
3165
3166
3167
3168
3169
3170
3171
3172
3173
3174
3175
3176
3177
3178
  {
  	struct hist_field *hist_field = NULL;
  	struct field_var *field_var;
  
  	/*
  	 * First try to create a field var on the target event (the
  	 * currently being defined).  This will create a variable for
  	 * unqualified fields on the target event, or if qualified,
  	 * target fields that have qualified names matching the target.
  	 */
  	field_var = create_target_field_var(hist_data, system, event, var);
  
  	if (field_var && !IS_ERR(field_var)) {
  		save_field_var(hist_data, field_var);
  		hist_field = field_var->var;
  	} else {
  		field_var = NULL;
  		/*
  		 * If no explicit system.event is specfied, default to
  		 * looking for fields on the onmatch(system.event.xxx)
  		 * event.
  		 */
7d18a10c3   Tom Zanussi   tracing: Refactor...
3179
  		if (!system && data->handler == HANDLER_ONMATCH) {
c3e49506a   Tom Zanussi   tracing: Split up...
3180
3181
  			system = data->match_data.event_system;
  			event = data->match_data.event;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3182
3183
3184
3185
3186
3187
3188
3189
3190
3191
3192
3193
3194
3195
3196
3197
3198
3199
3200
3201
  		}
  
  		/*
  		 * At this point, we're looking at a field on another
  		 * event.  Because we can't modify a hist trigger on
  		 * another event to add a variable for a field, we need
  		 * to create a new trigger on that event and create the
  		 * variable at the same time.
  		 */
  		hist_field = create_field_var_hist(hist_data, system, event, var);
  		if (IS_ERR(hist_field))
  			goto free;
  	}
   out:
  	return hist_field;
   free:
  	destroy_field_var(field_var);
  	hist_field = NULL;
  	goto out;
  }
7d18a10c3   Tom Zanussi   tracing: Refactor...
3202
3203
  static int trace_action_create(struct hist_trigger_data *hist_data,
  			       struct action_data *data)
c282a386a   Tom Zanussi   tracing: Add 'onm...
3204
  {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3205
  	struct trace_array *tr = hist_data->event_file->tr;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3206
3207
  	char *event_name, *param, *system = NULL;
  	struct hist_field *hist_field, *var_ref;
d380dcde9   Tom Zanussi   tracing: Fix now ...
3208
  	unsigned int i;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3209
3210
  	unsigned int field_pos = 0;
  	struct synth_event *event;
e91eefd73   Tom Zanussi   tracing: Add alte...
3211
  	char *synth_event_name;
d380dcde9   Tom Zanussi   tracing: Fix now ...
3212
  	int var_ref_idx, ret = 0;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3213

0e2b81f7b   Masami Hiramatsu   tracing: Remove u...
3214
  	lockdep_assert_held(&event_mutex);
e91eefd73   Tom Zanussi   tracing: Add alte...
3215
3216
3217
3218
3219
3220
  	if (data->use_trace_keyword)
  		synth_event_name = data->synth_event_name;
  	else
  		synth_event_name = data->action_name;
  
  	event = find_synth_event(synth_event_name);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3221
  	if (!event) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3222
  		hist_err(tr, HIST_ERR_SYNTH_EVENT_NOT_FOUND, errpos(synth_event_name));
c282a386a   Tom Zanussi   tracing: Add 'onm...
3223
3224
  		return -EINVAL;
  	}
7d18a10c3   Tom Zanussi   tracing: Refactor...
3225

c282a386a   Tom Zanussi   tracing: Add 'onm...
3226
  	event->ref++;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3227

c282a386a   Tom Zanussi   tracing: Add 'onm...
3228
3229
3230
3231
3232
3233
3234
3235
3236
3237
3238
3239
3240
3241
3242
3243
3244
3245
3246
3247
3248
3249
3250
  	for (i = 0; i < data->n_params; i++) {
  		char *p;
  
  		p = param = kstrdup(data->params[i], GFP_KERNEL);
  		if (!param) {
  			ret = -ENOMEM;
  			goto err;
  		}
  
  		system = strsep(&param, ".");
  		if (!param) {
  			param = (char *)system;
  			system = event_name = NULL;
  		} else {
  			event_name = strsep(&param, ".");
  			if (!param) {
  				kfree(p);
  				ret = -EINVAL;
  				goto err;
  			}
  		}
  
  		if (param[0] == '$')
7d18a10c3   Tom Zanussi   tracing: Refactor...
3251
3252
3253
  			hist_field = trace_action_find_var(hist_data, data,
  							   system, event_name,
  							   param);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3254
  		else
7d18a10c3   Tom Zanussi   tracing: Refactor...
3255
3256
3257
3258
3259
  			hist_field = trace_action_create_field_var(hist_data,
  								   data,
  								   system,
  								   event_name,
  								   param);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3260
3261
3262
3263
3264
3265
3266
3267
  
  		if (!hist_field) {
  			kfree(p);
  			ret = -EINVAL;
  			goto err;
  		}
  
  		if (check_synth_field(event, hist_field, field_pos) == 0) {
de40f033d   Tom Zanussi   tracing: Remove o...
3268
3269
  			var_ref = create_var_ref(hist_data, hist_field,
  						 system, event_name);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3270
3271
3272
3273
3274
  			if (!var_ref) {
  				kfree(p);
  				ret = -ENOMEM;
  				goto err;
  			}
d380dcde9   Tom Zanussi   tracing: Fix now ...
3275
3276
3277
3278
3279
3280
3281
  			var_ref_idx = find_var_ref_idx(hist_data, var_ref);
  			if (WARN_ON(var_ref_idx < 0)) {
  				ret = var_ref_idx;
  				goto err;
  			}
  
  			data->var_ref_idx[i] = var_ref_idx;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3282
3283
3284
3285
  			field_pos++;
  			kfree(p);
  			continue;
  		}
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3286
  		hist_err(tr, HIST_ERR_SYNTH_TYPE_MISMATCH, errpos(param));
c282a386a   Tom Zanussi   tracing: Add 'onm...
3287
3288
3289
3290
3291
3292
  		kfree(p);
  		ret = -EINVAL;
  		goto err;
  	}
  
  	if (field_pos != event->n_fields) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3293
  		hist_err(tr, HIST_ERR_SYNTH_COUNT_MISMATCH, errpos(event->name));
c282a386a   Tom Zanussi   tracing: Add 'onm...
3294
3295
3296
  		ret = -EINVAL;
  		goto err;
  	}
c3e49506a   Tom Zanussi   tracing: Split up...
3297
  	data->synth_event = event;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3298
3299
3300
   out:
  	return ret;
   err:
c282a386a   Tom Zanussi   tracing: Add 'onm...
3301
  	event->ref--;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3302
3303
3304
  
  	goto out;
  }
7d18a10c3   Tom Zanussi   tracing: Refactor...
3305
3306
3307
  static int action_create(struct hist_trigger_data *hist_data,
  			 struct action_data *data)
  {
a3785b7ec   Tom Zanussi   tracing: Add hist...
3308
  	struct trace_event_file *file = hist_data->event_file;
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3309
  	struct trace_array *tr = file->tr;
a3785b7ec   Tom Zanussi   tracing: Add hist...
3310
  	struct track_data *track_data;
7d18a10c3   Tom Zanussi   tracing: Refactor...
3311
3312
3313
3314
3315
3316
3317
  	struct field_var *field_var;
  	unsigned int i;
  	char *param;
  	int ret = 0;
  
  	if (data->action == ACTION_TRACE)
  		return trace_action_create(hist_data, data);
a3785b7ec   Tom Zanussi   tracing: Add hist...
3318
3319
3320
3321
3322
3323
3324
3325
3326
3327
3328
3329
3330
3331
  	if (data->action == ACTION_SNAPSHOT) {
  		track_data = track_data_alloc(hist_data->key_size, data, hist_data);
  		if (IS_ERR(track_data)) {
  			ret = PTR_ERR(track_data);
  			goto out;
  		}
  
  		ret = tracing_snapshot_cond_enable(file->tr, track_data,
  						   cond_snapshot_update);
  		if (ret)
  			track_data_free(track_data);
  
  		goto out;
  	}
7d18a10c3   Tom Zanussi   tracing: Refactor...
3332
3333
3334
  	if (data->action == ACTION_SAVE) {
  		if (hist_data->n_save_vars) {
  			ret = -EEXIST;
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3335
  			hist_err(tr, HIST_ERR_TOO_MANY_SAVE_ACTIONS, 0);
7d18a10c3   Tom Zanussi   tracing: Refactor...
3336
3337
3338
3339
3340
3341
3342
3343
3344
3345
3346
3347
  			goto out;
  		}
  
  		for (i = 0; i < data->n_params; i++) {
  			param = kstrdup(data->params[i], GFP_KERNEL);
  			if (!param) {
  				ret = -ENOMEM;
  				goto out;
  			}
  
  			field_var = create_target_field_var(hist_data, NULL, NULL, param);
  			if (IS_ERR(field_var)) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3348
3349
  				hist_err(tr, HIST_ERR_FIELD_VAR_CREATE_FAIL,
  					 errpos(param));
7d18a10c3   Tom Zanussi   tracing: Refactor...
3350
3351
3352
3353
3354
3355
3356
3357
3358
3359
3360
3361
3362
3363
3364
3365
3366
3367
3368
3369
  				ret = PTR_ERR(field_var);
  				kfree(param);
  				goto out;
  			}
  
  			hist_data->save_vars[hist_data->n_save_vars++] = field_var;
  			if (field_var->val->flags & HIST_FIELD_FL_STRING)
  				hist_data->n_save_var_str++;
  			kfree(param);
  		}
  	}
   out:
  	return ret;
  }
  
  static int onmatch_create(struct hist_trigger_data *hist_data,
  			  struct action_data *data)
  {
  	return action_create(hist_data, data);
  }
c282a386a   Tom Zanussi   tracing: Add 'onm...
3370
3371
3372
  static struct action_data *onmatch_parse(struct trace_array *tr, char *str)
  {
  	char *match_event, *match_event_system;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3373
3374
3375
3376
3377
3378
3379
3380
  	struct action_data *data;
  	int ret = -EINVAL;
  
  	data = kzalloc(sizeof(*data), GFP_KERNEL);
  	if (!data)
  		return ERR_PTR(-ENOMEM);
  
  	match_event = strsep(&str, ")");
f404da6e1   Tom Zanussi   tracing: Add 'las...
3381
  	if (!match_event || !str) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3382
  		hist_err(tr, HIST_ERR_NO_CLOSING_PAREN, errpos(match_event));
c282a386a   Tom Zanussi   tracing: Add 'onm...
3383
  		goto free;
f404da6e1   Tom Zanussi   tracing: Add 'las...
3384
  	}
c282a386a   Tom Zanussi   tracing: Add 'onm...
3385
3386
  
  	match_event_system = strsep(&match_event, ".");
f404da6e1   Tom Zanussi   tracing: Add 'las...
3387
  	if (!match_event) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3388
  		hist_err(tr, HIST_ERR_SUBSYS_NOT_FOUND, errpos(match_event_system));
c282a386a   Tom Zanussi   tracing: Add 'onm...
3389
  		goto free;
f404da6e1   Tom Zanussi   tracing: Add 'las...
3390
  	}
c282a386a   Tom Zanussi   tracing: Add 'onm...
3391

f404da6e1   Tom Zanussi   tracing: Add 'las...
3392
  	if (IS_ERR(event_file(tr, match_event_system, match_event))) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3393
  		hist_err(tr, HIST_ERR_INVALID_SUBSYS_EVENT, errpos(match_event));
c282a386a   Tom Zanussi   tracing: Add 'onm...
3394
  		goto free;
f404da6e1   Tom Zanussi   tracing: Add 'las...
3395
  	}
c282a386a   Tom Zanussi   tracing: Add 'onm...
3396

c3e49506a   Tom Zanussi   tracing: Split up...
3397
3398
  	data->match_data.event = kstrdup(match_event, GFP_KERNEL);
  	if (!data->match_data.event) {
c282a386a   Tom Zanussi   tracing: Add 'onm...
3399
3400
3401
  		ret = -ENOMEM;
  		goto free;
  	}
c3e49506a   Tom Zanussi   tracing: Split up...
3402
3403
  	data->match_data.event_system = kstrdup(match_event_system, GFP_KERNEL);
  	if (!data->match_data.event_system) {
c282a386a   Tom Zanussi   tracing: Add 'onm...
3404
3405
3406
  		ret = -ENOMEM;
  		goto free;
  	}
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3407
  	ret = action_parse(tr, str, data, HANDLER_ONMATCH);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3408
3409
3410
3411
3412
3413
3414
3415
3416
  	if (ret)
  		goto free;
   out:
  	return data;
   free:
  	onmatch_destroy(data);
  	data = ERR_PTR(ret);
  	goto out;
  }
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3417
3418
3419
  static int create_hitcount_val(struct hist_trigger_data *hist_data)
  {
  	hist_data->fields[HITCOUNT_IDX] =
30350d65a   Tom Zanussi   tracing: Add vari...
3420
  		create_hist_field(hist_data, NULL, HIST_FIELD_FL_HITCOUNT, NULL);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3421
3422
3423
3424
  	if (!hist_data->fields[HITCOUNT_IDX])
  		return -ENOMEM;
  
  	hist_data->n_vals++;
30350d65a   Tom Zanussi   tracing: Add vari...
3425
  	hist_data->n_fields++;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3426
3427
3428
3429
3430
3431
  
  	if (WARN_ON(hist_data->n_vals > TRACING_MAP_VALS_MAX))
  		return -EINVAL;
  
  	return 0;
  }
30350d65a   Tom Zanussi   tracing: Add vari...
3432
3433
3434
3435
3436
  static int __create_val_field(struct hist_trigger_data *hist_data,
  			      unsigned int val_idx,
  			      struct trace_event_file *file,
  			      char *var_name, char *field_str,
  			      unsigned long flags)
f2606835d   Tom Zanussi   tracing: Add hist...
3437
  {
100719dce   Tom Zanussi   tracing: Add simp...
3438
  	struct hist_field *hist_field;
f2606835d   Tom Zanussi   tracing: Add hist...
3439
  	int ret = 0;
100719dce   Tom Zanussi   tracing: Add simp...
3440
3441
3442
  	hist_field = parse_expr(hist_data, file, field_str, flags, var_name, 0);
  	if (IS_ERR(hist_field)) {
  		ret = PTR_ERR(hist_field);
f2606835d   Tom Zanussi   tracing: Add hist...
3443
3444
  		goto out;
  	}
100719dce   Tom Zanussi   tracing: Add simp...
3445
  	hist_data->fields[val_idx] = hist_field;
f2606835d   Tom Zanussi   tracing: Add hist...
3446
  	++hist_data->n_vals;
30350d65a   Tom Zanussi   tracing: Add vari...
3447
  	++hist_data->n_fields;
f2606835d   Tom Zanussi   tracing: Add hist...
3448

30350d65a   Tom Zanussi   tracing: Add vari...
3449
  	if (WARN_ON(hist_data->n_vals > TRACING_MAP_VALS_MAX + TRACING_MAP_VARS_MAX))
f2606835d   Tom Zanussi   tracing: Add hist...
3450
3451
3452
3453
  		ret = -EINVAL;
   out:
  	return ret;
  }
30350d65a   Tom Zanussi   tracing: Add vari...
3454
3455
3456
3457
3458
3459
3460
3461
3462
3463
3464
3465
3466
3467
3468
3469
  static int create_val_field(struct hist_trigger_data *hist_data,
  			    unsigned int val_idx,
  			    struct trace_event_file *file,
  			    char *field_str)
  {
  	if (WARN_ON(val_idx >= TRACING_MAP_VALS_MAX))
  		return -EINVAL;
  
  	return __create_val_field(hist_data, val_idx, file, NULL, field_str, 0);
  }
  
  static int create_var_field(struct hist_trigger_data *hist_data,
  			    unsigned int val_idx,
  			    struct trace_event_file *file,
  			    char *var_name, char *expr_str)
  {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3470
  	struct trace_array *tr = hist_data->event_file->tr;
30350d65a   Tom Zanussi   tracing: Add vari...
3471
  	unsigned long flags = 0;
63a1e5de3   Tom Zanussi   tracing: Save nor...
3472
  	int ret;
30350d65a   Tom Zanussi   tracing: Add vari...
3473
3474
3475
  
  	if (WARN_ON(val_idx >= TRACING_MAP_VALS_MAX + TRACING_MAP_VARS_MAX))
  		return -EINVAL;
f404da6e1   Tom Zanussi   tracing: Add 'las...
3476

30350d65a   Tom Zanussi   tracing: Add vari...
3477
  	if (find_var(hist_data, file, var_name) && !hist_data->remove) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3478
  		hist_err(tr, HIST_ERR_DUPLICATE_VAR, errpos(var_name));
30350d65a   Tom Zanussi   tracing: Add vari...
3479
3480
3481
3482
3483
3484
3485
  		return -EINVAL;
  	}
  
  	flags |= HIST_FIELD_FL_VAR;
  	hist_data->n_vars++;
  	if (WARN_ON(hist_data->n_vars > TRACING_MAP_VARS_MAX))
  		return -EINVAL;
63a1e5de3   Tom Zanussi   tracing: Save nor...
3486
  	ret = __create_val_field(hist_data, val_idx, file, var_name, expr_str, flags);
6d9bd1394   Steven Rostedt (VMware)   tracing: Check re...
3487
  	if (!ret && hist_data->fields[val_idx]->flags & HIST_FIELD_FL_STRING)
63a1e5de3   Tom Zanussi   tracing: Save nor...
3488
3489
3490
  		hist_data->fields[val_idx]->var_str_idx = hist_data->n_var_str++;
  
  	return ret;
30350d65a   Tom Zanussi   tracing: Add vari...
3491
  }
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3492
3493
3494
  static int create_val_fields(struct hist_trigger_data *hist_data,
  			     struct trace_event_file *file)
  {
f2606835d   Tom Zanussi   tracing: Add hist...
3495
  	char *fields_str, *field_str;
30350d65a   Tom Zanussi   tracing: Add vari...
3496
  	unsigned int i, j = 1;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3497
3498
3499
  	int ret;
  
  	ret = create_hitcount_val(hist_data);
f2606835d   Tom Zanussi   tracing: Add hist...
3500
3501
  	if (ret)
  		goto out;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3502

f2606835d   Tom Zanussi   tracing: Add hist...
3503
3504
3505
  	fields_str = hist_data->attrs->vals_str;
  	if (!fields_str)
  		goto out;
f2606835d   Tom Zanussi   tracing: Add hist...
3506
3507
3508
3509
3510
  	for (i = 0, j = 1; i < TRACING_MAP_VALS_MAX &&
  		     j < TRACING_MAP_VALS_MAX; i++) {
  		field_str = strsep(&fields_str, ",");
  		if (!field_str)
  			break;
30350d65a   Tom Zanussi   tracing: Add vari...
3511

f2606835d   Tom Zanussi   tracing: Add hist...
3512
3513
  		if (strcmp(field_str, "hitcount") == 0)
  			continue;
30350d65a   Tom Zanussi   tracing: Add vari...
3514

f2606835d   Tom Zanussi   tracing: Add hist...
3515
3516
3517
3518
  		ret = create_val_field(hist_data, j++, file, field_str);
  		if (ret)
  			goto out;
  	}
30350d65a   Tom Zanussi   tracing: Add vari...
3519

f2606835d   Tom Zanussi   tracing: Add hist...
3520
3521
3522
  	if (fields_str && (strcmp(fields_str, "hitcount") != 0))
  		ret = -EINVAL;
   out:
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3523
3524
3525
3526
3527
  	return ret;
  }
  
  static int create_key_field(struct hist_trigger_data *hist_data,
  			    unsigned int key_idx,
76a3b0c8a   Tom Zanussi   tracing: Add hist...
3528
  			    unsigned int key_offset,
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3529
3530
3531
  			    struct trace_event_file *file,
  			    char *field_str)
  {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3532
  	struct trace_array *tr = hist_data->event_file->tr;
30350d65a   Tom Zanussi   tracing: Add vari...
3533
  	struct hist_field *hist_field = NULL;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3534
3535
3536
  	unsigned long flags = 0;
  	unsigned int key_size;
  	int ret = 0;
30350d65a   Tom Zanussi   tracing: Add vari...
3537
  	if (WARN_ON(key_idx >= HIST_FIELDS_MAX))
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3538
3539
3540
  		return -EINVAL;
  
  	flags |= HIST_FIELD_FL_KEY;
69a0200c2   Tom Zanussi   tracing: Add hist...
3541
3542
3543
  	if (strcmp(field_str, "stacktrace") == 0) {
  		flags |= HIST_FIELD_FL_STACKTRACE;
  		key_size = sizeof(unsigned long) * HIST_STACKTRACE_DEPTH;
30350d65a   Tom Zanussi   tracing: Add vari...
3544
  		hist_field = create_hist_field(hist_data, NULL, flags, NULL);
69a0200c2   Tom Zanussi   tracing: Add hist...
3545
  	} else {
100719dce   Tom Zanussi   tracing: Add simp...
3546
3547
3548
3549
3550
  		hist_field = parse_expr(hist_data, file, field_str, flags,
  					NULL, 0);
  		if (IS_ERR(hist_field)) {
  			ret = PTR_ERR(hist_field);
  			goto out;
69a0200c2   Tom Zanussi   tracing: Add hist...
3551
  		}
c8d94a187   Tom Zanussi   tracing: Check ke...
3552
  		if (field_has_hist_vars(hist_field, 0))	{
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3553
  			hist_err(tr, HIST_ERR_INVALID_REF_KEY, errpos(field_str));
067fe038e   Tom Zanussi   tracing: Add vari...
3554
3555
3556
3557
  			destroy_hist_field(hist_field, 0);
  			ret = -EINVAL;
  			goto out;
  		}
100719dce   Tom Zanussi   tracing: Add simp...
3558
  		key_size = hist_field->size;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3559
  	}
100719dce   Tom Zanussi   tracing: Add simp...
3560
  	hist_data->fields[key_idx] = hist_field;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3561
3562
3563
  
  	key_size = ALIGN(key_size, sizeof(u64));
  	hist_data->fields[key_idx]->size = key_size;
76a3b0c8a   Tom Zanussi   tracing: Add hist...
3564
  	hist_data->fields[key_idx]->offset = key_offset;
100719dce   Tom Zanussi   tracing: Add simp...
3565

76a3b0c8a   Tom Zanussi   tracing: Add hist...
3566
  	hist_data->key_size += key_size;
100719dce   Tom Zanussi   tracing: Add simp...
3567

7ef224d1d   Tom Zanussi   tracing: Add 'his...
3568
3569
3570
3571
3572
3573
  	if (hist_data->key_size > HIST_KEY_SIZE_MAX) {
  		ret = -EINVAL;
  		goto out;
  	}
  
  	hist_data->n_keys++;
30350d65a   Tom Zanussi   tracing: Add vari...
3574
  	hist_data->n_fields++;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3575
3576
3577
3578
3579
3580
3581
3582
3583
3584
3585
3586
  
  	if (WARN_ON(hist_data->n_keys > TRACING_MAP_KEYS_MAX))
  		return -EINVAL;
  
  	ret = key_size;
   out:
  	return ret;
  }
  
  static int create_key_fields(struct hist_trigger_data *hist_data,
  			     struct trace_event_file *file)
  {
76a3b0c8a   Tom Zanussi   tracing: Add hist...
3587
  	unsigned int i, key_offset = 0, n_vals = hist_data->n_vals;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3588
3589
3590
3591
3592
3593
  	char *fields_str, *field_str;
  	int ret = -EINVAL;
  
  	fields_str = hist_data->attrs->keys_str;
  	if (!fields_str)
  		goto out;
76a3b0c8a   Tom Zanussi   tracing: Add hist...
3594
  	for (i = n_vals; i < n_vals + TRACING_MAP_KEYS_MAX; i++) {
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3595
3596
3597
  		field_str = strsep(&fields_str, ",");
  		if (!field_str)
  			break;
76a3b0c8a   Tom Zanussi   tracing: Add hist...
3598
3599
  		ret = create_key_field(hist_data, i, key_offset,
  				       file, field_str);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3600
3601
  		if (ret < 0)
  			goto out;
76a3b0c8a   Tom Zanussi   tracing: Add hist...
3602
  		key_offset += ret;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3603
3604
3605
3606
3607
3608
3609
3610
3611
  	}
  	if (fields_str) {
  		ret = -EINVAL;
  		goto out;
  	}
  	ret = 0;
   out:
  	return ret;
  }
30350d65a   Tom Zanussi   tracing: Add vari...
3612
3613
3614
3615
3616
3617
3618
3619
3620
3621
3622
3623
3624
3625
3626
3627
3628
3629
3630
3631
3632
3633
3634
3635
3636
3637
3638
3639
3640
3641
3642
3643
3644
3645
  static int create_var_fields(struct hist_trigger_data *hist_data,
  			     struct trace_event_file *file)
  {
  	unsigned int i, j = hist_data->n_vals;
  	int ret = 0;
  
  	unsigned int n_vars = hist_data->attrs->var_defs.n_vars;
  
  	for (i = 0; i < n_vars; i++) {
  		char *var_name = hist_data->attrs->var_defs.name[i];
  		char *expr = hist_data->attrs->var_defs.expr[i];
  
  		ret = create_var_field(hist_data, j++, file, var_name, expr);
  		if (ret)
  			goto out;
  	}
   out:
  	return ret;
  }
  
  static void free_var_defs(struct hist_trigger_data *hist_data)
  {
  	unsigned int i;
  
  	for (i = 0; i < hist_data->attrs->var_defs.n_vars; i++) {
  		kfree(hist_data->attrs->var_defs.name[i]);
  		kfree(hist_data->attrs->var_defs.expr[i]);
  	}
  
  	hist_data->attrs->var_defs.n_vars = 0;
  }
  
  static int parse_var_defs(struct hist_trigger_data *hist_data)
  {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3646
  	struct trace_array *tr = hist_data->event_file->tr;
30350d65a   Tom Zanussi   tracing: Add vari...
3647
3648
3649
3650
3651
3652
3653
3654
3655
3656
3657
3658
3659
  	char *s, *str, *var_name, *field_str;
  	unsigned int i, j, n_vars = 0;
  	int ret = 0;
  
  	for (i = 0; i < hist_data->attrs->n_assignments; i++) {
  		str = hist_data->attrs->assignment_str[i];
  		for (j = 0; j < TRACING_MAP_VARS_MAX; j++) {
  			field_str = strsep(&str, ",");
  			if (!field_str)
  				break;
  
  			var_name = strsep(&field_str, "=");
  			if (!var_name || !field_str) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3660
3661
  				hist_err(tr, HIST_ERR_MALFORMED_ASSIGNMENT,
  					 errpos(var_name));
30350d65a   Tom Zanussi   tracing: Add vari...
3662
3663
3664
3665
3666
  				ret = -EINVAL;
  				goto free;
  			}
  
  			if (n_vars == TRACING_MAP_VARS_MAX) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
3667
  				hist_err(tr, HIST_ERR_TOO_MANY_VARS, errpos(var_name));
30350d65a   Tom Zanussi   tracing: Add vari...
3668
3669
3670
3671
3672
3673
3674
3675
3676
3677
3678
3679
3680
  				ret = -EINVAL;
  				goto free;
  			}
  
  			s = kstrdup(var_name, GFP_KERNEL);
  			if (!s) {
  				ret = -ENOMEM;
  				goto free;
  			}
  			hist_data->attrs->var_defs.name[n_vars] = s;
  
  			s = kstrdup(field_str, GFP_KERNEL);
  			if (!s) {
30350d65a   Tom Zanussi   tracing: Add vari...
3681
3682
3683
3684
3685
3686
3687
3688
3689
3690
3691
3692
3693
3694
3695
  				ret = -ENOMEM;
  				goto free;
  			}
  			hist_data->attrs->var_defs.expr[n_vars++] = s;
  
  			hist_data->attrs->var_defs.n_vars = n_vars;
  		}
  	}
  
  	return ret;
   free:
  	free_var_defs(hist_data);
  
  	return ret;
  }
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3696
3697
3698
3699
  static int create_hist_fields(struct hist_trigger_data *hist_data,
  			      struct trace_event_file *file)
  {
  	int ret;
30350d65a   Tom Zanussi   tracing: Add vari...
3700
3701
3702
  	ret = parse_var_defs(hist_data);
  	if (ret)
  		goto out;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3703
3704
3705
  	ret = create_val_fields(hist_data, file);
  	if (ret)
  		goto out;
30350d65a   Tom Zanussi   tracing: Add vari...
3706
  	ret = create_var_fields(hist_data, file);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3707
3708
  	if (ret)
  		goto out;
30350d65a   Tom Zanussi   tracing: Add vari...
3709
3710
3711
  	ret = create_key_fields(hist_data, file);
  	if (ret)
  		goto out;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3712
   out:
30350d65a   Tom Zanussi   tracing: Add vari...
3713
  	free_var_defs(hist_data);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3714
3715
  	return ret;
  }
4de26c8c9   Tom Zanussi   tracing: Add hist...
3716
  static int is_descending(struct trace_array *tr, const char *str)
e62347d24   Tom Zanussi   tracing: Add hist...
3717
3718
3719
3720
3721
3722
3723
3724
3725
  {
  	if (!str)
  		return 0;
  
  	if (strcmp(str, "descending") == 0)
  		return 1;
  
  	if (strcmp(str, "ascending") == 0)
  		return 0;
4de26c8c9   Tom Zanussi   tracing: Add hist...
3726
  	hist_err(tr, HIST_ERR_INVALID_SORT_MODIFIER, errpos((char *)str));
e62347d24   Tom Zanussi   tracing: Add hist...
3727
3728
  	return -EINVAL;
  }
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3729
3730
  static int create_sort_keys(struct hist_trigger_data *hist_data)
  {
4de26c8c9   Tom Zanussi   tracing: Add hist...
3731
  	struct trace_array *tr = hist_data->event_file->tr;
e62347d24   Tom Zanussi   tracing: Add hist...
3732
  	char *fields_str = hist_data->attrs->sort_key_str;
e62347d24   Tom Zanussi   tracing: Add hist...
3733
3734
  	struct tracing_map_sort_key *sort_key;
  	int descending, ret = 0;
30350d65a   Tom Zanussi   tracing: Add vari...
3735
  	unsigned int i, j, k;
e62347d24   Tom Zanussi   tracing: Add hist...
3736
3737
3738
3739
3740
  
  	hist_data->n_sort_keys = 1; /* we always have at least one, hitcount */
  
  	if (!fields_str)
  		goto out;
e62347d24   Tom Zanussi   tracing: Add hist...
3741
  	for (i = 0; i < TRACING_MAP_SORT_KEYS_MAX; i++) {
85013256c   Tom Zanussi   tracing: Add hist...
3742
  		struct hist_field *hist_field;
e62347d24   Tom Zanussi   tracing: Add hist...
3743
  		char *field_str, *field_name;
85013256c   Tom Zanussi   tracing: Add hist...
3744
  		const char *test_name;
e62347d24   Tom Zanussi   tracing: Add hist...
3745
3746
3747
3748
  
  		sort_key = &hist_data->sort_keys[i];
  
  		field_str = strsep(&fields_str, ",");
b527b638f   Tom Zanussi   tracing: Simplify...
3749
3750
3751
3752
3753
  		if (!field_str)
  			break;
  
  		if (!*field_str) {
  			ret = -EINVAL;
4de26c8c9   Tom Zanussi   tracing: Add hist...
3754
  			hist_err(tr, HIST_ERR_EMPTY_SORT_FIELD, errpos("sort="));
e62347d24   Tom Zanussi   tracing: Add hist...
3755
3756
3757
3758
  			break;
  		}
  
  		if ((i == TRACING_MAP_SORT_KEYS_MAX - 1) && fields_str) {
4de26c8c9   Tom Zanussi   tracing: Add hist...
3759
  			hist_err(tr, HIST_ERR_TOO_MANY_SORT_FIELDS, errpos("sort="));
e62347d24   Tom Zanussi   tracing: Add hist...
3760
3761
3762
  			ret = -EINVAL;
  			break;
  		}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3763

e62347d24   Tom Zanussi   tracing: Add hist...
3764
  		field_name = strsep(&field_str, ".");
b527b638f   Tom Zanussi   tracing: Simplify...
3765
  		if (!field_name || !*field_name) {
e62347d24   Tom Zanussi   tracing: Add hist...
3766
  			ret = -EINVAL;
4de26c8c9   Tom Zanussi   tracing: Add hist...
3767
  			hist_err(tr, HIST_ERR_EMPTY_SORT_FIELD, errpos("sort="));
e62347d24   Tom Zanussi   tracing: Add hist...
3768
3769
3770
3771
  			break;
  		}
  
  		if (strcmp(field_name, "hitcount") == 0) {
4de26c8c9   Tom Zanussi   tracing: Add hist...
3772
  			descending = is_descending(tr, field_str);
e62347d24   Tom Zanussi   tracing: Add hist...
3773
3774
3775
3776
3777
3778
3779
  			if (descending < 0) {
  				ret = descending;
  				break;
  			}
  			sort_key->descending = descending;
  			continue;
  		}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3780

30350d65a   Tom Zanussi   tracing: Add vari...
3781
3782
  		for (j = 1, k = 1; j < hist_data->n_fields; j++) {
  			unsigned int idx;
85013256c   Tom Zanussi   tracing: Add hist...
3783
  			hist_field = hist_data->fields[j];
30350d65a   Tom Zanussi   tracing: Add vari...
3784
3785
3786
3787
  			if (hist_field->flags & HIST_FIELD_FL_VAR)
  				continue;
  
  			idx = k++;
85013256c   Tom Zanussi   tracing: Add hist...
3788
3789
3790
  			test_name = hist_field_name(hist_field, 0);
  
  			if (strcmp(field_name, test_name) == 0) {
30350d65a   Tom Zanussi   tracing: Add vari...
3791
  				sort_key->field_idx = idx;
4de26c8c9   Tom Zanussi   tracing: Add hist...
3792
  				descending = is_descending(tr, field_str);
e62347d24   Tom Zanussi   tracing: Add hist...
3793
3794
3795
3796
3797
3798
3799
3800
3801
3802
  				if (descending < 0) {
  					ret = descending;
  					goto out;
  				}
  				sort_key->descending = descending;
  				break;
  			}
  		}
  		if (j == hist_data->n_fields) {
  			ret = -EINVAL;
4de26c8c9   Tom Zanussi   tracing: Add hist...
3803
  			hist_err(tr, HIST_ERR_INVALID_SORT_FIELD, errpos(field_name));
e62347d24   Tom Zanussi   tracing: Add hist...
3804
3805
3806
  			break;
  		}
  	}
30350d65a   Tom Zanussi   tracing: Add vari...
3807

e62347d24   Tom Zanussi   tracing: Add hist...
3808
3809
  	hist_data->n_sort_keys = i;
   out:
7ef224d1d   Tom Zanussi   tracing: Add 'his...
3810
3811
  	return ret;
  }
0212e2aa3   Tom Zanussi   tracing: Add hist...
3812
3813
3814
3815
3816
3817
  static void destroy_actions(struct hist_trigger_data *hist_data)
  {
  	unsigned int i;
  
  	for (i = 0; i < hist_data->n_actions; i++) {
  		struct action_data *data = hist_data->actions[i];
7d18a10c3   Tom Zanussi   tracing: Refactor...
3818
  		if (data->handler == HANDLER_ONMATCH)
c282a386a   Tom Zanussi   tracing: Add 'onm...
3819
  			onmatch_destroy(data);
dff81f559   Tom Zanussi   tracing: Add hist...
3820
3821
  		else if (data->handler == HANDLER_ONMAX ||
  			 data->handler == HANDLER_ONCHANGE)
466f4528f   Tom Zanussi   tracing: Generali...
3822
  			track_data_destroy(hist_data, data);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3823
3824
  		else
  			kfree(data);
0212e2aa3   Tom Zanussi   tracing: Add hist...
3825
3826
3827
3828
3829
  	}
  }
  
  static int parse_actions(struct hist_trigger_data *hist_data)
  {
c282a386a   Tom Zanussi   tracing: Add 'onm...
3830
3831
  	struct trace_array *tr = hist_data->event_file->tr;
  	struct action_data *data;
0212e2aa3   Tom Zanussi   tracing: Add hist...
3832
3833
3834
  	unsigned int i;
  	int ret = 0;
  	char *str;
036876fa5   Steven Rostedt (VMware)   tracing: Have the...
3835
  	int len;
0212e2aa3   Tom Zanussi   tracing: Add hist...
3836
3837
3838
  
  	for (i = 0; i < hist_data->attrs->n_actions; i++) {
  		str = hist_data->attrs->action_str[i];
c282a386a   Tom Zanussi   tracing: Add 'onm...
3839

036876fa5   Steven Rostedt (VMware)   tracing: Have the...
3840
3841
  		if ((len = str_has_prefix(str, "onmatch("))) {
  			char *action_str = str + len;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3842
3843
3844
3845
3846
3847
  
  			data = onmatch_parse(tr, action_str);
  			if (IS_ERR(data)) {
  				ret = PTR_ERR(data);
  				break;
  			}
036876fa5   Steven Rostedt (VMware)   tracing: Have the...
3848
3849
  		} else if ((len = str_has_prefix(str, "onmax("))) {
  			char *action_str = str + len;
50450603e   Tom Zanussi   tracing: Add 'onm...
3850

466f4528f   Tom Zanussi   tracing: Generali...
3851
3852
  			data = track_data_parse(hist_data, action_str,
  						HANDLER_ONMAX);
50450603e   Tom Zanussi   tracing: Add 'onm...
3853
3854
3855
3856
  			if (IS_ERR(data)) {
  				ret = PTR_ERR(data);
  				break;
  			}
dff81f559   Tom Zanussi   tracing: Add hist...
3857
3858
3859
3860
3861
3862
3863
3864
3865
  		} else if ((len = str_has_prefix(str, "onchange("))) {
  			char *action_str = str + len;
  
  			data = track_data_parse(hist_data, action_str,
  						HANDLER_ONCHANGE);
  			if (IS_ERR(data)) {
  				ret = PTR_ERR(data);
  				break;
  			}
c282a386a   Tom Zanussi   tracing: Add 'onm...
3866
3867
3868
3869
3870
3871
  		} else {
  			ret = -EINVAL;
  			break;
  		}
  
  		hist_data->actions[hist_data->n_actions++] = data;
0212e2aa3   Tom Zanussi   tracing: Add hist...
3872
3873
3874
3875
  	}
  
  	return ret;
  }
7d18a10c3   Tom Zanussi   tracing: Refactor...
3876
  static int create_actions(struct hist_trigger_data *hist_data)
0212e2aa3   Tom Zanussi   tracing: Add hist...
3877
3878
3879
3880
3881
3882
3883
  {
  	struct action_data *data;
  	unsigned int i;
  	int ret = 0;
  
  	for (i = 0; i < hist_data->attrs->n_actions; i++) {
  		data = hist_data->actions[i];
c282a386a   Tom Zanussi   tracing: Add 'onm...
3884

7d18a10c3   Tom Zanussi   tracing: Refactor...
3885
3886
  		if (data->handler == HANDLER_ONMATCH) {
  			ret = onmatch_create(hist_data, data);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3887
  			if (ret)
7d18a10c3   Tom Zanussi   tracing: Refactor...
3888
  				break;
dff81f559   Tom Zanussi   tracing: Add hist...
3889
3890
  		} else if (data->handler == HANDLER_ONMAX ||
  			   data->handler == HANDLER_ONCHANGE) {
466f4528f   Tom Zanussi   tracing: Generali...
3891
  			ret = track_data_create(hist_data, data);
50450603e   Tom Zanussi   tracing: Add 'onm...
3892
  			if (ret)
7d18a10c3   Tom Zanussi   tracing: Refactor...
3893
3894
3895
3896
  				break;
  		} else {
  			ret = -EINVAL;
  			break;
c282a386a   Tom Zanussi   tracing: Add 'onm...
3897
  		}
0212e2aa3   Tom Zanussi   tracing: Add hist...
3898
3899
3900
3901
  	}
  
  	return ret;
  }
50450603e   Tom Zanussi   tracing: Add 'onm...
3902
3903
3904
3905
3906
3907
3908
3909
  static void print_actions(struct seq_file *m,
  			  struct hist_trigger_data *hist_data,
  			  struct tracing_map_elt *elt)
  {
  	unsigned int i;
  
  	for (i = 0; i < hist_data->n_actions; i++) {
  		struct action_data *data = hist_data->actions[i];
a3785b7ec   Tom Zanussi   tracing: Add hist...
3910
3911
  		if (data->action == ACTION_SNAPSHOT)
  			continue;
dff81f559   Tom Zanussi   tracing: Add hist...
3912
3913
  		if (data->handler == HANDLER_ONMAX ||
  		    data->handler == HANDLER_ONCHANGE)
466f4528f   Tom Zanussi   tracing: Generali...
3914
  			track_data_print(m, hist_data, elt, data);
50450603e   Tom Zanussi   tracing: Add 'onm...
3915
3916
  	}
  }
7d18a10c3   Tom Zanussi   tracing: Refactor...
3917
3918
3919
3920
3921
3922
3923
3924
3925
3926
3927
3928
3929
  static void print_action_spec(struct seq_file *m,
  			      struct hist_trigger_data *hist_data,
  			      struct action_data *data)
  {
  	unsigned int i;
  
  	if (data->action == ACTION_SAVE) {
  		for (i = 0; i < hist_data->n_save_vars; i++) {
  			seq_printf(m, "%s", hist_data->save_vars[i]->var->var.name);
  			if (i < hist_data->n_save_vars - 1)
  				seq_puts(m, ",");
  		}
  	} else if (data->action == ACTION_TRACE) {
e91eefd73   Tom Zanussi   tracing: Add alte...
3930
3931
  		if (data->use_trace_keyword)
  			seq_printf(m, "%s", data->synth_event_name);
7d18a10c3   Tom Zanussi   tracing: Refactor...
3932
  		for (i = 0; i < data->n_params; i++) {
e91eefd73   Tom Zanussi   tracing: Add alte...
3933
  			if (i || data->use_trace_keyword)
7d18a10c3   Tom Zanussi   tracing: Refactor...
3934
3935
3936
3937
3938
  				seq_puts(m, ",");
  			seq_printf(m, "%s", data->params[i]);
  		}
  	}
  }
466f4528f   Tom Zanussi   tracing: Generali...
3939
3940
3941
  static void print_track_data_spec(struct seq_file *m,
  				  struct hist_trigger_data *hist_data,
  				  struct action_data *data)
50450603e   Tom Zanussi   tracing: Add 'onm...
3942
  {
466f4528f   Tom Zanussi   tracing: Generali...
3943
3944
  	if (data->handler == HANDLER_ONMAX)
  		seq_puts(m, ":onmax(");
dff81f559   Tom Zanussi   tracing: Add hist...
3945
3946
  	else if (data->handler == HANDLER_ONCHANGE)
  		seq_puts(m, ":onchange(");
466f4528f   Tom Zanussi   tracing: Generali...
3947
  	seq_printf(m, "%s", data->track_data.var_str);
7d18a10c3   Tom Zanussi   tracing: Refactor...
3948
3949
3950
  	seq_printf(m, ").%s(", data->action_name);
  
  	print_action_spec(m, hist_data, data);
50450603e   Tom Zanussi   tracing: Add 'onm...
3951

50450603e   Tom Zanussi   tracing: Add 'onm...
3952
3953
  	seq_puts(m, ")");
  }
c282a386a   Tom Zanussi   tracing: Add 'onm...
3954
3955
3956
3957
  static void print_onmatch_spec(struct seq_file *m,
  			       struct hist_trigger_data *hist_data,
  			       struct action_data *data)
  {
c3e49506a   Tom Zanussi   tracing: Split up...
3958
3959
  	seq_printf(m, ":onmatch(%s.%s).", data->match_data.event_system,
  		   data->match_data.event);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3960

7d18a10c3   Tom Zanussi   tracing: Refactor...
3961
  	seq_printf(m, "%s(", data->action_name);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3962

7d18a10c3   Tom Zanussi   tracing: Refactor...
3963
  	print_action_spec(m, hist_data, data);
c282a386a   Tom Zanussi   tracing: Add 'onm...
3964
3965
3966
  
  	seq_puts(m, ")");
  }
48f794731   Tom Zanussi   tracing: Add acti...
3967
3968
3969
3970
3971
3972
3973
3974
3975
3976
3977
  static bool actions_match(struct hist_trigger_data *hist_data,
  			  struct hist_trigger_data *hist_data_test)
  {
  	unsigned int i, j;
  
  	if (hist_data->n_actions != hist_data_test->n_actions)
  		return false;
  
  	for (i = 0; i < hist_data->n_actions; i++) {
  		struct action_data *data = hist_data->actions[i];
  		struct action_data *data_test = hist_data_test->actions[i];
e91eefd73   Tom Zanussi   tracing: Add alte...
3978
  		char *action_name, *action_name_test;
48f794731   Tom Zanussi   tracing: Add acti...
3979

7d18a10c3   Tom Zanussi   tracing: Refactor...
3980
3981
3982
  		if (data->handler != data_test->handler)
  			return false;
  		if (data->action != data_test->action)
48f794731   Tom Zanussi   tracing: Add acti...
3983
3984
3985
3986
3987
3988
3989
3990
3991
  			return false;
  
  		if (data->n_params != data_test->n_params)
  			return false;
  
  		for (j = 0; j < data->n_params; j++) {
  			if (strcmp(data->params[j], data_test->params[j]) != 0)
  				return false;
  		}
e91eefd73   Tom Zanussi   tracing: Add alte...
3992
3993
3994
3995
3996
3997
3998
3999
4000
4001
4002
  		if (data->use_trace_keyword)
  			action_name = data->synth_event_name;
  		else
  			action_name = data->action_name;
  
  		if (data_test->use_trace_keyword)
  			action_name_test = data_test->synth_event_name;
  		else
  			action_name_test = data_test->action_name;
  
  		if (strcmp(action_name, action_name_test) != 0)
7d18a10c3   Tom Zanussi   tracing: Refactor...
4003
4004
4005
  			return false;
  
  		if (data->handler == HANDLER_ONMATCH) {
c3e49506a   Tom Zanussi   tracing: Split up...
4006
4007
  			if (strcmp(data->match_data.event_system,
  				   data_test->match_data.event_system) != 0)
48f794731   Tom Zanussi   tracing: Add acti...
4008
  				return false;
c3e49506a   Tom Zanussi   tracing: Split up...
4009
4010
  			if (strcmp(data->match_data.event,
  				   data_test->match_data.event) != 0)
48f794731   Tom Zanussi   tracing: Add acti...
4011
  				return false;
dff81f559   Tom Zanussi   tracing: Add hist...
4012
4013
  		} else if (data->handler == HANDLER_ONMAX ||
  			   data->handler == HANDLER_ONCHANGE) {
466f4528f   Tom Zanussi   tracing: Generali...
4014
4015
  			if (strcmp(data->track_data.var_str,
  				   data_test->track_data.var_str) != 0)
48f794731   Tom Zanussi   tracing: Add acti...
4016
  				return false;
48f794731   Tom Zanussi   tracing: Add acti...
4017
4018
4019
4020
4021
  		}
  	}
  
  	return true;
  }
c282a386a   Tom Zanussi   tracing: Add 'onm...
4022
4023
4024
4025
4026
4027
4028
  static void print_actions_spec(struct seq_file *m,
  			       struct hist_trigger_data *hist_data)
  {
  	unsigned int i;
  
  	for (i = 0; i < hist_data->n_actions; i++) {
  		struct action_data *data = hist_data->actions[i];
7d18a10c3   Tom Zanussi   tracing: Refactor...
4029
  		if (data->handler == HANDLER_ONMATCH)
c282a386a   Tom Zanussi   tracing: Add 'onm...
4030
  			print_onmatch_spec(m, hist_data, data);
dff81f559   Tom Zanussi   tracing: Add hist...
4031
4032
  		else if (data->handler == HANDLER_ONMAX ||
  			 data->handler == HANDLER_ONCHANGE)
466f4528f   Tom Zanussi   tracing: Generali...
4033
  			print_track_data_spec(m, hist_data, data);
c282a386a   Tom Zanussi   tracing: Add 'onm...
4034
4035
  	}
  }
02205a675   Tom Zanussi   tracing: Add supp...
4036
4037
4038
4039
4040
4041
4042
4043
4044
  static void destroy_field_var_hists(struct hist_trigger_data *hist_data)
  {
  	unsigned int i;
  
  	for (i = 0; i < hist_data->n_field_var_hists; i++) {
  		kfree(hist_data->field_var_hists[i]->cmd);
  		kfree(hist_data->field_var_hists[i]);
  	}
  }
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4045
4046
  static void destroy_hist_data(struct hist_trigger_data *hist_data)
  {
0212e2aa3   Tom Zanussi   tracing: Add hist...
4047
4048
  	if (!hist_data)
  		return;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4049
4050
4051
  	destroy_hist_trigger_attrs(hist_data->attrs);
  	destroy_hist_fields(hist_data);
  	tracing_map_destroy(hist_data->map);
0212e2aa3   Tom Zanussi   tracing: Add hist...
4052
4053
  
  	destroy_actions(hist_data);
02205a675   Tom Zanussi   tracing: Add supp...
4054
4055
  	destroy_field_vars(hist_data);
  	destroy_field_var_hists(hist_data);
0212e2aa3   Tom Zanussi   tracing: Add hist...
4056

7ef224d1d   Tom Zanussi   tracing: Add 'his...
4057
4058
4059
4060
4061
4062
4063
4064
  	kfree(hist_data);
  }
  
  static int create_tracing_map_fields(struct hist_trigger_data *hist_data)
  {
  	struct tracing_map *map = hist_data->map;
  	struct ftrace_event_field *field;
  	struct hist_field *hist_field;
b28d7b2dc   Dan Carpenter   tracing: Uninitia...
4065
  	int i, idx = 0;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4066
4067
4068
4069
4070
4071
4072
  
  	for_each_hist_field(i, hist_data) {
  		hist_field = hist_data->fields[i];
  		if (hist_field->flags & HIST_FIELD_FL_KEY) {
  			tracing_map_cmp_fn_t cmp_fn;
  
  			field = hist_field->field;
69a0200c2   Tom Zanussi   tracing: Add hist...
4073
4074
  			if (hist_field->flags & HIST_FIELD_FL_STACKTRACE)
  				cmp_fn = tracing_map_cmp_none;
ad42febe5   Tom Zanussi   tracing: Add hist...
4075
4076
4077
  			else if (!field)
  				cmp_fn = tracing_map_cmp_num(hist_field->size,
  							     hist_field->is_signed);
69a0200c2   Tom Zanussi   tracing: Add hist...
4078
  			else if (is_string_field(field))
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4079
4080
4081
4082
  				cmp_fn = tracing_map_cmp_string;
  			else
  				cmp_fn = tracing_map_cmp_num(field->size,
  							     field->is_signed);
76a3b0c8a   Tom Zanussi   tracing: Add hist...
4083
4084
4085
  			idx = tracing_map_add_key_field(map,
  							hist_field->offset,
  							cmp_fn);
30350d65a   Tom Zanussi   tracing: Add vari...
4086
  		} else if (!(hist_field->flags & HIST_FIELD_FL_VAR))
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4087
4088
4089
4090
  			idx = tracing_map_add_sum_field(map);
  
  		if (idx < 0)
  			return idx;
30350d65a   Tom Zanussi   tracing: Add vari...
4091
4092
4093
4094
4095
4096
4097
4098
  
  		if (hist_field->flags & HIST_FIELD_FL_VAR) {
  			idx = tracing_map_add_var(map);
  			if (idx < 0)
  				return idx;
  			hist_field->var.idx = idx;
  			hist_field->var.hist_data = hist_data;
  		}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4099
4100
4101
4102
4103
4104
4105
4106
  	}
  
  	return 0;
  }
  
  static struct hist_trigger_data *
  create_hist_data(unsigned int map_bits,
  		 struct hist_trigger_attrs *attrs,
30350d65a   Tom Zanussi   tracing: Add vari...
4107
4108
  		 struct trace_event_file *file,
  		 bool remove)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4109
  {
6b4827ad0   Tom Zanussi   tracing: Add hist...
4110
  	const struct tracing_map_ops *map_ops = NULL;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4111
4112
4113
4114
4115
4116
4117
4118
  	struct hist_trigger_data *hist_data;
  	int ret = 0;
  
  	hist_data = kzalloc(sizeof(*hist_data), GFP_KERNEL);
  	if (!hist_data)
  		return ERR_PTR(-ENOMEM);
  
  	hist_data->attrs = attrs;
30350d65a   Tom Zanussi   tracing: Add vari...
4119
  	hist_data->remove = remove;
067fe038e   Tom Zanussi   tracing: Add vari...
4120
  	hist_data->event_file = file;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4121

0212e2aa3   Tom Zanussi   tracing: Add hist...
4122
4123
4124
  	ret = parse_actions(hist_data);
  	if (ret)
  		goto free;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4125
4126
4127
4128
4129
4130
4131
  	ret = create_hist_fields(hist_data, file);
  	if (ret)
  		goto free;
  
  	ret = create_sort_keys(hist_data);
  	if (ret)
  		goto free;
af6a29bca   Tom Zanussi   tracing: Generali...
4132
  	map_ops = &hist_trigger_elt_data_ops;
6b4827ad0   Tom Zanussi   tracing: Add hist...
4133

7ef224d1d   Tom Zanussi   tracing: Add 'his...
4134
  	hist_data->map = tracing_map_create(map_bits, hist_data->key_size,
6b4827ad0   Tom Zanussi   tracing: Add hist...
4135
  					    map_ops, hist_data);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4136
4137
4138
4139
4140
4141
4142
4143
4144
  	if (IS_ERR(hist_data->map)) {
  		ret = PTR_ERR(hist_data->map);
  		hist_data->map = NULL;
  		goto free;
  	}
  
  	ret = create_tracing_map_fields(hist_data);
  	if (ret)
  		goto free;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4145
4146
4147
4148
4149
4150
4151
4152
4153
4154
4155
4156
4157
   out:
  	return hist_data;
   free:
  	hist_data->attrs = NULL;
  
  	destroy_hist_data(hist_data);
  
  	hist_data = ERR_PTR(ret);
  
  	goto out;
  }
  
  static void hist_trigger_elt_update(struct hist_trigger_data *hist_data,
fbd302cbe   Tom Zanussi   tracing: Add ring...
4158
  				    struct tracing_map_elt *elt, void *rec,
067fe038e   Tom Zanussi   tracing: Add vari...
4159
4160
  				    struct ring_buffer_event *rbe,
  				    u64 *var_ref_vals)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4161
  {
067fe038e   Tom Zanussi   tracing: Add vari...
4162
  	struct hist_elt_data *elt_data;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4163
  	struct hist_field *hist_field;
30350d65a   Tom Zanussi   tracing: Add vari...
4164
  	unsigned int i, var_idx;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4165
  	u64 hist_val;
067fe038e   Tom Zanussi   tracing: Add vari...
4166
4167
  	elt_data = elt->private_data;
  	elt_data->var_ref_vals = var_ref_vals;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4168
4169
  	for_each_hist_val_field(i, hist_data) {
  		hist_field = hist_data->fields[i];
df35d93bb   Tom Zanussi   tracing: Pass tra...
4170
  		hist_val = hist_field->fn(hist_field, elt, rbe, rec);
30350d65a   Tom Zanussi   tracing: Add vari...
4171
4172
  		if (hist_field->flags & HIST_FIELD_FL_VAR) {
  			var_idx = hist_field->var.idx;
63a1e5de3   Tom Zanussi   tracing: Save nor...
4173
4174
4175
4176
4177
4178
4179
4180
4181
4182
4183
4184
4185
4186
4187
4188
  
  			if (hist_field->flags & HIST_FIELD_FL_STRING) {
  				unsigned int str_start, var_str_idx, idx;
  				char *str, *val_str;
  
  				str_start = hist_data->n_field_var_str +
  					hist_data->n_save_var_str;
  				var_str_idx = hist_field->var_str_idx;
  				idx = str_start + var_str_idx;
  
  				str = elt_data->field_var_str[idx];
  				val_str = (char *)(uintptr_t)hist_val;
  				strscpy(str, val_str, STR_VAR_LEN_MAX);
  
  				hist_val = (u64)(uintptr_t)str;
  			}
30350d65a   Tom Zanussi   tracing: Add vari...
4189
4190
4191
  			tracing_map_set_var(elt, var_idx, hist_val);
  			continue;
  		}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4192
4193
  		tracing_map_update_sum(elt, i, hist_val);
  	}
30350d65a   Tom Zanussi   tracing: Add vari...
4194
4195
4196
4197
  
  	for_each_hist_key_field(i, hist_data) {
  		hist_field = hist_data->fields[i];
  		if (hist_field->flags & HIST_FIELD_FL_VAR) {
df35d93bb   Tom Zanussi   tracing: Pass tra...
4198
  			hist_val = hist_field->fn(hist_field, elt, rbe, rec);
30350d65a   Tom Zanussi   tracing: Add vari...
4199
4200
4201
4202
  			var_idx = hist_field->var.idx;
  			tracing_map_set_var(elt, var_idx, hist_val);
  		}
  	}
02205a675   Tom Zanussi   tracing: Add supp...
4203
4204
  
  	update_field_vars(hist_data, elt, rbe, rec);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4205
  }
6a475cb17   Tom Zanussi   tracing: Remove r...
4206
4207
4208
4209
4210
4211
4212
4213
4214
4215
4216
4217
4218
4219
4220
4221
4222
4223
4224
  static inline void add_to_key(char *compound_key, void *key,
  			      struct hist_field *key_field, void *rec)
  {
  	size_t size = key_field->size;
  
  	if (key_field->flags & HIST_FIELD_FL_STRING) {
  		struct ftrace_event_field *field;
  
  		field = key_field->field;
  		if (field->filter_type == FILTER_DYN_STRING)
  			size = *(u32 *)(rec + field->offset) >> 16;
  		else if (field->filter_type == FILTER_PTR_STRING)
  			size = strlen(key);
  		else if (field->filter_type == FILTER_STATIC_STRING)
  			size = field->size;
  
  		/* ensure NULL-termination */
  		if (size > key_field->size - 1)
  			size = key_field->size - 1;
6a475cb17   Tom Zanussi   tracing: Remove r...
4225

9f0bbf311   Tom Zanussi   tracing: Use strn...
4226
4227
4228
  		strncpy(compound_key + key_field->offset, (char *)key, size);
  	} else
  		memcpy(compound_key + key_field->offset, key, size);
6a475cb17   Tom Zanussi   tracing: Remove r...
4229
  }
0212e2aa3   Tom Zanussi   tracing: Add hist...
4230
4231
4232
  static void
  hist_trigger_actions(struct hist_trigger_data *hist_data,
  		     struct tracing_map_elt *elt, void *rec,
7d18a10c3   Tom Zanussi   tracing: Refactor...
4233
4234
  		     struct ring_buffer_event *rbe, void *key,
  		     u64 *var_ref_vals)
0212e2aa3   Tom Zanussi   tracing: Add hist...
4235
4236
4237
4238
4239
4240
  {
  	struct action_data *data;
  	unsigned int i;
  
  	for (i = 0; i < hist_data->n_actions; i++) {
  		data = hist_data->actions[i];
7d18a10c3   Tom Zanussi   tracing: Refactor...
4241
  		data->fn(hist_data, elt, rec, rbe, key, data, var_ref_vals);
0212e2aa3   Tom Zanussi   tracing: Add hist...
4242
4243
  	}
  }
1ac4f51c0   Tom Zanussi   tracing: Give eve...
4244
  static void event_hist_trigger(struct event_trigger_data *data, void *rec,
fbd302cbe   Tom Zanussi   tracing: Add ring...
4245
  			       struct ring_buffer_event *rbe)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4246
4247
  {
  	struct hist_trigger_data *hist_data = data->private_data;
6a475cb17   Tom Zanussi   tracing: Remove r...
4248
  	bool use_compound_key = (hist_data->n_keys > 1);
69a0200c2   Tom Zanussi   tracing: Add hist...
4249
  	unsigned long entries[HIST_STACKTRACE_DEPTH];
067fe038e   Tom Zanussi   tracing: Add vari...
4250
  	u64 var_ref_vals[TRACING_MAP_VARS_MAX];
76a3b0c8a   Tom Zanussi   tracing: Add hist...
4251
  	char compound_key[HIST_KEY_SIZE_MAX];
df35d93bb   Tom Zanussi   tracing: Pass tra...
4252
  	struct tracing_map_elt *elt = NULL;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4253
  	struct hist_field *key_field;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4254
4255
4256
  	u64 field_contents;
  	void *key = NULL;
  	unsigned int i;
6a475cb17   Tom Zanussi   tracing: Remove r...
4257
  	memset(compound_key, 0, hist_data->key_size);
76a3b0c8a   Tom Zanussi   tracing: Add hist...
4258

7ef224d1d   Tom Zanussi   tracing: Add 'his...
4259
4260
  	for_each_hist_key_field(i, hist_data) {
  		key_field = hist_data->fields[i];
69a0200c2   Tom Zanussi   tracing: Add hist...
4261
  		if (key_field->flags & HIST_FIELD_FL_STACKTRACE) {
e7d916632   Thomas Gleixner   tracing: Simplify...
4262
4263
4264
  			memset(entries, 0, HIST_STACKTRACE_SIZE);
  			stack_trace_save(entries, HIST_STACKTRACE_DEPTH,
  					 HIST_STACKTRACE_SKIP);
69a0200c2   Tom Zanussi   tracing: Add hist...
4265
4266
  			key = entries;
  		} else {
df35d93bb   Tom Zanussi   tracing: Pass tra...
4267
  			field_contents = key_field->fn(key_field, elt, rbe, rec);
6a475cb17   Tom Zanussi   tracing: Remove r...
4268
  			if (key_field->flags & HIST_FIELD_FL_STRING) {
69a0200c2   Tom Zanussi   tracing: Add hist...
4269
  				key = (void *)(unsigned long)field_contents;
6a475cb17   Tom Zanussi   tracing: Remove r...
4270
4271
  				use_compound_key = true;
  			} else
69a0200c2   Tom Zanussi   tracing: Add hist...
4272
  				key = (void *)&field_contents;
76a3b0c8a   Tom Zanussi   tracing: Add hist...
4273
  		}
6a475cb17   Tom Zanussi   tracing: Remove r...
4274
4275
4276
  
  		if (use_compound_key)
  			add_to_key(compound_key, key, key_field, rec);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4277
  	}
6a475cb17   Tom Zanussi   tracing: Remove r...
4278
  	if (use_compound_key)
76a3b0c8a   Tom Zanussi   tracing: Add hist...
4279
  		key = compound_key;
067fe038e   Tom Zanussi   tracing: Add vari...
4280
4281
4282
  	if (hist_data->n_var_refs &&
  	    !resolve_var_refs(hist_data, key, var_ref_vals, false))
  		return;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4283
  	elt = tracing_map_insert(hist_data->map, key);
067fe038e   Tom Zanussi   tracing: Add vari...
4284
4285
4286
4287
  	if (!elt)
  		return;
  
  	hist_trigger_elt_update(hist_data, elt, rec, rbe, var_ref_vals);
0212e2aa3   Tom Zanussi   tracing: Add hist...
4288
4289
  
  	if (resolve_var_refs(hist_data, key, var_ref_vals, true))
7d18a10c3   Tom Zanussi   tracing: Refactor...
4290
  		hist_trigger_actions(hist_data, elt, rec, rbe, key, var_ref_vals);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4291
  }
69a0200c2   Tom Zanussi   tracing: Add hist...
4292
4293
4294
4295
4296
4297
4298
4299
4300
  static void hist_trigger_stacktrace_print(struct seq_file *m,
  					  unsigned long *stacktrace_entries,
  					  unsigned int max_entries)
  {
  	char str[KSYM_SYMBOL_LEN];
  	unsigned int spaces = 8;
  	unsigned int i;
  
  	for (i = 0; i < max_entries; i++) {
4285f2fce   Thomas Gleixner   tracing: Remove t...
4301
  		if (!stacktrace_entries[i])
69a0200c2   Tom Zanussi   tracing: Add hist...
4302
4303
4304
4305
4306
4307
4308
4309
  			return;
  
  		seq_printf(m, "%*c", 1 + spaces, ' ');
  		sprint_symbol(str, stacktrace_entries[i]);
  		seq_printf(m, "%s
  ", str);
  	}
  }
a3785b7ec   Tom Zanussi   tracing: Add hist...
4310
4311
4312
4313
  static void hist_trigger_print_key(struct seq_file *m,
  				   struct hist_trigger_data *hist_data,
  				   void *key,
  				   struct tracing_map_elt *elt)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4314
4315
  {
  	struct hist_field *key_field;
c6afad49d   Tom Zanussi   tracing: Add hist...
4316
  	char str[KSYM_SYMBOL_LEN];
69a0200c2   Tom Zanussi   tracing: Add hist...
4317
  	bool multiline = false;
85013256c   Tom Zanussi   tracing: Add hist...
4318
  	const char *field_name;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4319
4320
4321
4322
4323
4324
4325
4326
4327
4328
  	unsigned int i;
  	u64 uval;
  
  	seq_puts(m, "{ ");
  
  	for_each_hist_key_field(i, hist_data) {
  		key_field = hist_data->fields[i];
  
  		if (i > hist_data->n_vals)
  			seq_puts(m, ", ");
85013256c   Tom Zanussi   tracing: Add hist...
4329
  		field_name = hist_field_name(key_field, 0);
0c4a6b466   Tom Zanussi   tracing: Add hist...
4330
4331
  		if (key_field->flags & HIST_FIELD_FL_HEX) {
  			uval = *(u64 *)(key + key_field->offset);
85013256c   Tom Zanussi   tracing: Add hist...
4332
  			seq_printf(m, "%s: %llx", field_name, uval);
c6afad49d   Tom Zanussi   tracing: Add hist...
4333
4334
4335
  		} else if (key_field->flags & HIST_FIELD_FL_SYM) {
  			uval = *(u64 *)(key + key_field->offset);
  			sprint_symbol_no_offset(str, uval);
85013256c   Tom Zanussi   tracing: Add hist...
4336
4337
  			seq_printf(m, "%s: [%llx] %-45s", field_name,
  				   uval, str);
c6afad49d   Tom Zanussi   tracing: Add hist...
4338
4339
4340
  		} else if (key_field->flags & HIST_FIELD_FL_SYM_OFFSET) {
  			uval = *(u64 *)(key + key_field->offset);
  			sprint_symbol(str, uval);
85013256c   Tom Zanussi   tracing: Add hist...
4341
4342
  			seq_printf(m, "%s: [%llx] %-55s", field_name,
  				   uval, str);
6b4827ad0   Tom Zanussi   tracing: Add hist...
4343
  		} else if (key_field->flags & HIST_FIELD_FL_EXECNAME) {
af6a29bca   Tom Zanussi   tracing: Generali...
4344
4345
4346
4347
4348
4349
4350
  			struct hist_elt_data *elt_data = elt->private_data;
  			char *comm;
  
  			if (WARN_ON_ONCE(!elt_data))
  				return;
  
  			comm = elt_data->comm;
6b4827ad0   Tom Zanussi   tracing: Add hist...
4351
4352
  
  			uval = *(u64 *)(key + key_field->offset);
85013256c   Tom Zanussi   tracing: Add hist...
4353
4354
  			seq_printf(m, "%s: %-16s[%10llu]", field_name,
  				   comm, uval);
316961988   Tom Zanussi   tracing: Add hist...
4355
4356
4357
4358
4359
4360
4361
  		} else if (key_field->flags & HIST_FIELD_FL_SYSCALL) {
  			const char *syscall_name;
  
  			uval = *(u64 *)(key + key_field->offset);
  			syscall_name = get_syscall_name(uval);
  			if (!syscall_name)
  				syscall_name = "unknown_syscall";
85013256c   Tom Zanussi   tracing: Add hist...
4362
4363
  			seq_printf(m, "%s: %-30s[%3llu]", field_name,
  				   syscall_name, uval);
69a0200c2   Tom Zanussi   tracing: Add hist...
4364
4365
4366
4367
4368
4369
4370
  		} else if (key_field->flags & HIST_FIELD_FL_STACKTRACE) {
  			seq_puts(m, "stacktrace:
  ");
  			hist_trigger_stacktrace_print(m,
  						      key + key_field->offset,
  						      HIST_STACKTRACE_DEPTH);
  			multiline = true;
4b94f5b7b   Namhyung Kim   tracing: Add hist...
4371
  		} else if (key_field->flags & HIST_FIELD_FL_LOG2) {
85013256c   Tom Zanussi   tracing: Add hist...
4372
  			seq_printf(m, "%s: ~ 2^%-2llu", field_name,
4b94f5b7b   Namhyung Kim   tracing: Add hist...
4373
  				   *(u64 *)(key + key_field->offset));
0c4a6b466   Tom Zanussi   tracing: Add hist...
4374
  		} else if (key_field->flags & HIST_FIELD_FL_STRING) {
85013256c   Tom Zanussi   tracing: Add hist...
4375
  			seq_printf(m, "%s: %-50s", field_name,
76a3b0c8a   Tom Zanussi   tracing: Add hist...
4376
  				   (char *)(key + key_field->offset));
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4377
  		} else {
76a3b0c8a   Tom Zanussi   tracing: Add hist...
4378
  			uval = *(u64 *)(key + key_field->offset);
85013256c   Tom Zanussi   tracing: Add hist...
4379
  			seq_printf(m, "%s: %10llu", field_name, uval);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4380
4381
  		}
  	}
69a0200c2   Tom Zanussi   tracing: Add hist...
4382
4383
4384
4385
  	if (!multiline)
  		seq_puts(m, " ");
  
  	seq_puts(m, "}");
a3785b7ec   Tom Zanussi   tracing: Add hist...
4386
4387
4388
4389
4390
4391
4392
4393
4394
4395
4396
  }
  
  static void hist_trigger_entry_print(struct seq_file *m,
  				     struct hist_trigger_data *hist_data,
  				     void *key,
  				     struct tracing_map_elt *elt)
  {
  	const char *field_name;
  	unsigned int i;
  
  	hist_trigger_print_key(m, hist_data, key, elt);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4397
4398
4399
  
  	seq_printf(m, " hitcount: %10llu",
  		   tracing_map_read_sum(elt, HITCOUNT_IDX));
f2606835d   Tom Zanussi   tracing: Add hist...
4400
  	for (i = 1; i < hist_data->n_vals; i++) {
85013256c   Tom Zanussi   tracing: Add hist...
4401
  		field_name = hist_field_name(hist_data->fields[i], 0);
100719dce   Tom Zanussi   tracing: Add simp...
4402
4403
  		if (hist_data->fields[i]->flags & HIST_FIELD_FL_VAR ||
  		    hist_data->fields[i]->flags & HIST_FIELD_FL_EXPR)
30350d65a   Tom Zanussi   tracing: Add vari...
4404
  			continue;
0c4a6b466   Tom Zanussi   tracing: Add hist...
4405
  		if (hist_data->fields[i]->flags & HIST_FIELD_FL_HEX) {
85013256c   Tom Zanussi   tracing: Add hist...
4406
  			seq_printf(m, "  %s: %10llx", field_name,
0c4a6b466   Tom Zanussi   tracing: Add hist...
4407
4408
  				   tracing_map_read_sum(elt, i));
  		} else {
85013256c   Tom Zanussi   tracing: Add hist...
4409
  			seq_printf(m, "  %s: %10llu", field_name,
0c4a6b466   Tom Zanussi   tracing: Add hist...
4410
4411
  				   tracing_map_read_sum(elt, i));
  		}
f2606835d   Tom Zanussi   tracing: Add hist...
4412
  	}
50450603e   Tom Zanussi   tracing: Add 'onm...
4413
  	print_actions(m, hist_data, elt);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4414
4415
4416
4417
4418
4419
4420
4421
4422
  	seq_puts(m, "
  ");
  }
  
  static int print_entries(struct seq_file *m,
  			 struct hist_trigger_data *hist_data)
  {
  	struct tracing_map_sort_entry **sort_entries = NULL;
  	struct tracing_map *map = hist_data->map;
d50c744ec   Steven Rostedt (Red Hat)   tracing: Fix unsi...
4423
  	int i, n_entries;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4424
4425
4426
4427
4428
4429
4430
4431
4432
4433
4434
4435
4436
4437
4438
4439
  
  	n_entries = tracing_map_sort_entries(map, hist_data->sort_keys,
  					     hist_data->n_sort_keys,
  					     &sort_entries);
  	if (n_entries < 0)
  		return n_entries;
  
  	for (i = 0; i < n_entries; i++)
  		hist_trigger_entry_print(m, hist_data,
  					 sort_entries[i]->key,
  					 sort_entries[i]->elt);
  
  	tracing_map_destroy_sort_entries(sort_entries, n_entries);
  
  	return n_entries;
  }
52a7f16de   Tom Zanussi   tracing: Add supp...
4440
4441
  static void hist_trigger_show(struct seq_file *m,
  			      struct event_trigger_data *data, int n)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4442
  {
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4443
  	struct hist_trigger_data *hist_data;
6e7a23981   Colin Ian King   tracing: Remove r...
4444
  	int n_entries;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4445

52a7f16de   Tom Zanussi   tracing: Add supp...
4446
4447
4448
4449
  	if (n > 0)
  		seq_puts(m, "
  
  ");
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4450
4451
4452
4453
4454
  
  	seq_puts(m, "# event histogram
  #
  # trigger info: ");
  	data->ops->print(m, data->ops, data);
52a7f16de   Tom Zanussi   tracing: Add supp...
4455
4456
4457
  	seq_puts(m, "#
  
  ");
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4458
4459
4460
  
  	hist_data = data->private_data;
  	n_entries = print_entries(m, hist_data);
6e7a23981   Colin Ian King   tracing: Remove r...
4461
  	if (n_entries < 0)
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4462
  		n_entries = 0;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4463

a3785b7ec   Tom Zanussi   tracing: Add hist...
4464
  	track_data_snapshot_print(m, hist_data);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4465
4466
4467
4468
4469
4470
4471
4472
  	seq_printf(m, "
  Totals:
      Hits: %llu
      Entries: %u
      Dropped: %llu
  ",
  		   (u64)atomic64_read(&hist_data->map->hits),
  		   n_entries, (u64)atomic64_read(&hist_data->map->drops));
52a7f16de   Tom Zanussi   tracing: Add supp...
4473
4474
4475
4476
4477
4478
4479
4480
4481
4482
4483
4484
4485
4486
4487
  }
  
  static int hist_show(struct seq_file *m, void *v)
  {
  	struct event_trigger_data *data;
  	struct trace_event_file *event_file;
  	int n = 0, ret = 0;
  
  	mutex_lock(&event_mutex);
  
  	event_file = event_file_data(m->private);
  	if (unlikely(!event_file)) {
  		ret = -ENODEV;
  		goto out_unlock;
  	}
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
4488
  	list_for_each_entry(data, &event_file->triggers, list) {
52a7f16de   Tom Zanussi   tracing: Add supp...
4489
4490
4491
  		if (data->cmd_ops->trigger_type == ETT_EVENT_HIST)
  			hist_trigger_show(m, data, n++);
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4492
4493
4494
4495
4496
4497
4498
4499
   out_unlock:
  	mutex_unlock(&event_mutex);
  
  	return ret;
  }
  
  static int event_hist_open(struct inode *inode, struct file *file)
  {
17911ff38   Steven Rostedt (VMware)   tracing: Add lock...
4500
4501
4502
4503
4504
  	int ret;
  
  	ret = security_locked_down(LOCKDOWN_TRACEFS);
  	if (ret)
  		return ret;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4505
4506
4507
4508
4509
4510
4511
4512
4513
  	return single_open(file, hist_show, file);
  }
  
  const struct file_operations event_hist_fops = {
  	.open = event_hist_open,
  	.read = seq_read,
  	.llseek = seq_lseek,
  	.release = single_release,
  };
2d19bd79a   Tom Zanussi   tracing: Add hist...
4514
4515
4516
4517
4518
4519
4520
4521
4522
4523
4524
4525
4526
4527
4528
4529
4530
4531
4532
4533
4534
4535
4536
4537
4538
4539
4540
4541
4542
4543
4544
4545
4546
4547
4548
4549
4550
4551
4552
4553
4554
4555
4556
4557
4558
4559
4560
4561
4562
4563
4564
4565
4566
4567
4568
4569
4570
4571
4572
4573
4574
4575
4576
4577
4578
4579
4580
4581
4582
4583
4584
4585
4586
4587
4588
4589
4590
4591
4592
4593
4594
4595
4596
4597
4598
4599
4600
4601
4602
4603
4604
4605
4606
4607
4608
4609
4610
4611
4612
4613
4614
4615
4616
4617
4618
4619
4620
4621
4622
4623
4624
4625
4626
4627
4628
4629
4630
4631
4632
4633
4634
4635
4636
4637
4638
4639
4640
4641
4642
4643
4644
4645
4646
4647
4648
4649
4650
4651
4652
4653
4654
4655
4656
4657
4658
4659
4660
4661
4662
4663
4664
4665
4666
4667
4668
4669
4670
4671
4672
4673
4674
4675
4676
4677
4678
4679
4680
4681
4682
4683
4684
4685
4686
4687
4688
4689
4690
4691
4692
4693
4694
4695
4696
4697
4698
4699
4700
4701
4702
4703
4704
4705
4706
4707
4708
4709
4710
4711
4712
4713
4714
4715
4716
4717
4718
4719
4720
4721
4722
4723
4724
4725
4726
4727
4728
4729
4730
4731
4732
4733
4734
4735
4736
4737
4738
4739
4740
4741
4742
4743
4744
4745
4746
4747
4748
4749
4750
4751
4752
4753
4754
4755
4756
4757
4758
4759
4760
4761
4762
4763
4764
4765
4766
4767
4768
4769
4770
4771
4772
4773
4774
4775
4776
4777
4778
4779
4780
4781
4782
4783
4784
4785
4786
4787
4788
4789
4790
4791
4792
4793
4794
4795
4796
4797
4798
4799
4800
4801
4802
4803
4804
4805
4806
4807
4808
4809
4810
4811
4812
4813
4814
4815
4816
4817
4818
4819
4820
4821
4822
4823
4824
4825
4826
4827
4828
4829
4830
4831
4832
4833
4834
4835
4836
4837
4838
4839
4840
4841
4842
4843
4844
4845
4846
4847
4848
4849
4850
4851
4852
4853
4854
4855
4856
  #ifdef CONFIG_HIST_TRIGGERS_DEBUG
  static void hist_field_debug_show_flags(struct seq_file *m,
  					unsigned long flags)
  {
  	seq_puts(m, "      flags:
  ");
  
  	if (flags & HIST_FIELD_FL_KEY)
  		seq_puts(m, "        HIST_FIELD_FL_KEY
  ");
  	else if (flags & HIST_FIELD_FL_HITCOUNT)
  		seq_puts(m, "        VAL: HIST_FIELD_FL_HITCOUNT
  ");
  	else if (flags & HIST_FIELD_FL_VAR)
  		seq_puts(m, "        HIST_FIELD_FL_VAR
  ");
  	else if (flags & HIST_FIELD_FL_VAR_REF)
  		seq_puts(m, "        HIST_FIELD_FL_VAR_REF
  ");
  	else
  		seq_puts(m, "        VAL: normal u64 value
  ");
  
  	if (flags & HIST_FIELD_FL_ALIAS)
  		seq_puts(m, "        HIST_FIELD_FL_ALIAS
  ");
  }
  
  static int hist_field_debug_show(struct seq_file *m,
  				 struct hist_field *field, unsigned long flags)
  {
  	if ((field->flags & flags) != flags) {
  		seq_printf(m, "ERROR: bad flags - %lx
  ", flags);
  		return -EINVAL;
  	}
  
  	hist_field_debug_show_flags(m, field->flags);
  	if (field->field)
  		seq_printf(m, "      ftrace_event_field name: %s
  ",
  			   field->field->name);
  
  	if (field->flags & HIST_FIELD_FL_VAR) {
  		seq_printf(m, "      var.name: %s
  ", field->var.name);
  		seq_printf(m, "      var.idx (into tracing_map_elt.vars[]): %u
  ",
  			   field->var.idx);
  	}
  
  	if (field->flags & HIST_FIELD_FL_ALIAS)
  		seq_printf(m, "      var_ref_idx (into hist_data->var_refs[]): %u
  ",
  			   field->var_ref_idx);
  
  	if (field->flags & HIST_FIELD_FL_VAR_REF) {
  		seq_printf(m, "      name: %s
  ", field->name);
  		seq_printf(m, "      var.idx (into tracing_map_elt.vars[]): %u
  ",
  			   field->var.idx);
  		seq_printf(m, "      var.hist_data: %p
  ", field->var.hist_data);
  		seq_printf(m, "      var_ref_idx (into hist_data->var_refs[]): %u
  ",
  			   field->var_ref_idx);
  		if (field->system)
  			seq_printf(m, "      system: %s
  ", field->system);
  		if (field->event_name)
  			seq_printf(m, "      event_name: %s
  ", field->event_name);
  	}
  
  	seq_printf(m, "      type: %s
  ", field->type);
  	seq_printf(m, "      size: %u
  ", field->size);
  	seq_printf(m, "      is_signed: %u
  ", field->is_signed);
  
  	return 0;
  }
  
  static int field_var_debug_show(struct seq_file *m,
  				struct field_var *field_var, unsigned int i,
  				bool save_vars)
  {
  	const char *vars_name = save_vars ? "save_vars" : "field_vars";
  	struct hist_field *field;
  	int ret = 0;
  
  	seq_printf(m, "
      hist_data->%s[%d]:
  ", vars_name, i);
  
  	field = field_var->var;
  
  	seq_printf(m, "
        %s[%d].var:
  ", vars_name, i);
  
  	hist_field_debug_show_flags(m, field->flags);
  	seq_printf(m, "      var.name: %s
  ", field->var.name);
  	seq_printf(m, "      var.idx (into tracing_map_elt.vars[]): %u
  ",
  		   field->var.idx);
  
  	field = field_var->val;
  
  	seq_printf(m, "
        %s[%d].val:
  ", vars_name, i);
  	if (field->field)
  		seq_printf(m, "      ftrace_event_field name: %s
  ",
  			   field->field->name);
  	else {
  		ret = -EINVAL;
  		goto out;
  	}
  
  	seq_printf(m, "      type: %s
  ", field->type);
  	seq_printf(m, "      size: %u
  ", field->size);
  	seq_printf(m, "      is_signed: %u
  ", field->is_signed);
  out:
  	return ret;
  }
  
  static int hist_action_debug_show(struct seq_file *m,
  				  struct action_data *data, int i)
  {
  	int ret = 0;
  
  	if (data->handler == HANDLER_ONMAX ||
  	    data->handler == HANDLER_ONCHANGE) {
  		seq_printf(m, "
      hist_data->actions[%d].track_data.var_ref:
  ", i);
  		ret = hist_field_debug_show(m, data->track_data.var_ref,
  					    HIST_FIELD_FL_VAR_REF);
  		if (ret)
  			goto out;
  
  		seq_printf(m, "
      hist_data->actions[%d].track_data.track_var:
  ", i);
  		ret = hist_field_debug_show(m, data->track_data.track_var,
  					    HIST_FIELD_FL_VAR);
  		if (ret)
  			goto out;
  	}
  
  	if (data->handler == HANDLER_ONMATCH) {
  		seq_printf(m, "
      hist_data->actions[%d].match_data.event_system: %s
  ",
  			   i, data->match_data.event_system);
  		seq_printf(m, "    hist_data->actions[%d].match_data.event: %s
  ",
  			   i, data->match_data.event);
  	}
  out:
  	return ret;
  }
  
  static int hist_actions_debug_show(struct seq_file *m,
  				   struct hist_trigger_data *hist_data)
  {
  	int i, ret = 0;
  
  	if (hist_data->n_actions)
  		seq_puts(m, "
    action tracking variables (for onmax()/onchange()/onmatch()):
  ");
  
  	for (i = 0; i < hist_data->n_actions; i++) {
  		struct action_data *action = hist_data->actions[i];
  
  		ret = hist_action_debug_show(m, action, i);
  		if (ret)
  			goto out;
  	}
  
  	if (hist_data->n_save_vars)
  		seq_puts(m, "
    save action variables (save() params):
  ");
  
  	for (i = 0; i < hist_data->n_save_vars; i++) {
  		ret = field_var_debug_show(m, hist_data->save_vars[i], i, true);
  		if (ret)
  			goto out;
  	}
  out:
  	return ret;
  }
  
  static void hist_trigger_debug_show(struct seq_file *m,
  				    struct event_trigger_data *data, int n)
  {
  	struct hist_trigger_data *hist_data;
  	int i, ret;
  
  	if (n > 0)
  		seq_puts(m, "
  
  ");
  
  	seq_puts(m, "# event histogram
  #
  # trigger info: ");
  	data->ops->print(m, data->ops, data);
  	seq_puts(m, "#
  
  ");
  
  	hist_data = data->private_data;
  
  	seq_printf(m, "hist_data: %p
  
  ", hist_data);
  	seq_printf(m, "  n_vals: %u
  ", hist_data->n_vals);
  	seq_printf(m, "  n_keys: %u
  ", hist_data->n_keys);
  	seq_printf(m, "  n_fields: %u
  ", hist_data->n_fields);
  
  	seq_puts(m, "
    val fields:
  
  ");
  
  	seq_puts(m, "    hist_data->fields[0]:
  ");
  	ret = hist_field_debug_show(m, hist_data->fields[0],
  				    HIST_FIELD_FL_HITCOUNT);
  	if (ret)
  		return;
  
  	for (i = 1; i < hist_data->n_vals; i++) {
  		seq_printf(m, "
      hist_data->fields[%d]:
  ", i);
  		ret = hist_field_debug_show(m, hist_data->fields[i], 0);
  		if (ret)
  			return;
  	}
  
  	seq_puts(m, "
    key fields:
  ");
  
  	for (i = hist_data->n_vals; i < hist_data->n_fields; i++) {
  		seq_printf(m, "
      hist_data->fields[%d]:
  ", i);
  		ret = hist_field_debug_show(m, hist_data->fields[i],
  					    HIST_FIELD_FL_KEY);
  		if (ret)
  			return;
  	}
  
  	if (hist_data->n_var_refs)
  		seq_puts(m, "
    variable reference fields:
  ");
  
  	for (i = 0; i < hist_data->n_var_refs; i++) {
  		seq_printf(m, "
      hist_data->var_refs[%d]:
  ", i);
  		ret = hist_field_debug_show(m, hist_data->var_refs[i],
  					    HIST_FIELD_FL_VAR_REF);
  		if (ret)
  			return;
  	}
  
  	if (hist_data->n_field_vars)
  		seq_puts(m, "
    field variables:
  ");
  
  	for (i = 0; i < hist_data->n_field_vars; i++) {
  		ret = field_var_debug_show(m, hist_data->field_vars[i], i, false);
  		if (ret)
  			return;
  	}
  
  	ret = hist_actions_debug_show(m, hist_data);
  	if (ret)
  		return;
  }
  
  static int hist_debug_show(struct seq_file *m, void *v)
  {
  	struct event_trigger_data *data;
  	struct trace_event_file *event_file;
  	int n = 0, ret = 0;
  
  	mutex_lock(&event_mutex);
  
  	event_file = event_file_data(m->private);
  	if (unlikely(!event_file)) {
  		ret = -ENODEV;
  		goto out_unlock;
  	}
  
  	list_for_each_entry(data, &event_file->triggers, list) {
  		if (data->cmd_ops->trigger_type == ETT_EVENT_HIST)
  			hist_trigger_debug_show(m, data, n++);
  	}
  
   out_unlock:
  	mutex_unlock(&event_mutex);
  
  	return ret;
  }
  
  static int event_hist_debug_open(struct inode *inode, struct file *file)
  {
  	int ret;
  
  	ret = security_locked_down(LOCKDOWN_TRACEFS);
  	if (ret)
  		return ret;
  
  	return single_open(file, hist_debug_show, file);
  }
  
  const struct file_operations event_hist_debug_fops = {
  	.open = event_hist_debug_open,
  	.read = seq_read,
  	.llseek = seq_lseek,
  	.release = single_release,
  };
  #endif
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4857
4858
  static void hist_field_print(struct seq_file *m, struct hist_field *hist_field)
  {
85013256c   Tom Zanussi   tracing: Add hist...
4859
  	const char *field_name = hist_field_name(hist_field, 0);
30350d65a   Tom Zanussi   tracing: Add vari...
4860
4861
  	if (hist_field->var.name)
  		seq_printf(m, "%s=", hist_field->var.name);
0ae7961e7   Tom Zanussi   tracing: Fix disp...
4862
  	if (hist_field->flags & HIST_FIELD_FL_CPU)
8b7622bf9   Tom Zanussi   tracing: Add cpu ...
4863
  		seq_puts(m, "cpu");
067fe038e   Tom Zanussi   tracing: Add vari...
4864
  	else if (field_name) {
7e8b88a30   Tom Zanussi   tracing: Add hist...
4865
4866
  		if (hist_field->flags & HIST_FIELD_FL_VAR_REF ||
  		    hist_field->flags & HIST_FIELD_FL_ALIAS)
067fe038e   Tom Zanussi   tracing: Add vari...
4867
  			seq_putc(m, '$');
ad42febe5   Tom Zanussi   tracing: Add hist...
4868
  		seq_printf(m, "%s", field_name);
0ae7961e7   Tom Zanussi   tracing: Fix disp...
4869
4870
  	} else if (hist_field->flags & HIST_FIELD_FL_TIMESTAMP)
  		seq_puts(m, "common_timestamp");
608940dab   Tom Zanussi   tracing: Restore ...
4871
4872
4873
4874
4875
4876
4877
4878
4879
4880
  
  	if (hist_field->flags) {
  		if (!(hist_field->flags & HIST_FIELD_FL_VAR_REF) &&
  		    !(hist_field->flags & HIST_FIELD_FL_EXPR)) {
  			const char *flags = get_hist_field_flags(hist_field);
  
  			if (flags)
  				seq_printf(m, ".%s", flags);
  		}
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4881
4882
4883
4884
4885
4886
4887
  }
  
  static int event_hist_trigger_print(struct seq_file *m,
  				    struct event_trigger_ops *ops,
  				    struct event_trigger_data *data)
  {
  	struct hist_trigger_data *hist_data = data->private_data;
30350d65a   Tom Zanussi   tracing: Add vari...
4888
4889
  	struct hist_field *field;
  	bool have_var = false;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4890
  	unsigned int i;
5463bfda3   Tom Zanussi   tracing: Add supp...
4891
4892
4893
4894
4895
4896
  	seq_puts(m, "hist:");
  
  	if (data->name)
  		seq_printf(m, "%s:", data->name);
  
  	seq_puts(m, "keys=");
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4897
4898
  
  	for_each_hist_key_field(i, hist_data) {
30350d65a   Tom Zanussi   tracing: Add vari...
4899
  		field = hist_data->fields[i];
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4900
4901
4902
  
  		if (i > hist_data->n_vals)
  			seq_puts(m, ",");
30350d65a   Tom Zanussi   tracing: Add vari...
4903
  		if (field->flags & HIST_FIELD_FL_STACKTRACE)
69a0200c2   Tom Zanussi   tracing: Add hist...
4904
4905
  			seq_puts(m, "stacktrace");
  		else
30350d65a   Tom Zanussi   tracing: Add vari...
4906
  			hist_field_print(m, field);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4907
4908
4909
  	}
  
  	seq_puts(m, ":vals=");
f2606835d   Tom Zanussi   tracing: Add hist...
4910
4911
  
  	for_each_hist_val_field(i, hist_data) {
30350d65a   Tom Zanussi   tracing: Add vari...
4912
4913
4914
4915
4916
  		field = hist_data->fields[i];
  		if (field->flags & HIST_FIELD_FL_VAR) {
  			have_var = true;
  			continue;
  		}
f2606835d   Tom Zanussi   tracing: Add hist...
4917
4918
4919
4920
  		if (i == HITCOUNT_IDX)
  			seq_puts(m, "hitcount");
  		else {
  			seq_puts(m, ",");
30350d65a   Tom Zanussi   tracing: Add vari...
4921
4922
4923
4924
4925
4926
4927
4928
4929
4930
4931
4932
4933
4934
4935
4936
4937
  			hist_field_print(m, field);
  		}
  	}
  
  	if (have_var) {
  		unsigned int n = 0;
  
  		seq_puts(m, ":");
  
  		for_each_hist_val_field(i, hist_data) {
  			field = hist_data->fields[i];
  
  			if (field->flags & HIST_FIELD_FL_VAR) {
  				if (n++)
  					seq_puts(m, ",");
  				hist_field_print(m, field);
  			}
f2606835d   Tom Zanussi   tracing: Add hist...
4938
4939
  		}
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4940
4941
  
  	seq_puts(m, ":sort=");
e62347d24   Tom Zanussi   tracing: Add hist...
4942
4943
4944
  
  	for (i = 0; i < hist_data->n_sort_keys; i++) {
  		struct tracing_map_sort_key *sort_key;
30350d65a   Tom Zanussi   tracing: Add vari...
4945
4946
4947
4948
  		unsigned int idx, first_key_idx;
  
  		/* skip VAR vals */
  		first_key_idx = hist_data->n_vals - hist_data->n_vars;
e62347d24   Tom Zanussi   tracing: Add hist...
4949
4950
  
  		sort_key = &hist_data->sort_keys[i];
ad42febe5   Tom Zanussi   tracing: Add hist...
4951
  		idx = sort_key->field_idx;
1a361dfcf   Tom Zanussi   tracing: Account ...
4952
  		if (WARN_ON(idx >= HIST_FIELDS_MAX))
ad42febe5   Tom Zanussi   tracing: Add hist...
4953
  			return -EINVAL;
e62347d24   Tom Zanussi   tracing: Add hist...
4954
4955
4956
  
  		if (i > 0)
  			seq_puts(m, ",");
ad42febe5   Tom Zanussi   tracing: Add hist...
4957
  		if (idx == HITCOUNT_IDX)
e62347d24   Tom Zanussi   tracing: Add hist...
4958
  			seq_puts(m, "hitcount");
30350d65a   Tom Zanussi   tracing: Add vari...
4959
4960
4961
  		else {
  			if (idx >= first_key_idx)
  				idx += hist_data->n_vars;
e62347d24   Tom Zanussi   tracing: Add hist...
4962
  			hist_field_print(m, hist_data->fields[idx]);
30350d65a   Tom Zanussi   tracing: Add vari...
4963
  		}
e62347d24   Tom Zanussi   tracing: Add hist...
4964
4965
4966
4967
  
  		if (sort_key->descending)
  			seq_puts(m, ".descending");
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4968
  	seq_printf(m, ":size=%u", (1 << hist_data->map->map_bits));
a4072fe85   Tom Zanussi   tracing: Add a cl...
4969
4970
  	if (hist_data->enable_timestamps)
  		seq_printf(m, ":clock=%s", hist_data->attrs->clock);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4971

c282a386a   Tom Zanussi   tracing: Add 'onm...
4972
  	print_actions_spec(m, hist_data);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4973
4974
  	if (data->filter_str)
  		seq_printf(m, " if %s", data->filter_str);
83e99914c   Tom Zanussi   tracing: Add hist...
4975
4976
4977
4978
  	if (data->paused)
  		seq_puts(m, " [paused]");
  	else
  		seq_puts(m, " [active]");
7ef224d1d   Tom Zanussi   tracing: Add 'his...
4979
4980
4981
4982
4983
4984
  
  	seq_putc(m, '
  ');
  
  	return 0;
  }
5463bfda3   Tom Zanussi   tracing: Add supp...
4985
4986
4987
4988
4989
4990
4991
4992
4993
4994
4995
4996
  static int event_hist_trigger_init(struct event_trigger_ops *ops,
  				   struct event_trigger_data *data)
  {
  	struct hist_trigger_data *hist_data = data->private_data;
  
  	if (!data->ref && hist_data->attrs->name)
  		save_named_trigger(hist_data->attrs->name, data);
  
  	data->ref++;
  
  	return 0;
  }
02205a675   Tom Zanussi   tracing: Add supp...
4997
4998
4999
5000
5001
5002
5003
5004
5005
5006
5007
5008
5009
5010
  static void unregister_field_var_hists(struct hist_trigger_data *hist_data)
  {
  	struct trace_event_file *file;
  	unsigned int i;
  	char *cmd;
  	int ret;
  
  	for (i = 0; i < hist_data->n_field_var_hists; i++) {
  		file = hist_data->field_var_hists[i]->hist_data->event_file;
  		cmd = hist_data->field_var_hists[i]->cmd;
  		ret = event_hist_trigger_func(&trigger_hist_cmd, file,
  					      "!hist", "hist", cmd);
  	}
  }
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5011
5012
5013
5014
5015
5016
5017
5018
5019
5020
  static void event_hist_trigger_free(struct event_trigger_ops *ops,
  				    struct event_trigger_data *data)
  {
  	struct hist_trigger_data *hist_data = data->private_data;
  
  	if (WARN_ON_ONCE(data->ref <= 0))
  		return;
  
  	data->ref--;
  	if (!data->ref) {
5463bfda3   Tom Zanussi   tracing: Add supp...
5021
5022
  		if (data->name)
  			del_named_trigger(data);
067fe038e   Tom Zanussi   tracing: Add vari...
5023

7ef224d1d   Tom Zanussi   tracing: Add 'his...
5024
  		trigger_data_free(data);
067fe038e   Tom Zanussi   tracing: Add vari...
5025
5026
  
  		remove_hist_vars(hist_data);
02205a675   Tom Zanussi   tracing: Add supp...
5027
  		unregister_field_var_hists(hist_data);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5028
5029
5030
5031
5032
5033
5034
  		destroy_hist_data(hist_data);
  	}
  }
  
  static struct event_trigger_ops event_hist_trigger_ops = {
  	.func			= event_hist_trigger,
  	.print			= event_hist_trigger_print,
5463bfda3   Tom Zanussi   tracing: Add supp...
5035
  	.init			= event_hist_trigger_init,
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5036
5037
  	.free			= event_hist_trigger_free,
  };
5463bfda3   Tom Zanussi   tracing: Add supp...
5038
5039
5040
5041
5042
5043
5044
5045
5046
5047
5048
5049
5050
5051
5052
5053
5054
5055
5056
5057
5058
5059
5060
5061
5062
5063
5064
5065
5066
5067
5068
5069
5070
  static int event_hist_trigger_named_init(struct event_trigger_ops *ops,
  					 struct event_trigger_data *data)
  {
  	data->ref++;
  
  	save_named_trigger(data->named_data->name, data);
  
  	event_hist_trigger_init(ops, data->named_data);
  
  	return 0;
  }
  
  static void event_hist_trigger_named_free(struct event_trigger_ops *ops,
  					  struct event_trigger_data *data)
  {
  	if (WARN_ON_ONCE(data->ref <= 0))
  		return;
  
  	event_hist_trigger_free(ops, data->named_data);
  
  	data->ref--;
  	if (!data->ref) {
  		del_named_trigger(data);
  		trigger_data_free(data);
  	}
  }
  
  static struct event_trigger_ops event_hist_trigger_named_ops = {
  	.func			= event_hist_trigger,
  	.print			= event_hist_trigger_print,
  	.init			= event_hist_trigger_named_init,
  	.free			= event_hist_trigger_named_free,
  };
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5071
5072
5073
5074
5075
  static struct event_trigger_ops *event_hist_get_trigger_ops(char *cmd,
  							    char *param)
  {
  	return &event_hist_trigger_ops;
  }
e86ae9baa   Tom Zanussi   tracing: Add hist...
5076
5077
5078
  static void hist_clear(struct event_trigger_data *data)
  {
  	struct hist_trigger_data *hist_data = data->private_data;
e86ae9baa   Tom Zanussi   tracing: Add hist...
5079

5463bfda3   Tom Zanussi   tracing: Add supp...
5080
5081
  	if (data->name)
  		pause_named_trigger(data);
e86ae9baa   Tom Zanussi   tracing: Add hist...
5082

e0a568dcd   Steven Rostedt (VMware)   tracing: Fix sync...
5083
  	tracepoint_synchronize_unregister();
e86ae9baa   Tom Zanussi   tracing: Add hist...
5084
5085
  
  	tracing_map_clear(hist_data->map);
5463bfda3   Tom Zanussi   tracing: Add supp...
5086
5087
5088
5089
5090
5091
5092
5093
5094
5095
5096
5097
5098
5099
5100
5101
5102
5103
5104
5105
5106
  	if (data->name)
  		unpause_named_trigger(data);
  }
  
  static bool compatible_field(struct ftrace_event_field *field,
  			     struct ftrace_event_field *test_field)
  {
  	if (field == test_field)
  		return true;
  	if (field == NULL || test_field == NULL)
  		return false;
  	if (strcmp(field->name, test_field->name) != 0)
  		return false;
  	if (strcmp(field->type, test_field->type) != 0)
  		return false;
  	if (field->size != test_field->size)
  		return false;
  	if (field->is_signed != test_field->is_signed)
  		return false;
  
  	return true;
e86ae9baa   Tom Zanussi   tracing: Add hist...
5107
  }
52a7f16de   Tom Zanussi   tracing: Add supp...
5108
  static bool hist_trigger_match(struct event_trigger_data *data,
5463bfda3   Tom Zanussi   tracing: Add supp...
5109
5110
5111
  			       struct event_trigger_data *data_test,
  			       struct event_trigger_data *named_data,
  			       bool ignore_filter)
52a7f16de   Tom Zanussi   tracing: Add supp...
5112
5113
5114
5115
5116
  {
  	struct tracing_map_sort_key *sort_key, *sort_key_test;
  	struct hist_trigger_data *hist_data, *hist_data_test;
  	struct hist_field *key_field, *key_field_test;
  	unsigned int i;
5463bfda3   Tom Zanussi   tracing: Add supp...
5117
5118
5119
5120
5121
5122
  	if (named_data && (named_data != data_test) &&
  	    (named_data != data_test->named_data))
  		return false;
  
  	if (!named_data && is_named_trigger(data_test))
  		return false;
52a7f16de   Tom Zanussi   tracing: Add supp...
5123
5124
5125
5126
5127
5128
5129
  	hist_data = data->private_data;
  	hist_data_test = data_test->private_data;
  
  	if (hist_data->n_vals != hist_data_test->n_vals ||
  	    hist_data->n_fields != hist_data_test->n_fields ||
  	    hist_data->n_sort_keys != hist_data_test->n_sort_keys)
  		return false;
5463bfda3   Tom Zanussi   tracing: Add supp...
5130
5131
5132
5133
5134
  	if (!ignore_filter) {
  		if ((data->filter_str && !data_test->filter_str) ||
  		   (!data->filter_str && data_test->filter_str))
  			return false;
  	}
52a7f16de   Tom Zanussi   tracing: Add supp...
5135
5136
5137
5138
5139
5140
5141
  
  	for_each_hist_field(i, hist_data) {
  		key_field = hist_data->fields[i];
  		key_field_test = hist_data_test->fields[i];
  
  		if (key_field->flags != key_field_test->flags)
  			return false;
5463bfda3   Tom Zanussi   tracing: Add supp...
5142
  		if (!compatible_field(key_field->field, key_field_test->field))
52a7f16de   Tom Zanussi   tracing: Add supp...
5143
5144
5145
  			return false;
  		if (key_field->offset != key_field_test->offset)
  			return false;
ad42febe5   Tom Zanussi   tracing: Add hist...
5146
5147
5148
5149
  		if (key_field->size != key_field_test->size)
  			return false;
  		if (key_field->is_signed != key_field_test->is_signed)
  			return false;
1a361dfcf   Tom Zanussi   tracing: Account ...
5150
5151
5152
5153
5154
  		if (!!key_field->var.name != !!key_field_test->var.name)
  			return false;
  		if (key_field->var.name &&
  		    strcmp(key_field->var.name, key_field_test->var.name) != 0)
  			return false;
52a7f16de   Tom Zanussi   tracing: Add supp...
5155
5156
5157
5158
5159
5160
5161
5162
5163
5164
  	}
  
  	for (i = 0; i < hist_data->n_sort_keys; i++) {
  		sort_key = &hist_data->sort_keys[i];
  		sort_key_test = &hist_data_test->sort_keys[i];
  
  		if (sort_key->field_idx != sort_key_test->field_idx ||
  		    sort_key->descending != sort_key_test->descending)
  			return false;
  	}
5463bfda3   Tom Zanussi   tracing: Add supp...
5165
  	if (!ignore_filter && data->filter_str &&
52a7f16de   Tom Zanussi   tracing: Add supp...
5166
5167
  	    (strcmp(data->filter_str, data_test->filter_str) != 0))
  		return false;
48f794731   Tom Zanussi   tracing: Add acti...
5168
5169
  	if (!actions_match(hist_data, hist_data_test))
  		return false;
52a7f16de   Tom Zanussi   tracing: Add supp...
5170
5171
  	return true;
  }
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5172
5173
5174
5175
  static int hist_register_trigger(char *glob, struct event_trigger_ops *ops,
  				 struct event_trigger_data *data,
  				 struct trace_event_file *file)
  {
83e99914c   Tom Zanussi   tracing: Add hist...
5176
  	struct hist_trigger_data *hist_data = data->private_data;
5463bfda3   Tom Zanussi   tracing: Add supp...
5177
  	struct event_trigger_data *test, *named_data = NULL;
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
5178
  	struct trace_array *tr = file->tr;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5179
  	int ret = 0;
5463bfda3   Tom Zanussi   tracing: Add supp...
5180
5181
5182
5183
5184
  	if (hist_data->attrs->name) {
  		named_data = find_named_trigger(hist_data->attrs->name);
  		if (named_data) {
  			if (!hist_trigger_match(data, named_data, named_data,
  						true)) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
5185
  				hist_err(tr, HIST_ERR_NAMED_MISMATCH, errpos(hist_data->attrs->name));
5463bfda3   Tom Zanussi   tracing: Add supp...
5186
5187
5188
5189
5190
5191
5192
5193
  				ret = -EINVAL;
  				goto out;
  			}
  		}
  	}
  
  	if (hist_data->attrs->name && !named_data)
  		goto new;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
5194
5195
5196
  	lockdep_assert_held(&event_mutex);
  
  	list_for_each_entry(test, &file->triggers, list) {
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5197
  		if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
5463bfda3   Tom Zanussi   tracing: Add supp...
5198
  			if (!hist_trigger_match(data, test, named_data, false))
52a7f16de   Tom Zanussi   tracing: Add supp...
5199
  				continue;
83e99914c   Tom Zanussi   tracing: Add hist...
5200
5201
5202
5203
  			if (hist_data->attrs->pause)
  				test->paused = true;
  			else if (hist_data->attrs->cont)
  				test->paused = false;
e86ae9baa   Tom Zanussi   tracing: Add hist...
5204
5205
  			else if (hist_data->attrs->clear)
  				hist_clear(test);
f404da6e1   Tom Zanussi   tracing: Add 'las...
5206
  			else {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
5207
  				hist_err(tr, HIST_ERR_TRIGGER_EEXIST, 0);
83e99914c   Tom Zanussi   tracing: Add hist...
5208
  				ret = -EEXIST;
f404da6e1   Tom Zanussi   tracing: Add 'las...
5209
  			}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5210
5211
5212
  			goto out;
  		}
  	}
5463bfda3   Tom Zanussi   tracing: Add supp...
5213
   new:
e86ae9baa   Tom Zanussi   tracing: Add hist...
5214
  	if (hist_data->attrs->cont || hist_data->attrs->clear) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
5215
  		hist_err(tr, HIST_ERR_TRIGGER_ENOENT_CLEAR, 0);
83e99914c   Tom Zanussi   tracing: Add hist...
5216
5217
5218
  		ret = -ENOENT;
  		goto out;
  	}
7522c03ae   Tom Zanussi   tracing: Fix use-...
5219
5220
  	if (hist_data->attrs->pause)
  		data->paused = true;
5463bfda3   Tom Zanussi   tracing: Add supp...
5221
  	if (named_data) {
5463bfda3   Tom Zanussi   tracing: Add supp...
5222
5223
5224
5225
  		data->private_data = named_data->private_data;
  		set_named_trigger_data(data, named_data);
  		data->ops = &event_hist_trigger_named_ops;
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5226
5227
5228
5229
5230
  	if (data->ops->init) {
  		ret = data->ops->init(data->ops, data);
  		if (ret < 0)
  			goto out;
  	}
a4072fe85   Tom Zanussi   tracing: Add a cl...
5231
5232
5233
5234
5235
  	if (hist_data->enable_timestamps) {
  		char *clock = hist_data->attrs->clock;
  
  		ret = tracing_set_clock(file->tr, hist_data->attrs->clock);
  		if (ret) {
d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
5236
  			hist_err(tr, HIST_ERR_SET_CLOCK_FAIL, errpos(clock));
a4072fe85   Tom Zanussi   tracing: Add a cl...
5237
5238
  			goto out;
  		}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5239

ad42febe5   Tom Zanussi   tracing: Add hist...
5240
  		tracing_set_time_stamp_abs(file->tr, true);
a4072fe85   Tom Zanussi   tracing: Add a cl...
5241
5242
5243
5244
5245
5246
  	}
  
  	if (named_data)
  		destroy_hist_data(hist_data);
  
  	ret++;
067fe038e   Tom Zanussi   tracing: Add vari...
5247
5248
5249
5250
5251
5252
5253
5254
5255
5256
5257
5258
   out:
  	return ret;
  }
  
  static int hist_trigger_enable(struct event_trigger_data *data,
  			       struct trace_event_file *file)
  {
  	int ret = 0;
  
  	list_add_tail_rcu(&data->list, &file->triggers);
  
  	update_cond_flag(file);
ad42febe5   Tom Zanussi   tracing: Add hist...
5259

7ef224d1d   Tom Zanussi   tracing: Add 'his...
5260
5261
5262
5263
5264
  	if (trace_event_trigger_enable_disable(file, 1) < 0) {
  		list_del_rcu(&data->list);
  		update_cond_flag(file);
  		ret--;
  	}
067fe038e   Tom Zanussi   tracing: Add vari...
5265

7ef224d1d   Tom Zanussi   tracing: Add 'his...
5266
5267
  	return ret;
  }
4b147936f   Tom Zanussi   tracing: Add supp...
5268
5269
5270
5271
5272
5273
  static bool have_hist_trigger_match(struct event_trigger_data *data,
  				    struct trace_event_file *file)
  {
  	struct hist_trigger_data *hist_data = data->private_data;
  	struct event_trigger_data *test, *named_data = NULL;
  	bool match = false;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
5274
  	lockdep_assert_held(&event_mutex);
4b147936f   Tom Zanussi   tracing: Add supp...
5275
5276
  	if (hist_data->attrs->name)
  		named_data = find_named_trigger(hist_data->attrs->name);
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
5277
  	list_for_each_entry(test, &file->triggers, list) {
4b147936f   Tom Zanussi   tracing: Add supp...
5278
5279
5280
5281
5282
5283
5284
5285
5286
5287
  		if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
  			if (hist_trigger_match(data, test, named_data, false)) {
  				match = true;
  				break;
  			}
  		}
  	}
  
  	return match;
  }
067fe038e   Tom Zanussi   tracing: Add vari...
5288
5289
5290
5291
5292
  static bool hist_trigger_check_refs(struct event_trigger_data *data,
  				    struct trace_event_file *file)
  {
  	struct hist_trigger_data *hist_data = data->private_data;
  	struct event_trigger_data *test, *named_data = NULL;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
5293
  	lockdep_assert_held(&event_mutex);
067fe038e   Tom Zanussi   tracing: Add vari...
5294
5295
  	if (hist_data->attrs->name)
  		named_data = find_named_trigger(hist_data->attrs->name);
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
5296
  	list_for_each_entry(test, &file->triggers, list) {
067fe038e   Tom Zanussi   tracing: Add vari...
5297
5298
5299
5300
5301
5302
5303
5304
5305
5306
5307
5308
  		if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
  			if (!hist_trigger_match(data, test, named_data, false))
  				continue;
  			hist_data = test->private_data;
  			if (check_var_refs(hist_data))
  				return true;
  			break;
  		}
  	}
  
  	return false;
  }
52a7f16de   Tom Zanussi   tracing: Add supp...
5309
5310
5311
5312
  static void hist_unregister_trigger(char *glob, struct event_trigger_ops *ops,
  				    struct event_trigger_data *data,
  				    struct trace_event_file *file)
  {
5463bfda3   Tom Zanussi   tracing: Add supp...
5313
5314
  	struct hist_trigger_data *hist_data = data->private_data;
  	struct event_trigger_data *test, *named_data = NULL;
52a7f16de   Tom Zanussi   tracing: Add supp...
5315
  	bool unregistered = false;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
5316
  	lockdep_assert_held(&event_mutex);
5463bfda3   Tom Zanussi   tracing: Add supp...
5317
5318
  	if (hist_data->attrs->name)
  		named_data = find_named_trigger(hist_data->attrs->name);
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
5319
  	list_for_each_entry(test, &file->triggers, list) {
52a7f16de   Tom Zanussi   tracing: Add supp...
5320
  		if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
5463bfda3   Tom Zanussi   tracing: Add supp...
5321
  			if (!hist_trigger_match(data, test, named_data, false))
52a7f16de   Tom Zanussi   tracing: Add supp...
5322
5323
5324
5325
5326
5327
5328
5329
5330
5331
5332
  				continue;
  			unregistered = true;
  			list_del_rcu(&test->list);
  			trace_event_trigger_enable_disable(file, 0);
  			update_cond_flag(file);
  			break;
  		}
  	}
  
  	if (unregistered && test->ops->free)
  		test->ops->free(test->ops, test);
ad42febe5   Tom Zanussi   tracing: Add hist...
5333
5334
  
  	if (hist_data->enable_timestamps) {
30350d65a   Tom Zanussi   tracing: Add vari...
5335
  		if (!hist_data->remove || unregistered)
ad42febe5   Tom Zanussi   tracing: Add hist...
5336
5337
  			tracing_set_time_stamp_abs(file->tr, false);
  	}
52a7f16de   Tom Zanussi   tracing: Add supp...
5338
  }
067fe038e   Tom Zanussi   tracing: Add vari...
5339
5340
5341
5342
  static bool hist_file_check_refs(struct trace_event_file *file)
  {
  	struct hist_trigger_data *hist_data;
  	struct event_trigger_data *test;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
5343
5344
5345
  	lockdep_assert_held(&event_mutex);
  
  	list_for_each_entry(test, &file->triggers, list) {
067fe038e   Tom Zanussi   tracing: Add vari...
5346
5347
5348
5349
5350
5351
5352
5353
5354
  		if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
  			hist_data = test->private_data;
  			if (check_var_refs(hist_data))
  				return true;
  		}
  	}
  
  	return false;
  }
52a7f16de   Tom Zanussi   tracing: Add supp...
5355
5356
  static void hist_unreg_all(struct trace_event_file *file)
  {
47c185697   Steven Rostedt   tracing: Fix use-...
5357
  	struct event_trigger_data *test, *n;
ad42febe5   Tom Zanussi   tracing: Add hist...
5358
  	struct hist_trigger_data *hist_data;
4b147936f   Tom Zanussi   tracing: Add supp...
5359
5360
  	struct synth_event *se;
  	const char *se_name;
52a7f16de   Tom Zanussi   tracing: Add supp...
5361

0e2b81f7b   Masami Hiramatsu   tracing: Remove u...
5362
  	lockdep_assert_held(&event_mutex);
067fe038e   Tom Zanussi   tracing: Add vari...
5363
5364
  	if (hist_file_check_refs(file))
  		return;
47c185697   Steven Rostedt   tracing: Fix use-...
5365
  	list_for_each_entry_safe(test, n, &file->triggers, list) {
52a7f16de   Tom Zanussi   tracing: Add supp...
5366
  		if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
ad42febe5   Tom Zanussi   tracing: Add hist...
5367
  			hist_data = test->private_data;
52a7f16de   Tom Zanussi   tracing: Add supp...
5368
5369
  			list_del_rcu(&test->list);
  			trace_event_trigger_enable_disable(file, 0);
4b147936f   Tom Zanussi   tracing: Add supp...
5370

4b147936f   Tom Zanussi   tracing: Add supp...
5371
5372
5373
5374
  			se_name = trace_event_name(file->event_call);
  			se = find_synth_event(se_name);
  			if (se)
  				se->ref--;
4b147936f   Tom Zanussi   tracing: Add supp...
5375

52a7f16de   Tom Zanussi   tracing: Add supp...
5376
  			update_cond_flag(file);
ad42febe5   Tom Zanussi   tracing: Add hist...
5377
5378
  			if (hist_data->enable_timestamps)
  				tracing_set_time_stamp_abs(file->tr, false);
52a7f16de   Tom Zanussi   tracing: Add supp...
5379
5380
5381
5382
5383
  			if (test->ops->free)
  				test->ops->free(test->ops, test);
  		}
  	}
  }
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5384
5385
5386
5387
5388
5389
5390
5391
5392
  static int event_hist_trigger_func(struct event_command *cmd_ops,
  				   struct trace_event_file *file,
  				   char *glob, char *cmd, char *param)
  {
  	unsigned int hist_trigger_bits = TRACING_MAP_BITS_DEFAULT;
  	struct event_trigger_data *trigger_data;
  	struct hist_trigger_attrs *attrs;
  	struct event_trigger_ops *trigger_ops;
  	struct hist_trigger_data *hist_data;
4b147936f   Tom Zanussi   tracing: Add supp...
5393
5394
  	struct synth_event *se;
  	const char *se_name;
30350d65a   Tom Zanussi   tracing: Add vari...
5395
  	bool remove = false;
ec5ce0987   Tom Zanussi   tracing: Allow wh...
5396
  	char *trigger, *p;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5397
  	int ret = 0;
0e2b81f7b   Masami Hiramatsu   tracing: Remove u...
5398
  	lockdep_assert_held(&event_mutex);
f404da6e1   Tom Zanussi   tracing: Add 'las...
5399
  	if (glob && strlen(glob)) {
f404da6e1   Tom Zanussi   tracing: Add 'las...
5400
  		hist_err_clear();
a1a05bb40   Tom Zanussi   tracing: Save the...
5401
  		last_cmd_set(file, param);
f404da6e1   Tom Zanussi   tracing: Add 'las...
5402
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5403
5404
  	if (!param)
  		return -EINVAL;
30350d65a   Tom Zanussi   tracing: Add vari...
5405
5406
  	if (glob[0] == '!')
  		remove = true;
ec5ce0987   Tom Zanussi   tracing: Allow wh...
5407
5408
5409
5410
5411
5412
5413
5414
5415
5416
5417
5418
5419
5420
5421
  	/*
  	 * separate the trigger from the filter (k:v [if filter])
  	 * allowing for whitespace in the trigger
  	 */
  	p = trigger = param;
  	do {
  		p = strstr(p, "if");
  		if (!p)
  			break;
  		if (p == param)
  			return -EINVAL;
  		if (*(p - 1) != ' ' && *(p - 1) != '\t') {
  			p++;
  			continue;
  		}
2f31ed930   Tom Zanussi   tracing: Change s...
5422
  		if (p >= param + strlen(param) - (sizeof("if") - 1) - 1)
ec5ce0987   Tom Zanussi   tracing: Allow wh...
5423
  			return -EINVAL;
2f31ed930   Tom Zanussi   tracing: Change s...
5424
  		if (*(p + sizeof("if") - 1) != ' ' && *(p + sizeof("if") - 1) != '\t') {
ec5ce0987   Tom Zanussi   tracing: Allow wh...
5425
5426
5427
5428
5429
5430
5431
5432
5433
5434
5435
5436
5437
  			p++;
  			continue;
  		}
  		break;
  	} while (p);
  
  	if (!p)
  		param = NULL;
  	else {
  		*(p - 1) = '\0';
  		param = strstrip(p);
  		trigger = strstrip(trigger);
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5438

d0cd871ba   Steven Rostedt (VMware)   tracing: Have his...
5439
  	attrs = parse_hist_trigger_attrs(file->tr, trigger);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5440
5441
5442
5443
5444
  	if (IS_ERR(attrs))
  		return PTR_ERR(attrs);
  
  	if (attrs->map_bits)
  		hist_trigger_bits = attrs->map_bits;
30350d65a   Tom Zanussi   tracing: Add vari...
5445
  	hist_data = create_hist_data(hist_trigger_bits, attrs, file, remove);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5446
5447
5448
5449
5450
5451
  	if (IS_ERR(hist_data)) {
  		destroy_hist_trigger_attrs(attrs);
  		return PTR_ERR(hist_data);
  	}
  
  	trigger_ops = cmd_ops->get_trigger_ops(cmd, trigger);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5452
  	trigger_data = kzalloc(sizeof(*trigger_data), GFP_KERNEL);
4b147936f   Tom Zanussi   tracing: Add supp...
5453
5454
  	if (!trigger_data) {
  		ret = -ENOMEM;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5455
  		goto out_free;
4b147936f   Tom Zanussi   tracing: Add supp...
5456
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5457
5458
5459
5460
5461
5462
5463
5464
5465
  
  	trigger_data->count = -1;
  	trigger_data->ops = trigger_ops;
  	trigger_data->cmd_ops = cmd_ops;
  
  	INIT_LIST_HEAD(&trigger_data->list);
  	RCU_INIT_POINTER(trigger_data->filter, NULL);
  
  	trigger_data->private_data = hist_data;
52a7f16de   Tom Zanussi   tracing: Add supp...
5466
5467
5468
5469
5470
5471
  	/* if param is non-empty, it's supposed to be a filter */
  	if (param && cmd_ops->set_filter) {
  		ret = cmd_ops->set_filter(param, trigger_data, file);
  		if (ret < 0)
  			goto out_free;
  	}
30350d65a   Tom Zanussi   tracing: Add vari...
5472
  	if (remove) {
4b147936f   Tom Zanussi   tracing: Add supp...
5473
5474
  		if (!have_hist_trigger_match(trigger_data, file))
  			goto out_free;
067fe038e   Tom Zanussi   tracing: Add vari...
5475
5476
5477
5478
  		if (hist_trigger_check_refs(trigger_data, file)) {
  			ret = -EBUSY;
  			goto out_free;
  		}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5479
  		cmd_ops->unreg(glob+1, trigger_ops, trigger_data, file);
4b147936f   Tom Zanussi   tracing: Add supp...
5480
5481
5482
5483
  		se_name = trace_event_name(file->event_call);
  		se = find_synth_event(se_name);
  		if (se)
  			se->ref--;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5484
5485
5486
  		ret = 0;
  		goto out_free;
  	}
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5487
5488
5489
5490
5491
5492
5493
  	ret = cmd_ops->reg(glob, trigger_ops, trigger_data, file);
  	/*
  	 * The above returns on success the # of triggers registered,
  	 * but if it didn't register any it returns zero.  Consider no
  	 * triggers registered a failure too.
  	 */
  	if (!ret) {
e86ae9baa   Tom Zanussi   tracing: Add hist...
5494
  		if (!(attrs->pause || attrs->cont || attrs->clear))
83e99914c   Tom Zanussi   tracing: Add hist...
5495
  			ret = -ENOENT;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5496
5497
5498
  		goto out_free;
  	} else if (ret < 0)
  		goto out_free;
067fe038e   Tom Zanussi   tracing: Add vari...
5499
5500
5501
5502
5503
5504
  
  	if (get_named_trigger_data(trigger_data))
  		goto enable;
  
  	if (has_hist_vars(hist_data))
  		save_hist_vars(hist_data);
7d18a10c3   Tom Zanussi   tracing: Refactor...
5505
  	ret = create_actions(hist_data);
0212e2aa3   Tom Zanussi   tracing: Add hist...
5506
5507
  	if (ret)
  		goto out_unreg;
067fe038e   Tom Zanussi   tracing: Add vari...
5508
5509
5510
5511
5512
5513
5514
  	ret = tracing_map_init(hist_data->map);
  	if (ret)
  		goto out_unreg;
  enable:
  	ret = hist_trigger_enable(trigger_data, file);
  	if (ret)
  		goto out_unreg;
4b147936f   Tom Zanussi   tracing: Add supp...
5515
5516
5517
5518
  	se_name = trace_event_name(file->event_call);
  	se = find_synth_event(se_name);
  	if (se)
  		se->ref++;
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5519
5520
5521
  	/* Just return zero, not the number of registered triggers */
  	ret = 0;
   out:
f404da6e1   Tom Zanussi   tracing: Add 'las...
5522
5523
  	if (ret == 0)
  		hist_err_clear();
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5524
  	return ret;
067fe038e   Tom Zanussi   tracing: Add vari...
5525
5526
   out_unreg:
  	cmd_ops->unreg(glob+1, trigger_ops, trigger_data, file);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5527
5528
5529
   out_free:
  	if (cmd_ops->set_filter)
  		cmd_ops->set_filter(NULL, trigger_data, NULL);
067fe038e   Tom Zanussi   tracing: Add vari...
5530
  	remove_hist_vars(hist_data);
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5531
5532
5533
5534
5535
5536
5537
5538
5539
5540
5541
5542
  	kfree(trigger_data);
  
  	destroy_hist_data(hist_data);
  	goto out;
  }
  
  static struct event_command trigger_hist_cmd = {
  	.name			= "hist",
  	.trigger_type		= ETT_EVENT_HIST,
  	.flags			= EVENT_CMD_FL_NEEDS_REC,
  	.func			= event_hist_trigger_func,
  	.reg			= hist_register_trigger,
52a7f16de   Tom Zanussi   tracing: Add supp...
5543
5544
  	.unreg			= hist_unregister_trigger,
  	.unreg_all		= hist_unreg_all,
7ef224d1d   Tom Zanussi   tracing: Add 'his...
5545
5546
5547
5548
5549
5550
5551
5552
5553
5554
5555
5556
5557
  	.get_trigger_ops	= event_hist_get_trigger_ops,
  	.set_filter		= set_trigger_filter,
  };
  
  __init int register_trigger_hist_cmd(void)
  {
  	int ret;
  
  	ret = register_event_command(&trigger_hist_cmd);
  	WARN_ON(ret < 0);
  
  	return ret;
  }
d0bad49bb   Tom Zanussi   tracing: Add enab...
5558
5559
  
  static void
1ac4f51c0   Tom Zanussi   tracing: Give eve...
5560
5561
  hist_enable_trigger(struct event_trigger_data *data, void *rec,
  		    struct ring_buffer_event *event)
d0bad49bb   Tom Zanussi   tracing: Add enab...
5562
5563
5564
  {
  	struct enable_trigger_data *enable_data = data->private_data;
  	struct event_trigger_data *test;
aeed8aa38   Masami Hiramatsu   tracing: trigger:...
5565
5566
  	list_for_each_entry_rcu(test, &enable_data->file->triggers, list,
  				lockdep_is_held(&event_mutex)) {
d0bad49bb   Tom Zanussi   tracing: Add enab...
5567
5568
5569
5570
5571
  		if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
  			if (enable_data->enable)
  				test->paused = false;
  			else
  				test->paused = true;
d0bad49bb   Tom Zanussi   tracing: Add enab...
5572
5573
5574
5575
5576
  		}
  	}
  }
  
  static void
1ac4f51c0   Tom Zanussi   tracing: Give eve...
5577
5578
  hist_enable_count_trigger(struct event_trigger_data *data, void *rec,
  			  struct ring_buffer_event *event)
d0bad49bb   Tom Zanussi   tracing: Add enab...
5579
5580
5581
5582
5583
5584
  {
  	if (!data->count)
  		return;
  
  	if (data->count != -1)
  		(data->count)--;
1ac4f51c0   Tom Zanussi   tracing: Give eve...
5585
  	hist_enable_trigger(data, rec, event);
d0bad49bb   Tom Zanussi   tracing: Add enab...
5586
5587
5588
5589
5590
5591
5592
5593
5594
5595
5596
5597
5598
5599
5600
5601
5602
5603
5604
5605
5606
5607
5608
5609
5610
5611
5612
5613
5614
5615
5616
5617
5618
5619
5620
5621
5622
5623
5624
5625
5626
5627
5628
5629
5630
5631
5632
  }
  
  static struct event_trigger_ops hist_enable_trigger_ops = {
  	.func			= hist_enable_trigger,
  	.print			= event_enable_trigger_print,
  	.init			= event_trigger_init,
  	.free			= event_enable_trigger_free,
  };
  
  static struct event_trigger_ops hist_enable_count_trigger_ops = {
  	.func			= hist_enable_count_trigger,
  	.print			= event_enable_trigger_print,
  	.init			= event_trigger_init,
  	.free			= event_enable_trigger_free,
  };
  
  static struct event_trigger_ops hist_disable_trigger_ops = {
  	.func			= hist_enable_trigger,
  	.print			= event_enable_trigger_print,
  	.init			= event_trigger_init,
  	.free			= event_enable_trigger_free,
  };
  
  static struct event_trigger_ops hist_disable_count_trigger_ops = {
  	.func			= hist_enable_count_trigger,
  	.print			= event_enable_trigger_print,
  	.init			= event_trigger_init,
  	.free			= event_enable_trigger_free,
  };
  
  static struct event_trigger_ops *
  hist_enable_get_trigger_ops(char *cmd, char *param)
  {
  	struct event_trigger_ops *ops;
  	bool enable;
  
  	enable = (strcmp(cmd, ENABLE_HIST_STR) == 0);
  
  	if (enable)
  		ops = param ? &hist_enable_count_trigger_ops :
  			&hist_enable_trigger_ops;
  	else
  		ops = param ? &hist_disable_count_trigger_ops :
  			&hist_disable_trigger_ops;
  
  	return ops;
  }
52a7f16de   Tom Zanussi   tracing: Add supp...
5633
5634
  static void hist_enable_unreg_all(struct trace_event_file *file)
  {
47c185697   Steven Rostedt   tracing: Fix use-...
5635
  	struct event_trigger_data *test, *n;
52a7f16de   Tom Zanussi   tracing: Add supp...
5636

47c185697   Steven Rostedt   tracing: Fix use-...
5637
  	list_for_each_entry_safe(test, n, &file->triggers, list) {
52a7f16de   Tom Zanussi   tracing: Add supp...
5638
5639
5640
5641
5642
5643
5644
5645
5646
  		if (test->cmd_ops->trigger_type == ETT_HIST_ENABLE) {
  			list_del_rcu(&test->list);
  			update_cond_flag(file);
  			trace_event_trigger_enable_disable(file, 0);
  			if (test->ops->free)
  				test->ops->free(test->ops, test);
  		}
  	}
  }
d0bad49bb   Tom Zanussi   tracing: Add enab...
5647
5648
5649
5650
5651
5652
  static struct event_command trigger_hist_enable_cmd = {
  	.name			= ENABLE_HIST_STR,
  	.trigger_type		= ETT_HIST_ENABLE,
  	.func			= event_enable_trigger_func,
  	.reg			= event_enable_register_trigger,
  	.unreg			= event_enable_unregister_trigger,
52a7f16de   Tom Zanussi   tracing: Add supp...
5653
  	.unreg_all		= hist_enable_unreg_all,
d0bad49bb   Tom Zanussi   tracing: Add enab...
5654
5655
5656
5657
5658
5659
5660
5661
5662
5663
  	.get_trigger_ops	= hist_enable_get_trigger_ops,
  	.set_filter		= set_trigger_filter,
  };
  
  static struct event_command trigger_hist_disable_cmd = {
  	.name			= DISABLE_HIST_STR,
  	.trigger_type		= ETT_HIST_ENABLE,
  	.func			= event_enable_trigger_func,
  	.reg			= event_enable_register_trigger,
  	.unreg			= event_enable_unregister_trigger,
52a7f16de   Tom Zanussi   tracing: Add supp...
5664
  	.unreg_all		= hist_enable_unreg_all,
d0bad49bb   Tom Zanussi   tracing: Add enab...
5665
5666
5667
5668
5669
5670
5671
5672
5673
5674
5675
5676
5677
5678
5679
5680
5681
5682
5683
5684
5685
5686
5687
  	.get_trigger_ops	= hist_enable_get_trigger_ops,
  	.set_filter		= set_trigger_filter,
  };
  
  static __init void unregister_trigger_hist_enable_disable_cmds(void)
  {
  	unregister_event_command(&trigger_hist_enable_cmd);
  	unregister_event_command(&trigger_hist_disable_cmd);
  }
  
  __init int register_trigger_hist_enable_disable_cmds(void)
  {
  	int ret;
  
  	ret = register_event_command(&trigger_hist_enable_cmd);
  	if (WARN_ON(ret < 0))
  		return ret;
  	ret = register_event_command(&trigger_hist_disable_cmd);
  	if (WARN_ON(ret < 0))
  		unregister_trigger_hist_enable_disable_cmds();
  
  	return ret;
  }