Received: by 2002:a25:ca44:0:0:0:0:0 with SMTP id a65csp967786ybg; Wed, 29 Jul 2020 02:20:55 -0700 (PDT) X-Google-Smtp-Source: ABdhPJx516p2v7/Rvx3HP1XYiyos2BnUhU6lIw3HSnISi1MD+wLALDpTxfCsQpnKnbDQmSp8S3eH X-Received: by 2002:a05:6402:28f:: with SMTP id l15mr17453911edv.233.1596014455768; Wed, 29 Jul 2020 02:20:55 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1596014455; cv=none; d=google.com; s=arc-20160816; b=wFRXjz6cmUFvSo+ITfAFwE94OIaTAD0RwPks3Sdx6ZZyXiCZXqIwUNfRLQ+fv2pQ5q ctTvly/kSgZ/KojFE9TWT592c9gh0o7twU6nq6QUyxfdAn8OoLDw59r4tppGhDiYp6q3 wu7ImTFh/cfsp+g9s0iEMxu52RLA4Bk/J667N2dD+yhv0U6f2Y7PSA1B+DqNq0t1w0Ak 5ga+xqESM68SjXgB+w7EcvN5e4XMiCDRXtkm4bVLJp1z+YeZoIo7L9cLcL3vQrc//NLn FULhO8OkB0PbxclJIrjAhdkAc5OJ42hjrWQrEKIX+/0CVFDETch+1/M4Pio0R2IGcvnt NTIg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from; bh=sxQkR/lyQMyoOIrR1sRlvGw0N2VaitezyC+CRB7H3RY=; b=C3Rm7cP4V1GVfQzuw6LTg59wZCWQHqVbCCF9iZXG4gTjpugw8h4LIfW2JYNNWk6SXX ++GzcnMc02faTiFYZ/0kOd5zITO95Z+uoi3IBM9CUJJcssGTjAE6pgXX/YHvqtr30aA5 uT75yQtPixkJwpeUyn0OF4DS9wjDbj/aw/5djZTdnuqAl7CiaYOrBrl7dpCRyx7Y2zPH elEk5tDmnC9yj0Pyhy8SWLAy4qkvBYEgztk2j6FCLOi6C1pndalYM0ip0zTbt+MZtHG2 iO6M7ClC6bY7CO1BDwwRRleo/bKGMnB3YBy4Imfuoqb7eJ+2kZoFpKrmO7sD2rlFX8nz LJrw== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id g20si935639edh.258.2020.07.29.02.20.33; Wed, 29 Jul 2020 02:20:55 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1728255AbgG2JTz convert rfc822-to-8bit (ORCPT + 99 others); Wed, 29 Jul 2020 05:19:55 -0400 Received: from us-smtp-1.mimecast.com ([205.139.110.61]:40525 "EHLO us-smtp-delivery-1.mimecast.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S1727794AbgG2JTy (ORCPT ); Wed, 29 Jul 2020 05:19:54 -0400 Received: from mimecast-mx01.redhat.com (mimecast-mx01.redhat.com [209.132.183.4]) (Using TLS) by relay.mimecast.com with ESMTP id us-mta-24-Xedtg9E9PECK-Wgv5OogaA-1; Wed, 29 Jul 2020 05:19:46 -0400 X-MC-Unique: Xedtg9E9PECK-Wgv5OogaA-1 Received: from smtp.corp.redhat.com (int-mx06.intmail.prod.int.phx2.redhat.com [10.5.11.16]) (using TLSv1.2 with cipher AECDH-AES256-SHA (256/256 bits)) (No client certificate requested) by mimecast-mx01.redhat.com (Postfix) with ESMTPS id 5476118C63C5; Wed, 29 Jul 2020 09:19:44 +0000 (UTC) Received: from krava.redhat.com (unknown [10.40.193.247]) by smtp.corp.redhat.com (Postfix) with ESMTP id 0C1E069332; Wed, 29 Jul 2020 09:19:40 +0000 (UTC) From: Jiri Olsa To: Arnaldo Carvalho de Melo Cc: Alexander Shishkin , Andi Kleen , John Garry , Michael Petlan , Namhyung Kim , Paul Clarke , Peter Zijlstra , Stephane Eranian , lkml , Ingo Molnar , Peter Zijlstra , Kajol Jain , Ian Rogers Subject: [PATCH 08/19] perf metric: Collect referenced metrics in struct metric_ref_node Date: Wed, 29 Jul 2020 11:18:57 +0200 Message-Id: <20200729091908.1378911-9-jolsa@kernel.org> In-Reply-To: <20200729091908.1378911-1-jolsa@kernel.org> References: <20200729091908.1378911-1-jolsa@kernel.org> MIME-Version: 1.0 X-Scanned-By: MIMEDefang 2.79 on 10.5.11.16 Authentication-Results: relay.mimecast.com; auth=pass smtp.auth=CUSA124A263 smtp.mailfrom=jolsa@kernel.org X-Mimecast-Spam-Score: 0 X-Mimecast-Originator: kernel.org Content-Type: text/plain; charset=WINDOWS-1252 Content-Transfer-Encoding: 8BIT Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Collecting referenced metrics in struct metric_ref_node object, so we can process them later on. The change will parse nested metric names out of expression and 'resolve' them. All referenced metrics are dissolved into one context, meaning all nested metrics events and added to the parent context. Signed-off-by: Jiri Olsa Cc: Alexander Shishkin Cc: Andi Kleen Cc: John Garry Cc: Michael Petlan Cc: Namhyung Kim Cc: Paul Clarke Cc: Peter Zijlstra Cc: Stephane Eranian Link: http://lore.kernel.org/lkml/20200719181320.785305-9-jolsa@kernel.org Signed-off-by: Arnaldo Carvalho de Melo --- tools/perf/util/metricgroup.c | 170 +++++++++++++++++++++++++++++++--- 1 file changed, 156 insertions(+), 14 deletions(-) diff --git a/tools/perf/util/metricgroup.c b/tools/perf/util/metricgroup.c index ccd80538a6ae..0997df4e4f52 100644 --- a/tools/perf/util/metricgroup.c +++ b/tools/perf/util/metricgroup.c @@ -102,12 +102,25 @@ void metricgroup__rblist_exit(struct rblist *metric_events) rblist__exit(metric_events); } +/* + * A node in the list of referenced metrics. metric_expr + * is held as a convenience to avoid a search through the + * metric list. + */ +struct metric_ref_node { + const char *metric_name; + const char *metric_expr; + struct list_head list; +}; + struct egroup { struct list_head nd; struct expr_parse_ctx pctx; const char *metric_name; const char *metric_expr; const char *metric_unit; + struct list_head metric_refs; + int metric_refs_cnt; int runtime; bool has_constraint; }; @@ -574,27 +587,72 @@ int __weak arch_get_runtimeparam(void) static int __add_metric(struct list_head *group_list, struct pmu_event *pe, bool metric_no_group, - int runtime) + int runtime, + struct egroup **egp) { + struct metric_ref_node *ref; struct egroup *eg; - eg = malloc(sizeof(*eg)); - if (!eg) - return -ENOMEM; + if (*egp == NULL) { + /* + * We got in here for the parent group, + * allocate it and put it on the list. + */ + eg = malloc(sizeof(*eg)); + if (!eg) + return -ENOMEM; + + expr__ctx_init(&eg->pctx); + eg->metric_name = pe->metric_name; + eg->metric_expr = pe->metric_expr; + eg->metric_unit = pe->unit; + eg->runtime = runtime; + eg->has_constraint = metric_no_group || metricgroup__has_constraint(pe); + INIT_LIST_HEAD(&eg->metric_refs); + eg->metric_refs_cnt = 0; + *egp = eg; + } else { + /* + * We got here for the referenced metric, via the + * recursive metricgroup__add_metric call, add + * it to the parent group. + */ + eg = *egp; + + ref = malloc(sizeof(*ref)); + if (!ref) + return -ENOMEM; + + /* + * Intentionally passing just const char pointers, + * from 'pe' object, so they never go away. We don't + * need to change them, so there's no need to create + * our own copy. + */ + ref->metric_name = pe->metric_name; + ref->metric_expr = pe->metric_expr; - expr__ctx_init(&eg->pctx); - eg->metric_name = pe->metric_name; - eg->metric_expr = pe->metric_expr; - eg->metric_unit = pe->unit; - eg->runtime = runtime; - eg->has_constraint = metric_no_group || metricgroup__has_constraint(pe); + list_add(&ref->list, &eg->metric_refs); + eg->metric_refs_cnt++; + } + /* + * For both the parent and referenced metrics, we parse + * all the metric's IDs and add it to the parent context. + */ if (expr__find_other(pe->metric_expr, NULL, &eg->pctx, runtime) < 0) { expr__ctx_clear(&eg->pctx); free(eg); return -EINVAL; } + /* + * We add new group only in the 'parent' call, + * so bail out for referenced metric case. + */ + if (eg->metric_refs_cnt) + return 0; + if (list_empty(group_list)) list_add(&eg->nd, group_list); else { @@ -625,16 +683,78 @@ static int __add_metric(struct list_head *group_list, (match_metric(__pe->metric_group, __metric) || \ match_metric(__pe->metric_name, __metric))) +static struct pmu_event *find_metric(const char *metric, struct pmu_events_map *map) +{ + struct pmu_event *pe; + int i; + + map_for_each_event(pe, i, map) { + if (match_metric(pe->metric_name, metric)) + return pe; + } + + return NULL; +} + +static int add_metric(struct list_head *group_list, + struct pmu_event *pe, + bool metric_no_group, + struct egroup **egp); + +static int resolve_metric(struct egroup *eg, + bool metric_no_group, + struct list_head *group_list, + struct pmu_events_map *map) +{ + struct hashmap_entry *cur; + size_t bkt; + bool all; + int ret; + + /* + * Iterate all the parsed IDs and if there's metric, + * add it to the context. + */ + do { + all = true; + hashmap__for_each_entry((&eg->pctx.ids), cur, bkt) { + struct pmu_event *pe; + + pe = find_metric(cur->key, map); + if (!pe) + continue; + + all = false; + /* The metric key itself needs to go out.. */ + expr__del_id(&eg->pctx, cur->key); + + /* ... and it gets resolved to the parent context. */ + ret = add_metric(group_list, pe, metric_no_group, &eg); + if (ret) + return ret; + + /* + * We added new metric to hashmap, so we need + * to break the iteration and start over. + */ + break; + } + } while (!all); + + return 0; +} + static int add_metric(struct list_head *group_list, struct pmu_event *pe, - bool metric_no_group) + bool metric_no_group, + struct egroup **egp) { int ret = 0; pr_debug("metric expr %s for %s\n", pe->metric_expr, pe->metric_name); if (!strstr(pe->metric_expr, "?")) { - ret = __add_metric(group_list, pe, metric_no_group, 1); + ret = __add_metric(group_list, pe, metric_no_group, 1, egp); } else { int j, count; @@ -646,7 +766,7 @@ static int add_metric(struct list_head *group_list, */ for (j = 0; j < count && !ret; j++) { - ret = __add_metric(group_list, pe, metric_no_group, j); + ret = __add_metric(group_list, pe, metric_no_group, j, egp); } } @@ -657,6 +777,7 @@ static int metricgroup__add_metric(const char *metric, bool metric_no_group, struct strbuf *events, struct list_head *group_list, struct pmu_events_map *map) + { struct pmu_event *pe; struct egroup *eg; @@ -665,8 +786,18 @@ static int metricgroup__add_metric(const char *metric, bool metric_no_group, map_for_each_metric(pe, i, map, metric) { has_match = true; + eg = NULL; - ret = add_metric(group_list, pe, metric_no_group); + ret = add_metric(group_list, pe, metric_no_group, &eg); + if (ret) + return ret; + + /* + * Process any possible referenced metrics + * included in the expression. + */ + ret = resolve_metric(eg, metric_no_group, + group_list, map); if (ret) return ret; } @@ -723,11 +854,22 @@ static int metricgroup__add_metric_list(const char *list, bool metric_no_group, return ret; } +static void egroup__free_refs(struct egroup *egroup) +{ + struct metric_ref_node *ref, *tmp; + + list_for_each_entry_safe(ref, tmp, &egroup->metric_refs, list) { + list_del(&ref->list); + free(ref); + } +} + static void metricgroup__free_egroups(struct list_head *group_list) { struct egroup *eg, *egtmp; list_for_each_entry_safe (eg, egtmp, group_list, nd) { + egroup__free_refs(eg); expr__ctx_clear(&eg->pctx); list_del_init(&eg->nd); free(eg); -- 2.25.4