aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--tools/perf/builtin-record.c15
-rw-r--r--tools/perf/util/parse-events.c26
-rw-r--r--tools/perf/util/parse-events.h2
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
375static void create_counter(int counter, int cpu, pid_t pid) 375static 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
11int nr_counters; 11int nr_counters;
12 12
13struct perf_event_attr attrs[MAX_COUNTERS]; 13struct perf_event_attr attrs[MAX_COUNTERS];
14char *filters[MAX_COUNTERS];
14 15
15struct event_symbol { 16struct 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
712int parse_events(const struct option *opt __used, const char *str, int unset __used) 712int 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
748int 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
748static const char * const event_type_descriptors[] = { 770static 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);
17extern int nr_counters; 17extern int nr_counters;
18 18
19extern struct perf_event_attr attrs[MAX_COUNTERS]; 19extern struct perf_event_attr attrs[MAX_COUNTERS];
20extern char *filters[MAX_COUNTERS];
20 21
21extern const char *event_name(int ctr); 22extern const char *event_name(int ctr);
22extern const char *__event_name(int type, u64 config); 23extern const char *__event_name(int type, u64 config);
23 24
24extern int parse_events(const struct option *opt, const char *str, int unset); 25extern int parse_events(const struct option *opt, const char *str, int unset);
26extern 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