diff options
| -rw-r--r-- | tools/perf/builtin-record.c | 15 | ||||
| -rw-r--r-- | tools/perf/util/parse-events.c | 26 | ||||
| -rw-r--r-- | tools/perf/util/parse-events.h | 2 |
3 files changed, 40 insertions, 3 deletions
diff --git a/tools/perf/builtin-record.c b/tools/perf/builtin-record.c index 4e3a374e7aa7..8b2c860c49a2 100644 --- a/tools/perf/builtin-record.c +++ b/tools/perf/builtin-record.c | |||
| @@ -374,9 +374,11 @@ static struct perf_header_attr *get_header_attr(struct perf_event_attr *a, int n | |||
| 374 | 374 | ||
| 375 | static void create_counter(int counter, int cpu, pid_t pid) | 375 | static void create_counter(int counter, int cpu, pid_t pid) |
| 376 | { | 376 | { |
| 377 | char *filter = filters[counter]; | ||
| 377 | struct perf_event_attr *attr = attrs + counter; | 378 | struct perf_event_attr *attr = attrs + counter; |
| 378 | struct perf_header_attr *h_attr; | 379 | struct perf_header_attr *h_attr; |
| 379 | int track = !counter; /* only the first counter needs these */ | 380 | int track = !counter; /* only the first counter needs these */ |
| 381 | int ret; | ||
| 380 | struct { | 382 | struct { |
| 381 | u64 count; | 383 | u64 count; |
| 382 | u64 time_enabled; | 384 | u64 time_enabled; |
| @@ -479,7 +481,6 @@ try_again: | |||
| 479 | multiplex_fd = fd[nr_cpu][counter]; | 481 | multiplex_fd = fd[nr_cpu][counter]; |
| 480 | 482 | ||
| 481 | if (multiplex && fd[nr_cpu][counter] != multiplex_fd) { | 483 | if (multiplex && fd[nr_cpu][counter] != multiplex_fd) { |
| 482 | int ret; | ||
| 483 | 484 | ||
| 484 | ret = ioctl(fd[nr_cpu][counter], PERF_EVENT_IOC_SET_OUTPUT, multiplex_fd); | 485 | ret = ioctl(fd[nr_cpu][counter], PERF_EVENT_IOC_SET_OUTPUT, multiplex_fd); |
| 485 | assert(ret != -1); | 486 | assert(ret != -1); |
| @@ -499,6 +500,16 @@ try_again: | |||
| 499 | } | 500 | } |
| 500 | } | 501 | } |
| 501 | 502 | ||
| 503 | if (filter != NULL) { | ||
| 504 | ret = ioctl(fd[nr_cpu][counter], | ||
| 505 | PERF_EVENT_IOC_SET_FILTER, filter); | ||
| 506 | if (ret) { | ||
| 507 | error("failed to set filter with %d (%s)\n", errno, | ||
| 508 | strerror(errno)); | ||
| 509 | exit(-1); | ||
| 510 | } | ||
| 511 | } | ||
| 512 | |||
| 502 | ioctl(fd[nr_cpu][counter], PERF_EVENT_IOC_ENABLE); | 513 | ioctl(fd[nr_cpu][counter], PERF_EVENT_IOC_ENABLE); |
| 503 | } | 514 | } |
| 504 | 515 | ||
| @@ -676,6 +687,8 @@ static const struct option options[] = { | |||
| 676 | OPT_CALLBACK('e', "event", NULL, "event", | 687 | OPT_CALLBACK('e', "event", NULL, "event", |
| 677 | "event selector. use 'perf list' to list available events", | 688 | "event selector. use 'perf list' to list available events", |
| 678 | parse_events), | 689 | parse_events), |
| 690 | OPT_CALLBACK(0, "filter", NULL, "filter", | ||
| 691 | "event filter", parse_filter), | ||
| 679 | OPT_INTEGER('p', "pid", &target_pid, | 692 | OPT_INTEGER('p', "pid", &target_pid, |
| 680 | "record events on existing pid"), | 693 | "record events on existing pid"), |
| 681 | OPT_INTEGER('r', "realtime", &realtime_prio, | 694 | OPT_INTEGER('r', "realtime", &realtime_prio, |
diff --git a/tools/perf/util/parse-events.c b/tools/perf/util/parse-events.c index 8cfb48cbbea0..b097570e9623 100644 --- a/tools/perf/util/parse-events.c +++ b/tools/perf/util/parse-events.c | |||
| @@ -8,9 +8,10 @@ | |||
| 8 | #include "cache.h" | 8 | #include "cache.h" |
| 9 | #include "header.h" | 9 | #include "header.h" |
| 10 | 10 | ||
| 11 | int nr_counters; | 11 | int nr_counters; |
| 12 | 12 | ||
| 13 | struct perf_event_attr attrs[MAX_COUNTERS]; | 13 | struct perf_event_attr attrs[MAX_COUNTERS]; |
| 14 | char *filters[MAX_COUNTERS]; | ||
| 14 | 15 | ||
| 15 | struct event_symbol { | 16 | struct event_symbol { |
| 16 | u8 type; | 17 | u8 type; |
| @@ -708,7 +709,6 @@ static void store_event_type(const char *orgname) | |||
| 708 | perf_header__push_event(id, orgname); | 709 | perf_header__push_event(id, orgname); |
| 709 | } | 710 | } |
| 710 | 711 | ||
| 711 | |||
| 712 | int parse_events(const struct option *opt __used, const char *str, int unset __used) | 712 | int parse_events(const struct option *opt __used, const char *str, int unset __used) |
| 713 | { | 713 | { |
| 714 | struct perf_event_attr attr; | 714 | struct perf_event_attr attr; |
| @@ -745,6 +745,28 @@ int parse_events(const struct option *opt __used, const char *str, int unset __u | |||
| 745 | return 0; | 745 | return 0; |
| 746 | } | 746 | } |
| 747 | 747 | ||
| 748 | int parse_filter(const struct option *opt __used, const char *str, | ||
| 749 | int unset __used) | ||
| 750 | { | ||
| 751 | int i = nr_counters - 1; | ||
| 752 | int len = strlen(str); | ||
| 753 | |||
| 754 | if (i < 0 || attrs[i].type != PERF_TYPE_TRACEPOINT) { | ||
| 755 | fprintf(stderr, | ||
| 756 | "-F option should follow a -e tracepoint option\n"); | ||
| 757 | return -1; | ||
| 758 | } | ||
| 759 | |||
| 760 | filters[i] = malloc(len + 1); | ||
| 761 | if (!filters[i]) { | ||
| 762 | fprintf(stderr, "not enough memory to hold filter string\n"); | ||
| 763 | return -1; | ||
| 764 | } | ||
| 765 | strcpy(filters[i], str); | ||
| 766 | |||
| 767 | return 0; | ||
| 768 | } | ||
| 769 | |||
| 748 | static const char * const event_type_descriptors[] = { | 770 | static const char * const event_type_descriptors[] = { |
| 749 | "", | 771 | "", |
| 750 | "Hardware event", | 772 | "Hardware event", |
diff --git a/tools/perf/util/parse-events.h b/tools/perf/util/parse-events.h index 8626a439033d..b8c1f64bc935 100644 --- a/tools/perf/util/parse-events.h +++ b/tools/perf/util/parse-events.h | |||
| @@ -17,11 +17,13 @@ extern struct tracepoint_path *tracepoint_id_to_path(u64 config); | |||
| 17 | extern int nr_counters; | 17 | extern int nr_counters; |
| 18 | 18 | ||
| 19 | extern struct perf_event_attr attrs[MAX_COUNTERS]; | 19 | extern struct perf_event_attr attrs[MAX_COUNTERS]; |
| 20 | extern char *filters[MAX_COUNTERS]; | ||
| 20 | 21 | ||
| 21 | extern const char *event_name(int ctr); | 22 | extern const char *event_name(int ctr); |
| 22 | extern const char *__event_name(int type, u64 config); | 23 | extern const char *__event_name(int type, u64 config); |
| 23 | 24 | ||
| 24 | extern int parse_events(const struct option *opt, const char *str, int unset); | 25 | extern int parse_events(const struct option *opt, const char *str, int unset); |
| 26 | extern int parse_filter(const struct option *opt, const char *str, int unset); | ||
| 25 | 27 | ||
| 26 | #define EVENTS_HELP_MAX (128*1024) | 28 | #define EVENTS_HELP_MAX (128*1024) |
| 27 | 29 | ||
