perf tools: Add support for event post configuration

Add support to overload any global settings for event and force user
specified term value. It will be useful for new time and backtrace
terms.

Signed-off-by: Jiri Olsa <jolsa@kernel.org>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Kan Liang <kan.liang@intel.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Link: http://lkml.kernel.org/r/1438162936-59698-2-git-send-email-kan.liang@intel.com
Signed-off-by: Kan Liang <kan.liang@intel.com>
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
This commit is contained in:
Jiri Olsa 2015-07-29 05:42:10 -04:00 committed by Arnaldo Carvalho de Melo
parent 4c7de49a29
commit 930a2e2975
3 changed files with 106 additions and 11 deletions

View File

@ -207,6 +207,7 @@ void perf_evsel__init(struct perf_evsel *evsel,
evsel->unit = ""; evsel->unit = "";
evsel->scale = 1.0; evsel->scale = 1.0;
INIT_LIST_HEAD(&evsel->node); INIT_LIST_HEAD(&evsel->node);
INIT_LIST_HEAD(&evsel->config_terms);
perf_evsel__object.init(evsel); perf_evsel__object.init(evsel);
evsel->sample_size = __perf_evsel__sample_size(attr->sample_type); evsel->sample_size = __perf_evsel__sample_size(attr->sample_type);
perf_evsel__calc_id_pos(evsel); perf_evsel__calc_id_pos(evsel);
@ -586,6 +587,19 @@ perf_evsel__config_callgraph(struct perf_evsel *evsel,
} }
} }
static void apply_config_terms(struct perf_event_attr *attr __maybe_unused,
struct list_head *config_terms)
{
struct perf_evsel_config_term *term;
list_for_each_entry(term, config_terms, list) {
switch (term->type) {
default:
break;
}
}
}
/* /*
* The enable_on_exec/disabled value strategy: * The enable_on_exec/disabled value strategy:
* *
@ -777,6 +791,12 @@ void perf_evsel__config(struct perf_evsel *evsel, struct record_opts *opts)
attr->use_clockid = 1; attr->use_clockid = 1;
attr->clockid = opts->clockid; attr->clockid = opts->clockid;
} }
/*
* Apply event specific term settings,
* it overloads any global configuration.
*/
apply_config_terms(attr, &evsel->config_terms);
} }
static int perf_evsel__alloc_fd(struct perf_evsel *evsel, int ncpus, int nthreads) static int perf_evsel__alloc_fd(struct perf_evsel *evsel, int ncpus, int nthreads)
@ -900,6 +920,16 @@ static void perf_evsel__free_id(struct perf_evsel *evsel)
zfree(&evsel->id); zfree(&evsel->id);
} }
static void perf_evsel__free_config_terms(struct perf_evsel *evsel)
{
struct perf_evsel_config_term *term, *h;
list_for_each_entry_safe(term, h, &evsel->config_terms, list) {
list_del(&term->list);
free(term);
}
}
void perf_evsel__close_fd(struct perf_evsel *evsel, int ncpus, int nthreads) void perf_evsel__close_fd(struct perf_evsel *evsel, int ncpus, int nthreads)
{ {
int cpu, thread; int cpu, thread;
@ -919,6 +949,7 @@ void perf_evsel__exit(struct perf_evsel *evsel)
assert(list_empty(&evsel->node)); assert(list_empty(&evsel->node));
perf_evsel__free_fd(evsel); perf_evsel__free_fd(evsel);
perf_evsel__free_id(evsel); perf_evsel__free_id(evsel);
perf_evsel__free_config_terms(evsel);
close_cgroup(evsel->cgrp); close_cgroup(evsel->cgrp);
cpu_map__put(evsel->cpus); cpu_map__put(evsel->cpus);
thread_map__put(evsel->threads); thread_map__put(evsel->threads);

View File

@ -31,6 +31,24 @@ struct perf_sample_id {
struct cgroup_sel; struct cgroup_sel;
/*
* The 'struct perf_evsel_config_term' is used to pass event
* specific configuration data to perf_evsel__config routine.
* It is allocated within event parsing and attached to
* perf_evsel::config_terms list head.
*/
enum {
PERF_EVSEL__CONFIG_TERM_MAX,
};
struct perf_evsel_config_term {
struct list_head list;
int type;
union {
u64 period;
} val;
};
/** struct perf_evsel - event selector /** struct perf_evsel - event selector
* *
* @name - Can be set to retain the original event name passed by the user, * @name - Can be set to retain the original event name passed by the user,
@ -87,6 +105,7 @@ struct perf_evsel {
struct perf_evsel *leader; struct perf_evsel *leader;
char *group_name; char *group_name;
bool cmdline_group_boundary; bool cmdline_group_boundary;
struct list_head config_terms;
}; };
union u64_swap { union u64_swap {

View File

@ -276,7 +276,8 @@ const char *event_type(int type)
static struct perf_evsel * static struct perf_evsel *
__add_event(struct list_head *list, int *idx, __add_event(struct list_head *list, int *idx,
struct perf_event_attr *attr, struct perf_event_attr *attr,
char *name, struct cpu_map *cpus) char *name, struct cpu_map *cpus,
struct list_head *config_terms)
{ {
struct perf_evsel *evsel; struct perf_evsel *evsel;
@ -291,14 +292,19 @@ __add_event(struct list_head *list, int *idx,
if (name) if (name)
evsel->name = strdup(name); evsel->name = strdup(name);
if (config_terms)
list_splice(config_terms, &evsel->config_terms);
list_add_tail(&evsel->node, list); list_add_tail(&evsel->node, list);
return evsel; return evsel;
} }
static int add_event(struct list_head *list, int *idx, static int add_event(struct list_head *list, int *idx,
struct perf_event_attr *attr, char *name) struct perf_event_attr *attr, char *name,
struct list_head *config_terms)
{ {
return __add_event(list, idx, attr, name, NULL) ? 0 : -ENOMEM; return __add_event(list, idx, attr, name, NULL, config_terms) ? 0 : -ENOMEM;
} }
static int parse_aliases(char *str, const char *names[][PERF_EVSEL__MAX_ALIASES], int size) static int parse_aliases(char *str, const char *names[][PERF_EVSEL__MAX_ALIASES], int size)
@ -377,7 +383,7 @@ int parse_events_add_cache(struct list_head *list, int *idx,
memset(&attr, 0, sizeof(attr)); memset(&attr, 0, sizeof(attr));
attr.config = cache_type | (cache_op << 8) | (cache_result << 16); attr.config = cache_type | (cache_op << 8) | (cache_result << 16);
attr.type = PERF_TYPE_HW_CACHE; attr.type = PERF_TYPE_HW_CACHE;
return add_event(list, idx, &attr, name); return add_event(list, idx, &attr, name, NULL);
} }
static int add_tracepoint(struct list_head *list, int *idx, static int add_tracepoint(struct list_head *list, int *idx,
@ -539,7 +545,7 @@ int parse_events_add_breakpoint(struct list_head *list, int *idx,
attr.type = PERF_TYPE_BREAKPOINT; attr.type = PERF_TYPE_BREAKPOINT;
attr.sample_period = 1; attr.sample_period = 1;
return add_event(list, idx, &attr, NULL); return add_event(list, idx, &attr, NULL, NULL);
} }
static int check_type_val(struct parse_events_term *term, static int check_type_val(struct parse_events_term *term,
@ -622,22 +628,56 @@ static int config_attr(struct perf_event_attr *attr,
return 0; return 0;
} }
static int get_config_terms(struct list_head *head_config,
struct list_head *head_terms __maybe_unused)
{
#define ADD_CONFIG_TERM(__type, __name, __val) \
do { \
struct perf_evsel_config_term *__t; \
\
__t = zalloc(sizeof(*__t)); \
if (!__t) \
return -ENOMEM; \
\
INIT_LIST_HEAD(&__t->list); \
__t->type = PERF_EVSEL__CONFIG_TERM_ ## __type; \
__t->val.__name = __val; \
list_add_tail(&__t->list, head_terms); \
} while (0)
struct parse_events_term *term;
list_for_each_entry(term, head_config, list) {
switch (term->type_term) {
default:
break;
}
}
#undef ADD_EVSEL_CONFIG
return 0;
}
int parse_events_add_numeric(struct parse_events_evlist *data, int parse_events_add_numeric(struct parse_events_evlist *data,
struct list_head *list, struct list_head *list,
u32 type, u64 config, u32 type, u64 config,
struct list_head *head_config) struct list_head *head_config)
{ {
struct perf_event_attr attr; struct perf_event_attr attr;
LIST_HEAD(config_terms);
memset(&attr, 0, sizeof(attr)); memset(&attr, 0, sizeof(attr));
attr.type = type; attr.type = type;
attr.config = config; attr.config = config;
if (head_config && if (head_config) {
config_attr(&attr, head_config, data->error)) if (config_attr(&attr, head_config, data->error))
return -EINVAL; return -EINVAL;
return add_event(list, &data->idx, &attr, NULL); if (get_config_terms(head_config, &config_terms))
return -ENOMEM;
}
return add_event(list, &data->idx, &attr, NULL, &config_terms);
} }
static int parse_events__is_name_term(struct parse_events_term *term) static int parse_events__is_name_term(struct parse_events_term *term)
@ -664,6 +704,7 @@ int parse_events_add_pmu(struct parse_events_evlist *data,
struct perf_pmu_info info; struct perf_pmu_info info;
struct perf_pmu *pmu; struct perf_pmu *pmu;
struct perf_evsel *evsel; struct perf_evsel *evsel;
LIST_HEAD(config_terms);
pmu = perf_pmu__find(name); pmu = perf_pmu__find(name);
if (!pmu) if (!pmu)
@ -678,7 +719,7 @@ int parse_events_add_pmu(struct parse_events_evlist *data,
if (!head_config) { if (!head_config) {
attr.type = pmu->type; attr.type = pmu->type;
evsel = __add_event(list, &data->idx, &attr, NULL, pmu->cpus); evsel = __add_event(list, &data->idx, &attr, NULL, pmu->cpus, NULL);
return evsel ? 0 : -ENOMEM; return evsel ? 0 : -ENOMEM;
} }
@ -692,11 +733,15 @@ int parse_events_add_pmu(struct parse_events_evlist *data,
if (config_attr(&attr, head_config, data->error)) if (config_attr(&attr, head_config, data->error))
return -EINVAL; return -EINVAL;
if (get_config_terms(head_config, &config_terms))
return -ENOMEM;
if (perf_pmu__config(pmu, &attr, head_config, data->error)) if (perf_pmu__config(pmu, &attr, head_config, data->error))
return -EINVAL; return -EINVAL;
evsel = __add_event(list, &data->idx, &attr, evsel = __add_event(list, &data->idx, &attr,
pmu_event_name(head_config), pmu->cpus); pmu_event_name(head_config), pmu->cpus,
&config_terms);
if (evsel) { if (evsel) {
evsel->unit = info.unit; evsel->unit = info.unit;
evsel->scale = info.scale; evsel->scale = info.scale;