mirror of
https://github.com/Fishwaldo/Star64_linux.git
synced 2025-06-25 16:11:45 +00:00
perf probe: Add --list option for listing current probe events
Add --list option for listing currently defined probe events in the kernel. This shows events in below format; [group:event] <perf-probe probe-definition> for example: [probe:schedule_0] schedule+30 cpu Note that source file/line information is not supported yet. So even if you added a probe by line, it will be shown in <symbol+offset>. Signed-off-by: Masami Hiramatsu <mhiramat@redhat.com> Cc: systemtap <systemtap@sources.redhat.com> Cc: DLE <dle-develop@lists.sourceforge.net> 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: Frederic Weisbecker <fweisbec@gmail.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: Arnaldo Carvalho de Melo <acme@redhat.com> Cc: Frederic Weisbecker <fweisbec@gmail.com> LKML-Reference: <20091201002017.10235.76575.stgit@harusame> Signed-off-by: Ingo Molnar <mingo@elte.hu>
This commit is contained in:
parent
e1c01d61a9
commit
4de189fe6e
3 changed files with 231 additions and 19 deletions
|
@ -62,6 +62,8 @@ static struct {
|
||||||
struct probe_point probes[MAX_PROBES];
|
struct probe_point probes[MAX_PROBES];
|
||||||
} session;
|
} session;
|
||||||
|
|
||||||
|
static bool listing;
|
||||||
|
|
||||||
/* Parse an event definition. Note that any error must die. */
|
/* Parse an event definition. Note that any error must die. */
|
||||||
static void parse_probe_event(const char *str)
|
static void parse_probe_event(const char *str)
|
||||||
{
|
{
|
||||||
|
@ -119,6 +121,7 @@ static int open_default_vmlinux(void)
|
||||||
static const char * const probe_usage[] = {
|
static const char * const probe_usage[] = {
|
||||||
"perf probe [<options>] 'PROBEDEF' ['PROBEDEF' ...]",
|
"perf probe [<options>] 'PROBEDEF' ['PROBEDEF' ...]",
|
||||||
"perf probe [<options>] --add 'PROBEDEF' [--add 'PROBEDEF' ...]",
|
"perf probe [<options>] --add 'PROBEDEF' [--add 'PROBEDEF' ...]",
|
||||||
|
"perf probe --list",
|
||||||
NULL
|
NULL
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -129,6 +132,7 @@ static const struct option options[] = {
|
||||||
OPT_STRING('k', "vmlinux", &session.vmlinux, "file",
|
OPT_STRING('k', "vmlinux", &session.vmlinux, "file",
|
||||||
"vmlinux/module pathname"),
|
"vmlinux/module pathname"),
|
||||||
#endif
|
#endif
|
||||||
|
OPT_BOOLEAN('l', "list", &listing, "list up current probes"),
|
||||||
OPT_CALLBACK('a', "add", NULL,
|
OPT_CALLBACK('a', "add", NULL,
|
||||||
#ifdef NO_LIBDWARF
|
#ifdef NO_LIBDWARF
|
||||||
"FUNC[+OFFS|%return] [ARG ...]",
|
"FUNC[+OFFS|%return] [ARG ...]",
|
||||||
|
@ -164,9 +168,15 @@ int cmd_probe(int argc, const char **argv, const char *prefix __used)
|
||||||
for (i = 0; i < argc; i++)
|
for (i = 0; i < argc; i++)
|
||||||
parse_probe_event(argv[i]);
|
parse_probe_event(argv[i]);
|
||||||
|
|
||||||
if (session.nr_probe == 0)
|
if ((session.nr_probe == 0 && !listing) ||
|
||||||
|
(session.nr_probe != 0 && listing))
|
||||||
usage_with_options(probe_usage, options);
|
usage_with_options(probe_usage, options);
|
||||||
|
|
||||||
|
if (listing) {
|
||||||
|
show_perf_probe_events();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (session.need_dwarf)
|
if (session.need_dwarf)
|
||||||
#ifdef NO_LIBDWARF
|
#ifdef NO_LIBDWARF
|
||||||
die("Debuginfo-analysis is not supported");
|
die("Debuginfo-analysis is not supported");
|
||||||
|
|
|
@ -29,10 +29,13 @@
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <limits.h>
|
||||||
|
|
||||||
#undef _GNU_SOURCE
|
#undef _GNU_SOURCE
|
||||||
#include "event.h"
|
#include "event.h"
|
||||||
#include "string.h"
|
#include "string.h"
|
||||||
|
#include "strlist.h"
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
#include "parse-events.h" /* For debugfs_path */
|
#include "parse-events.h" /* For debugfs_path */
|
||||||
#include "probe-event.h"
|
#include "probe-event.h"
|
||||||
|
@ -43,6 +46,19 @@
|
||||||
|
|
||||||
#define semantic_error(msg ...) die("Semantic error :" msg)
|
#define semantic_error(msg ...) die("Semantic error :" msg)
|
||||||
|
|
||||||
|
/* If there is no space to write, returns -E2BIG. */
|
||||||
|
static int e_snprintf(char *str, size_t size, const char *format, ...)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
va_list ap;
|
||||||
|
va_start(ap, format);
|
||||||
|
ret = vsnprintf(str, size, format, ap);
|
||||||
|
va_end(ap);
|
||||||
|
if (ret >= (int)size)
|
||||||
|
ret = -E2BIG;
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
/* Parse probepoint definition. */
|
/* Parse probepoint definition. */
|
||||||
static void parse_perf_probe_probepoint(char *arg, struct probe_point *pp)
|
static void parse_perf_probe_probepoint(char *arg, struct probe_point *pp)
|
||||||
{
|
{
|
||||||
|
@ -166,23 +182,91 @@ int parse_perf_probe_event(const char *str, struct probe_point *pp)
|
||||||
return need_dwarf;
|
return need_dwarf;
|
||||||
}
|
}
|
||||||
|
|
||||||
int synthesize_trace_kprobe_event(struct probe_point *pp)
|
/* Parse kprobe_events event into struct probe_point */
|
||||||
|
void parse_trace_kprobe_event(const char *str, char **group, char **event,
|
||||||
|
struct probe_point *pp)
|
||||||
|
{
|
||||||
|
char pr;
|
||||||
|
char *p;
|
||||||
|
int ret, i, argc;
|
||||||
|
char **argv;
|
||||||
|
|
||||||
|
pr_debug("Parsing kprobe_events: %s\n", str);
|
||||||
|
argv = argv_split(str, &argc);
|
||||||
|
if (!argv)
|
||||||
|
die("argv_split failed.");
|
||||||
|
if (argc < 2)
|
||||||
|
semantic_error("Too less arguments.");
|
||||||
|
|
||||||
|
/* Scan event and group name. */
|
||||||
|
ret = sscanf(argv[0], "%c:%m[^/ \t]/%m[^ \t]",
|
||||||
|
&pr, group, event);
|
||||||
|
if (ret != 3)
|
||||||
|
semantic_error("Failed to parse event name: %s", argv[0]);
|
||||||
|
pr_debug("Group:%s Event:%s probe:%c\n", *group, *event, pr);
|
||||||
|
|
||||||
|
if (!pp)
|
||||||
|
goto end;
|
||||||
|
|
||||||
|
pp->retprobe = (pr == 'r');
|
||||||
|
|
||||||
|
/* Scan function name and offset */
|
||||||
|
ret = sscanf(argv[1], "%m[^+]+%d", &pp->function, &pp->offset);
|
||||||
|
if (ret == 1)
|
||||||
|
pp->offset = 0;
|
||||||
|
|
||||||
|
/* kprobe_events doesn't have this information */
|
||||||
|
pp->line = 0;
|
||||||
|
pp->file = NULL;
|
||||||
|
|
||||||
|
pp->nr_args = argc - 2;
|
||||||
|
pp->args = zalloc(sizeof(char *) * pp->nr_args);
|
||||||
|
for (i = 0; i < pp->nr_args; i++) {
|
||||||
|
p = strchr(argv[i + 2], '=');
|
||||||
|
if (p) /* We don't need which register is assigned. */
|
||||||
|
*p = '\0';
|
||||||
|
pp->args[i] = strdup(argv[i + 2]);
|
||||||
|
if (!pp->args[i])
|
||||||
|
die("Failed to copy argument.");
|
||||||
|
}
|
||||||
|
|
||||||
|
end:
|
||||||
|
argv_free(argv);
|
||||||
|
}
|
||||||
|
|
||||||
|
int synthesize_perf_probe_event(struct probe_point *pp)
|
||||||
{
|
{
|
||||||
char *buf;
|
char *buf;
|
||||||
|
char offs[64] = "", line[64] = "";
|
||||||
int i, len, ret;
|
int i, len, ret;
|
||||||
|
|
||||||
pp->probes[0] = buf = zalloc(MAX_CMDLEN);
|
pp->probes[0] = buf = zalloc(MAX_CMDLEN);
|
||||||
if (!buf)
|
if (!buf)
|
||||||
die("Failed to allocate memory by zalloc.");
|
die("Failed to allocate memory by zalloc.");
|
||||||
ret = snprintf(buf, MAX_CMDLEN, "%s+%d", pp->function, pp->offset);
|
if (pp->offset) {
|
||||||
if (ret <= 0 || ret >= MAX_CMDLEN)
|
ret = e_snprintf(offs, 64, "+%d", pp->offset);
|
||||||
|
if (ret <= 0)
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
if (pp->line) {
|
||||||
|
ret = e_snprintf(line, 64, ":%d", pp->line);
|
||||||
|
if (ret <= 0)
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pp->function)
|
||||||
|
ret = e_snprintf(buf, MAX_CMDLEN, "%s%s%s%s", pp->function,
|
||||||
|
offs, pp->retprobe ? "%return" : "", line);
|
||||||
|
else
|
||||||
|
ret = e_snprintf(buf, MAX_CMDLEN, "%s%s%s%s", pp->file, line);
|
||||||
|
if (ret <= 0)
|
||||||
goto error;
|
goto error;
|
||||||
len = ret;
|
len = ret;
|
||||||
|
|
||||||
for (i = 0; i < pp->nr_args; i++) {
|
for (i = 0; i < pp->nr_args; i++) {
|
||||||
ret = snprintf(&buf[len], MAX_CMDLEN - len, " %s",
|
ret = e_snprintf(&buf[len], MAX_CMDLEN - len, " %s",
|
||||||
pp->args[i]);
|
pp->args[i]);
|
||||||
if (ret <= 0 || ret >= MAX_CMDLEN - len)
|
if (ret <= 0)
|
||||||
goto error;
|
goto error;
|
||||||
len += ret;
|
len += ret;
|
||||||
}
|
}
|
||||||
|
@ -191,12 +275,134 @@ int synthesize_trace_kprobe_event(struct probe_point *pp)
|
||||||
return pp->found;
|
return pp->found;
|
||||||
error:
|
error:
|
||||||
free(pp->probes[0]);
|
free(pp->probes[0]);
|
||||||
if (ret > 0)
|
|
||||||
ret = -E2BIG;
|
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int synthesize_trace_kprobe_event(struct probe_point *pp)
|
||||||
|
{
|
||||||
|
char *buf;
|
||||||
|
int i, len, ret;
|
||||||
|
|
||||||
|
pp->probes[0] = buf = zalloc(MAX_CMDLEN);
|
||||||
|
if (!buf)
|
||||||
|
die("Failed to allocate memory by zalloc.");
|
||||||
|
ret = e_snprintf(buf, MAX_CMDLEN, "%s+%d", pp->function, pp->offset);
|
||||||
|
if (ret <= 0)
|
||||||
|
goto error;
|
||||||
|
len = ret;
|
||||||
|
|
||||||
|
for (i = 0; i < pp->nr_args; i++) {
|
||||||
|
ret = e_snprintf(&buf[len], MAX_CMDLEN - len, " %s",
|
||||||
|
pp->args[i]);
|
||||||
|
if (ret <= 0)
|
||||||
|
goto error;
|
||||||
|
len += ret;
|
||||||
|
}
|
||||||
|
pp->found = 1;
|
||||||
|
|
||||||
|
return pp->found;
|
||||||
|
error:
|
||||||
|
free(pp->probes[0]);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int open_kprobe_events(int flags, int mode)
|
||||||
|
{
|
||||||
|
char buf[PATH_MAX];
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = e_snprintf(buf, PATH_MAX, "%s/../kprobe_events", debugfs_path);
|
||||||
|
if (ret < 0)
|
||||||
|
die("Failed to make kprobe_events path.");
|
||||||
|
|
||||||
|
ret = open(buf, flags, mode);
|
||||||
|
if (ret < 0) {
|
||||||
|
if (errno == ENOENT)
|
||||||
|
die("kprobe_events file does not exist -"
|
||||||
|
" please rebuild with CONFIG_KPROBE_TRACER.");
|
||||||
|
else
|
||||||
|
die("Could not open kprobe_events file: %s",
|
||||||
|
strerror(errno));
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get raw string list of current kprobe_events */
|
||||||
|
static struct strlist *get_trace_kprobe_event_rawlist(int fd)
|
||||||
|
{
|
||||||
|
int ret, idx;
|
||||||
|
FILE *fp;
|
||||||
|
char buf[MAX_CMDLEN];
|
||||||
|
char *p;
|
||||||
|
struct strlist *sl;
|
||||||
|
|
||||||
|
sl = strlist__new(true, NULL);
|
||||||
|
|
||||||
|
fp = fdopen(dup(fd), "r");
|
||||||
|
while (!feof(fp)) {
|
||||||
|
p = fgets(buf, MAX_CMDLEN, fp);
|
||||||
|
if (!p)
|
||||||
|
break;
|
||||||
|
|
||||||
|
idx = strlen(p) - 1;
|
||||||
|
if (p[idx] == '\n')
|
||||||
|
p[idx] = '\0';
|
||||||
|
ret = strlist__add(sl, buf);
|
||||||
|
if (ret < 0)
|
||||||
|
die("strlist__add failed: %s", strerror(-ret));
|
||||||
|
}
|
||||||
|
fclose(fp);
|
||||||
|
|
||||||
|
return sl;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Free and zero clear probe_point */
|
||||||
|
static void clear_probe_point(struct probe_point *pp)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (pp->function)
|
||||||
|
free(pp->function);
|
||||||
|
if (pp->file)
|
||||||
|
free(pp->file);
|
||||||
|
for (i = 0; i < pp->nr_args; i++)
|
||||||
|
free(pp->args[i]);
|
||||||
|
if (pp->args)
|
||||||
|
free(pp->args);
|
||||||
|
for (i = 0; i < pp->found; i++)
|
||||||
|
free(pp->probes[i]);
|
||||||
|
memset(pp, 0, sizeof(pp));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* List up current perf-probe events */
|
||||||
|
void show_perf_probe_events(void)
|
||||||
|
{
|
||||||
|
unsigned int i;
|
||||||
|
int fd;
|
||||||
|
char *group, *event;
|
||||||
|
struct probe_point pp;
|
||||||
|
struct strlist *rawlist;
|
||||||
|
struct str_node *ent;
|
||||||
|
|
||||||
|
fd = open_kprobe_events(O_RDONLY, 0);
|
||||||
|
rawlist = get_trace_kprobe_event_rawlist(fd);
|
||||||
|
close(fd);
|
||||||
|
|
||||||
|
for (i = 0; i < strlist__nr_entries(rawlist); i++) {
|
||||||
|
ent = strlist__entry(rawlist, i);
|
||||||
|
parse_trace_kprobe_event(ent->s, &group, &event, &pp);
|
||||||
|
synthesize_perf_probe_event(&pp);
|
||||||
|
printf("[%s:%s]\t%s\n", group, event, pp.probes[0]);
|
||||||
|
free(group);
|
||||||
|
free(event);
|
||||||
|
clear_probe_point(&pp);
|
||||||
|
}
|
||||||
|
|
||||||
|
strlist__delete(rawlist);
|
||||||
|
}
|
||||||
|
|
||||||
static int write_trace_kprobe_event(int fd, const char *buf)
|
static int write_trace_kprobe_event(int fd, const char *buf)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
@ -216,16 +422,7 @@ void add_trace_kprobe_events(struct probe_point *probes, int nr_probes)
|
||||||
struct probe_point *pp;
|
struct probe_point *pp;
|
||||||
char buf[MAX_CMDLEN];
|
char buf[MAX_CMDLEN];
|
||||||
|
|
||||||
snprintf(buf, MAX_CMDLEN, "%s/../kprobe_events", debugfs_path);
|
fd = open_kprobe_events(O_WRONLY, O_APPEND);
|
||||||
fd = open(buf, O_WRONLY, O_APPEND);
|
|
||||||
if (fd < 0) {
|
|
||||||
if (errno == ENOENT)
|
|
||||||
die("kprobe_events file does not exist -"
|
|
||||||
" please rebuild with CONFIG_KPROBE_TRACER.");
|
|
||||||
else
|
|
||||||
die("Could not open kprobe_events file: %s",
|
|
||||||
strerror(errno));
|
|
||||||
}
|
|
||||||
|
|
||||||
for (j = 0; j < nr_probes; j++) {
|
for (j = 0; j < nr_probes; j++) {
|
||||||
pp = probes + j;
|
pp = probes + j;
|
||||||
|
|
|
@ -2,9 +2,14 @@
|
||||||
#define _PROBE_EVENT_H
|
#define _PROBE_EVENT_H
|
||||||
|
|
||||||
#include "probe-finder.h"
|
#include "probe-finder.h"
|
||||||
|
#include "strlist.h"
|
||||||
|
|
||||||
extern int parse_perf_probe_event(const char *str, struct probe_point *pp);
|
extern int parse_perf_probe_event(const char *str, struct probe_point *pp);
|
||||||
|
extern int synthesize_perf_probe_event(struct probe_point *pp);
|
||||||
|
extern void parse_trace_kprobe_event(const char *str, char **group,
|
||||||
|
char **event, struct probe_point *pp);
|
||||||
extern int synthesize_trace_kprobe_event(struct probe_point *pp);
|
extern int synthesize_trace_kprobe_event(struct probe_point *pp);
|
||||||
extern void add_trace_kprobe_events(struct probe_point *probes, int nr_probes);
|
extern void add_trace_kprobe_events(struct probe_point *probes, int nr_probes);
|
||||||
|
extern void show_perf_probe_events(void);
|
||||||
|
|
||||||
#endif /*_PROBE_EVENT_H */
|
#endif /*_PROBE_EVENT_H */
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue