tracing: Generalize hist trigger onmax and save action
The action refactor code allowed actions and handlers to be separated, but the existing onmax handler and save action code is still not flexible enough to handle arbitrary coupling. This change generalizes them and in the process makes additional handlers and actions easier to implement. The onmax action can be broken up and thought of as two separate components - a variable to be tracked (the parameter given to the onmax($var_to_track) function) and an invisible variable created to save the ongoing result of doing something with that variable, such as saving the max value of that variable so far seen. Separating it out like this and renaming it appropriately allows us to use the same code for similar tracking functions such as onchange($var_to_track), which would just track the last value seen rather than the max seen so far, which is useful in some situations. Additionally, because different handlers and actions may want to save and access data differently e.g. save and retrieve tracking values as local variables vs something more global, save_val() and get_val() interface functions are introduced and max-specific implementations are used instead. The same goes for the code that checks whether a maximum has been hit - a generic check_val() interface and max-checking implementation is used instead, which allows future patches to make use of he same code using their own implemetations of similar functionality. Link: http://lkml.kernel.org/r/980ea73dd8e3f36db3d646f99652f8fed42b77d4.1550100284.git.tom.zanussi@linux.intel.com Signed-off-by: Tom Zanussi <tom.zanussi@linux.intel.com> Signed-off-by: Steven Rostedt (VMware) <rostedt@goodmis.org>
This commit is contained in:
Родитель
c3e49506a0
Коммит
466f4528fb
|
@ -386,6 +386,8 @@ typedef void (*action_fn_t) (struct hist_trigger_data *hist_data,
|
|||
struct ring_buffer_event *rbe, void *key,
|
||||
struct action_data *data, u64 *var_ref_vals);
|
||||
|
||||
typedef bool (*check_track_val_fn_t) (u64 track_val, u64 var_val);
|
||||
|
||||
enum handler_id {
|
||||
HANDLER_ONMATCH = 1,
|
||||
HANDLER_ONMAX,
|
||||
|
@ -423,15 +425,35 @@ struct action_data {
|
|||
} match_data;
|
||||
|
||||
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;
|
||||
unsigned int max_var_ref_idx;
|
||||
struct hist_field *max_var;
|
||||
struct hist_field *var;
|
||||
} onmax;
|
||||
|
||||
/*
|
||||
* var_ref refers to the variable being
|
||||
* tracked e.g onmax($var).
|
||||
*/
|
||||
struct hist_field *var_ref;
|
||||
|
||||
/*
|
||||
* track_var contains the 'invisible' tracking
|
||||
* variable created to keep the current
|
||||
* e.g. max value.
|
||||
*/
|
||||
struct hist_field *track_var;
|
||||
|
||||
check_track_val_fn_t check_val;
|
||||
action_fn_t save_data;
|
||||
} track_data;
|
||||
};
|
||||
};
|
||||
|
||||
|
||||
static char last_hist_cmd[MAX_FILTER_STR_VAL];
|
||||
static char hist_err_str[MAX_FILTER_STR_VAL];
|
||||
|
||||
|
@ -3238,10 +3260,10 @@ static void update_field_vars(struct hist_trigger_data *hist_data,
|
|||
hist_data->n_field_vars, 0);
|
||||
}
|
||||
|
||||
static void update_max_vars(struct hist_trigger_data *hist_data,
|
||||
struct tracing_map_elt *elt,
|
||||
struct ring_buffer_event *rbe,
|
||||
void *rec)
|
||||
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)
|
||||
{
|
||||
__update_field_vars(elt, rbe, rec, hist_data->save_vars,
|
||||
hist_data->n_save_vars, hist_data->n_field_var_str);
|
||||
|
@ -3379,14 +3401,67 @@ create_target_field_var(struct hist_trigger_data *target_hist_data,
|
|||
return create_field_var(target_hist_data, file, var_name);
|
||||
}
|
||||
|
||||
static void onmax_print(struct seq_file *m,
|
||||
struct hist_trigger_data *hist_data,
|
||||
struct tracing_map_elt *elt,
|
||||
struct action_data *data)
|
||||
static bool check_track_val_max(u64 track_val, u64 var_val)
|
||||
{
|
||||
unsigned int i, save_var_idx, max_idx = data->onmax.max_var->var.idx;
|
||||
if (var_val <= track_val)
|
||||
return false;
|
||||
|
||||
seq_printf(m, "\n\tmax: %10llu", tracing_map_read_var(elt, max_idx));
|
||||
return true;
|
||||
}
|
||||
|
||||
static u64 get_track_val(struct hist_trigger_data *hist_data,
|
||||
struct tracing_map_elt *elt,
|
||||
struct action_data *data)
|
||||
{
|
||||
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);
|
||||
|
||||
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);
|
||||
}
|
||||
|
||||
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, "\n\tmax: %10llu", track_val);
|
||||
|
||||
for (i = 0; i < hist_data->n_save_vars; i++) {
|
||||
struct hist_field *save_val = hist_data->save_vars[i]->val;
|
||||
|
@ -3405,25 +3480,17 @@ static void onmax_print(struct seq_file *m,
|
|||
}
|
||||
}
|
||||
|
||||
static void onmax_save(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 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)
|
||||
{
|
||||
unsigned int max_idx = data->onmax.max_var->var.idx;
|
||||
unsigned int max_var_ref_idx = data->onmax.max_var_ref_idx;
|
||||
u64 var_val = var_ref_vals[data->track_data.var_ref->var_ref_idx];
|
||||
|
||||
u64 var_val, max_val;
|
||||
|
||||
var_val = var_ref_vals[max_var_ref_idx];
|
||||
max_val = tracing_map_read_var(elt, max_idx);
|
||||
|
||||
if (var_val <= max_val)
|
||||
return;
|
||||
|
||||
tracing_map_set_var(elt, max_idx, var_val);
|
||||
|
||||
update_max_vars(hist_data, elt, rbe, rec);
|
||||
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);
|
||||
}
|
||||
}
|
||||
|
||||
static void action_data_destroy(struct action_data *data)
|
||||
|
@ -3443,12 +3510,13 @@ static void action_data_destroy(struct action_data *data)
|
|||
kfree(data);
|
||||
}
|
||||
|
||||
static void onmax_destroy(struct action_data *data)
|
||||
static void track_data_destroy(struct hist_trigger_data *hist_data,
|
||||
struct action_data *data)
|
||||
{
|
||||
destroy_hist_field(data->onmax.max_var, 0);
|
||||
destroy_hist_field(data->onmax.var, 0);
|
||||
destroy_hist_field(data->track_data.track_var, 0);
|
||||
destroy_hist_field(data->track_data.var_ref, 0);
|
||||
|
||||
kfree(data->onmax.var_str);
|
||||
kfree(data->track_data.var_str);
|
||||
|
||||
action_data_destroy(data);
|
||||
}
|
||||
|
@ -3456,25 +3524,24 @@ static void onmax_destroy(struct action_data *data)
|
|||
static int action_create(struct hist_trigger_data *hist_data,
|
||||
struct action_data *data);
|
||||
|
||||
static int onmax_create(struct hist_trigger_data *hist_data,
|
||||
struct action_data *data)
|
||||
static int track_data_create(struct hist_trigger_data *hist_data,
|
||||
struct action_data *data)
|
||||
{
|
||||
struct hist_field *var_field, *ref_field, *max_var = NULL;
|
||||
struct hist_field *var_field, *ref_field, *track_var = NULL;
|
||||
struct trace_event_file *file = hist_data->event_file;
|
||||
unsigned int var_ref_idx = hist_data->n_var_refs;
|
||||
char *onmax_var_str;
|
||||
char *track_data_var_str;
|
||||
int ret = 0;
|
||||
|
||||
onmax_var_str = data->onmax.var_str;
|
||||
if (onmax_var_str[0] != '$') {
|
||||
hist_err("onmax: For onmax(x), x must be a variable: ", onmax_var_str);
|
||||
track_data_var_str = data->track_data.var_str;
|
||||
if (track_data_var_str[0] != '$') {
|
||||
hist_err("For onmax(x), x must be a variable: ", track_data_var_str);
|
||||
return -EINVAL;
|
||||
}
|
||||
onmax_var_str++;
|
||||
track_data_var_str++;
|
||||
|
||||
var_field = find_target_event_var(hist_data, NULL, NULL, onmax_var_str);
|
||||
var_field = find_target_event_var(hist_data, NULL, NULL, track_data_var_str);
|
||||
if (!var_field) {
|
||||
hist_err("onmax: Couldn't find onmax variable: ", onmax_var_str);
|
||||
hist_err("Couldn't find onmax variable: ", track_data_var_str);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
|
@ -3482,17 +3549,16 @@ static int onmax_create(struct hist_trigger_data *hist_data,
|
|||
if (!ref_field)
|
||||
return -ENOMEM;
|
||||
|
||||
data->onmax.var = ref_field;
|
||||
data->track_data.var_ref = ref_field;
|
||||
|
||||
data->onmax.max_var_ref_idx = var_ref_idx;
|
||||
|
||||
max_var = create_var(hist_data, file, "max", sizeof(u64), "u64");
|
||||
if (IS_ERR(max_var)) {
|
||||
hist_err("onmax: Couldn't create onmax variable: ", "max");
|
||||
ret = PTR_ERR(max_var);
|
||||
if (data->handler == HANDLER_ONMAX)
|
||||
track_var = create_var(hist_data, file, "__max", sizeof(u64), "u64");
|
||||
if (IS_ERR(track_var)) {
|
||||
hist_err("Couldn't create onmax variable: ", "__max");
|
||||
ret = PTR_ERR(track_var);
|
||||
goto out;
|
||||
}
|
||||
data->onmax.max_var = max_var;
|
||||
data->track_data.track_var = track_var;
|
||||
|
||||
ret = action_create(hist_data, data);
|
||||
out:
|
||||
|
@ -3570,8 +3636,15 @@ static int action_parse(char *str, struct action_data *data,
|
|||
goto out;
|
||||
|
||||
if (handler == HANDLER_ONMAX)
|
||||
data->fn = onmax_save;
|
||||
data->track_data.check_val = check_track_val_max;
|
||||
else {
|
||||
hist_err("action parsing: Handler doesn't support action: ", action_name);
|
||||
ret = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
data->track_data.save_data = save_track_data_vars;
|
||||
data->fn = ontrack_action;
|
||||
data->action = ACTION_SAVE;
|
||||
} else {
|
||||
char *params = strsep(&str, ")");
|
||||
|
@ -3582,7 +3655,15 @@ static int action_parse(char *str, struct action_data *data,
|
|||
goto out;
|
||||
}
|
||||
|
||||
data->fn = action_trace;
|
||||
if (handler == HANDLER_ONMAX)
|
||||
data->track_data.check_val = check_track_val_max;
|
||||
|
||||
if (handler != HANDLER_ONMATCH) {
|
||||
data->track_data.save_data = action_trace;
|
||||
data->fn = ontrack_action;
|
||||
} else
|
||||
data->fn = action_trace;
|
||||
|
||||
data->action = ACTION_TRACE;
|
||||
}
|
||||
|
||||
|
@ -3597,24 +3678,25 @@ static int action_parse(char *str, struct action_data *data,
|
|||
return ret;
|
||||
}
|
||||
|
||||
static struct action_data *onmax_parse(char *str, enum handler_id handler)
|
||||
static struct action_data *track_data_parse(struct hist_trigger_data *hist_data,
|
||||
char *str, enum handler_id handler)
|
||||
{
|
||||
struct action_data *data;
|
||||
char *onmax_var_str;
|
||||
int ret = -EINVAL;
|
||||
char *var_str;
|
||||
|
||||
data = kzalloc(sizeof(*data), GFP_KERNEL);
|
||||
if (!data)
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
||||
onmax_var_str = strsep(&str, ")");
|
||||
if (!onmax_var_str || !str) {
|
||||
var_str = strsep(&str, ")");
|
||||
if (!var_str || !str) {
|
||||
ret = -EINVAL;
|
||||
goto free;
|
||||
}
|
||||
|
||||
data->onmax.var_str = kstrdup(onmax_var_str, GFP_KERNEL);
|
||||
if (!data->onmax.var_str) {
|
||||
data->track_data.var_str = kstrdup(var_str, GFP_KERNEL);
|
||||
if (!data->track_data.var_str) {
|
||||
ret = -ENOMEM;
|
||||
goto free;
|
||||
}
|
||||
|
@ -3625,7 +3707,7 @@ static struct action_data *onmax_parse(char *str, enum handler_id handler)
|
|||
out:
|
||||
return data;
|
||||
free:
|
||||
onmax_destroy(data);
|
||||
track_data_destroy(hist_data, data);
|
||||
data = ERR_PTR(ret);
|
||||
goto out;
|
||||
}
|
||||
|
@ -4389,7 +4471,7 @@ static void destroy_actions(struct hist_trigger_data *hist_data)
|
|||
if (data->handler == HANDLER_ONMATCH)
|
||||
onmatch_destroy(data);
|
||||
else if (data->handler == HANDLER_ONMAX)
|
||||
onmax_destroy(data);
|
||||
track_data_destroy(hist_data, data);
|
||||
else
|
||||
kfree(data);
|
||||
}
|
||||
|
@ -4418,7 +4500,8 @@ static int parse_actions(struct hist_trigger_data *hist_data)
|
|||
} else if ((len = str_has_prefix(str, "onmax("))) {
|
||||
char *action_str = str + len;
|
||||
|
||||
data = onmax_parse(action_str, HANDLER_ONMAX);
|
||||
data = track_data_parse(hist_data, action_str,
|
||||
HANDLER_ONMAX);
|
||||
if (IS_ERR(data)) {
|
||||
ret = PTR_ERR(data);
|
||||
break;
|
||||
|
@ -4448,7 +4531,7 @@ static int create_actions(struct hist_trigger_data *hist_data)
|
|||
if (ret)
|
||||
break;
|
||||
} else if (data->handler == HANDLER_ONMAX) {
|
||||
ret = onmax_create(hist_data, data);
|
||||
ret = track_data_create(hist_data, data);
|
||||
if (ret)
|
||||
break;
|
||||
} else {
|
||||
|
@ -4470,7 +4553,7 @@ static void print_actions(struct seq_file *m,
|
|||
struct action_data *data = hist_data->actions[i];
|
||||
|
||||
if (data->handler == HANDLER_ONMAX)
|
||||
onmax_print(m, hist_data, elt, data);
|
||||
track_data_print(m, hist_data, elt, data);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -4495,12 +4578,13 @@ static void print_action_spec(struct seq_file *m,
|
|||
}
|
||||
}
|
||||
|
||||
static void print_onmax_spec(struct seq_file *m,
|
||||
struct hist_trigger_data *hist_data,
|
||||
struct action_data *data)
|
||||
static void print_track_data_spec(struct seq_file *m,
|
||||
struct hist_trigger_data *hist_data,
|
||||
struct action_data *data)
|
||||
{
|
||||
seq_puts(m, ":onmax(");
|
||||
seq_printf(m, "%s", data->onmax.var_str);
|
||||
if (data->handler == HANDLER_ONMAX)
|
||||
seq_puts(m, ":onmax(");
|
||||
seq_printf(m, "%s", data->track_data.var_str);
|
||||
seq_printf(m, ").%s(", data->action_name);
|
||||
|
||||
print_action_spec(m, hist_data, data);
|
||||
|
@ -4558,8 +4642,8 @@ static bool actions_match(struct hist_trigger_data *hist_data,
|
|||
data_test->match_data.event) != 0)
|
||||
return false;
|
||||
} else if (data->handler == HANDLER_ONMAX) {
|
||||
if (strcmp(data->onmax.var_str,
|
||||
data_test->onmax.var_str) != 0)
|
||||
if (strcmp(data->track_data.var_str,
|
||||
data_test->track_data.var_str) != 0)
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
@ -4579,7 +4663,7 @@ static void print_actions_spec(struct seq_file *m,
|
|||
if (data->handler == HANDLER_ONMATCH)
|
||||
print_onmatch_spec(m, hist_data, data);
|
||||
else if (data->handler == HANDLER_ONMAX)
|
||||
print_onmax_spec(m, hist_data, data);
|
||||
print_track_data_spec(m, hist_data, data);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
Загрузка…
Ссылка в новой задаче