Adding formula object to interface formula definitions like: set { events = {cycles,instructions,branch-instructions}:u
cpi { formula = cycles / instructions desc = cycles per instruction } branch-rate { formula = branch-instructions / instructions desc = branch rate } } The 'set' defines set of counter that share same events. Each 'set' defines: events - event string that would go into stat/record -e option counters - any number of counters based on above events Each counter (cpi/branch-rate) defines formula - formula with that produce the counter number event names and numbers could be used desc - text description of the counter Interface: perf_formula__init - initialize perf_formula handler perf_formula__load - load file into the handler perf_formula__free - cleanup perf_formula__set - get 'set' handler perf_formula__evlist - update perf_evlist with needed events perf_formula__print - display output ratios Signed-off-by: Jiri Olsa <jo...@redhat.com> Cc: Arnaldo Carvalho de Melo <a...@redhat.com> Cc: Namhyung Kim <namhy...@kernel.org> Cc: Corey Ashford <cjash...@linux.vnet.ibm.com> Cc: Frederic Weisbecker <fweis...@gmail.com> Cc: Ingo Molnar <mi...@elte.hu> Cc: Namhyung Kim <namhy...@kernel.org> Cc: Paul Mackerras <pau...@samba.org> Cc: Peter Zijlstra <a.p.zijls...@chello.nl> Cc: Andi Kleen <a...@firstfloor.org> Cc: David Ahern <dsah...@gmail.com> Cc: Ulrich Drepper <drep...@gmail.com> --- tools/perf/Makefile | 11 ++ tools/perf/util/evlist.c | 13 ++ tools/perf/util/evlist.h | 4 + tools/perf/util/formula.c | 387 ++++++++++++++++++++++++++++++++++++++++++++++ tools/perf/util/formula.h | 113 ++++++++++++++ tools/perf/util/formula.l | 119 ++++++++++++++ tools/perf/util/formula.y | 248 +++++++++++++++++++++++++++++ 7 files changed, 895 insertions(+) create mode 100644 tools/perf/util/formula.c create mode 100644 tools/perf/util/formula.h create mode 100644 tools/perf/util/formula.l create mode 100644 tools/perf/util/formula.y diff --git a/tools/perf/Makefile b/tools/perf/Makefile index 103ed95..0f120a4 100644 --- a/tools/perf/Makefile +++ b/tools/perf/Makefile @@ -296,8 +296,15 @@ $(OUTPUT)util/pmu-flex.c: util/pmu.l $(OUTPUT)util/pmu-bison.c $(OUTPUT)util/pmu-bison.c: util/pmu.y $(QUIET_BISON)$(BISON) -v util/pmu.y -d -o $(OUTPUT)util/pmu-bison.c +$(OUTPUT)util/formula-flex.c: util/formula.l util/formula.y $(OUTPUT)util/formula-bison.c + $(QUIET_FLEX)$(FLEX) --header-file=$(OUTPUT)util/formula-flex.h $(PARSER_DEBUG_FLEX) -t util/formula.l > $(OUTPUT)util/formula-flex.c + +$(OUTPUT)util/formula-bison.c: util/formula.y + $(QUIET_BISON)$(BISON) -v util/formula.y -d $(PARSER_DEBUG_BISON) -o $(OUTPUT)util/formula-bison.c + $(OUTPUT)util/parse-events.o: $(OUTPUT)util/parse-events-flex.c $(OUTPUT)util/parse-events-bison.c $(OUTPUT)util/pmu.o: $(OUTPUT)util/pmu-flex.c $(OUTPUT)util/pmu-bison.c +$(OUTPUT)util/formula.o: $(OUTPUT)util/formula-flex.c $(OUTPUT)util/formula-bison.c LIB_FILE=$(OUTPUT)libperf.a @@ -391,6 +398,7 @@ LIB_H += util/intlist.h LIB_H += util/perf_regs.h LIB_H += util/unwind.h LIB_H += util/vdso.h +LIB_H += util/formula.h LIB_H += ui/helpline.h LIB_H += ui/progress.h LIB_H += ui/util.h @@ -464,6 +472,9 @@ LIB_OBJS += $(OUTPUT)util/rblist.o LIB_OBJS += $(OUTPUT)util/intlist.o LIB_OBJS += $(OUTPUT)util/vdso.o LIB_OBJS += $(OUTPUT)util/stat.o +LIB_OBJS += $(OUTPUT)util/formula.o +LIB_OBJS += $(OUTPUT)util/formula-flex.o +LIB_OBJS += $(OUTPUT)util/formula-bison.o LIB_OBJS += $(OUTPUT)ui/setup.o LIB_OBJS += $(OUTPUT)ui/helpline.o diff --git a/tools/perf/util/evlist.c b/tools/perf/util/evlist.c index dc8aee9..e1a6126 100644 --- a/tools/perf/util/evlist.c +++ b/tools/perf/util/evlist.c @@ -845,3 +845,16 @@ size_t perf_evlist__fprintf(struct perf_evlist *evlist, FILE *fp) return printed + fprintf(fp, "\n");; } + +struct perf_evsel* +perf_evlist__find_evsel_name(struct perf_evlist *evlist, char *name) +{ + struct perf_evsel *evsel; + + list_for_each_entry(evsel, &evlist->entries, node) { + if (!strcmp(name, perf_evsel__name(evsel))) + return evsel; + } + + return NULL; +} diff --git a/tools/perf/util/evlist.h b/tools/perf/util/evlist.h index 457e235..a8630f2 100644 --- a/tools/perf/util/evlist.h +++ b/tools/perf/util/evlist.h @@ -135,4 +135,8 @@ static inline struct perf_evsel *perf_evlist__last(struct perf_evlist *evlist) } size_t perf_evlist__fprintf(struct perf_evlist *evlist, FILE *fp); + +struct perf_evsel* +perf_evlist__find_evsel_name(struct perf_evlist *evlist, char *name); + #endif /* __PERF_EVLIST_H */ diff --git a/tools/perf/util/formula.c b/tools/perf/util/formula.c new file mode 100644 index 0000000..e9fa05b --- /dev/null +++ b/tools/perf/util/formula.c @@ -0,0 +1,387 @@ + +#include <linux/compiler.h> +#include <stdio.h> +#include "stat.h" +#include "parse-events.h" +#include "formula.h" +#include "formula-bison.h" +#define YY_EXTRA_TYPE int +#include "formula-flex.h" + +#ifdef PARSER_DEBUG +extern int perf_formula_debug; +#endif +int perf_formula_parse(void *_data, void *scanner); + +void perf_formula__init(struct perf_formula *f) +{ + memset(f, 0x0, sizeof(*f)); + INIT_LIST_HEAD(&f->head_files); +} + +static int scanner_expr(const char *str, void *data) +{ + YY_BUFFER_STATE buffer; + void *scanner; + int ret; + + ret = perf_formula_lex_init_extra(PF_START_EXPR, &scanner); + if (ret) + return ret; + + buffer = perf_formula__scan_string(str, scanner); + +#ifdef PARSER_DEBUG + perf_formula_debug = 1; +#endif + ret = perf_formula_parse(data, scanner); + + perf_formula__flush_buffer(buffer, scanner); + perf_formula__delete_buffer(buffer, scanner); + perf_formula_lex_destroy(scanner); + return ret; +} + +static int scanner_config(FILE *file, void *data) +{ + void *scanner; + int ret; + + ret = perf_formula_lex_init_extra(PF_START_CONFIG, &scanner); + if (ret) + return ret; + + perf_formula_set_in(file, scanner); + +#ifdef PARSER_DEBUG + perf_formula_debug = 1; +#endif + ret = perf_formula_parse(data, scanner); + + perf_formula_lex_destroy(scanner); + return ret; +} + +static int config_parse(struct perf_formula__file *file) +{ + FILE *f; + int ret; + + f = fopen(file->path, "r"); + if (!f) + return -EINVAL; + + ret = scanner_config(f, file); + + fclose(f); + return ret; +} + +static int counter_init(struct perf_formula__counter *counter) +{ + struct perf_formula__expr expr = { + .test_only = true, + }; + + return scanner_expr(counter->formula, &expr); +} + +static int set_init(struct perf_formula__set *set) +{ + struct perf_formula__counter *counter; + int ret = 0; + + list_for_each_entry(counter, &set->head_counters, list) { + ret = counter_init(counter); + if (ret) + break; + + counter->set = set; + } + + return ret; +} + +static int file_init(struct perf_formula__file *file) +{ + struct perf_formula__set *set; + int ret; + + ret = config_parse(file); + + list_for_each_entry(set, &file->head_sets, list) { + ret = set_init(set); + if (ret) + break; + } + + return ret; +} + +static void file_free(struct perf_formula__file *file) +{ + struct perf_formula__set *set; + + list_for_each_entry(set, &file->head_sets, list) { + struct perf_formula__counter *counter; + + list_for_each_entry(counter, &set->head_counters, list) + free(counter); + + free(set); + } + + free(file->path); + free(file); +} + +int perf_formula__load(struct perf_formula *f, char *path) +{ + struct perf_formula__file *file; + int ret; + + file = zalloc(sizeof(*file)); + if (!file) + return -ENOMEM; + + INIT_LIST_HEAD(&file->list); + INIT_LIST_HEAD(&file->head_sets); + file->path = strdup(path); + + ret = file_init(file); + if (ret) + file_free(file); + else + list_add_tail(&file->list, &f->head_files); + + return ret; +} + +int perf_formula__free(struct perf_formula *f) +{ + struct perf_formula__file *file; + + list_for_each_entry(file, &f->head_files, list) + file_free(file); + + return 0; +} + +enum { + CB_NEXT, + CB_OK, + CB_FAIL, +}; + +typedef int (*set_cb)(struct perf_formula__set *set, void *data); + +static int for_each_set(struct perf_formula *formula, + set_cb cb, void *data) +{ + struct perf_formula__file *file; + + list_for_each_entry(file, &formula->head_files, list) { + struct perf_formula__set *set; + + list_for_each_entry(set, &file->head_sets, list) { + int ret = cb(set, data); + + if (ret == CB_NEXT) + continue; + else if (ret == CB_OK) + return 0; + else if (ret == CB_FAIL) + return -1; + } + } + + return 0; +} +struct find_set_data { + struct perf_formula__set *set; + char *name; +}; + +static int find_set_cb(struct perf_formula__set *set, void *data) +{ + struct find_set_data *d = data; + + if (strcmp(set->name, d->name)) + return CB_NEXT; + + d->set = set; + return CB_OK; +} + +struct perf_formula__set* +perf_formula__set(struct perf_formula *f, char *name) +{ + struct find_set_data data = { + .name = name, + }; + + if (for_each_set(f, find_set_cb, &data)) + return NULL; + + return data.set; +} + +static int counter_print(FILE *out, struct perf_formula__counter *counter, + struct perf_formula__expr *expr) +{ + int ret; + + ret = scanner_expr(counter->formula, expr); + + if (ret) { + fprintf(stderr, "failed to process counter %s\n", + counter->name); + return -1; + } + + fprintf(out, "%'18.8F %-25s # %s\n", + expr->result, counter->name, counter->desc); + + return ret; +} + +int perf_formula__print(FILE *file, + struct perf_formula__set *set, + struct perf_evlist *evlist, + struct perf_formula__value **values) +{ + struct perf_formula__counter *counter; + struct perf_formula__expr expr = { + .evlist = evlist, + .values = values, + }; + + list_for_each_entry(counter, &set->head_counters, list) + if (counter_print(file, counter, &expr)) + break; + + fprintf(file, "\n"); + return 0; +} + +static int set_evlist(struct perf_formula__set *set, + struct perf_evlist *evlist) +{ + struct perf_evlist *evlist_tmp = evlist; + + if (set->loaded) + return 0; + + if (parse_events(evlist_tmp, set->events)) + return -1; + + set->loaded = true; + + return 0; +} + +int perf_formula__evlist(struct perf_formula__set *set, + struct perf_evlist *evlist) +{ + return set_evlist(set, evlist); +} + +struct perf_formula__counter* +perf_formula__counter_new(char *name, struct list_head *head) +{ + struct perf_formula__counter *counter; + struct perf_formula__config *config; + + counter = zalloc(sizeof(*counter)); + if (!counter) + return NULL; + + INIT_LIST_HEAD(&counter->list); \ + + list_for_each_entry(config, head, list) { + switch (config->type) { + case PERF_FORMULA__CONFIG_FORMULA: + counter->formula = config->formula; + break; + + case PERF_FORMULA__CONFIG_DESC: + counter->desc = config->desc; + break; + + case PERF_FORMULA__CONFIG_COUNTER: + case PERF_FORMULA__CONFIG_EVENTS: + default: + BUG_ON(1); + } + } + + counter->name = strdup(name); + return counter; +} + +struct perf_formula__set* +perf_formula__set_new(char *name, struct list_head *head) +{ + struct perf_formula__set *set; + struct perf_formula__config *config; + + set = zalloc(sizeof(*set)); + if (!set) + return NULL; + + INIT_LIST_HEAD(&set->list); + INIT_LIST_HEAD(&set->head_counters); + + list_for_each_entry(config, head, list) { + struct perf_formula__counter *counter; + + switch (config->type) { + case PERF_FORMULA__CONFIG_COUNTER: + counter = config->counter; + + list_add_tail(&counter->list, + &set->head_counters); + break; + + case PERF_FORMULA__CONFIG_EVENTS: + if (set->events) { + free(set); + return NULL; + } + + set->events = config->events; + break; + + case PERF_FORMULA__CONFIG_FORMULA: + case PERF_FORMULA__CONFIG_DESC: + default: + BUG_ON(1); + } + } + + set->name = strdup(name); + return set; +} + +struct perf_stat { + struct stats res_stats[3]; +}; + +double perf_formula__expr_resolve(struct perf_formula__expr *expr, + char *name) +{ + struct perf_evsel *evsel; + double res = 0; + + if (expr->test_only) + return 1; + + evsel = perf_evlist__find_evsel_name(expr->evlist, name); + if (evsel) { + struct perf_stat *ps = evsel->priv; + res = avg_stats(&ps->res_stats[0]); + } else + pr_err("failed to resolve event '%s'\n", name); + + return res; +} diff --git a/tools/perf/util/formula.h b/tools/perf/util/formula.h new file mode 100644 index 0000000..e7d988b --- /dev/null +++ b/tools/perf/util/formula.h @@ -0,0 +1,113 @@ +#ifndef __PERF_FORMULA +#define __PERF_FORMULA + +/* + * format: + * + * IPC { + * events = cycles,instructions + * + * counter IPC { + * formula = instructions/cycles + * desc = krava + * } + * } + * + */ + +#include <linux/list.h> +#include "evlist.h" + +struct perf_formula { + struct list_head head_files; +}; + +struct perf_formula__file { + char *path; + + struct list_head head_sets; + struct list_head list; +}; + +struct perf_formula__set { + char *name; + char *events; + bool loaded; + + struct list_head head_counters; + struct list_head list; +}; + +struct perf_formula__counter { + char *name; + char *formula; + char *desc; + + struct perf_formula__set *set; + + struct list_head list; +}; + +#define PERF_FORMULA__VALUE(n, p) \ +{ \ + .name = n, \ + .ptr = p, \ +} + +struct perf_formula__value { + char *name; + double *ptr; +}; + +struct perf_formula__expr { + struct perf_evlist *evlist; + struct perf_formula__value **values; + double result; + bool test_only; +}; + +struct perf_formula__config { + enum { + PERF_FORMULA__CONFIG_COUNTER, + PERF_FORMULA__CONFIG_EVENTS, + PERF_FORMULA__CONFIG_FORMULA, + PERF_FORMULA__CONFIG_DESC, + } type; + + union { + struct perf_formula__counter *counter; + char *events; + char *formula; + char *desc; + }; + + struct list_head list; +}; + + +void perf_formula__init(struct perf_formula *f); + +int perf_formula__load(struct perf_formula *f, char *path); +int perf_formula__free(struct perf_formula *f); + +struct perf_formula__set* +perf_formula__set(struct perf_formula *f, char *name); + +int perf_formula__evlist(struct perf_formula__set *set, + struct perf_evlist *evlist); + +int perf_formula__print(FILE *file, + struct perf_formula__set *set, + struct perf_evlist *evlist, + struct perf_formula__value **values); + +struct perf_formula__counter* +perf_formula__counter_new(char *name, struct list_head *head); + +struct perf_formula__set* +perf_formula__set_new(char *name, struct list_head *head); + +double perf_formula__expr_resolve(struct perf_formula__expr *expr, + char *name); + +#endif /* __PERF_FORMULA */ diff --git a/tools/perf/util/formula.l b/tools/perf/util/formula.l new file mode 100644 index 0000000..e7669a5 --- /dev/null +++ b/tools/perf/util/formula.l @@ -0,0 +1,119 @@ + +%option reentrant +%option bison-bridge +%option prefix="perf_formula_" +%option stack + +%{ +#include "formula-bison.h" +#include "formula.h" + +char *perf_formula_get_text(yyscan_t yyscanner); +YYSTYPE *perf_formula_get_lval(yyscan_t yyscanner); + +static int __value(YYSTYPE *yylval, char *str, int base, int token) +{ + double num; + + errno = 0; + num = strtoull(str, NULL, base); + if (errno) + return PF_ERROR; + + yylval->num = num; + return token; +} + +static int value(yyscan_t scanner, int base) +{ + YYSTYPE *yylval = perf_formula_get_lval(scanner); + char *text = perf_formula_get_text(scanner); + + return __value(yylval, text, base, PF_VALUE); +} + +static int str(yyscan_t scanner, int token) +{ + YYSTYPE *yylval = perf_formula_get_lval(scanner); + char *text = perf_formula_get_text(scanner); + + yylval->str = strdup(text); + return token; +} + +%} + +num_dec [0-9]+ +num_hex 0x[a-fA-F0-9]+ +name [a-zA-Z_*?][a-zA-Z0-9_*?\.-]* + +%x config +%x expr +%x config_eoln_str + +%% + +%{ + { + int start_token; + + start_token = perf_formula_get_extra(yyscanner); + + if (start_token == PF_START_CONFIG) + BEGIN(config); + else if (start_token == PF_START_EXPR) + BEGIN(expr); + + if (start_token) { + perf_formula_set_extra(NULL, yyscanner); + return start_token; + } + } +%} + +<config>{ +"{" { return '{'; } +"}" { return '}'; } +"=" { return '='; } + +events { BEGIN(config_eoln_str); return PF_EVENTS; } +formula { BEGIN(config_eoln_str); return PF_FORMULA; } +desc { BEGIN(config_eoln_str); return PF_DESC; } + +{name} { return str(yyscanner, PF_NAME); } + +\n { } +} + +<config_eoln_str>{ +[ \t]*= { return '='; } +[^=\n]+ { + str(yyscanner, PF_EOLN_STR); + BEGIN(config); + return PF_EOLN_STR; + } + +<<EOF>> { + BEGIN(config); + } +} + +<expr>{ +"*" { return '*'; } +"-" { return '-'; } +"+" { return '+'; } +"/" { return '/'; } + +{name} { return str(yyscanner, PF_NAME); } +{num_dec} { return value(yyscanner, 10); } +{num_hex} { return value(yyscanner, 16); } + +. { } +} + +%% + +int perf_formula_wrap(void *scanner __maybe_unused) +{ + return 1; +} diff --git a/tools/perf/util/formula.y b/tools/perf/util/formula.y new file mode 100644 index 0000000..42b6ebf --- /dev/null +++ b/tools/perf/util/formula.y @@ -0,0 +1,248 @@ +%pure-parser +%name-prefix "perf_formula_" +%parse-param {void *_data} +%parse-param {void *scanner} +%lex-param {void* scanner} + +%left '+' '-' '*' '/' + +%{ + +#define YYDEBUG 1 + +#include "util.h" +#include "formula.h" +#include "formula-bison.h" + +extern int formula_lex(YYSTYPE* lvalp, void* scanner); + + +#define ABORT() YYABORT + +#define ABORT_ON(val) \ +do { \ + if (val) \ + YYABORT; \ +} while (0) + +#define HEAD() ({ \ + struct list_head *__head = zalloc(sizeof(*__head)); \ + ABORT_ON(!__head); \ + INIT_LIST_HEAD(__head); \ + __head; \ +}) + +#define CONFIG() ({ \ + struct perf_formula__config *__config; \ + __config = zalloc(sizeof(*__config)); \ + ABORT_ON(!__config); \ + INIT_LIST_HEAD(&__config->list); \ + __config; \ +}) + +%} + +%token PF_START_CONFIG PF_START_EXPR +%token PF_NAME +%token PF_VALUE +%token PF_FORMULA +%token PF_DESC +%token PF_EVENTS +%token PF_EOLN_STR +%token PF_ERROR + +%type <str> PF_NAME +%type <num> PF_VALUE +%type <str> PF_EOLN_STR +%type <head> set_def +%type <head> counter_def +%type <config> set_token +%type <config> counter_token +%type <counter> counter +%type <num> expr + +%union +{ + char *str; + double num; + struct list_head *head; + struct config *config; + struct perf_formula__counter *counter; +} + +%% + +start: +PF_START_CONFIG start_config +| +PF_START_EXPR start_expr + +start_config: sets + +sets: +sets set | set + +set: +PF_NAME '{' set_def '}' +{ + struct perf_formula__file *file = _data; + struct perf_formula__set *set; + + set = perf_formula__set_new($1, $3); + ABORT_ON(!set); + + list_add_tail(&set->list, &file->head_sets); +} + +set_def: +set_def set_token +{ + struct list_head *head = $1; + struct perf_formula__config *config = $2; + + list_add_tail(&config->list, head); + $$ = head; +} +| +set_token +{ + struct list_head *head = HEAD(); + struct perf_formula__config *config = $1; + + list_add_tail(&config->list, head); + $$ = head; +} + +set_token: +counter +{ + struct perf_formula__config *config = CONFIG(); + + config->type = PERF_FORMULA__CONFIG_COUNTER; + config->counter = $1; + + $$ = config; +} +| +PF_EVENTS '=' PF_EOLN_STR +{ + struct perf_formula__config *config = CONFIG(); + + config->type = PERF_FORMULA__CONFIG_EVENTS; + config->counter = $3; + + $$ = config; +} + +counter: +PF_NAME '{' counter_def '}' +{ + struct perf_formula__counter *counter; + struct config *config; + + counter = perf_formula__counter_new($1, $3); + ABORT_ON(!counter); + + $$ = counter; +} + +counter_def: +counter_def counter_token +{ + struct list_head *head = $1; + struct perf_formula__config *config = $2; + + list_add_tail(&config->list, head); + $$ = head; +} +| +counter_token +{ + struct list_head *head = HEAD(); + struct perf_formula__config *config = $1; + + list_add_tail(&config->list, head); + $$ = head; +} + +counter_token: +PF_FORMULA '=' PF_EOLN_STR +{ + struct perf_formula__config *config = CONFIG(); + + config->type = PERF_FORMULA__CONFIG_FORMULA; + config->counter = $3; + + $$ = config; +} +| +PF_DESC '=' PF_EOLN_STR +{ + struct perf_formula__config *config = CONFIG(); + + config->type = PERF_FORMULA__CONFIG_DESC; + config->counter = $3; + + $$ = config; +} + +start_expr: +expr +{ + struct perf_formula__expr *expr = _data; + + expr->result = $1; +} + +expr: +PF_VALUE +{ + $$ = $1; +} +| +PF_NAME +{ + $$ = perf_formula__expr_resolve(_data, $1); +} +| +'-' expr +{ + $$ = - $2; +} +| +expr '+' expr +{ + $$ = $1 + $3; +} +| +expr '-' expr +{ + $$ = $1 - $3; +} +| +expr '*' expr +{ + $$ = $1 * $3; +} +| +expr '/' expr +{ + if (!$3) { + pr_err("division by zero\n"); + ABORT(); + } + + $$ = $1 / $3; +} +| +'(' expr ')' +{ +} + +%% + +void perf_formula_error(void *data __maybe_unused, + void *scanner __maybe_unused, + char const *msg __maybe_unused) +{ +} -- 1.7.11.7 -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/