Commit 69a0200c authored by Tom Zanussi's avatar Tom Zanussi Committed by Steven Rostedt

tracing: Add hist trigger support for stacktraces as keys

It's often useful to be able to use a stacktrace as a hash key, for
keeping a count of the number of times a particular call path resulted
in a trace event, for instance.  Add a special key named 'stacktrace'
which can be used as key in a 'keys=' param for this purpose:

    # echo hist:keys=stacktrace ... \
               [ if filter] > event/trigger

Link: http://lkml.kernel.org/r/87515e90b3785232a874a12156174635a348edb1.1457029949.git.tom.zanussi@linux.intel.comSigned-off-by: default avatarTom Zanussi <tom.zanussi@linux.intel.com>
Tested-by: default avatarMasami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
Reviewed-by: default avatarNamhyung Kim <namhyung@kernel.org>
Signed-off-by: default avatarSteven Rostedt <rostedt@goodmis.org>
parent 31696198
...@@ -3843,14 +3843,14 @@ static const char readme_msg[] = ...@@ -3843,14 +3843,14 @@ static const char readme_msg[] =
"\t table using the key(s) and value(s) named, and the value of a\n" "\t table using the key(s) and value(s) named, and the value of a\n"
"\t sum called 'hitcount' is incremented. Keys and values\n" "\t sum called 'hitcount' is incremented. Keys and values\n"
"\t correspond to fields in the event's format description. Keys\n" "\t correspond to fields in the event's format description. Keys\n"
"\t can be any field. Compound keys consisting of up to two\n" "\t can be any field, or the special string 'stacktrace'.\n"
"\t fields can be specified by the 'keys' keyword. Values must\n" "\t Compound keys consisting of up to two fields can be specified\n"
"\t correspond to numeric fields. Sort keys consisting of up to\n" "\t by the 'keys' keyword. Values must correspond to numeric\n"
"\t two fields can be specified using the 'sort' keyword. The\n" "\t fields. Sort keys consisting of up to two fields can be\n"
"\t sort direction can be modified by appending '.descending' or\n" "\t specified using the 'sort' keyword. The sort direction can\n"
"\t '.ascending' to a sort field. The 'size' parameter can be\n" "\t be modified by appending '.descending' or '.ascending' to a\n"
"\t used to specify more or fewer than the default 2048 entries\n" "\t sort field. The 'size' parameter can be used to specify more\n"
"\t for the hashtable size.\n\n" "\t or fewer than the default 2048 entries for the hashtable size.\n\n"
"\t Reading the 'hist' file for the event will dump the hash\n" "\t Reading the 'hist' file for the event will dump the hash\n"
"\t table in its entirety to stdout. The default format used to\n" "\t table in its entirety to stdout. The default format used to\n"
"\t display a given field can be modified by appending any of the\n" "\t display a given field can be modified by appending any of the\n"
......
...@@ -35,6 +35,11 @@ struct hist_field { ...@@ -35,6 +35,11 @@ struct hist_field {
unsigned int offset; unsigned int offset;
}; };
static u64 hist_field_none(struct hist_field *field, void *event)
{
return 0;
}
static u64 hist_field_counter(struct hist_field *field, void *event) static u64 hist_field_counter(struct hist_field *field, void *event)
{ {
return 1; return 1;
...@@ -73,8 +78,12 @@ DEFINE_HIST_FIELD_FN(u8); ...@@ -73,8 +78,12 @@ DEFINE_HIST_FIELD_FN(u8);
#define for_each_hist_key_field(i, hist_data) \ #define for_each_hist_key_field(i, hist_data) \
for ((i) = (hist_data)->n_vals; (i) < (hist_data)->n_fields; (i)++) for ((i) = (hist_data)->n_vals; (i) < (hist_data)->n_fields; (i)++)
#define HIST_STACKTRACE_DEPTH 16
#define HIST_STACKTRACE_SIZE (HIST_STACKTRACE_DEPTH * sizeof(unsigned long))
#define HIST_STACKTRACE_SKIP 5
#define HITCOUNT_IDX 0 #define HITCOUNT_IDX 0
#define HIST_KEY_SIZE_MAX (MAX_FILTER_STR_VAL + sizeof(u64)) #define HIST_KEY_SIZE_MAX (MAX_FILTER_STR_VAL + HIST_STACKTRACE_SIZE)
enum hist_field_flags { enum hist_field_flags {
HIST_FIELD_FL_HITCOUNT = 1, HIST_FIELD_FL_HITCOUNT = 1,
...@@ -85,6 +94,7 @@ enum hist_field_flags { ...@@ -85,6 +94,7 @@ enum hist_field_flags {
HIST_FIELD_FL_SYM_OFFSET = 32, HIST_FIELD_FL_SYM_OFFSET = 32,
HIST_FIELD_FL_EXECNAME = 64, HIST_FIELD_FL_EXECNAME = 64,
HIST_FIELD_FL_SYSCALL = 128, HIST_FIELD_FL_SYSCALL = 128,
HIST_FIELD_FL_STACKTRACE = 256,
}; };
struct hist_trigger_attrs { struct hist_trigger_attrs {
...@@ -323,6 +333,11 @@ static struct hist_field *create_hist_field(struct ftrace_event_field *field, ...@@ -323,6 +333,11 @@ static struct hist_field *create_hist_field(struct ftrace_event_field *field,
goto out; goto out;
} }
if (flags & HIST_FIELD_FL_STACKTRACE) {
hist_field->fn = hist_field_none;
goto out;
}
if (is_string_field(field)) { if (is_string_field(field)) {
flags |= HIST_FIELD_FL_STRING; flags |= HIST_FIELD_FL_STRING;
hist_field->fn = hist_field_string; hist_field->fn = hist_field_string;
...@@ -456,7 +471,6 @@ static int create_key_field(struct hist_trigger_data *hist_data, ...@@ -456,7 +471,6 @@ static int create_key_field(struct hist_trigger_data *hist_data,
struct ftrace_event_field *field = NULL; struct ftrace_event_field *field = NULL;
unsigned long flags = 0; unsigned long flags = 0;
unsigned int key_size; unsigned int key_size;
char *field_name;
int ret = 0; int ret = 0;
if (WARN_ON(key_idx >= TRACING_MAP_FIELDS_MAX)) if (WARN_ON(key_idx >= TRACING_MAP_FIELDS_MAX))
...@@ -464,33 +478,39 @@ static int create_key_field(struct hist_trigger_data *hist_data, ...@@ -464,33 +478,39 @@ static int create_key_field(struct hist_trigger_data *hist_data,
flags |= HIST_FIELD_FL_KEY; flags |= HIST_FIELD_FL_KEY;
field_name = strsep(&field_str, "."); if (strcmp(field_str, "stacktrace") == 0) {
if (field_str) { flags |= HIST_FIELD_FL_STACKTRACE;
if (strcmp(field_str, "hex") == 0) key_size = sizeof(unsigned long) * HIST_STACKTRACE_DEPTH;
flags |= HIST_FIELD_FL_HEX; } else {
else if (strcmp(field_str, "sym") == 0) char *field_name = strsep(&field_str, ".");
flags |= HIST_FIELD_FL_SYM;
else if (strcmp(field_str, "sym-offset") == 0) if (field_str) {
flags |= HIST_FIELD_FL_SYM_OFFSET; if (strcmp(field_str, "hex") == 0)
else if ((strcmp(field_str, "execname") == 0) && flags |= HIST_FIELD_FL_HEX;
(strcmp(field_name, "common_pid") == 0)) else if (strcmp(field_str, "sym") == 0)
flags |= HIST_FIELD_FL_EXECNAME; flags |= HIST_FIELD_FL_SYM;
else if (strcmp(field_str, "syscall") == 0) else if (strcmp(field_str, "sym-offset") == 0)
flags |= HIST_FIELD_FL_SYSCALL; flags |= HIST_FIELD_FL_SYM_OFFSET;
else { else if ((strcmp(field_str, "execname") == 0) &&
(strcmp(field_name, "common_pid") == 0))
flags |= HIST_FIELD_FL_EXECNAME;
else if (strcmp(field_str, "syscall") == 0)
flags |= HIST_FIELD_FL_SYSCALL;
else {
ret = -EINVAL;
goto out;
}
}
field = trace_find_event_field(file->event_call, field_name);
if (!field) {
ret = -EINVAL; ret = -EINVAL;
goto out; goto out;
} }
}
field = trace_find_event_field(file->event_call, field_name); key_size = field->size;
if (!field) {
ret = -EINVAL;
goto out;
} }
key_size = field->size;
hist_data->fields[key_idx] = create_hist_field(field, flags); hist_data->fields[key_idx] = create_hist_field(field, flags);
if (!hist_data->fields[key_idx]) { if (!hist_data->fields[key_idx]) {
ret = -ENOMEM; ret = -ENOMEM;
...@@ -679,7 +699,9 @@ static int create_tracing_map_fields(struct hist_trigger_data *hist_data) ...@@ -679,7 +699,9 @@ static int create_tracing_map_fields(struct hist_trigger_data *hist_data)
field = hist_field->field; field = hist_field->field;
if (is_string_field(field)) if (hist_field->flags & HIST_FIELD_FL_STACKTRACE)
cmp_fn = tracing_map_cmp_none;
else if (is_string_field(field))
cmp_fn = tracing_map_cmp_string; cmp_fn = tracing_map_cmp_string;
else else
cmp_fn = tracing_map_cmp_num(field->size, cmp_fn = tracing_map_cmp_num(field->size,
...@@ -786,7 +808,9 @@ static void hist_trigger_elt_update(struct hist_trigger_data *hist_data, ...@@ -786,7 +808,9 @@ static void hist_trigger_elt_update(struct hist_trigger_data *hist_data,
static void event_hist_trigger(struct event_trigger_data *data, void *rec) static void event_hist_trigger(struct event_trigger_data *data, void *rec)
{ {
struct hist_trigger_data *hist_data = data->private_data; struct hist_trigger_data *hist_data = data->private_data;
unsigned long entries[HIST_STACKTRACE_DEPTH];
char compound_key[HIST_KEY_SIZE_MAX]; char compound_key[HIST_KEY_SIZE_MAX];
struct stack_trace stacktrace;
struct hist_field *key_field; struct hist_field *key_field;
struct tracing_map_elt *elt; struct tracing_map_elt *elt;
u64 field_contents; u64 field_contents;
...@@ -799,15 +823,27 @@ static void event_hist_trigger(struct event_trigger_data *data, void *rec) ...@@ -799,15 +823,27 @@ static void event_hist_trigger(struct event_trigger_data *data, void *rec)
for_each_hist_key_field(i, hist_data) { for_each_hist_key_field(i, hist_data) {
key_field = hist_data->fields[i]; key_field = hist_data->fields[i];
field_contents = key_field->fn(key_field, rec); if (key_field->flags & HIST_FIELD_FL_STACKTRACE) {
if (key_field->flags & HIST_FIELD_FL_STRING) stacktrace.max_entries = HIST_STACKTRACE_DEPTH;
key = (void *)(unsigned long)field_contents; stacktrace.entries = entries;
else stacktrace.nr_entries = 0;
key = (void *)&field_contents; stacktrace.skip = HIST_STACKTRACE_SKIP;
if (hist_data->n_keys > 1) { memset(stacktrace.entries, 0, HIST_STACKTRACE_SIZE);
memcpy(compound_key + key_field->offset, key, save_stack_trace(&stacktrace);
key_field->size);
key = entries;
} else {
field_contents = key_field->fn(key_field, rec);
if (key_field->flags & HIST_FIELD_FL_STRING)
key = (void *)(unsigned long)field_contents;
else
key = (void *)&field_contents;
if (hist_data->n_keys > 1) {
memcpy(compound_key + key_field->offset, key,
key_field->size);
}
} }
} }
...@@ -819,6 +855,24 @@ static void event_hist_trigger(struct event_trigger_data *data, void *rec) ...@@ -819,6 +855,24 @@ static void event_hist_trigger(struct event_trigger_data *data, void *rec)
hist_trigger_elt_update(hist_data, elt, rec); hist_trigger_elt_update(hist_data, elt, rec);
} }
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++) {
if (stacktrace_entries[i] == ULONG_MAX)
return;
seq_printf(m, "%*c", 1 + spaces, ' ');
sprint_symbol(str, stacktrace_entries[i]);
seq_printf(m, "%s\n", str);
}
}
static void static void
hist_trigger_entry_print(struct seq_file *m, hist_trigger_entry_print(struct seq_file *m,
struct hist_trigger_data *hist_data, void *key, struct hist_trigger_data *hist_data, void *key,
...@@ -826,6 +880,7 @@ hist_trigger_entry_print(struct seq_file *m, ...@@ -826,6 +880,7 @@ hist_trigger_entry_print(struct seq_file *m,
{ {
struct hist_field *key_field; struct hist_field *key_field;
char str[KSYM_SYMBOL_LEN]; char str[KSYM_SYMBOL_LEN];
bool multiline = false;
unsigned int i; unsigned int i;
u64 uval; u64 uval;
...@@ -867,6 +922,12 @@ hist_trigger_entry_print(struct seq_file *m, ...@@ -867,6 +922,12 @@ hist_trigger_entry_print(struct seq_file *m,
seq_printf(m, "%s: %-30s[%3llu]", seq_printf(m, "%s: %-30s[%3llu]",
key_field->field->name, syscall_name, uval); key_field->field->name, syscall_name, uval);
} else if (key_field->flags & HIST_FIELD_FL_STACKTRACE) {
seq_puts(m, "stacktrace:\n");
hist_trigger_stacktrace_print(m,
key + key_field->offset,
HIST_STACKTRACE_DEPTH);
multiline = true;
} else if (key_field->flags & HIST_FIELD_FL_STRING) { } else if (key_field->flags & HIST_FIELD_FL_STRING) {
seq_printf(m, "%s: %-50s", key_field->field->name, seq_printf(m, "%s: %-50s", key_field->field->name,
(char *)(key + key_field->offset)); (char *)(key + key_field->offset));
...@@ -877,7 +938,10 @@ hist_trigger_entry_print(struct seq_file *m, ...@@ -877,7 +938,10 @@ hist_trigger_entry_print(struct seq_file *m,
} }
} }
seq_puts(m, " }"); if (!multiline)
seq_puts(m, " ");
seq_puts(m, "}");
seq_printf(m, " hitcount: %10llu", seq_printf(m, " hitcount: %10llu",
tracing_map_read_sum(elt, HITCOUNT_IDX)); tracing_map_read_sum(elt, HITCOUNT_IDX));
...@@ -1021,7 +1085,10 @@ static int event_hist_trigger_print(struct seq_file *m, ...@@ -1021,7 +1085,10 @@ static int event_hist_trigger_print(struct seq_file *m,
if (i > hist_data->n_vals) if (i > hist_data->n_vals)
seq_puts(m, ","); seq_puts(m, ",");
hist_field_print(m, key_field); if (key_field->flags & HIST_FIELD_FL_STACKTRACE)
seq_puts(m, "stacktrace");
else
hist_field_print(m, key_field);
} }
seq_puts(m, ":vals="); seq_puts(m, ":vals=");
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment