Commit 2b62b3a6 authored by Ian Rogers's avatar Ian Rogers Committed by Arnaldo Carvalho de Melo

perf parse-events: Add new "metric-id" term

Add a new "metric-id" term to events so that metric parsing can set an
ID that can be reliably looked up.

Metric parsing currently will turn a metric like "instructions/cycles"
into a parse events string of "{instructions,cycles}:W".

However, parse-events may change "instructions" into "instructions:u" if
perf_event_paranoid=2.

When this happens expr__resolve_id currently fails as stat-shadow adds
the ID "instructions:u" to match with the counter value and the metric
tries to look up the ID just "instructions".

A later patch will use the new term.

An example of the current problem:

  $ echo -1 > /proc/sys/kernel/perf_event_paranoid
  $ perf stat -M IPC /bin/true
   Performance counter stats for '/bin/true':

           1,217,161      inst_retired.any          #     0.97 IPC
           1,250,389      cpu_clk_unhalted.thread

         0.002064773 seconds time elapsed

         0.002378000 seconds user
         0.000000000 seconds sys

  $ echo 2 > /proc/sys/kernel/perf_event_paranoid
  $ perf stat -M IPC /bin/true
   Performance counter stats for '/bin/true':

             150,298      inst_retired.any:u        #      nan IPC
             187,095      cpu_clk_unhalted.thread:u

         0.002042731 seconds time elapsed

         0.000000000 seconds user
         0.002377000 seconds sys

