Commit 61a75954 authored by Linus Torvalds's avatar Linus Torvalds
Browse files
Pull tracing fixes from Steven Rostedt:
 "Various fixes:

   - Fix an uninitialized variable

   - Fix compile bug to bootconfig userspace tool (in tools directory)

   - Suppress some error messages of bootconfig userspace tool

   - Remove unneded CONFIG_LIBXBC from bootconfig

   - Allocate bootconfig xbc_nodes dynamically. To ease complaints about
     taking up static memory at boot up

   - Use of parse_args() to parse bootconfig instead of strstr() usage
     Prevents issues of double quotes containing the interested string

   - Fix missing ring_buffer_nest_end() on synthetic event error path

   - Return zero not -EINVAL on soft disabled synthetic event (soft
     disabling must be the same as hard disabling, which returns zero)

   - Consolidate synthetic event code (remove duplicate code)"

* tag 'trace-v5.6-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/rostedt/linux-trace:
  tracing: Consolidate trace() functions
  tracing: Don't return -EINVAL when tracing soft disabled synth events
  tracing: Add missing nest end to synth_event_trace_start() error case
  tools/bootconfig: Suppress non-error messages
  bootconfig: Allocate xbc_nodes array dynamically
  bootconfig: Use parse_args() to find bootconfig and '--'
  tracing/kprobe: Fix uninitialized variable bug
  bootconfig: Remove unneeded CONFIG_LIBXBC
  tools/bootconfig: Fix wrong __VA_ARGS__ usage
parents 0a679e13 7276531d
Loading
Loading
Loading
Loading
+1 −1
Original line number Diff line number Diff line
@@ -440,7 +440,7 @@ struct synth_event_trace_state {
	struct synth_event *event;
	unsigned int cur_field;
	unsigned int n_u64;
	bool enabled;
	bool disabled;
	bool add_next;
	bool add_name;
};
+0 −1
Original line number Diff line number Diff line
@@ -1227,7 +1227,6 @@ endif
config BOOT_CONFIG
	bool "Boot config support"
	depends on BLK_DEV_INITRD
	select LIBXBC
	default y
	help
	  Extra boot config allows system admin to pass a config file as
+30 −7
Original line number Diff line number Diff line
@@ -142,6 +142,15 @@ static char *extra_command_line;
/* Extra init arguments */
static char *extra_init_args;

#ifdef CONFIG_BOOT_CONFIG
/* Is bootconfig on command line? */
static bool bootconfig_found;
static bool initargs_found;
#else
# define bootconfig_found false
# define initargs_found false
#endif

static char *execute_command;
static char *ramdisk_execute_command;

@@ -336,17 +345,30 @@ u32 boot_config_checksum(unsigned char *p, u32 size)
	return ret;
}

static int __init bootconfig_params(char *param, char *val,
				    const char *unused, void *arg)
{
	if (strcmp(param, "bootconfig") == 0) {
		bootconfig_found = true;
	} else if (strcmp(param, "--") == 0) {
		initargs_found = true;
	}
	return 0;
}

static void __init setup_boot_config(const char *cmdline)
{
	static char tmp_cmdline[COMMAND_LINE_SIZE] __initdata;
	u32 size, csum;
	char *data, *copy;
	const char *p;
	u32 *hdr;
	int ret;

	p = strstr(cmdline, "bootconfig");
	if (!p || (p != cmdline && !isspace(*(p-1))) ||
	    (p[10] && !isspace(p[10])))
	strlcpy(tmp_cmdline, boot_command_line, COMMAND_LINE_SIZE);
	parse_args("bootconfig", tmp_cmdline, NULL, 0, 0, 0, NULL,
		   bootconfig_params);

	if (!bootconfig_found)
		return;

	if (!initrd_end)
@@ -562,11 +584,12 @@ static void __init setup_command_line(char *command_line)
		 * to init.
		 */
		len = strlen(saved_command_line);
		if (!strstr(boot_command_line, " -- ")) {
		if (initargs_found) {
			saved_command_line[len++] = ' ';
		} else {
			strcpy(saved_command_line + len, " -- ");
			len += 4;
		} else
			saved_command_line[len++] = ' ';
		}

		strcpy(saved_command_line + len, extra_init_args);
	}
+86 −141
Original line number Diff line number Diff line
@@ -1798,6 +1798,60 @@ void synth_event_cmd_init(struct dynevent_cmd *cmd, char *buf, int maxlen)
}
EXPORT_SYMBOL_GPL(synth_event_cmd_init);

static inline int
__synth_event_trace_start(struct trace_event_file *file,
			  struct synth_event_trace_state *trace_state)
{
	int entry_size, fields_size = 0;
	int ret = 0;

