From: Arnaldo Carvalho de Melo <acme@kernel.org>
To: James Clark <james.clark@linaro.org>
Cc: Peter Zijlstra <peterz@infradead.org>,
Ingo Molnar <mingo@redhat.com>,
Namhyung Kim <namhyung@kernel.org>,
Mark Rutland <mark.rutland@arm.com>,
Alexander Shishkin <alexander.shishkin@linux.intel.com>,
Jiri Olsa <jolsa@kernel.org>, Ian Rogers <irogers@google.com>,
Adrian Hunter <adrian.hunter@intel.com>,
Suzuki K Poulose <suzuki.poulose@arm.com>,
Mike Leach <mike.leach@linaro.org>,
John Garry <john.g.garry@oracle.com>,
Will Deacon <will@kernel.org>, Leo Yan <leo.yan@linux.dev>,
linux-perf-users@vger.kernel.org, linux-kernel@vger.kernel.org,
coresight@lists.linaro.org, linux-arm-kernel@lists.infradead.org
Subject: Re: [PATCH v4 01/14] perf parse-events: Refactor get_config_terms() to remove macros
Date: Tue, 13 Jan 2026 17:53:13 -0300 [thread overview]
Message-ID: <aWawuXJU0BVE4q-v@x1> (raw)
In-Reply-To: <20251222-james-perf-config-bits-v4-1-0608438186fc@linaro.org>
On Mon, Dec 22, 2025 at 03:14:26PM +0000, James Clark wrote:
> The ADD_CONFIG_TERM() macros build the __type argument out of a partial
> EVSEL__CONFIG_TERM_x enum name. This means that they can't be called
> from a function where __type is a variable and it's also impossible to
> grep the codebase to find usages of these enums as they're never typed
> in full.
>
> Fix this by removing the macros and replacing them with an
> add_config_term() function. It seems the main reason these existed in
> the first place was to avoid type punning and to write to a specific
> field in the union, but the same thing can be achieved with a single
> write to a u64 'val' field.
>
> Running the Perf tests with "-fsanitize=undefined -fno-sanitize-recover"
> results in no new issues as a result of this change.
>
> Reviewed-by: Ian Rogers <irogers@google.com>
So I see the tags here, will try applying...
- Arnaldo
> Signed-off-by: James Clark <james.clark@linaro.org>
> ---
> tools/perf/util/evsel_config.h | 1 +
> tools/perf/util/parse-events.c | 146 ++++++++++++++++++++++++-----------------
> 2 files changed, 86 insertions(+), 61 deletions(-)
>
> diff --git a/tools/perf/util/evsel_config.h b/tools/perf/util/evsel_config.h
> index bcd3a978f0c4..685fd8d5c4a8 100644
> --- a/tools/perf/util/evsel_config.h
> +++ b/tools/perf/util/evsel_config.h
> @@ -50,6 +50,7 @@ struct evsel_config_term {
> u64 cfg_chg;
> char *str;
> int cpu;
> + u64 val;
> } val;
> bool weak;
> };
> diff --git a/tools/perf/util/parse-events.c b/tools/perf/util/parse-events.c
> index 17c1c36a7bf9..46422286380f 100644
> --- a/tools/perf/util/parse-events.c
> +++ b/tools/perf/util/parse-events.c
> @@ -1116,105 +1116,107 @@ static int config_attr(struct perf_event_attr *attr,
> return 0;
> }
>
> -static int get_config_terms(const struct parse_events_terms *head_config,
> - struct list_head *head_terms)
> +static struct evsel_config_term *add_config_term(enum evsel_term_type type,
> + struct list_head *head_terms,
> + bool weak)
> {
> -#define ADD_CONFIG_TERM(__type, __weak) \
> - struct evsel_config_term *__t; \
> - \
> - __t = zalloc(sizeof(*__t)); \
> - if (!__t) \
> - return -ENOMEM; \
> - \
> - INIT_LIST_HEAD(&__t->list); \
> - __t->type = EVSEL__CONFIG_TERM_ ## __type; \
> - __t->weak = __weak; \
> - list_add_tail(&__t->list, head_terms)
> -
> -#define ADD_CONFIG_TERM_VAL(__type, __name, __val, __weak) \
> -do { \
> - ADD_CONFIG_TERM(__type, __weak); \
> - __t->val.__name = __val; \
> -} while (0)
> + struct evsel_config_term *t;
>
> -#define ADD_CONFIG_TERM_STR(__type, __val, __weak) \
> -do { \
> - ADD_CONFIG_TERM(__type, __weak); \
> - __t->val.str = strdup(__val); \
> - if (!__t->val.str) { \
> - zfree(&__t); \
> - return -ENOMEM; \
> - } \
> - __t->free_str = true; \
> -} while (0)
> + t = zalloc(sizeof(*t));
> + if (!t)
> + return NULL;
> +
> + INIT_LIST_HEAD(&t->list);
> + t->type = type;
> + t->weak = weak;
> + list_add_tail(&t->list, head_terms);
>
> + return t;
> +}
> +
> +static int get_config_terms(const struct parse_events_terms *head_config,
> + struct list_head *head_terms)
> +{
> struct parse_events_term *term;
>
> list_for_each_entry(term, &head_config->terms, list) {
> + struct evsel_config_term *new_term;
> + enum evsel_term_type new_type;
> + bool str_type = false;
> + u64 val;
> +
> switch (term->type_term) {
> case PARSE_EVENTS__TERM_TYPE_SAMPLE_PERIOD:
> - ADD_CONFIG_TERM_VAL(PERIOD, period, term->val.num, term->weak);
> + new_type = EVSEL__CONFIG_TERM_PERIOD;
> + val = term->val.num;
> break;
> case PARSE_EVENTS__TERM_TYPE_SAMPLE_FREQ:
> - ADD_CONFIG_TERM_VAL(FREQ, freq, term->val.num, term->weak);
> + new_type = EVSEL__CONFIG_TERM_FREQ;
> + val = term->val.num;
> break;
> case PARSE_EVENTS__TERM_TYPE_TIME:
> - ADD_CONFIG_TERM_VAL(TIME, time, term->val.num, term->weak);
> + new_type = EVSEL__CONFIG_TERM_TIME;
> + val = term->val.num;
> break;
> case PARSE_EVENTS__TERM_TYPE_CALLGRAPH:
> - ADD_CONFIG_TERM_STR(CALLGRAPH, term->val.str, term->weak);
> + new_type = EVSEL__CONFIG_TERM_CALLGRAPH;
> + str_type = true;
> break;
> case PARSE_EVENTS__TERM_TYPE_BRANCH_SAMPLE_TYPE:
> - ADD_CONFIG_TERM_STR(BRANCH, term->val.str, term->weak);
> + new_type = EVSEL__CONFIG_TERM_BRANCH;
> + str_type = true;
> break;
> case PARSE_EVENTS__TERM_TYPE_STACKSIZE:
> - ADD_CONFIG_TERM_VAL(STACK_USER, stack_user,
> - term->val.num, term->weak);
> + new_type = EVSEL__CONFIG_TERM_STACK_USER;
> + val = term->val.num;
> break;
> case PARSE_EVENTS__TERM_TYPE_INHERIT:
> - ADD_CONFIG_TERM_VAL(INHERIT, inherit,
> - term->val.num ? 1 : 0, term->weak);
> + new_type = EVSEL__CONFIG_TERM_INHERIT;
> + val = term->val.num ? 1 : 0;
> break;
> case PARSE_EVENTS__TERM_TYPE_NOINHERIT:
> - ADD_CONFIG_TERM_VAL(INHERIT, inherit,
> - term->val.num ? 0 : 1, term->weak);
> + new_type = EVSEL__CONFIG_TERM_INHERIT;
> + val = term->val.num ? 0 : 1;
> break;
> case PARSE_EVENTS__TERM_TYPE_MAX_STACK:
> - ADD_CONFIG_TERM_VAL(MAX_STACK, max_stack,
> - term->val.num, term->weak);
> + new_type = EVSEL__CONFIG_TERM_MAX_STACK;
> + val = term->val.num;
> break;
> case PARSE_EVENTS__TERM_TYPE_MAX_EVENTS:
> - ADD_CONFIG_TERM_VAL(MAX_EVENTS, max_events,
> - term->val.num, term->weak);
> + new_type = EVSEL__CONFIG_TERM_MAX_EVENTS;
> + val = term->val.num;
> break;
> case PARSE_EVENTS__TERM_TYPE_OVERWRITE:
> - ADD_CONFIG_TERM_VAL(OVERWRITE, overwrite,
> - term->val.num ? 1 : 0, term->weak);
> + new_type = EVSEL__CONFIG_TERM_OVERWRITE;
> + val = term->val.num ? 1 : 0;
> break;
> case PARSE_EVENTS__TERM_TYPE_NOOVERWRITE:
> - ADD_CONFIG_TERM_VAL(OVERWRITE, overwrite,
> - term->val.num ? 0 : 1, term->weak);
> + new_type = EVSEL__CONFIG_TERM_OVERWRITE;
> + val = term->val.num ? 0 : 1;
> break;
> case PARSE_EVENTS__TERM_TYPE_DRV_CFG:
> - ADD_CONFIG_TERM_STR(DRV_CFG, term->val.str, term->weak);
> + new_type = EVSEL__CONFIG_TERM_DRV_CFG;
> + str_type = true;
> break;
> case PARSE_EVENTS__TERM_TYPE_PERCORE:
> - ADD_CONFIG_TERM_VAL(PERCORE, percore,
> - term->val.num ? true : false, term->weak);
> + new_type = EVSEL__CONFIG_TERM_PERCORE;
> + val = term->val.num ? true : false;
> break;
> case PARSE_EVENTS__TERM_TYPE_AUX_OUTPUT:
> - ADD_CONFIG_TERM_VAL(AUX_OUTPUT, aux_output,
> - term->val.num ? 1 : 0, term->weak);
> + new_type = EVSEL__CONFIG_TERM_AUX_OUTPUT;
> + val = term->val.num ? 1 : 0;
> break;
> case PARSE_EVENTS__TERM_TYPE_AUX_ACTION:
> - ADD_CONFIG_TERM_STR(AUX_ACTION, term->val.str, term->weak);
> + new_type = EVSEL__CONFIG_TERM_AUX_ACTION;
> + str_type = true;
> break;
> case PARSE_EVENTS__TERM_TYPE_AUX_SAMPLE_SIZE:
> - ADD_CONFIG_TERM_VAL(AUX_SAMPLE_SIZE, aux_sample_size,
> - term->val.num, term->weak);
> + new_type = EVSEL__CONFIG_TERM_AUX_SAMPLE_SIZE;
> + val = term->val.num;
> break;
> case PARSE_EVENTS__TERM_TYPE_RATIO_TO_PREV:
> - ADD_CONFIG_TERM_STR(RATIO_TO_PREV, term->val.str, term->weak);
> + new_type = EVSEL__CONFIG_TERM_RATIO_TO_PREV;
> + str_type = true;
> break;
> case PARSE_EVENTS__TERM_TYPE_USER:
> case PARSE_EVENTS__TERM_TYPE_CONFIG:
> @@ -1229,7 +1231,23 @@ do { \
> case PARSE_EVENTS__TERM_TYPE_RAW:
> case PARSE_EVENTS__TERM_TYPE_CPU:
> default:
> - break;
> + /* Don't add a new term for these ones */
> + continue;
> + }
> +
> + new_term = add_config_term(new_type, head_terms, term->weak);
> + if (!new_term)
> + return -ENOMEM;
> +
> + if (str_type) {
> + new_term->val.str = strdup(term->val.str);
> + if (!new_term->val.str) {
> + zfree(&new_term);
> + return -ENOMEM;
> + }
> + new_term->free_str = true;
> + } else {
> + new_term->val.val = val;
> }
> }
> return 0;
> @@ -1290,10 +1308,16 @@ static int get_config_chgs(struct perf_pmu *pmu, struct parse_events_terms *head
> }
> }
>
> - if (bits)
> - ADD_CONFIG_TERM_VAL(CFG_CHG, cfg_chg, bits, false);
> + if (bits) {
> + struct evsel_config_term *new_term;
> +
> + new_term = add_config_term(EVSEL__CONFIG_TERM_CFG_CHG,
> + head_terms, false);
> + if (!new_term)
> + return -ENOMEM;
> + new_term->val.cfg_chg = bits;
> + }
>
> -#undef ADD_CONFIG_TERM
> return 0;
> }
>
>
> --
> 2.34.1
>
next prev parent reply other threads:[~2026-01-13 20:53 UTC|newest]
Thread overview: 24+ messages / expand[flat|nested] mbox.gz Atom feed top
2025-12-22 15:14 [PATCH v4 00/14] perf cs-etm/arm-spe: Remove hard coded config fields James Clark
2025-12-22 15:14 ` [PATCH v4 01/14] perf parse-events: Refactor get_config_terms() to remove macros James Clark
2026-01-13 20:53 ` Arnaldo Carvalho de Melo [this message]
2025-12-22 15:14 ` [PATCH v4 02/14] perf evsel: Refactor evsel__set_config_if_unset() arguments James Clark
2026-01-13 22:13 ` Arnaldo Carvalho de Melo
2026-01-14 12:14 ` James Clark
2026-01-14 13:33 ` James Clark
2026-01-14 15:47 ` Arnaldo Carvalho de Melo
2026-01-14 15:58 ` James Clark
2025-12-22 15:14 ` [PATCH v4 03/14] perf evsel: Move evsel__* functions to evsel.c James Clark
2025-12-22 15:14 ` [PATCH v4 04/14] perf evsel: Support sparse fields in evsel__set_config_if_unset() James Clark
2025-12-22 15:14 ` [PATCH v4 05/14] perf parse-events: Track all user changed config bits James Clark
2025-12-22 15:14 ` [PATCH v4 06/14] perf evsel: apply evsel__set_config_if_unset() to all config fields James Clark
2025-12-22 15:14 ` [PATCH v4 07/14] perf evsel: Add a helper to get the value of a config field James Clark
2025-12-22 15:14 ` [PATCH v4 08/14] perf parse-events: Always track user config changes James Clark
2025-12-22 15:14 ` [PATCH v4 09/14] perf tests: Test evsel__set_config_if_unset() and config change tracking James Clark
2025-12-22 15:14 ` [PATCH v4 10/14] perf cs-etm: Make a helper to find the Coresight evsel James Clark
2025-12-22 15:14 ` [PATCH v4 11/14] perf cs-etm: Don't use hard coded config bits when setting up ETMCR James Clark
2025-12-22 15:14 ` [PATCH v4 12/14] perf cs-etm: Don't use hard coded config bits when setting up TRCCONFIGR James Clark
2025-12-22 15:14 ` [PATCH v4 13/14] perf cs-etm: Don't hard code config attribute when configuring the event James Clark
2025-12-22 15:14 ` [PATCH v4 14/14] perf arm-spe: Don't hard code config attribute James Clark
2026-01-13 20:58 ` [PATCH v4 00/14] perf cs-etm/arm-spe: Remove hard coded config fields Ian Rogers
2026-01-13 21:03 ` Arnaldo Carvalho de Melo
2026-01-13 21:01 ` Arnaldo Carvalho de Melo
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=aWawuXJU0BVE4q-v@x1 \
--to=acme@kernel.org \
--cc=adrian.hunter@intel.com \
--cc=alexander.shishkin@linux.intel.com \
--cc=coresight@lists.linaro.org \
--cc=irogers@google.com \
--cc=james.clark@linaro.org \
--cc=john.g.garry@oracle.com \
--cc=jolsa@kernel.org \
--cc=leo.yan@linux.dev \
--cc=linux-arm-kernel@lists.infradead.org \
--cc=linux-kernel@vger.kernel.org \
--cc=linux-perf-users@vger.kernel.org \
--cc=mark.rutland@arm.com \
--cc=mike.leach@linaro.org \
--cc=mingo@redhat.com \
--cc=namhyung@kernel.org \
--cc=peterz@infradead.org \
--cc=suzuki.poulose@arm.com \
--cc=will@kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.