Note: nan IPC is printed as an effect of "perf metric: Use NAN for
missing event IDs." but earlier versions of perf just fail with a parse
error and display no value.
Signed-off-by: default avatarIan Rogers <irogers@google.com>
Acked-by: default avatarAndi Kleen <ak@linux.intel.com>
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: Alexander Antonov <alexander.antonov@linux.intel.com>
Cc: Alexander Shishkin <alexander.shishkin@linux.intel.com>
Cc: Andrew Kilroy <andrew.kilroy@arm.com>
Cc: Andrew Morton <akpm@linux-foundation.org>
Cc: Changbin Du <changbin.du@intel.com>
Cc: Denys Zagorui <dzagorui@cisco.com>
Cc: Fabian Hemmer <copy@copy.sh>
Cc: Felix Fietkau <nbd@nbd.name>
Cc: Heiko Carstens <hca@linux.ibm.com>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Jacob Keller <jacob.e.keller@intel.com>
Cc: Jiapeng Chong <jiapeng.chong@linux.alibaba.com>
Cc: Jin Yao <yao.jin@linux.intel.com>
Cc: Jiri Olsa <jolsa@redhat.com>
Cc: Joakim Zhang <qiangqing.zhang@nxp.com>
Cc: John Garry <john.garry@huawei.com>
Cc: Kajol Jain <kjain@linux.ibm.com>
Cc: Kan Liang <kan.liang@linux.intel.com>
Cc: Kees Kook <keescook@chromium.org>
Cc: Mark Rutland <mark.rutland@arm.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Nicholas Fraser <nfraser@codeweavers.com>
Cc: Nick Desaulniers <ndesaulniers@google.com>
Cc: Paul Clarke <pc@us.ibm.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Riccardo Mancini <rickyman7@gmail.com>
Cc: Sami Tolvanen <samitolvanen@google.com>
Cc: ShihCheng Tu <mrtoastcheng@gmail.com>
Cc: Song Liu <songliubraving@fb.com>
Cc: Stephane Eranian <eranian@google.com>
Cc: Sumanth Korikkar <sumanthk@linux.ibm.com>
Cc: Thomas Richter <tmricht@linux.ibm.com>
Cc: Wan Jiabing <wanjiabing@vivo.com>
Cc: Zhen Lei <thunder.leizhen@huawei.com>
Link: https://lore.kernel.org/r/20211015172132.1162559-15-irogers@google.comSigned-off-by: default avatarArnaldo Carvalho de Melo <acme@redhat.com>
parent 8e8bbfb3
...@@ -410,6 +410,11 @@ struct evsel *evsel__clone(struct evsel *orig) ...@@ -410,6 +410,11 @@ struct evsel *evsel__clone(struct evsel *orig)
if (evsel->filter == NULL) if (evsel->filter == NULL)
goto out_err; goto out_err;
} }
if (orig->metric_id) {
evsel->metric_id = strdup(orig->metric_id);
if (evsel->metric_id == NULL)
goto out_err;
}
evsel->cgrp = cgroup__get(orig->cgrp); evsel->cgrp = cgroup__get(orig->cgrp);
evsel->tp_format = orig->tp_format; evsel->tp_format = orig->tp_format;
evsel->handler = orig->handler; evsel->handler = orig->handler;
...@@ -779,6 +784,17 @@ const char *evsel__name(struct evsel *evsel) ...@@ -779,6 +784,17 @@ const char *evsel__name(struct evsel *evsel)
return "unknown"; return "unknown";
} }
const char *evsel__metric_id(const struct evsel *evsel)
{
if (evsel->metric_id)
return evsel->metric_id;
if (evsel->core.attr.type == PERF_TYPE_SOFTWARE && evsel->tool_event)
return "duration_time";
return "unknown";
}
const char *evsel__group_name(struct evsel *evsel) const char *evsel__group_name(struct evsel *evsel)
{ {
return evsel->group_name ?: "anon group"; return evsel->group_name ?: "anon group";
...@@ -1423,6 +1439,7 @@ void evsel__exit(struct evsel *evsel) ...@@ -1423,6 +1439,7 @@ void evsel__exit(struct evsel *evsel)
zfree(&evsel->group_name); zfree(&evsel->group_name);
zfree(&evsel->name); zfree(&evsel->name);
zfree(&evsel->pmu_name); zfree(&evsel->pmu_name);
zfree(&evsel->metric_id);
evsel__zero_per_pkg(evsel); evsel__zero_per_pkg(evsel);
hashmap__free(evsel->per_pkg_mask); hashmap__free(evsel->per_pkg_mask);
evsel->per_pkg_mask = NULL; evsel->per_pkg_mask = NULL;
......
...@@ -68,6 +68,7 @@ struct evsel { ...@@ -68,6 +68,7 @@ struct evsel {
double scale; double scale;
const char *unit; const char *unit;
struct cgroup *cgrp; struct cgroup *cgrp;
const char *metric_id;
enum perf_tool_event tool_event; enum perf_tool_event tool_event;
/* parse modifier helper */ /* parse modifier helper */
int exclude_GH; int exclude_GH;
...@@ -261,6 +262,7 @@ bool evsel__match_bpf_counter_events(const char *name); ...@@ -261,6 +262,7 @@ bool evsel__match_bpf_counter_events(const char *name);
int __evsel__hw_cache_type_op_res_name(u8 type, u8 op, u8 result, char *bf, size_t size); int __evsel__hw_cache_type_op_res_name(u8 type, u8 op, u8 result, char *bf, size_t size);
const char *evsel__name(struct evsel *evsel); const char *evsel__name(struct evsel *evsel);
const char *evsel__metric_id(const struct evsel *evsel);
const char *evsel__group_name(struct evsel *evsel); const char *evsel__group_name(struct evsel *evsel);
int evsel__group_desc(struct evsel *evsel, char *buf, size_t size); int evsel__group_desc(struct evsel *evsel, char *buf, size_t size);
......
...@@ -39,6 +39,7 @@ static void config_hybrid_attr(struct perf_event_attr *attr, ...@@ -39,6 +39,7 @@ static void config_hybrid_attr(struct perf_event_attr *attr,
static int create_event_hybrid(__u32 config_type, int *idx, static int create_event_hybrid(__u32 config_type, int *idx,
struct list_head *list, struct list_head *list,
struct perf_event_attr *attr, const char *name, struct perf_event_attr *attr, const char *name,
const char *metric_id,
struct list_head *config_terms, struct list_head *config_terms,
struct perf_pmu *pmu) struct perf_pmu *pmu)
{ {
...@@ -47,7 +48,7 @@ static int create_event_hybrid(__u32 config_type, int *idx, ...@@ -47,7 +48,7 @@ static int create_event_hybrid(__u32 config_type, int *idx,
__u64 config = attr->config; __u64 config = attr->config;
config_hybrid_attr(attr, config_type, pmu->type); config_hybrid_attr(attr, config_type, pmu->type);
evsel = parse_events__add_event_hybrid(list, idx, attr, name, evsel = parse_events__add_event_hybrid(list, idx, attr, name, metric_id,
pmu, config_terms); pmu, config_terms);
if (evsel) if (evsel)
evsel->pmu_name = strdup(pmu->name); evsel->pmu_name = strdup(pmu->name);
...@@ -70,7 +71,8 @@ static int pmu_cmp(struct parse_events_state *parse_state, ...@@ -70,7 +71,8 @@ static int pmu_cmp(struct parse_events_state *parse_state,
static int add_hw_hybrid(struct parse_events_state *parse_state, static int add_hw_hybrid(struct parse_events_state *parse_state,
struct list_head *list, struct perf_event_attr *attr, struct list_head *list, struct perf_event_attr *attr,
const char *name, struct list_head *config_terms) const char *name, const char *metric_id,
struct list_head *config_terms)
{ {
struct perf_pmu *pmu; struct perf_pmu *pmu;
int ret; int ret;
...@@ -84,7 +86,7 @@ static int add_hw_hybrid(struct parse_events_state *parse_state, ...@@ -84,7 +86,7 @@ static int add_hw_hybrid(struct parse_events_state *parse_state,
copy_config_terms(&terms, config_terms); copy_config_terms(&terms, config_terms);
ret = create_event_hybrid(PERF_TYPE_HARDWARE, ret = create_event_hybrid(PERF_TYPE_HARDWARE,
&parse_state->idx, list, attr, name, &parse_state->idx, list, attr, name,
&terms, pmu); metric_id, &terms, pmu);
free_config_terms(&terms); free_config_terms(&terms);
if (ret) if (ret)
return ret; return ret;
...@@ -96,13 +98,14 @@ static int add_hw_hybrid(struct parse_events_state *parse_state, ...@@ -96,13 +98,14 @@ static int add_hw_hybrid(struct parse_events_state *parse_state,
static int create_raw_event_hybrid(int *idx, struct list_head *list, static int create_raw_event_hybrid(int *idx, struct list_head *list,
struct perf_event_attr *attr, struct perf_event_attr *attr,
const char *name, const char *name,
const char *metric_id,
struct list_head *config_terms, struct list_head *config_terms,
struct perf_pmu *pmu) struct perf_pmu *pmu)
{ {
struct evsel *evsel; struct evsel *evsel;
attr->type = pmu->type; attr->type = pmu->type;
evsel = parse_events__add_event_hybrid(list, idx, attr, name, evsel = parse_events__add_event_hybrid(list, idx, attr, name, metric_id,
pmu, config_terms); pmu, config_terms);
if (evsel) if (evsel)
evsel->pmu_name = strdup(pmu->name); evsel->pmu_name = strdup(pmu->name);
...@@ -114,7 +117,8 @@ static int create_raw_event_hybrid(int *idx, struct list_head *list, ...@@ -114,7 +117,8 @@ static int create_raw_event_hybrid(int *idx, struct list_head *list,
static int add_raw_hybrid(struct parse_events_state *parse_state, static int add_raw_hybrid(struct parse_events_state *parse_state,
struct list_head *list, struct perf_event_attr *attr, struct list_head *list, struct perf_event_attr *attr,
const char *name, struct list_head *config_terms) const char *name, const char *metric_id,
struct list_head *config_terms)
{ {
struct perf_pmu *pmu; struct perf_pmu *pmu;
int ret; int ret;
...@@ -127,7 +131,7 @@ static int add_raw_hybrid(struct parse_events_state *parse_state, ...@@ -127,7 +131,7 @@ static int add_raw_hybrid(struct parse_events_state *parse_state,
copy_config_terms(&terms, config_terms); copy_config_terms(&terms, config_terms);
ret = create_raw_event_hybrid(&parse_state->idx, list, attr, ret = create_raw_event_hybrid(&parse_state->idx, list, attr,
name, &terms, pmu); name, metric_id, &terms, pmu);
free_config_terms(&terms); free_config_terms(&terms);
if (ret) if (ret)
return ret; return ret;
...@@ -139,7 +143,7 @@ static int add_raw_hybrid(struct parse_events_state *parse_state, ...@@ -139,7 +143,7 @@ static int add_raw_hybrid(struct parse_events_state *parse_state,
int parse_events__add_numeric_hybrid(struct parse_events_state *parse_state, int parse_events__add_numeric_hybrid(struct parse_events_state *parse_state,
struct list_head *list, struct list_head *list,
struct perf_event_attr *attr, struct perf_event_attr *attr,
const char *name, const char *name, const char *metric_id,
struct list_head *config_terms, struct list_head *config_terms,
bool *hybrid) bool *hybrid)
{ {
...@@ -152,17 +156,18 @@ int parse_events__add_numeric_hybrid(struct parse_events_state *parse_state, ...@@ -152,17 +156,18 @@ int parse_events__add_numeric_hybrid(struct parse_events_state *parse_state,
*hybrid = true; *hybrid = true;
if (attr->type != PERF_TYPE_RAW) { if (attr->type != PERF_TYPE_RAW) {
return add_hw_hybrid(parse_state, list, attr, name, return add_hw_hybrid(parse_state, list, attr, name, metric_id,
config_terms); config_terms);
} }
return add_raw_hybrid(parse_state, list, attr, name, return add_raw_hybrid(parse_state, list, attr, name, metric_id,
config_terms); config_terms);
} }
int parse_events__add_cache_hybrid(struct list_head *list, int *idx, int parse_events__add_cache_hybrid(struct list_head *list, int *idx,
struct perf_event_attr *attr, struct perf_event_attr *attr,
const char *name, const char *name,
const char *metric_id,
struct list_head *config_terms, struct list_head *config_terms,
bool *hybrid, bool *hybrid,
struct parse_events_state *parse_state) struct parse_events_state *parse_state)
...@@ -183,7 +188,7 @@ int parse_events__add_cache_hybrid(struct list_head *list, int *idx, ...@@ -183,7 +188,7 @@ int parse_events__add_cache_hybrid(struct list_head *list, int *idx,
copy_config_terms(&terms, config_terms); copy_config_terms(&terms, config_terms);
ret = create_event_hybrid(PERF_TYPE_HW_CACHE, idx, list, ret = create_event_hybrid(PERF_TYPE_HW_CACHE, idx, list,
attr, name, &terms, pmu); attr, name, metric_id, &terms, pmu);
free_config_terms(&terms); free_config_terms(&terms);
if (ret) if (ret)
return ret; return ret;
......
...@@ -11,13 +11,13 @@ ...@@ -11,13 +11,13 @@
int parse_events__add_numeric_hybrid(struct parse_events_state *parse_state, int parse_events__add_numeric_hybrid(struct parse_events_state *parse_state,
struct list_head *list, struct list_head *list,
struct perf_event_attr *attr, struct perf_event_attr *attr,
const char *name, const char *name, const char *metric_id,
struct list_head *config_terms, struct list_head *config_terms,
bool *hybrid); bool *hybrid);
int parse_events__add_cache_hybrid(struct list_head *list, int *idx, int parse_events__add_cache_hybrid(struct list_head *list, int *idx,
struct perf_event_attr *attr, struct perf_event_attr *attr,
const char *name, const char *name, const char *metric_id,
struct list_head *config_terms, struct list_head *config_terms,
bool *hybrid, bool *hybrid,
struct parse_events_state *parse_state); struct parse_events_state *parse_state);
......
...@@ -332,12 +332,7 @@ const char *event_type(int type) ...@@ -332,12 +332,7 @@ const char *event_type(int type)
return "unknown"; return "unknown";
} }
static int parse_events__is_name_term(struct parse_events_term *term) static char *get_config_str(struct list_head *head_terms, int type_term)
{
return term->type_term == PARSE_EVENTS__TERM_TYPE_NAME;
}
static const char *get_config_name(struct list_head *head_terms)
{ {
struct parse_events_term *term; struct parse_events_term *term;
...@@ -345,17 +340,27 @@ static const char *get_config_name(struct list_head *head_terms) ...@@ -345,17 +340,27 @@ static const char *get_config_name(struct list_head *head_terms)
return NULL; return NULL;
list_for_each_entry(term, head_terms, list) list_for_each_entry(term, head_terms, list)
if (parse_events__is_name_term(term)) if (term->type_term == type_term)
return term->val.str; return term->val.str;
return NULL; return NULL;
} }
static char *get_config_metric_id(struct list_head *head_terms)
{
return get_config_str(head_terms, PARSE_EVENTS__TERM_TYPE_METRIC_ID);
}
static char *get_config_name(struct list_head *head_terms)
{
return get_config_str(head_terms, PARSE_EVENTS__TERM_TYPE_NAME);
}
static struct evsel * static struct 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,
bool init_attr, bool init_attr,
const char *name, struct perf_pmu *pmu, const char *name, const char *metric_id, struct perf_pmu *pmu,
struct list_head *config_terms, bool auto_merge_stats, struct list_head *config_terms, bool auto_merge_stats,
const char *cpu_list) const char *cpu_list)
{ {
...@@ -384,6 +389,9 @@ __add_event(struct list_head *list, int *idx, ...@@ -384,6 +389,9 @@ __add_event(struct list_head *list, int *idx,
if (name) if (name)
evsel->name = strdup(name); evsel->name = strdup(name);
if (metric_id)
evsel->metric_id = strdup(metric_id);
if (config_terms) if (config_terms)
list_splice_init(config_terms, &evsel->config_terms); list_splice_init(config_terms, &evsel->config_terms);
...@@ -394,18 +402,21 @@ __add_event(struct list_head *list, int *idx, ...@@ -394,18 +402,21 @@ __add_event(struct list_head *list, int *idx,
} }
struct evsel *parse_events__add_event(int idx, struct perf_event_attr *attr, struct evsel *parse_events__add_event(int idx, struct perf_event_attr *attr,
const char *name, struct perf_pmu *pmu) const char *name, const char *metric_id,
struct perf_pmu *pmu)
{ {
return __add_event(NULL, &idx, attr, false, name, pmu, NULL, false, return __add_event(/*list=*/NULL, &idx, attr, /*init_attr=*/false, name,
NULL); metric_id, pmu, /*config_terms=*/NULL,
/*auto_merge_stats=*/false, /*cpu_list=*/NULL);
} }
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, const char *name, struct perf_event_attr *attr, const char *name,
struct list_head *config_terms) const char *metric_id, struct list_head *config_terms)
{ {
return __add_event(list, idx, attr, true, name, NULL, config_terms, return __add_event(list, idx, attr, /*init_attr*/true, name, metric_id,
false, NULL) ? 0 : -ENOMEM; /*pmu=*/NULL, config_terms,
/*auto_merge_stats=*/false, /*cpu_list=*/NULL) ? 0 : -ENOMEM;
} }
static int add_event_tool(struct list_head *list, int *idx, static int add_event_tool(struct list_head *list, int *idx,
...@@ -417,8 +428,10 @@ static int add_event_tool(struct list_head *list, int *idx, ...@@ -417,8 +428,10 @@ static int add_event_tool(struct list_head *list, int *idx,
.config = PERF_COUNT_SW_DUMMY, .config = PERF_COUNT_SW_DUMMY,
}; };
evsel = __add_event(list, idx, &attr, true, NULL, NULL, NULL, false, evsel = __add_event(list, idx, &attr, /*init_attr=*/true, /*name=*/NULL,
"0"); /*metric_id=*/NULL, /*pmu=*/NULL,
/*config_terms=*/NULL, /*auto_merge_stats=*/false,
/*cpu_list=*/"0");
if (!evsel) if (!evsel)
return -ENOMEM; return -ENOMEM;
evsel->tool_event = tool_event; evsel->tool_event = tool_event;
...@@ -465,7 +478,7 @@ int parse_events_add_cache(struct list_head *list, int *idx, ...@@ -465,7 +478,7 @@ int parse_events_add_cache(struct list_head *list, int *idx,
struct perf_event_attr attr; struct perf_event_attr attr;
LIST_HEAD(config_terms); LIST_HEAD(config_terms);
char name[MAX_NAME_LEN]; char name[MAX_NAME_LEN];
const char *config_name; const char *config_name, *metric_id;
int cache_type = -1, cache_op = -1, cache_result = -1; int cache_type = -1, cache_op = -1, cache_result = -1;
char *op_result[2] = { op_result1, op_result2 }; char *op_result[2] = { op_result1, op_result2 };
int i, n, ret; int i, n, ret;
...@@ -530,13 +543,17 @@ int parse_events_add_cache(struct list_head *list, int *idx, ...@@ -530,13 +543,17 @@ int parse_events_add_cache(struct list_head *list, int *idx,
return -ENOMEM; return -ENOMEM;
} }
metric_id = get_config_metric_id(head_config);
ret = parse_events__add_cache_hybrid(list, idx, &attr, ret = parse_events__add_cache_hybrid(list, idx, &attr,
config_name ? : name, &config_terms, config_name ? : name,
metric_id,
&config_terms,
&hybrid, parse_state); &hybrid, parse_state);
if (hybrid) if (hybrid)
goto out_free_terms; goto out_free_terms;
ret = add_event(list, idx, &attr, config_name ? : name, &config_terms); ret = add_event(list, idx, &attr, config_name ? : name, metric_id,
&config_terms);
out_free_terms: out_free_terms:
free_config_terms(&config_terms); free_config_terms(&config_terms);
return ret; return ret;
...@@ -1013,7 +1030,8 @@ int parse_events_add_breakpoint(struct list_head *list, int *idx, ...@@ -1013,7 +1030,8 @@ 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, NULL); return add_event(list, idx, &attr, /*name=*/NULL, /*mertic_id=*/NULL,
/*config_terms=*/NULL);
} }
static int check_type_val(struct parse_events_term *term, static int check_type_val(struct parse_events_term *term,
...@@ -1058,6 +1076,7 @@ static const char *config_term_names[__PARSE_EVENTS__TERM_TYPE_NR] = { ...@@ -1058,6 +1076,7 @@ static const char *config_term_names[__PARSE_EVENTS__TERM_TYPE_NR] = {
[PARSE_EVENTS__TERM_TYPE_PERCORE] = "percore", [PARSE_EVENTS__TERM_TYPE_PERCORE] = "percore",
[PARSE_EVENTS__TERM_TYPE_AUX_OUTPUT] = "aux-output", [PARSE_EVENTS__TERM_TYPE_AUX_OUTPUT] = "aux-output",
[PARSE_EVENTS__TERM_TYPE_AUX_SAMPLE_SIZE] = "aux-sample-size", [PARSE_EVENTS__TERM_TYPE_AUX_SAMPLE_SIZE] = "aux-sample-size",
[PARSE_EVENTS__TERM_TYPE_METRIC_ID] = "metric-id",
}; };
static bool config_term_shrinked; static bool config_term_shrinked;
...@@ -1080,6 +1099,7 @@ config_term_avail(int term_type, struct parse_events_error *err) ...@@ -1080,6 +1099,7 @@ config_term_avail(int term_type, struct parse_events_error *err)
case PARSE_EVENTS__TERM_TYPE_CONFIG1: case PARSE_EVENTS__TERM_TYPE_CONFIG1:
case PARSE_EVENTS__TERM_TYPE_CONFIG2: case PARSE_EVENTS__TERM_TYPE_CONFIG2:
case PARSE_EVENTS__TERM_TYPE_NAME: case PARSE_EVENTS__TERM_TYPE_NAME:
case PARSE_EVENTS__TERM_TYPE_METRIC_ID:
case PARSE_EVENTS__TERM_TYPE_SAMPLE_PERIOD: case PARSE_EVENTS__TERM_TYPE_SAMPLE_PERIOD:
case PARSE_EVENTS__TERM_TYPE_PERCORE: case PARSE_EVENTS__TERM_TYPE_PERCORE:
return true; return true;
...@@ -1170,6 +1190,9 @@ do { \ ...@@ -1170,6 +1190,9 @@ do { \
case PARSE_EVENTS__TERM_TYPE_NAME: case PARSE_EVENTS__TERM_TYPE_NAME:
CHECK_TYPE_VAL(STR); CHECK_TYPE_VAL(STR);
break; break;
case PARSE_EVENTS__TERM_TYPE_METRIC_ID:
CHECK_TYPE_VAL(STR);
break;
case PARSE_EVENTS__TERM_TYPE_MAX_STACK: case PARSE_EVENTS__TERM_TYPE_MAX_STACK:
CHECK_TYPE_VAL(NUM); CHECK_TYPE_VAL(NUM);
break; break;
...@@ -1439,6 +1462,7 @@ int parse_events_add_numeric(struct parse_events_state *parse_state, ...@@ -1439,6 +1462,7 @@ int parse_events_add_numeric(struct parse_events_state *parse_state,
{ {
struct perf_event_attr attr; struct perf_event_attr attr;
LIST_HEAD(config_terms); LIST_HEAD(config_terms);
const char *name, *metric_id;
bool hybrid; bool hybrid;
int ret; int ret;
...@@ -1455,14 +1479,16 @@ int parse_events_add_numeric(struct parse_events_state *parse_state, ...@@ -1455,14 +1479,16 @@ int parse_events_add_numeric(struct parse_events_state *parse_state,
return -ENOMEM; return -ENOMEM;
} }
name = get_config_name(head_config);
metric_id = get_config_metric_id(head_config);
ret = parse_events__add_numeric_hybrid(parse_state, list, &attr, ret = parse_events__add_numeric_hybrid(parse_state, list, &attr,
get_config_name(head_config), name, metric_id,
&config_terms, &hybrid); &config_terms, &hybrid);
if (hybrid) if (hybrid)
goto out_free_terms; goto out_free_terms;
ret = add_event(list, &parse_state->idx, &attr, ret = add_event(list, &parse_state->idx, &attr, name, metric_id,
get_config_name(head_config), &config_terms); &config_terms);
out_free_terms: out_free_terms:
free_config_terms(&config_terms); free_config_terms(&config_terms);
return ret; return ret;
...@@ -1563,8 +1589,11 @@ int parse_events_add_pmu(struct parse_events_state *parse_state, ...@@ -1563,8 +1589,11 @@ int parse_events_add_pmu(struct parse_events_state *parse_state,
if (!head_config) { if (!head_config) {
attr.type = pmu->type; attr.type = pmu->type;
evsel = __add_event(list, &parse_state->idx, &attr, true, NULL, evsel = __add_event(list, &parse_state->idx, &attr,
pmu, NULL, auto_merge_stats, NULL); /*init_attr=*/true, /*name=*/NULL,
/*metric_id=*/NULL, pmu,
/*config_terms=*/NULL, auto_merge_stats,
/*cpu_list=*/NULL);
if (evsel) { if (evsel) {
evsel->pmu_name = name ? strdup(name) : NULL; evsel->pmu_name = name ? strdup(name) : NULL;
evsel->use_uncore_alias = use_uncore_alias; evsel->use_uncore_alias = use_uncore_alias;
...@@ -1617,9 +1646,10 @@ int parse_events_add_pmu(struct parse_events_state *parse_state, ...@@ -1617,9 +1646,10 @@ int parse_events_add_pmu(struct parse_events_state *parse_state,
return -EINVAL; return -EINVAL;
} }
evsel = __add_event(list, &parse_state->idx, &attr, true, evsel = __add_event(list, &parse_state->idx, &attr, /*init_attr=*/true,
get_config_name(head_config), pmu, get_config_name(head_config),
&config_terms, auto_merge_stats, NULL); get_config_metric_id(head_config), pmu,
&config_terms, auto_merge_stats, /*cpu_list=*/NULL);
if (!evsel) if (!evsel)
return -ENOMEM; return -ENOMEM;
...@@ -3345,9 +3375,12 @@ char *parse_events_formats_error_string(char *additional_terms) ...@@ -3345,9 +3375,12 @@ char *parse_events_formats_error_string(char *additional_terms)
struct evsel *parse_events__add_event_hybrid(struct list_head *list, int *idx, struct evsel *parse_events__add_event_hybrid(struct list_head *list, int *idx,
struct perf_event_attr *attr, struct perf_event_attr *attr,
const char *name, struct perf_pmu *pmu, const char *name,
const char *metric_id,
struct perf_pmu *pmu,
struct list_head *config_terms) struct list_head *config_terms)
{ {
return __add_event(list, idx, attr, true, name, pmu, return __add_event(list, idx, attr, /*init_attr=*/true, name, metric_id,
config_terms, false, NULL); pmu, config_terms, /*auto_merge_stats=*/false,
/*cpu_list=*/NULL);
} }
...@@ -87,6 +87,7 @@ enum { ...@@ -87,6 +87,7 @@ enum {
PARSE_EVENTS__TERM_TYPE_PERCORE, PARSE_EVENTS__TERM_TYPE_PERCORE,
PARSE_EVENTS__TERM_TYPE_AUX_OUTPUT, PARSE_EVENTS__TERM_TYPE_AUX_OUTPUT,
PARSE_EVENTS__TERM_TYPE_AUX_SAMPLE_SIZE, PARSE_EVENTS__TERM_TYPE_AUX_SAMPLE_SIZE,
PARSE_EVENTS__TERM_TYPE_METRIC_ID,
__PARSE_EVENTS__TERM_TYPE_NR, __PARSE_EVENTS__TERM_TYPE_NR,
}; };
...@@ -199,7 +200,8 @@ int parse_events_add_pmu(struct parse_events_state *parse_state, ...@@ -199,7 +200,8 @@ int parse_events_add_pmu(struct parse_events_state *parse_state,
bool use_alias); bool use_alias);
struct evsel *parse_events__add_event(int idx, struct perf_event_attr *attr, struct evsel *parse_events__add_event(int idx, struct perf_event_attr *attr,
const char *name, struct perf_pmu *pmu); const char *name, const char *metric_id,
struct perf_pmu *pmu);
int parse_events_multi_pmu_add(struct parse_events_state *parse_state, int parse_events_multi_pmu_add(struct parse_events_state *parse_state,
char *str, char *str,
...@@ -267,6 +269,7 @@ int perf_pmu__test_parse_init(void); ...@@ -267,6 +269,7 @@ int perf_pmu__test_parse_init(void);
struct evsel *parse_events__add_event_hybrid(struct list_head *list, int *idx, struct evsel *parse_events__add_event_hybrid(struct list_head *list, int *idx,
struct perf_event_attr *attr, struct perf_event_attr *attr,
const char *name, const char *name,
const char *metric_id,
struct perf_pmu *pmu, struct perf_pmu *pmu,
struct list_head *config_terms); struct list_head *config_terms);
......
...@@ -294,6 +294,7 @@ no-overwrite { return term(yyscanner, PARSE_EVENTS__TERM_TYPE_NOOVERWRITE); } ...@@ -294,6 +294,7 @@ no-overwrite { return term(yyscanner, PARSE_EVENTS__TERM_TYPE_NOOVERWRITE); }
percore { return term(yyscanner, PARSE_EVENTS__TERM_TYPE_PERCORE); } percore { return term(yyscanner, PARSE_EVENTS__TERM_TYPE_PERCORE); }
aux-output { return term(yyscanner, PARSE_EVENTS__TERM_TYPE_AUX_OUTPUT); } aux-output { return term(yyscanner, PARSE_EVENTS__TERM_TYPE_AUX_OUTPUT); }
aux-sample-size { return term(yyscanner, PARSE_EVENTS__TERM_TYPE_AUX_SAMPLE_SIZE); } aux-sample-size { return term(yyscanner, PARSE_EVENTS__TERM_TYPE_AUX_SAMPLE_SIZE); }
metric-id { return term(yyscanner, PARSE_EVENTS__TERM_TYPE_METRIC_ID); }
r{num_raw_hex} { return raw(yyscanner); } r{num_raw_hex} { return raw(yyscanner); }
r0x{num_raw_hex} { return raw(yyscanner); } r0x{num_raw_hex} { return raw(yyscanner); }
, { return ','; } , { return ','; }
......
...@@ -87,7 +87,8 @@ int parse_libpfm_events_option(const struct option *opt, const char *str, ...@@ -87,7 +87,8 @@ int parse_libpfm_events_option(const struct option *opt, const char *str,
pmu = perf_pmu__find_by_type((unsigned int)attr.type); pmu = perf_pmu__find_by_type((unsigned int)attr.type);
evsel = parse_events__add_event(evlist->core.nr_entries, evsel = parse_events__add_event(evlist->core.nr_entries,
&attr, q, pmu); &attr, q, /*metric_id=*/NULL,
pmu);
if (evsel == NULL) if (evsel == NULL)
goto error; goto error;
......
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