	/*
	 * Normal event tracing doesn't get called at all unless the
	 * ENABLED bit is set (which attaches the probe thus allowing
	 * this code to be called, etc).  Because this is called
	 * directly by the user, we don't have that but we still need
	 * to honor not logging when disabled.  For the the iterated
	 * trace case, we save the enabed state upon start and just
	 * ignore the following data calls.
	 */
	if (!(file->flags & EVENT_FILE_FL_ENABLED) ||
	    trace_trigger_soft_disabled(file)) {
		trace_state->disabled = true;
		ret = -ENOENT;
		goto out;
	}

	trace_state->event = file->event_call->data;

	fields_size = trace_state->event->n_u64 * sizeof(u64);

	/*
	 * Avoid ring buffer recursion detection, as this event
	 * is being performed within another event.
	 */
	trace_state->buffer = file->tr->array_buffer.buffer;
	ring_buffer_nest_start(trace_state->buffer);

	entry_size = sizeof(*trace_state->entry) + fields_size;
	trace_state->entry = trace_event_buffer_reserve(&trace_state->fbuffer,
							file,
							entry_size);
	if (!trace_state->entry) {
		ring_buffer_nest_end(trace_state->buffer);
		ret = -EINVAL;
	}
out:
	return ret;
}

static inline void
__synth_event_trace_end(struct synth_event_trace_state *trace_state)
{
	trace_event_buffer_commit(&trace_state->fbuffer);

	ring_buffer_nest_end(trace_state->buffer);
}

/**
 * synth_event_trace - Trace a synthetic event
 * @file: The trace_event_file representing the synthetic event
@@ -1819,71 +1873,38 @@ EXPORT_SYMBOL_GPL(synth_event_cmd_init);
 */
