mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	tracing: Have existing event_command.parse() implementations use helpers
Simplify the existing event_command.parse() implementations by having them make use of the helper functions previously introduced. Link: https://lkml.kernel.org/r/b353e3427a81f9d3adafd98fd7d73e78a8209f43.1644010576.git.zanussi@kernel.org Signed-off-by: Tom Zanussi <zanussi@kernel.org> Signed-off-by: Steven Rostedt (Google) <rostedt@goodmis.org>
This commit is contained in:
		
							parent
							
								
									4767054195
								
							
						
					
					
						commit
						e1f187d09e
					
				
					 4 changed files with 71 additions and 153 deletions
				
			
		| 
						 | 
					@ -1577,7 +1577,8 @@ extern int event_enable_trigger_print(struct seq_file *m,
 | 
				
			||||||
extern void event_enable_trigger_free(struct event_trigger_data *data);
 | 
					extern void event_enable_trigger_free(struct event_trigger_data *data);
 | 
				
			||||||
extern int event_enable_trigger_parse(struct event_command *cmd_ops,
 | 
					extern int event_enable_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
				      struct trace_event_file *file,
 | 
									      struct trace_event_file *file,
 | 
				
			||||||
				      char *glob, char *cmd, char *param);
 | 
									      char *glob, char *cmd,
 | 
				
			||||||
 | 
									      char *param_and_filter);
 | 
				
			||||||
extern int event_enable_register_trigger(char *glob,
 | 
					extern int event_enable_register_trigger(char *glob,
 | 
				
			||||||
					 struct event_trigger_data *data,
 | 
										 struct event_trigger_data *data,
 | 
				
			||||||
					 struct trace_event_file *file);
 | 
										 struct trace_event_file *file);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -546,7 +546,8 @@ static struct event_trigger_ops eprobe_trigger_ops = {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int eprobe_trigger_cmd_parse(struct event_command *cmd_ops,
 | 
					static int eprobe_trigger_cmd_parse(struct event_command *cmd_ops,
 | 
				
			||||||
				    struct trace_event_file *file,
 | 
									    struct trace_event_file *file,
 | 
				
			||||||
				    char *glob, char *cmd, char *param)
 | 
									    char *glob, char *cmd,
 | 
				
			||||||
 | 
									    char *param_and_filter)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	return -1;
 | 
						return -1;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2785,7 +2785,8 @@ static char *find_trigger_filter(struct hist_trigger_data *hist_data,
 | 
				
			||||||
static struct event_command trigger_hist_cmd;
 | 
					static struct event_command trigger_hist_cmd;
 | 
				
			||||||
static int event_hist_trigger_parse(struct event_command *cmd_ops,
 | 
					static int event_hist_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
				    struct trace_event_file *file,
 | 
									    struct trace_event_file *file,
 | 
				
			||||||
				    char *glob, char *cmd, char *param);
 | 
									    char *glob, char *cmd,
 | 
				
			||||||
 | 
									    char *param_and_filter);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static bool compatible_keys(struct hist_trigger_data *target_hist_data,
 | 
					static bool compatible_keys(struct hist_trigger_data *target_hist_data,
 | 
				
			||||||
			    struct hist_trigger_data *hist_data,
 | 
								    struct hist_trigger_data *hist_data,
 | 
				
			||||||
| 
						 | 
					@ -6166,17 +6167,17 @@ static void hist_unreg_all(struct trace_event_file *file)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int event_hist_trigger_parse(struct event_command *cmd_ops,
 | 
					static int event_hist_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
				    struct trace_event_file *file,
 | 
									    struct trace_event_file *file,
 | 
				
			||||||
				    char *glob, char *cmd, char *param)
 | 
									    char *glob, char *cmd,
 | 
				
			||||||
 | 
									    char *param_and_filter)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	unsigned int hist_trigger_bits = TRACING_MAP_BITS_DEFAULT;
 | 
						unsigned int hist_trigger_bits = TRACING_MAP_BITS_DEFAULT;
 | 
				
			||||||
	struct event_trigger_data *trigger_data;
 | 
						struct event_trigger_data *trigger_data;
 | 
				
			||||||
	struct hist_trigger_attrs *attrs;
 | 
						struct hist_trigger_attrs *attrs;
 | 
				
			||||||
	struct event_trigger_ops *trigger_ops;
 | 
					 | 
				
			||||||
	struct hist_trigger_data *hist_data;
 | 
						struct hist_trigger_data *hist_data;
 | 
				
			||||||
 | 
						char *param, *filter, *p, *start;
 | 
				
			||||||
	struct synth_event *se;
 | 
						struct synth_event *se;
 | 
				
			||||||
	const char *se_name;
 | 
						const char *se_name;
 | 
				
			||||||
	bool remove = false;
 | 
						bool remove;
 | 
				
			||||||
	char *trigger, *p, *start;
 | 
					 | 
				
			||||||
	int ret = 0;
 | 
						int ret = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	lockdep_assert_held(&event_mutex);
 | 
						lockdep_assert_held(&event_mutex);
 | 
				
			||||||
| 
						 | 
					@ -6185,31 +6186,30 @@ static int event_hist_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (strlen(glob)) {
 | 
						if (strlen(glob)) {
 | 
				
			||||||
		hist_err_clear();
 | 
							hist_err_clear();
 | 
				
			||||||
		last_cmd_set(file, param);
 | 
							last_cmd_set(file, param_and_filter);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (!param)
 | 
						remove = event_trigger_check_remove(glob);
 | 
				
			||||||
		return -EINVAL;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (glob[0] == '!')
 | 
						if (event_trigger_empty_param(param_and_filter))
 | 
				
			||||||
		remove = true;
 | 
							return -EINVAL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/*
 | 
						/*
 | 
				
			||||||
	 * separate the trigger from the filter (k:v [if filter])
 | 
						 * separate the trigger from the filter (k:v [if filter])
 | 
				
			||||||
	 * allowing for whitespace in the trigger
 | 
						 * allowing for whitespace in the trigger
 | 
				
			||||||
	 */
 | 
						 */
 | 
				
			||||||
	p = trigger = param;
 | 
						p = param = param_and_filter;
 | 
				
			||||||
	do {
 | 
						do {
 | 
				
			||||||
		p = strstr(p, "if");
 | 
							p = strstr(p, "if");
 | 
				
			||||||
		if (!p)
 | 
							if (!p)
 | 
				
			||||||
			break;
 | 
								break;
 | 
				
			||||||
		if (p == param)
 | 
							if (p == param_and_filter)
 | 
				
			||||||
			return -EINVAL;
 | 
								return -EINVAL;
 | 
				
			||||||
		if (*(p - 1) != ' ' && *(p - 1) != '\t') {
 | 
							if (*(p - 1) != ' ' && *(p - 1) != '\t') {
 | 
				
			||||||
			p++;
 | 
								p++;
 | 
				
			||||||
			continue;
 | 
								continue;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		if (p >= param + strlen(param) - (sizeof("if") - 1) - 1)
 | 
							if (p >= param_and_filter + strlen(param_and_filter) - (sizeof("if") - 1) - 1)
 | 
				
			||||||
			return -EINVAL;
 | 
								return -EINVAL;
 | 
				
			||||||
		if (*(p + sizeof("if") - 1) != ' ' && *(p + sizeof("if") - 1) != '\t') {
 | 
							if (*(p + sizeof("if") - 1) != ' ' && *(p + sizeof("if") - 1) != '\t') {
 | 
				
			||||||
			p++;
 | 
								p++;
 | 
				
			||||||
| 
						 | 
					@ -6219,24 +6219,24 @@ static int event_hist_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
	} while (1);
 | 
						} while (1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (!p)
 | 
						if (!p)
 | 
				
			||||||
		param = NULL;
 | 
							filter = NULL;
 | 
				
			||||||
	else {
 | 
						else {
 | 
				
			||||||
		*(p - 1) = '\0';
 | 
							*(p - 1) = '\0';
 | 
				
			||||||
		param = strstrip(p);
 | 
							filter = strstrip(p);
 | 
				
			||||||
		trigger = strstrip(trigger);
 | 
							param = strstrip(param);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/*
 | 
						/*
 | 
				
			||||||
	 * To simplify arithmetic expression parsing, replace occurrences of
 | 
						 * To simplify arithmetic expression parsing, replace occurrences of
 | 
				
			||||||
	 * '.sym-offset' modifier with '.symXoffset'
 | 
						 * '.sym-offset' modifier with '.symXoffset'
 | 
				
			||||||
	 */
 | 
						 */
 | 
				
			||||||
	start = strstr(trigger, ".sym-offset");
 | 
						start = strstr(param, ".sym-offset");
 | 
				
			||||||
	while (start) {
 | 
						while (start) {
 | 
				
			||||||
		*(start + 4) = 'X';
 | 
							*(start + 4) = 'X';
 | 
				
			||||||
		start = strstr(start + 11, ".sym-offset");
 | 
							start = strstr(start + 11, ".sym-offset");
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	attrs = parse_hist_trigger_attrs(file->tr, trigger);
 | 
						attrs = parse_hist_trigger_attrs(file->tr, param);
 | 
				
			||||||
	if (IS_ERR(attrs))
 | 
						if (IS_ERR(attrs))
 | 
				
			||||||
		return PTR_ERR(attrs);
 | 
							return PTR_ERR(attrs);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -6249,29 +6249,15 @@ static int event_hist_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
		return PTR_ERR(hist_data);
 | 
							return PTR_ERR(hist_data);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	trigger_ops = cmd_ops->get_trigger_ops(cmd, trigger);
 | 
						trigger_data = event_trigger_alloc(cmd_ops, cmd, param, hist_data);
 | 
				
			||||||
 | 
					 | 
				
			||||||
	trigger_data = kzalloc(sizeof(*trigger_data), GFP_KERNEL);
 | 
					 | 
				
			||||||
	if (!trigger_data) {
 | 
						if (!trigger_data) {
 | 
				
			||||||
		ret = -ENOMEM;
 | 
							ret = -ENOMEM;
 | 
				
			||||||
		goto out_free;
 | 
							goto out_free;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	trigger_data->count = -1;
 | 
						ret = event_trigger_set_filter(cmd_ops, file, filter, trigger_data);
 | 
				
			||||||
	trigger_data->ops = trigger_ops;
 | 
						if (ret < 0)
 | 
				
			||||||
	trigger_data->cmd_ops = cmd_ops;
 | 
							goto out_free;
 | 
				
			||||||
 | 
					 | 
				
			||||||
	INIT_LIST_HEAD(&trigger_data->list);
 | 
					 | 
				
			||||||
	RCU_INIT_POINTER(trigger_data->filter, NULL);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	trigger_data->private_data = hist_data;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	/* 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;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (remove) {
 | 
						if (remove) {
 | 
				
			||||||
		if (!have_hist_trigger_match(trigger_data, file))
 | 
							if (!have_hist_trigger_match(trigger_data, file))
 | 
				
			||||||
| 
						 | 
					@ -6298,8 +6284,7 @@ static int event_hist_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
		if (!(attrs->pause || attrs->cont || attrs->clear))
 | 
							if (!(attrs->pause || attrs->cont || attrs->clear))
 | 
				
			||||||
			ret = -ENOENT;
 | 
								ret = -ENOENT;
 | 
				
			||||||
		goto out_free;
 | 
							goto out_free;
 | 
				
			||||||
	} else if (ret < 0)
 | 
						}
 | 
				
			||||||
		goto out_free;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (get_named_trigger_data(trigger_data))
 | 
						if (get_named_trigger_data(trigger_data))
 | 
				
			||||||
		goto enable;
 | 
							goto enable;
 | 
				
			||||||
| 
						 | 
					@ -6331,8 +6316,7 @@ static int event_hist_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
 out_unreg:
 | 
					 out_unreg:
 | 
				
			||||||
	event_trigger_unregister(cmd_ops, file, glob+1, trigger_data);
 | 
						event_trigger_unregister(cmd_ops, file, glob+1, trigger_data);
 | 
				
			||||||
 out_free:
 | 
					 out_free:
 | 
				
			||||||
	if (cmd_ops->set_filter)
 | 
						event_trigger_reset_filter(cmd_ops, trigger_data);
 | 
				
			||||||
		cmd_ops->set_filter(NULL, trigger_data, NULL);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	remove_hist_vars(hist_data);
 | 
						remove_hist_vars(hist_data);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -965,7 +965,7 @@ void event_trigger_unregister(struct event_command *cmd_ops,
 | 
				
			||||||
 * @file: The trace_event_file associated with the event
 | 
					 * @file: The trace_event_file associated with the event
 | 
				
			||||||
 * @glob: The raw string used to register the trigger
 | 
					 * @glob: The raw string used to register the trigger
 | 
				
			||||||
 * @cmd: The cmd portion of the string used to register the trigger
 | 
					 * @cmd: The cmd portion of the string used to register the trigger
 | 
				
			||||||
 * @param: The params portion of the string used to register the trigger
 | 
					 * @param_and_filter: The param and filter portion of the string used to register the trigger
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * Common implementation for event command parsing and trigger
 | 
					 * Common implementation for event command parsing and trigger
 | 
				
			||||||
 * instantiation.
 | 
					 * instantiation.
 | 
				
			||||||
| 
						 | 
					@ -978,72 +978,39 @@ void event_trigger_unregister(struct event_command *cmd_ops,
 | 
				
			||||||
static int
 | 
					static int
 | 
				
			||||||
event_trigger_parse(struct event_command *cmd_ops,
 | 
					event_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
		    struct trace_event_file *file,
 | 
							    struct trace_event_file *file,
 | 
				
			||||||
		    char *glob, char *cmd, char *param)
 | 
							    char *glob, char *cmd, char *param_and_filter)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct event_trigger_data *trigger_data;
 | 
						struct event_trigger_data *trigger_data;
 | 
				
			||||||
	struct event_trigger_ops *trigger_ops;
 | 
						char *param, *filter;
 | 
				
			||||||
	char *trigger = NULL;
 | 
						bool remove;
 | 
				
			||||||
	char *number;
 | 
					 | 
				
			||||||
	int ret;
 | 
						int ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* separate the trigger from the filter (t:n [if filter]) */
 | 
						remove = event_trigger_check_remove(glob);
 | 
				
			||||||
	if (param && isdigit(param[0])) {
 | 
					 | 
				
			||||||
		trigger = strsep(¶m, " \t");
 | 
					 | 
				
			||||||
		if (param) {
 | 
					 | 
				
			||||||
			param = skip_spaces(param);
 | 
					 | 
				
			||||||
			if (!*param)
 | 
					 | 
				
			||||||
				param = NULL;
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	trigger_ops = cmd_ops->get_trigger_ops(cmd, trigger);
 | 
						ret = event_trigger_separate_filter(param_and_filter, ¶m, &filter, false);
 | 
				
			||||||
 | 
						if (ret)
 | 
				
			||||||
 | 
							return ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ret = -ENOMEM;
 | 
						ret = -ENOMEM;
 | 
				
			||||||
	trigger_data = kzalloc(sizeof(*trigger_data), GFP_KERNEL);
 | 
						trigger_data = event_trigger_alloc(cmd_ops, cmd, param, file);
 | 
				
			||||||
	if (!trigger_data)
 | 
						if (!trigger_data)
 | 
				
			||||||
		goto out;
 | 
							goto out;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	trigger_data->count = -1;
 | 
						if (remove) {
 | 
				
			||||||
	trigger_data->ops = trigger_ops;
 | 
					 | 
				
			||||||
	trigger_data->cmd_ops = cmd_ops;
 | 
					 | 
				
			||||||
	trigger_data->private_data = file;
 | 
					 | 
				
			||||||
	INIT_LIST_HEAD(&trigger_data->list);
 | 
					 | 
				
			||||||
	INIT_LIST_HEAD(&trigger_data->named_list);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (glob[0] == '!') {
 | 
					 | 
				
			||||||
		event_trigger_unregister(cmd_ops, file, glob+1, trigger_data);
 | 
							event_trigger_unregister(cmd_ops, file, glob+1, trigger_data);
 | 
				
			||||||
		kfree(trigger_data);
 | 
							kfree(trigger_data);
 | 
				
			||||||
		ret = 0;
 | 
							ret = 0;
 | 
				
			||||||
		goto out;
 | 
							goto out;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (trigger) {
 | 
						ret = event_trigger_parse_num(param, trigger_data);
 | 
				
			||||||
		number = strsep(&trigger, ":");
 | 
						if (ret)
 | 
				
			||||||
 | 
							goto out_free;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		ret = -EINVAL;
 | 
						ret = event_trigger_set_filter(cmd_ops, file, filter, trigger_data);
 | 
				
			||||||
		if (!strlen(number))
 | 
					 | 
				
			||||||
			goto out_free;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		/*
 | 
					 | 
				
			||||||
		 * We use the callback data field (which is a pointer)
 | 
					 | 
				
			||||||
		 * as our counter.
 | 
					 | 
				
			||||||
		 */
 | 
					 | 
				
			||||||
		ret = kstrtoul(number, 0, &trigger_data->count);
 | 
					 | 
				
			||||||
		if (ret)
 | 
					 | 
				
			||||||
			goto out_free;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (!param) /* if param is non-empty, it's supposed to be a filter */
 | 
					 | 
				
			||||||
		goto out_reg;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (!cmd_ops->set_filter)
 | 
					 | 
				
			||||||
		goto out_reg;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	ret = cmd_ops->set_filter(param, trigger_data, file);
 | 
					 | 
				
			||||||
	if (ret < 0)
 | 
						if (ret < 0)
 | 
				
			||||||
		goto out_free;
 | 
							goto out_free;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 out_reg:
 | 
					 | 
				
			||||||
	/* Up the trigger_data count to make sure reg doesn't free it on failure */
 | 
						/* Up the trigger_data count to make sure reg doesn't free it on failure */
 | 
				
			||||||
	event_trigger_init(trigger_data);
 | 
						event_trigger_init(trigger_data);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1057,8 +1024,7 @@ event_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
	return ret;
 | 
						return ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 out_free:
 | 
					 out_free:
 | 
				
			||||||
	if (cmd_ops->set_filter)
 | 
						event_trigger_reset_filter(cmd_ops, trigger_data);
 | 
				
			||||||
		cmd_ops->set_filter(NULL, trigger_data, NULL);
 | 
					 | 
				
			||||||
	kfree(trigger_data);
 | 
						kfree(trigger_data);
 | 
				
			||||||
	goto out;
 | 
						goto out;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -1752,39 +1718,33 @@ static struct event_trigger_ops event_disable_count_trigger_ops = {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int event_enable_trigger_parse(struct event_command *cmd_ops,
 | 
					int event_enable_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
			       struct trace_event_file *file,
 | 
								       struct trace_event_file *file,
 | 
				
			||||||
			       char *glob, char *cmd, char *param)
 | 
								       char *glob, char *cmd, char *param_and_filter)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct trace_event_file *event_enable_file;
 | 
						struct trace_event_file *event_enable_file;
 | 
				
			||||||
	struct enable_trigger_data *enable_data;
 | 
						struct enable_trigger_data *enable_data;
 | 
				
			||||||
	struct event_trigger_data *trigger_data;
 | 
						struct event_trigger_data *trigger_data;
 | 
				
			||||||
	struct event_trigger_ops *trigger_ops;
 | 
					 | 
				
			||||||
	struct trace_array *tr = file->tr;
 | 
						struct trace_array *tr = file->tr;
 | 
				
			||||||
 | 
						char *param, *filter;
 | 
				
			||||||
 | 
						bool enable, remove;
 | 
				
			||||||
	const char *system;
 | 
						const char *system;
 | 
				
			||||||
	const char *event;
 | 
						const char *event;
 | 
				
			||||||
	bool hist = false;
 | 
						bool hist = false;
 | 
				
			||||||
	char *trigger;
 | 
					 | 
				
			||||||
	char *number;
 | 
					 | 
				
			||||||
	bool enable;
 | 
					 | 
				
			||||||
	int ret;
 | 
						int ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						remove = event_trigger_check_remove(glob);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (event_trigger_empty_param(param_and_filter))
 | 
				
			||||||
 | 
							return -EINVAL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ret = event_trigger_separate_filter(param_and_filter, ¶m, &filter, true);
 | 
				
			||||||
 | 
						if (ret)
 | 
				
			||||||
 | 
							return ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						system = strsep(¶m, ":");
 | 
				
			||||||
	if (!param)
 | 
						if (!param)
 | 
				
			||||||
		return -EINVAL;
 | 
							return -EINVAL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* separate the trigger from the filter (s:e:n [if filter]) */
 | 
						event = strsep(¶m, ":");
 | 
				
			||||||
	trigger = strsep(¶m, " \t");
 | 
					 | 
				
			||||||
	if (!trigger)
 | 
					 | 
				
			||||||
		return -EINVAL;
 | 
					 | 
				
			||||||
	if (param) {
 | 
					 | 
				
			||||||
		param = skip_spaces(param);
 | 
					 | 
				
			||||||
		if (!*param)
 | 
					 | 
				
			||||||
			param = NULL;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	system = strsep(&trigger, ":");
 | 
					 | 
				
			||||||
	if (!trigger)
 | 
					 | 
				
			||||||
		return -EINVAL;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	event = strsep(&trigger, ":");
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ret = -EINVAL;
 | 
						ret = -EINVAL;
 | 
				
			||||||
	event_enable_file = find_event_file(tr, system, event);
 | 
						event_enable_file = find_event_file(tr, system, event);
 | 
				
			||||||
| 
						 | 
					@ -1800,31 +1760,23 @@ int event_enable_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
#else
 | 
					#else
 | 
				
			||||||
	enable = strcmp(cmd, ENABLE_EVENT_STR) == 0;
 | 
						enable = strcmp(cmd, ENABLE_EVENT_STR) == 0;
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
	trigger_ops = cmd_ops->get_trigger_ops(cmd, trigger);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	ret = -ENOMEM;
 | 
						ret = -ENOMEM;
 | 
				
			||||||
	trigger_data = kzalloc(sizeof(*trigger_data), GFP_KERNEL);
 | 
					 | 
				
			||||||
	if (!trigger_data)
 | 
					 | 
				
			||||||
		goto out;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	enable_data = kzalloc(sizeof(*enable_data), GFP_KERNEL);
 | 
						enable_data = kzalloc(sizeof(*enable_data), GFP_KERNEL);
 | 
				
			||||||
	if (!enable_data) {
 | 
						if (!enable_data)
 | 
				
			||||||
		kfree(trigger_data);
 | 
					 | 
				
			||||||
		goto out;
 | 
							goto out;
 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	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);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	enable_data->hist = hist;
 | 
						enable_data->hist = hist;
 | 
				
			||||||
	enable_data->enable = enable;
 | 
						enable_data->enable = enable;
 | 
				
			||||||
	enable_data->file = event_enable_file;
 | 
						enable_data->file = event_enable_file;
 | 
				
			||||||
	trigger_data->private_data = enable_data;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (glob[0] == '!') {
 | 
						trigger_data = event_trigger_alloc(cmd_ops, cmd, param, enable_data);
 | 
				
			||||||
 | 
						if (!trigger_data) {
 | 
				
			||||||
 | 
							kfree(enable_data);
 | 
				
			||||||
 | 
							goto out;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (remove) {
 | 
				
			||||||
		event_trigger_unregister(cmd_ops, file, glob+1, trigger_data);
 | 
							event_trigger_unregister(cmd_ops, file, glob+1, trigger_data);
 | 
				
			||||||
		kfree(trigger_data);
 | 
							kfree(trigger_data);
 | 
				
			||||||
		kfree(enable_data);
 | 
							kfree(enable_data);
 | 
				
			||||||
| 
						 | 
					@ -1835,33 +1787,14 @@ int event_enable_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
	/* Up the trigger_data count to make sure nothing frees it on failure */
 | 
						/* Up the trigger_data count to make sure nothing frees it on failure */
 | 
				
			||||||
	event_trigger_init(trigger_data);
 | 
						event_trigger_init(trigger_data);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (trigger) {
 | 
						ret = event_trigger_parse_num(param, trigger_data);
 | 
				
			||||||
		number = strsep(&trigger, ":");
 | 
						if (ret)
 | 
				
			||||||
 | 
							goto out_free;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		ret = -EINVAL;
 | 
						ret = event_trigger_set_filter(cmd_ops, file, filter, trigger_data);
 | 
				
			||||||
		if (!strlen(number))
 | 
					 | 
				
			||||||
			goto out_free;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		/*
 | 
					 | 
				
			||||||
		 * We use the callback data field (which is a pointer)
 | 
					 | 
				
			||||||
		 * as our counter.
 | 
					 | 
				
			||||||
		 */
 | 
					 | 
				
			||||||
		ret = kstrtoul(number, 0, &trigger_data->count);
 | 
					 | 
				
			||||||
		if (ret)
 | 
					 | 
				
			||||||
			goto out_free;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (!param) /* if param is non-empty, it's supposed to be a filter */
 | 
					 | 
				
			||||||
		goto out_reg;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (!cmd_ops->set_filter)
 | 
					 | 
				
			||||||
		goto out_reg;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	ret = cmd_ops->set_filter(param, trigger_data, file);
 | 
					 | 
				
			||||||
	if (ret < 0)
 | 
						if (ret < 0)
 | 
				
			||||||
		goto out_free;
 | 
							goto out_free;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 out_reg:
 | 
					 | 
				
			||||||
	/* Don't let event modules unload while probe registered */
 | 
						/* Don't let event modules unload while probe registered */
 | 
				
			||||||
	ret = trace_event_try_get_ref(event_enable_file->event_call);
 | 
						ret = trace_event_try_get_ref(event_enable_file->event_call);
 | 
				
			||||||
	if (!ret) {
 | 
						if (!ret) {
 | 
				
			||||||
| 
						 | 
					@ -1880,16 +1813,15 @@ int event_enable_trigger_parse(struct event_command *cmd_ops,
 | 
				
			||||||
	event_trigger_free(trigger_data);
 | 
						event_trigger_free(trigger_data);
 | 
				
			||||||
 out:
 | 
					 out:
 | 
				
			||||||
	return ret;
 | 
						return ret;
 | 
				
			||||||
 | 
					 | 
				
			||||||
 out_disable:
 | 
					 out_disable:
 | 
				
			||||||
	trace_event_enable_disable(event_enable_file, 0, 1);
 | 
						trace_event_enable_disable(event_enable_file, 0, 1);
 | 
				
			||||||
 out_put:
 | 
					 out_put:
 | 
				
			||||||
	trace_event_put_ref(event_enable_file->event_call);
 | 
						trace_event_put_ref(event_enable_file->event_call);
 | 
				
			||||||
 out_free:
 | 
					 out_free:
 | 
				
			||||||
	if (cmd_ops->set_filter)
 | 
						event_trigger_reset_filter(cmd_ops, trigger_data);
 | 
				
			||||||
		cmd_ops->set_filter(NULL, trigger_data, NULL);
 | 
					 | 
				
			||||||
	event_trigger_free(trigger_data);
 | 
						event_trigger_free(trigger_data);
 | 
				
			||||||
	kfree(enable_data);
 | 
						kfree(enable_data);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	goto out;
 | 
						goto out;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue