linux/debian/patches/features/all/rt/0011-tracing-Add-hist-trigg...

244 lines
7.4 KiB
Diff
Raw Normal View History

2017-07-09 11:42:39 +00:00
From: Tom Zanussi <tom.zanussi@linux.intel.com>
2018-01-31 19:59:07 +00:00
Date: Mon, 15 Jan 2018 20:51:45 -0600
2017-11-29 23:05:39 +00:00
Subject: [PATCH 11/37] tracing: Add hist trigger timestamp support
2018-01-31 19:59:07 +00:00
Origin: https://www.kernel.org/pub/linux/kernel/projects/rt/4.14/older/patches-4.14.15-rt13.tar.xz
2017-07-09 11:42:39 +00:00
Add support for a timestamp event field. This is actually a 'pseudo-'
event field in that it behaves like it's part of the event record, but
is really part of the corresponding ring buffer event.
To make use of the timestamp field, users can specify
2017-12-20 18:53:59 +00:00
"common_timestamp" as a field name for any histogram. Note that this
2017-07-09 11:42:39 +00:00
doesn't make much sense on its own either as either a key or value,
but needs to be supported even so, since follow-on patches will add
2017-12-20 18:53:59 +00:00
support for making use of this field in time deltas. The
common_timestamp 'field' is not a bona fide event field - so you won't
find it in the event description - but rather it's a synthetic field
that can be used like a real field.
2017-07-09 11:42:39 +00:00
Note that the use of this field requires the ring buffer be put into
2017-11-29 23:05:39 +00:00
'absolute timestamp' mode, which saves the complete timestamp for each
2017-07-09 11:42:39 +00:00
event rather than an offset. This mode will be enabled if and only if
2017-12-20 18:53:59 +00:00
a histogram makes use of the "common_timestamp" field.
2017-07-09 11:42:39 +00:00
Signed-off-by: Tom Zanussi <tom.zanussi@linux.intel.com>
Signed-off-by: Baohong Liu <baohong.liu@intel.com>
2017-12-20 18:53:59 +00:00
[kasan use-after-free fix]
Signed-off-by: Vedang Patel <vedang.patel@intel.com>
2017-07-09 11:42:39 +00:00
Signed-off-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
---
2017-11-29 23:05:39 +00:00
kernel/trace/trace_events_hist.c | 94 +++++++++++++++++++++++++++++----------
1 file changed, 71 insertions(+), 23 deletions(-)
2017-07-09 11:42:39 +00:00
--- a/kernel/trace/trace_events_hist.c
+++ b/kernel/trace/trace_events_hist.c
@@ -89,6 +89,12 @@ static u64 hist_field_log2(struct hist_f
return (u64) ilog2(roundup_pow_of_two(val));
}
+static u64 hist_field_timestamp(struct hist_field *hist_field, void *event,
+ struct ring_buffer_event *rbe)
+{
+ return ring_buffer_event_time_stamp(rbe);
+}
+
#define DEFINE_HIST_FIELD_FN(type) \
static u64 hist_field_##type(struct hist_field *hist_field, \
void *event, \
@@ -135,6 +141,7 @@ enum hist_field_flags {
HIST_FIELD_FL_SYSCALL = 1 << 7,
HIST_FIELD_FL_STACKTRACE = 1 << 8,
HIST_FIELD_FL_LOG2 = 1 << 9,
+ HIST_FIELD_FL_TIMESTAMP = 1 << 10,
2017-07-09 11:42:39 +00:00
};
struct hist_trigger_attrs {
@@ -159,6 +166,7 @@ struct hist_trigger_data {
struct trace_event_file *event_file;
struct hist_trigger_attrs *attrs;
struct tracing_map *map;
+ bool enable_timestamps;
};
static const char *hist_field_name(struct hist_field *field,
@@ -173,6 +181,8 @@ static const char *hist_field_name(struc
field_name = field->field->name;
else if (field->flags & HIST_FIELD_FL_LOG2)
field_name = hist_field_name(field->operands[0], ++level);
+ else if (field->flags & HIST_FIELD_FL_TIMESTAMP)
2017-12-20 18:53:59 +00:00
+ field_name = "common_timestamp";
2017-07-09 11:42:39 +00:00
if (field_name == NULL)
field_name = "";
@@ -440,6 +450,12 @@ static struct hist_field *create_hist_fi
2017-07-09 11:42:39 +00:00
goto out;
}
+ if (flags & HIST_FIELD_FL_TIMESTAMP) {
+ hist_field->fn = hist_field_timestamp;
+ hist_field->size = sizeof(u64);
+ goto out;
+ }
+
if (WARN_ON_ONCE(!field))
goto out;
@@ -517,10 +533,15 @@ static int create_val_field(struct hist_
2017-07-09 11:42:39 +00:00
}
}
- field = trace_find_event_field(file->event_call, field_name);
- if (!field || !field->size) {
2017-07-09 11:42:39 +00:00
- ret = -EINVAL;
- goto out;
2017-12-20 18:53:59 +00:00
+ if (strcmp(field_name, "common_timestamp") == 0) {
2017-07-09 11:42:39 +00:00
+ flags |= HIST_FIELD_FL_TIMESTAMP;
+ hist_data->enable_timestamps = true;
+ } else {
+ field = trace_find_event_field(file->event_call, field_name);
+ if (!field || !field->size) {
2017-07-09 11:42:39 +00:00
+ ret = -EINVAL;
+ goto out;
+ }
}
hist_data->fields[val_idx] = create_hist_field(field, flags);
@@ -615,16 +636,22 @@ static int create_key_field(struct hist_
2017-07-09 11:42:39 +00:00
}
}
- field = trace_find_event_field(file->event_call, field_name);
- if (!field || !field->size) {
2017-07-09 11:42:39 +00:00
- ret = -EINVAL;
- goto out;
- }
2017-12-20 18:53:59 +00:00
+ if (strcmp(field_name, "common_timestamp") == 0) {
2017-07-09 11:42:39 +00:00
+ flags |= HIST_FIELD_FL_TIMESTAMP;
+ hist_data->enable_timestamps = true;
+ key_size = sizeof(u64);
+ } else {
+ field = trace_find_event_field(file->event_call, field_name);
+ if (!field || !field->size) {
2017-07-09 11:42:39 +00:00
+ ret = -EINVAL;
+ goto out;
+ }
- if (is_string_field(field))
- key_size = MAX_FILTER_STR_VAL;
- else
- key_size = field->size;
+ if (is_string_field(field))
+ key_size = MAX_FILTER_STR_VAL;
+ else
+ key_size = field->size;
+ }
}
hist_data->fields[key_idx] = create_hist_field(field, flags);
@@ -820,6 +847,9 @@ static int create_tracing_map_fields(str
2017-07-09 11:42:39 +00:00
if (hist_field->flags & HIST_FIELD_FL_STACKTRACE)
cmp_fn = tracing_map_cmp_none;
+ else if (!field)
+ cmp_fn = tracing_map_cmp_num(hist_field->size,
+ hist_field->is_signed);
else if (is_string_field(field))
cmp_fn = tracing_map_cmp_string;
else
@@ -1217,7 +1247,11 @@ static void hist_field_print(struct seq_
2017-07-09 11:42:39 +00:00
{
const char *field_name = hist_field_name(hist_field, 0);
- seq_printf(m, "%s", field_name);
+ if (hist_field->flags & HIST_FIELD_FL_TIMESTAMP)
2017-12-20 18:53:59 +00:00
+ seq_puts(m, "common_timestamp");
2017-07-09 11:42:39 +00:00
+ else if (field_name)
+ seq_printf(m, "%s", field_name);
+
if (hist_field->flags) {
const char *flags_str = get_hist_field_flags(hist_field);
@@ -1268,27 +1302,25 @@ static int event_hist_trigger_print(stru
2017-07-09 11:42:39 +00:00
for (i = 0; i < hist_data->n_sort_keys; i++) {
struct tracing_map_sort_key *sort_key;
+ unsigned int idx;
sort_key = &hist_data->sort_keys[i];
+ idx = sort_key->field_idx;
+
+ if (WARN_ON(idx >= TRACING_MAP_FIELDS_MAX))
+ return -EINVAL;
if (i > 0)
seq_puts(m, ",");
- if (sort_key->field_idx == HITCOUNT_IDX)
+ if (idx == HITCOUNT_IDX)
seq_puts(m, "hitcount");
- else {
- unsigned int idx = sort_key->field_idx;
-
- if (WARN_ON(idx >= TRACING_MAP_FIELDS_MAX))
- return -EINVAL;
-
+ else
hist_field_print(m, hist_data->fields[idx]);
- }
if (sort_key->descending)
seq_puts(m, ".descending");
}
-
seq_printf(m, ":size=%u", (1 << hist_data->map->map_bits));
if (data->filter_str)
@@ -1456,6 +1488,10 @@ static bool hist_trigger_match(struct ev
2017-07-09 11:42:39 +00:00
return false;
if (key_field->offset != key_field_test->offset)
return false;
+ if (key_field->size != key_field_test->size)
+ return false;
+ if (key_field->is_signed != key_field_test->is_signed)
+ return false;
}
for (i = 0; i < hist_data->n_sort_keys; i++) {
@@ -1538,6 +1574,9 @@ static int hist_register_trigger(char *g
2017-07-09 11:42:39 +00:00
update_cond_flag(file);
+ if (hist_data->enable_timestamps)
+ tracing_set_time_stamp_abs(file->tr, true);
+
if (trace_event_trigger_enable_disable(file, 1) < 0) {
list_del_rcu(&data->list);
update_cond_flag(file);
2017-12-20 18:53:59 +00:00
@@ -1572,17 +1611,26 @@ static void hist_unregister_trigger(char
2017-07-09 11:42:39 +00:00
if (unregistered && test->ops->free)
test->ops->free(test->ops, test);
+
+ if (hist_data->enable_timestamps) {
+ if (unregistered)
+ tracing_set_time_stamp_abs(file->tr, false);
+ }
2017-07-09 11:42:39 +00:00
}
static void hist_unreg_all(struct trace_event_file *file)
2017-11-29 23:05:39 +00:00
{
struct event_trigger_data *test, *n;
+ struct hist_trigger_data *hist_data;
list_for_each_entry_safe(test, n, &file->triggers, list) {
if (test->cmd_ops->trigger_type == ETT_EVENT_HIST) {
+ hist_data = test->private_data;
list_del_rcu(&test->list);
trace_event_trigger_enable_disable(file, 0);
update_cond_flag(file);
+ if (hist_data->enable_timestamps)
+ tracing_set_time_stamp_abs(file->tr, false);
2017-12-20 18:53:59 +00:00
if (test->ops->free)
test->ops->free(test->ops, test);
2017-11-29 23:05:39 +00:00
}