int synth_event_trace(struct trace_event_file *file, unsigned int n_vals, ...)
{
	struct trace_event_buffer fbuffer;
	struct synth_trace_event *entry;
	struct trace_buffer *buffer;
	struct synth_event *event;
	struct synth_event_trace_state state;
	unsigned int i, n_u64;
	int fields_size = 0;
	va_list args;
	int ret = 0;

	/*
	 * Normal event generation doesn't get called at all unless
	 * the ENABLED bit is set (which attaches the probe thus
	 * allowing this code to be called, etc).  Because this is
	 * called directly by the user, we don't have that but we
	 * still need to honor not logging when disabled.
	 */
	if (!(file->flags & EVENT_FILE_FL_ENABLED))
		return 0;

	event = file->event_call->data;

	if (n_vals != event->n_fields)
		return -EINVAL;

	if (trace_trigger_soft_disabled(file))
		return -EINVAL;

	fields_size = event->n_u64 * sizeof(u64);

	/*
	 * Avoid ring buffer recursion detection, as this event
	 * is being performed within another event.
	 */
	buffer = file->tr->array_buffer.buffer;
	ring_buffer_nest_start(buffer);
	int ret;

	entry = trace_event_buffer_reserve(&fbuffer, file,
					   sizeof(*entry) + fields_size);
	if (!entry) {
		ret = -EINVAL;
		goto out;
	ret = __synth_event_trace_start(file, &state);
	if (ret) {
		if (ret == -ENOENT)
			ret = 0; /* just disabled, not really an error */
		return ret;
	}

	va_start(args, n_vals);
	for (i = 0, n_u64 = 0; i < event->n_fields; i++) {
	for (i = 0, n_u64 = 0; i < state.event->n_fields; i++) {
		u64 val;

		val = va_arg(args, u64);

		if (event->fields[i]->is_string) {
		if (state.event->fields[i]->is_string) {
			char *str_val = (char *)(long)val;
			char *str_field = (char *)&entry->fields[n_u64];
			char *str_field = (char *)&state.entry->fields[n_u64];

			strscpy(str_field, str_val, STR_VAR_LEN_MAX);
			n_u64 += STR_VAR_LEN_MAX / sizeof(u64);
		} else {
			entry->fields[n_u64] = val;
			state.entry->fields[n_u64] = val;
			n_u64++;
		}
	}
	va_end(args);

	trace_event_buffer_commit(&fbuffer);
out:
	ring_buffer_nest_end(buffer);
	__synth_event_trace_end(&state);

	return ret;
}
@@ -1910,64 +1931,31 @@ EXPORT_SYMBOL_GPL(synth_event_trace);
int synth_event_trace_array(struct trace_event_file *file, u64 *vals,
			    unsigned int n_vals)
{
	struct trace_event_buffer fbuffer;
	struct synth_trace_event *entry;
	struct trace_buffer *buffer;
	struct synth_event *event;
	struct synth_event_trace_state state;
	unsigned int i, n_u64;
	int fields_size = 0;
	int ret = 0;

	/*
	 * Normal event generation doesn't get called at all unless
	 * the ENABLED bit is set (which attaches the probe thus
	 * allowing this code to be called, etc).  Because this is
	 * called directly by the user, we don't have that but we
	 * still need to honor not logging when disabled.
	 */
	if (!(file->flags & EVENT_FILE_FL_ENABLED))
		return 0;

	event = file->event_call->data;

	if (n_vals != event->n_fields)
		return -EINVAL;

	if (trace_trigger_soft_disabled(file))
		return -EINVAL;

	fields_size = event->n_u64 * sizeof(u64);

	/*
	 * Avoid ring buffer recursion detection, as this event
	 * is being performed within another event.
	 */
	buffer = file->tr->array_buffer.buffer;
	ring_buffer_nest_start(buffer);
	int ret;

	entry = trace_event_buffer_reserve(&fbuffer, file,
					   sizeof(*entry) + fields_size);
	if (!entry) {
		ret = -EINVAL;
		goto out;
	ret = __synth_event_trace_start(file, &state);
	if (ret) {
		if (ret == -ENOENT)
			ret = 0; /* just disabled, not really an error */
		return ret;
	}

	for (i = 0, n_u64 = 0; i < event->n_fields; i++) {
		if (event->fields[i]->is_string) {
	for (i = 0, n_u64 = 0; i < state.event->n_fields; i++) {
		if (state.event->fields[i]->is_string) {
			char *str_val = (char *)(long)vals[i];
			char *str_field = (char *)&entry->fields[n_u64];
			char *str_field = (char *)&state.entry->fields[n_u64];

			strscpy(str_field, str_val, STR_VAR_LEN_MAX);
			n_u64 += STR_VAR_LEN_MAX / sizeof(u64);
		} else {
			entry->fields[n_u64] = vals[i];
			state.entry->fields[n_u64] = vals[i];
			n_u64++;
		}
	}

	trace_event_buffer_commit(&fbuffer);
out:
	ring_buffer_nest_end(buffer);
	__synth_event_trace_end(&state);

	return ret;
}
@@ -2004,58 +1992,17 @@ EXPORT_SYMBOL_GPL(synth_event_trace_array);
int synth_event_trace_start(struct trace_event_file *file,
			    struct synth_event_trace_state *trace_state)
{
	struct synth_trace_event *entry;
	int fields_size = 0;
	int ret = 0;
	int ret;

	if (!trace_state) {
		ret = -EINVAL;
		goto out;
	}
	if (!trace_state)
		return -EINVAL;

	memset(trace_state, '\0', sizeof(*trace_state));

	/*
	 * Normal event tracing doesn't get called at all unless the
	 * ENABLED bit is set (which attaches the probe thus allowing
	 * this code to be called, etc).  Because this is called
	 * directly by the user, we don't have that but we still need
	 * to honor not logging when disabled.  For the the iterated
	 * trace case, we save the enabed state upon start and just
	 * ignore the following data calls.
	 */
	if (!(file->flags & EVENT_FILE_FL_ENABLED)) {
		trace_state->enabled = false;
		goto out;
	}

	trace_state->enabled = true;

	trace_state->event = file->event_call->data;

	if (trace_trigger_soft_disabled(file)) {
		ret = -EINVAL;
		goto out;
	}

	fields_size = trace_state->event->n_u64 * sizeof(u64);

	/*
	 * Avoid ring buffer recursion detection, as this event
	 * is being performed within another event.
	 */
	trace_state->buffer = file->tr->array_buffer.buffer;
	ring_buffer_nest_start(trace_state->buffer);

	entry = trace_event_buffer_reserve(&trace_state->fbuffer, file,
					   sizeof(*entry) + fields_size);
	if (!entry) {
		ret = -EINVAL;
		goto out;
	}
	ret = __synth_event_trace_start(file, trace_state);
	if (ret == -ENOENT)
		ret = 0; /* just disabled, not really an error */

	trace_state->entry = entry;
out:
	return ret;
}
EXPORT_SYMBOL_GPL(synth_event_trace_start);
@@ -2088,7 +2035,7 @@ static int __synth_event_add_val(const char *field_name, u64 val,
		trace_state->add_next = true;
	}

	if (!trace_state->enabled)
	if (trace_state->disabled)
		goto out;

	event = trace_state->event;
@@ -2223,9 +2170,7 @@ int synth_event_trace_end(struct synth_event_trace_state *trace_state)
	if (!trace_state)
		return -EINVAL;

	trace_event_buffer_commit(&trace_state->fbuffer);

	ring_buffer_nest_end(trace_state->buffer);
	__synth_event_trace_end(trace_state);

	return 0;
}
+1 −1
Original line number Diff line number Diff line
@@ -1012,7 +1012,7 @@ int __kprobe_event_add_fields(struct dynevent_cmd *cmd, ...)
{
	struct dynevent_arg arg;
	va_list args;
	int ret;
	int ret = 0;

	if (cmd->type != DYNEVENT_TYPE_KPROBE)
		return -EINVAL;
Loading