parse_events_option),
        OPT_CALLBACK(0, "filter", &record.evlist, "filter",
                     "event filter", parse_filter),
+       OPT_CALLBACK_NOOPT(0, "exclude-perf", &record.evlist,
+                          NULL, "don't record events from perf itself",
+                          exclude_perf),
        OPT_STRING('p', "pid", &record.opts.target.pid, "pid",
                    "record events on existing process id"),
        OPT_STRING('t', "tid", &record.opts.target.tid, "tid",
 
        return ret;
 }
 
-int parse_filter(const struct option *opt, const char *str,
-                int unset __maybe_unused)
+static int
+foreach_evsel_in_last_glob(struct perf_evlist *evlist,
+                          int (*func)(struct perf_evsel *evsel,
+                                      const void *arg),
+                          const void *arg)
 {
-       struct perf_evlist *evlist = *(struct perf_evlist **)opt->value;
        struct perf_evsel *last = NULL;
+       int err;
 
        if (evlist->nr_entries > 0)
                last = perf_evlist__last(evlist);
 
        do {
-               if (last == NULL || last->attr.type != PERF_TYPE_TRACEPOINT) {
-                       fprintf(stderr,
-                               "--filter option should follow a -e tracepoint option\n");
-                       return -1;
-               }
-
-               if (perf_evsel__set_filter(last, str) < 0) {
-                       fprintf(stderr,
-                               "not enough memory to hold filter string\n");
+               err = (*func)(last, arg);
+               if (err)
                        return -1;
-               }
+               if (!last)
+                       return 0;
 
                if (last->node.prev == &evlist->entries)
                        return 0;
        return 0;
 }
 
+static int set_filter(struct perf_evsel *evsel, const void *arg)
+{
+       const char *str = arg;
+
+       if (evsel == NULL || evsel->attr.type != PERF_TYPE_TRACEPOINT) {
+               fprintf(stderr,
+                       "--filter option should follow a -e tracepoint option\n");
+               return -1;
+       }
+
+       if (perf_evsel__append_filter(evsel, "&&", str) < 0) {
+               fprintf(stderr,
+                       "not enough memory to hold filter string\n");
+               return -1;
+       }
+
+       return 0;
+}
+
+int parse_filter(const struct option *opt, const char *str,
+                int unset __maybe_unused)
+{
+       struct perf_evlist *evlist = *(struct perf_evlist **)opt->value;
+
+       return foreach_evsel_in_last_glob(evlist, set_filter,
+                                         (const void *)str);
+}
+
+static int add_exclude_perf_filter(struct perf_evsel *evsel,
+                                  const void *arg __maybe_unused)
+{
+       char new_filter[64];
+
+       if (evsel == NULL || evsel->attr.type != PERF_TYPE_TRACEPOINT) {
+               fprintf(stderr,
+                       "--exclude-perf option should follow a -e tracepoint option\n");
+               return -1;
+       }
+
+       snprintf(new_filter, sizeof(new_filter), "common_pid != %d", getpid());
+
+       if (perf_evsel__append_filter(evsel, "&&", new_filter) < 0) {
+               fprintf(stderr,
+                       "not enough memory to hold filter string\n");
+               return -1;
+       }
+
+       return 0;
+}
+
+int exclude_perf(const struct option *opt,
+                const char *arg __maybe_unused,
+                int unset __maybe_unused)
+{
+       struct perf_evlist *evlist = *(struct perf_evlist **)opt->value;
+
+       return foreach_evsel_in_last_glob(evlist, add_exclude_perf_filter,
+                                         NULL);
+}
+
 static const char * const event_type_descriptors[] = {
        "Hardware event",
        "Software event",