Commit bbbb521b authored by Masami Hiramatsu's avatar Masami Hiramatsu Committed by Ingo Molnar

perf probe: Add glob matching support on --del

Add glob-expression matching support on --del option.
You can use wildcards for specifying deleting events.
e.g.

 Clear all probe events:

 # perf probe --del '*'

 Clear probes on schedule():

 # perf probe --del 'schedule*'
Signed-off-by: default avatarMasami Hiramatsu <mhiramat@redhat.com>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Jim Keniston <jkenisto@us.ibm.com>
Cc: Ananth N Mavinakayanahalli <ananth@in.ibm.com>
Cc: Christoph Hellwig <hch@infradead.org>
Cc: Frank Ch. Eigler <fche@redhat.com>
Cc: Jason Baron <jbaron@redhat.com>
Cc: K.Prasad <prasad@linux.vnet.ibm.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Srikar Dronamraju <srikar@linux.vnet.ibm.com>
Cc: systemtap <systemtap@sources.redhat.com>
Cc: DLE <dle-develop@lists.sourceforge.net>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
LKML-Reference: <20091215153210.17436.12327.stgit@dhcp-100-2-132.bos.redhat.com>
Signed-off-by: default avatarIngo Molnar <mingo@elte.hu>
parent adf365f4
...@@ -532,26 +532,50 @@ void add_trace_kprobe_events(struct probe_point *probes, int nr_probes) ...@@ -532,26 +532,50 @@ void add_trace_kprobe_events(struct probe_point *probes, int nr_probes)
close(fd); close(fd);
} }
static void __del_trace_kprobe_event(int fd, struct str_node *ent)
{
char *p;
char buf[128];
/* Convert from perf-probe event to trace-kprobe event */
if (e_snprintf(buf, 128, "-:%s", ent->s) < 0)
die("Failed to copy event.");
p = strchr(buf + 2, ':');
if (!p)
die("Internal error: %s should have ':' but not.", ent->s);
*p = '/';
write_trace_kprobe_event(fd, buf);
printf("Remove event: %s\n", ent->s);
}
static void del_trace_kprobe_event(int fd, const char *group, static void del_trace_kprobe_event(int fd, const char *group,
const char *event, struct strlist *namelist) const char *event, struct strlist *namelist)
{ {
char buf[128]; char buf[128];
struct str_node *ent; struct str_node *ent, *n;
int found = 0;
if (e_snprintf(buf, 128, "%s:%s", group, event) < 0) if (e_snprintf(buf, 128, "%s:%s", group, event) < 0)
die("Failed to copy event."); die("Failed to copy event.");
ent = strlist__find(namelist, buf);
if (!ent) {
pr_info("Info: event \"%s\" does not exist, could not remove it.\n", buf);
return;
}
/* Convert from perf-probe event to trace-kprobe event */
if (e_snprintf(buf, 128, "-:%s/%s", group, event) < 0)
die("Failed to copy event.");
write_trace_kprobe_event(fd, buf); if (strpbrk(buf, "*?")) { /* Glob-exp */
printf("Remove event: %s:%s\n", group, event); strlist__for_each_safe(ent, n, namelist)
strlist__remove(namelist, ent); if (strglobmatch(ent->s, buf)) {
found++;
__del_trace_kprobe_event(fd, ent);
strlist__remove(namelist, ent);
}
} else {
ent = strlist__find(namelist, buf);
if (ent) {
found++;
__del_trace_kprobe_event(fd, ent);
strlist__remove(namelist, ent);
}
}
if (found == 0)
pr_info("Info: event \"%s\" does not exist, could not remove it.\n", buf);
} }
void del_trace_kprobe_events(struct strlist *dellist) void del_trace_kprobe_events(struct strlist *dellist)
...@@ -570,15 +594,17 @@ void del_trace_kprobe_events(struct strlist *dellist) ...@@ -570,15 +594,17 @@ void del_trace_kprobe_events(struct strlist *dellist)
str = strdup(ent->s); str = strdup(ent->s);
if (!str) if (!str)
die("Failed to copy event."); die("Failed to copy event.");
pr_debug("Parsing: %s\n", str);
p = strchr(str, ':'); p = strchr(str, ':');
if (p) { if (p) {
group = str; group = str;
*p = '\0'; *p = '\0';
event = p + 1; event = p + 1;
} else { } else {
group = PERFPROBE_GROUP; group = "*";
event = str; event = str;
} }
pr_debug("Group: %s, Event: %s\n", group, event);
del_trace_kprobe_event(fd, group, event, namelist); del_trace_kprobe_event(fd, group, event, namelist);
free(str); free(str);
} }
......
...@@ -226,3 +226,28 @@ char **argv_split(const char *str, int *argcp) ...@@ -226,3 +226,28 @@ char **argv_split(const char *str, int *argcp)
argv_free(argv); argv_free(argv);
return NULL; return NULL;
} }
/* Glob expression pattern matching */
bool strglobmatch(const char *str, const char *pat)
{
while (*str && *pat && *pat != '*') {
if (*pat == '?') {
str++;
pat++;
} else
if (*str++ != *pat++)
return false;
}
/* Check wild card */
if (*pat == '*') {
while (*pat == '*')
pat++;
if (!*pat) /* Tail wild card matches all */
return true;
while (*str)
if (strglobmatch(str++, pat))
return true;
}
return !*str && !*pat;
}
#ifndef __PERF_STRING_H_ #ifndef __PERF_STRING_H_
#define __PERF_STRING_H_ #define __PERF_STRING_H_
#include <stdbool.h>
#include "types.h" #include "types.h"
int hex2u64(const char *ptr, u64 *val); int hex2u64(const char *ptr, u64 *val);
...@@ -8,6 +9,7 @@ char *strxfrchar(char *s, char from, char to); ...@@ -8,6 +9,7 @@ char *strxfrchar(char *s, char from, char to);
s64 perf_atoll(const char *str); s64 perf_atoll(const char *str);
char **argv_split(const char *str, int *argcp); char **argv_split(const char *str, int *argcp);
void argv_free(char **argv); void argv_free(char **argv);
bool strglobmatch(const char *str, const char *pat);
#define _STR(x) #x #define _STR(x) #x
#define STR(x) _STR(x) #define STR(x) _STR(x)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment