Re: [PATCH v3 19/33] tracing: Add variable reference handling to hist triggers

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Fri, Sep 22, 2017 at 02:59:59PM -0500, Tom Zanussi wrote:
> Add the necessary infrastructure to allow the variables defined on one
> event to be referenced in another.  This allows variables set by a
> previous event to be referenced and used in expressions combining the
> variable values saved by that previous event and the event fields of
> the current event.  For example, here's how a latency can be
> calculated and saved into yet another variable named 'wakeup_lat':
> 
>     # echo 'hist:keys=pid,prio:ts0=common_timestamp ...
>     # echo 'hist:keys=next_pid:wakeup_lat=common_timestamp-$ts0 ...
> 
> In the first event, the event's timetamp is saved into the variable
> ts0.  In the next line, ts0 is subtracted from the second event's
> timestamp to produce the latency.
> 
> Further users of variable references will be described in subsequent
> patches, such as for instance how the 'wakeup_lat' variable above can
> be displayed in a latency histogram.
> 
> Signed-off-by: Tom Zanussi <tom.zanussi@xxxxxxxxxxxxxxx>
> ---
>  kernel/trace/trace.c                |   2 +
>  kernel/trace/trace.h                |   3 +
>  kernel/trace/trace_events_hist.c    | 613 ++++++++++++++++++++++++++++++++----
>  kernel/trace/trace_events_trigger.c |   6 +
>  4 files changed, 568 insertions(+), 56 deletions(-)
> 
> diff --git a/kernel/trace/trace.c b/kernel/trace/trace.c
> index 19ffbe1..5eb313a 100644
> --- a/kernel/trace/trace.c
> +++ b/kernel/trace/trace.c
> @@ -7753,6 +7753,7 @@ static int instance_mkdir(const char *name)
>  
>  	INIT_LIST_HEAD(&tr->systems);
>  	INIT_LIST_HEAD(&tr->events);
> +	INIT_LIST_HEAD(&tr->hist_vars);
>  
>  	if (allocate_trace_buffers(tr, trace_buf_size) < 0)
>  		goto out_free_tr;
> @@ -8500,6 +8501,7 @@ __init static int tracer_alloc_buffers(void)
>  
>  	INIT_LIST_HEAD(&global_trace.systems);
>  	INIT_LIST_HEAD(&global_trace.events);
> +	INIT_LIST_HEAD(&global_trace.hist_vars);
>  	list_add(&global_trace.list, &ftrace_trace_arrays);
>  
>  	apply_trace_boot_options();
> diff --git a/kernel/trace/trace.h b/kernel/trace/trace.h
> index 02bfd5c..7b78762 100644
> --- a/kernel/trace/trace.h
> +++ b/kernel/trace/trace.h
> @@ -273,6 +273,7 @@ struct trace_array {
>  	int			function_enabled;
>  #endif
>  	int			time_stamp_abs_ref;
> +	struct list_head	hist_vars;
>  };
>  
>  enum {
> @@ -1547,6 +1548,8 @@ extern int save_named_trigger(const char *name,
>  extern void unpause_named_trigger(struct event_trigger_data *data);
>  extern void set_named_trigger_data(struct event_trigger_data *data,
>  				   struct event_trigger_data *named_data);
> +extern struct event_trigger_data *
> +get_named_trigger_data(struct event_trigger_data *data);
>  extern int register_event_command(struct event_command *cmd);
>  extern int unregister_event_command(struct event_command *cmd);
>  extern int register_trigger_hist_enable_disable_cmds(void);
> diff --git a/kernel/trace/trace_events_hist.c b/kernel/trace/trace_events_hist.c
> index 02170df..6dcef22 100644
> --- a/kernel/trace/trace_events_hist.c
> +++ b/kernel/trace/trace_events_hist.c
> @@ -60,6 +60,9 @@ struct hist_field {
>  	struct hist_var			var;
>  	enum field_op_id		operator;
>  	char				*name;
> +	unsigned int			var_idx;
> +	unsigned int			var_ref_idx;
> +	bool                            read_once;
>  };
>  
>  static u64 hist_field_none(struct hist_field *field,
> @@ -215,6 +218,7 @@ enum hist_field_flags {
>  	HIST_FIELD_FL_VAR		= 1 << 12,
>  	HIST_FIELD_FL_VAR_ONLY		= 1 << 13,
>  	HIST_FIELD_FL_EXPR		= 1 << 14,
> +	HIST_FIELD_FL_VAR_REF		= 1 << 15,
>  };
>  
>  struct var_defs {
> @@ -255,6 +259,8 @@ struct hist_trigger_data {
>  	struct tracing_map		*map;
>  	bool				enable_timestamps;
>  	bool				remove;
> +	struct hist_field               *var_refs[TRACING_MAP_VARS_MAX];
> +	unsigned int			n_var_refs;
>  };
>  
>  static u64 hist_field_timestamp(struct hist_field *hist_field,
> @@ -273,10 +279,344 @@ static u64 hist_field_timestamp(struct hist_field *hist_field,
>  	return ts;
>  }
>  
> +struct hist_var_data {
> +	struct list_head list;
> +	struct hist_trigger_data *hist_data;
> +};
> +
> +static struct hist_field *check_var_ref(struct hist_field *hist_field,
> +					struct hist_trigger_data *var_data,
> +					unsigned int var_idx)
> +{
> +	struct hist_field *found = NULL;
> +
> +	if (hist_field && hist_field->flags & HIST_FIELD_FL_VAR_REF) {
> +		if (hist_field->var.idx == var_idx &&
> +		    hist_field->var.hist_data == var_data) {
> +			found = hist_field;
> +		}
> +	}
> +
> +	return found;
> +}
> +
> +static struct hist_field *find_var_ref(struct hist_trigger_data *hist_data,
> +				       struct hist_trigger_data *var_data,
> +				       unsigned int var_idx)
> +{
> +	struct hist_field *hist_field, *found = NULL;
> +	unsigned int i, j;
> +
> +	for_each_hist_field(i, hist_data) {
> +		hist_field = hist_data->fields[i];
> +		found = check_var_ref(hist_field, var_data, var_idx);
> +		if (found)
> +			return found;
> +
> +		for (j = 0; j < HIST_FIELD_OPERANDS_MAX; j++) {
> +			struct hist_field *operand;
> +
> +			operand = hist_field->operands[j];
> +			found = check_var_ref(operand, var_data, var_idx);
> +			if (found)
> +				return found;
> +		}

Hmm.. IIUC expression can be nested but it seems to miss var-ref at
level 2 - something like "a + b + $c", no?


> +	}
> +
> +	return found;
> +}
> +
> +static struct hist_field *find_any_var_ref(struct hist_trigger_data *hist_data,
> +					   unsigned int var_idx)
> +{
> +	struct trace_array *tr = hist_data->event_file->tr;
> +	struct hist_field *found = NULL;
> +	struct hist_var_data *var_data;
> +
> +	list_for_each_entry(var_data, &tr->hist_vars, list) {
> +		found = find_var_ref(var_data->hist_data, hist_data, var_idx);
> +		if (found)
> +			break;

Shouldn't it check self-references - i.e. ref from the same hist_data?


> +	}
> +
> +	return found;
> +}
> +
> +static bool check_var_refs(struct hist_trigger_data *hist_data)
> +{
> +	struct hist_field *field;
> +	bool found = false;
> +	int i;
> +
> +	for_each_hist_field(i, hist_data) {
> +		field = hist_data->fields[i];
> +		if (field && field->flags & HIST_FIELD_FL_VAR) {
> +			if (find_any_var_ref(hist_data, field->var.idx)) {
> +				found = true;
> +				break;
> +			}
> +		}
> +	}
> +
> +	return found;
> +}
> +
> +static struct hist_var_data *find_hist_vars(struct hist_trigger_data *hist_data)
> +{
> +	struct trace_array *tr = hist_data->event_file->tr;
> +	struct hist_var_data *var_data, *found = NULL;
> +
> +	list_for_each_entry(var_data, &tr->hist_vars, list) {
> +		if (var_data->hist_data == hist_data) {
> +			found = var_data;
> +			break;
> +		}
> +	}
> +
> +	return found;
> +}
> +
> +static bool has_hist_vars(struct hist_trigger_data *hist_data)
> +{
> +	struct hist_field *hist_field;
> +	int i, j;
> +
> +	for_each_hist_field(i, hist_data) {
> +		hist_field = hist_data->fields[i];
> +		if (hist_field &&
> +		    (hist_field->flags & HIST_FIELD_FL_VAR ||
> +		     hist_field->flags & HIST_FIELD_FL_VAR_REF))
> +			return true;
> +
> +		for (j = 0; j < HIST_FIELD_OPERANDS_MAX; j++) {
> +			struct hist_field *operand;
> +
> +			operand = hist_field->operands[j];
> +			if (operand &&
> +			    (operand->flags & HIST_FIELD_FL_VAR ||
> +			     operand->flags & HIST_FIELD_FL_VAR_REF))
> +				return true;

Same with the above.

Thanks,
Namhyung


> +		}
> +	}
> +
> +	return false;
> +}
> +
--
To unsubscribe from this list: send the line "unsubscribe linux-rt-users" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [RT Stable]     [Kernel Newbies]     [IDE]     [Security]     [Git]     [Netfilter]     [Bugtraq]     [Yosemite]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux ATA RAID]     [Samba]     [Video 4 Linux]     [Device Mapper]

  Powered by Linux