2015-07-15 17:14:07 +08:00
|
|
|
/*
|
|
|
|
* probe-file.c : operate ftrace k/uprobe events files
|
|
|
|
*
|
|
|
|
* Written by Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
*/
|
2016-06-15 11:28:30 +08:00
|
|
|
#include <sys/uio.h>
|
2015-07-15 17:14:07 +08:00
|
|
|
#include "util.h"
|
|
|
|
#include "event.h"
|
|
|
|
#include "strlist.h"
|
|
|
|
#include "debug.h"
|
|
|
|
#include "cache.h"
|
|
|
|
#include "color.h"
|
|
|
|
#include "symbol.h"
|
|
|
|
#include "thread.h"
|
2015-09-02 15:56:45 +08:00
|
|
|
#include <api/fs/tracing_path.h>
|
2015-07-15 17:14:07 +08:00
|
|
|
#include "probe-event.h"
|
|
|
|
#include "probe-file.h"
|
|
|
|
#include "session.h"
|
|
|
|
|
|
|
|
#define MAX_CMDLEN 256
|
|
|
|
|
|
|
|
static void print_open_warning(int err, bool uprobe)
|
|
|
|
{
|
|
|
|
char sbuf[STRERR_BUFSIZE];
|
|
|
|
|
|
|
|
if (err == -ENOENT) {
|
|
|
|
const char *config;
|
|
|
|
|
|
|
|
if (uprobe)
|
|
|
|
config = "CONFIG_UPROBE_EVENTS";
|
|
|
|
else
|
|
|
|
config = "CONFIG_KPROBE_EVENTS";
|
|
|
|
|
|
|
|
pr_warning("%cprobe_events file does not exist"
|
|
|
|
" - please rebuild kernel with %s.\n",
|
|
|
|
uprobe ? 'u' : 'k', config);
|
|
|
|
} else if (err == -ENOTSUP)
|
|
|
|
pr_warning("Tracefs or debugfs is not mounted.\n");
|
|
|
|
else
|
|
|
|
pr_warning("Failed to open %cprobe_events: %s\n",
|
|
|
|
uprobe ? 'u' : 'k',
|
tools: Introduce str_error_r()
The tools so far have been using the strerror_r() GNU variant, that
returns a string, be it the buffer passed or something else.
But that, besides being tricky in cases where we expect that the
function using strerror_r() returns the error formatted in a provided
buffer (we have to check if it returned something else and copy that
instead), breaks the build on systems not using glibc, like Alpine
Linux, where musl libc is used.
So, introduce yet another wrapper, str_error_r(), that has the GNU
interface, but uses the portable XSI variant of strerror_r(), so that
users rest asured that the provided buffer is used and it is what is
returned.
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: David Ahern <dsahern@gmail.com>
Cc: Jiri Olsa <jolsa@kernel.org>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Wang Nan <wangnan0@huawei.com>
Link: http://lkml.kernel.org/n/tip-d4t42fnf48ytlk8rjxs822tf@git.kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2016-07-06 22:56:20 +08:00
|
|
|
str_error_r(-err, sbuf, sizeof(sbuf)));
|
2015-07-15 17:14:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void print_both_open_warning(int kerr, int uerr)
|
|
|
|
{
|
|
|
|
/* Both kprobes and uprobes are disabled, warn it. */
|
|
|
|
if (kerr == -ENOTSUP && uerr == -ENOTSUP)
|
|
|
|
pr_warning("Tracefs or debugfs is not mounted.\n");
|
|
|
|
else if (kerr == -ENOENT && uerr == -ENOENT)
|
|
|
|
pr_warning("Please rebuild kernel with CONFIG_KPROBE_EVENTS "
|
|
|
|
"or/and CONFIG_UPROBE_EVENTS.\n");
|
|
|
|
else {
|
|
|
|
char sbuf[STRERR_BUFSIZE];
|
|
|
|
pr_warning("Failed to open kprobe events: %s.\n",
|
tools: Introduce str_error_r()
The tools so far have been using the strerror_r() GNU variant, that
returns a string, be it the buffer passed or something else.
But that, besides being tricky in cases where we expect that the
function using strerror_r() returns the error formatted in a provided
buffer (we have to check if it returned something else and copy that
instead), breaks the build on systems not using glibc, like Alpine
Linux, where musl libc is used.
So, introduce yet another wrapper, str_error_r(), that has the GNU
interface, but uses the portable XSI variant of strerror_r(), so that
users rest asured that the provided buffer is used and it is what is
returned.
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: David Ahern <dsahern@gmail.com>
Cc: Jiri Olsa <jolsa@kernel.org>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Wang Nan <wangnan0@huawei.com>
Link: http://lkml.kernel.org/n/tip-d4t42fnf48ytlk8rjxs822tf@git.kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2016-07-06 22:56:20 +08:00
|
|
|
str_error_r(-kerr, sbuf, sizeof(sbuf)));
|
2015-07-15 17:14:07 +08:00
|
|
|
pr_warning("Failed to open uprobe events: %s.\n",
|
tools: Introduce str_error_r()
The tools so far have been using the strerror_r() GNU variant, that
returns a string, be it the buffer passed or something else.
But that, besides being tricky in cases where we expect that the
function using strerror_r() returns the error formatted in a provided
buffer (we have to check if it returned something else and copy that
instead), breaks the build on systems not using glibc, like Alpine
Linux, where musl libc is used.
So, introduce yet another wrapper, str_error_r(), that has the GNU
interface, but uses the portable XSI variant of strerror_r(), so that
users rest asured that the provided buffer is used and it is what is
returned.
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: David Ahern <dsahern@gmail.com>
Cc: Jiri Olsa <jolsa@kernel.org>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Wang Nan <wangnan0@huawei.com>
Link: http://lkml.kernel.org/n/tip-d4t42fnf48ytlk8rjxs822tf@git.kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2016-07-06 22:56:20 +08:00
|
|
|
str_error_r(-uerr, sbuf, sizeof(sbuf)));
|
2015-07-15 17:14:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int open_probe_events(const char *trace_file, bool readwrite)
|
|
|
|
{
|
|
|
|
char buf[PATH_MAX];
|
|
|
|
const char *tracing_dir = "";
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = e_snprintf(buf, PATH_MAX, "%s/%s%s",
|
2015-09-02 15:56:45 +08:00
|
|
|
tracing_path, tracing_dir, trace_file);
|
2015-07-15 17:14:07 +08:00
|
|
|
if (ret >= 0) {
|
|
|
|
pr_debug("Opening %s write=%d\n", buf, readwrite);
|
|
|
|
if (readwrite && !probe_event_dry_run)
|
|
|
|
ret = open(buf, O_RDWR | O_APPEND, 0);
|
|
|
|
else
|
|
|
|
ret = open(buf, O_RDONLY, 0);
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
ret = -errno;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int open_kprobe_events(bool readwrite)
|
|
|
|
{
|
|
|
|
return open_probe_events("kprobe_events", readwrite);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int open_uprobe_events(bool readwrite)
|
|
|
|
{
|
|
|
|
return open_probe_events("uprobe_events", readwrite);
|
|
|
|
}
|
|
|
|
|
|
|
|
int probe_file__open(int flag)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
if (flag & PF_FL_UPROBE)
|
|
|
|
fd = open_uprobe_events(flag & PF_FL_RW);
|
|
|
|
else
|
|
|
|
fd = open_kprobe_events(flag & PF_FL_RW);
|
|
|
|
if (fd < 0)
|
|
|
|
print_open_warning(fd, flag & PF_FL_UPROBE);
|
|
|
|
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
int probe_file__open_both(int *kfd, int *ufd, int flag)
|
|
|
|
{
|
|
|
|
if (!kfd || !ufd)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
*kfd = open_kprobe_events(flag & PF_FL_RW);
|
|
|
|
*ufd = open_uprobe_events(flag & PF_FL_RW);
|
|
|
|
if (*kfd < 0 && *ufd < 0) {
|
|
|
|
print_both_open_warning(*kfd, *ufd);
|
|
|
|
return *kfd;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get raw string list of current kprobe_events or uprobe_events */
|
|
|
|
struct strlist *probe_file__get_rawlist(int fd)
|
|
|
|
{
|
2016-08-13 05:44:56 +08:00
|
|
|
int ret, idx, fddup;
|
2015-07-15 17:14:07 +08:00
|
|
|
FILE *fp;
|
|
|
|
char buf[MAX_CMDLEN];
|
|
|
|
char *p;
|
|
|
|
struct strlist *sl;
|
|
|
|
|
2015-11-06 17:50:15 +08:00
|
|
|
if (fd < 0)
|
|
|
|
return NULL;
|
|
|
|
|
2015-07-15 17:14:07 +08:00
|
|
|
sl = strlist__new(NULL, NULL);
|
2016-08-16 04:06:47 +08:00
|
|
|
if (sl == NULL)
|
|
|
|
return NULL;
|
2015-07-15 17:14:07 +08:00
|
|
|
|
2016-08-13 05:44:56 +08:00
|
|
|
fddup = dup(fd);
|
|
|
|
if (fddup < 0)
|
|
|
|
goto out_free_sl;
|
|
|
|
|
|
|
|
fp = fdopen(fddup, "r");
|
|
|
|
if (!fp)
|
|
|
|
goto out_close_fddup;
|
|
|
|
|
2015-07-15 17:14:07 +08:00
|
|
|
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) {
|
|
|
|
pr_debug("strlist__add failed (%d)\n", ret);
|
2016-08-16 04:06:47 +08:00
|
|
|
goto out_close_fp;
|
2015-07-15 17:14:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
|
|
|
|
return sl;
|
2016-08-13 05:44:56 +08:00
|
|
|
|
2016-08-16 04:06:47 +08:00
|
|
|
out_close_fp:
|
|
|
|
fclose(fp);
|
|
|
|
goto out_free_sl;
|
2016-08-13 05:44:56 +08:00
|
|
|
out_close_fddup:
|
|
|
|
close(fddup);
|
|
|
|
out_free_sl:
|
|
|
|
strlist__delete(sl);
|
|
|
|
return NULL;
|
2015-07-15 17:14:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct strlist *__probe_file__get_namelist(int fd, bool include_group)
|
|
|
|
{
|
|
|
|
char buf[128];
|
|
|
|
struct strlist *sl, *rawlist;
|
|
|
|
struct str_node *ent;
|
|
|
|
struct probe_trace_event tev;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
memset(&tev, 0, sizeof(tev));
|
|
|
|
rawlist = probe_file__get_rawlist(fd);
|
|
|
|
if (!rawlist)
|
|
|
|
return NULL;
|
|
|
|
sl = strlist__new(NULL, NULL);
|
2016-06-23 22:31:20 +08:00
|
|
|
strlist__for_each_entry(ent, rawlist) {
|
2015-07-15 17:14:07 +08:00
|
|
|
ret = parse_probe_trace_command(ent->s, &tev);
|
|
|
|
if (ret < 0)
|
|
|
|
break;
|
|
|
|
if (include_group) {
|
|
|
|
ret = e_snprintf(buf, 128, "%s:%s", tev.group,
|
|
|
|
tev.event);
|
|
|
|
if (ret >= 0)
|
|
|
|
ret = strlist__add(sl, buf);
|
|
|
|
} else
|
|
|
|
ret = strlist__add(sl, tev.event);
|
|
|
|
clear_probe_trace_event(&tev);
|
|
|
|
if (ret < 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
strlist__delete(rawlist);
|
|
|
|
|
|
|
|
if (ret < 0) {
|
|
|
|
strlist__delete(sl);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return sl;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get current perf-probe event names */
|
|
|
|
struct strlist *probe_file__get_namelist(int fd)
|
|
|
|
{
|
|
|
|
return __probe_file__get_namelist(fd, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
int probe_file__add_event(int fd, struct probe_trace_event *tev)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
char *buf = synthesize_probe_trace_command(tev);
|
|
|
|
char sbuf[STRERR_BUFSIZE];
|
|
|
|
|
|
|
|
if (!buf) {
|
|
|
|
pr_debug("Failed to synthesize probe trace event.\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
pr_debug("Writing event: %s\n", buf);
|
|
|
|
if (!probe_event_dry_run) {
|
2016-04-26 17:03:04 +08:00
|
|
|
if (write(fd, buf, strlen(buf)) < (int)strlen(buf)) {
|
2015-07-15 17:14:07 +08:00
|
|
|
ret = -errno;
|
|
|
|
pr_warning("Failed to write event: %s\n",
|
tools: Introduce str_error_r()
The tools so far have been using the strerror_r() GNU variant, that
returns a string, be it the buffer passed or something else.
But that, besides being tricky in cases where we expect that the
function using strerror_r() returns the error formatted in a provided
buffer (we have to check if it returned something else and copy that
instead), breaks the build on systems not using glibc, like Alpine
Linux, where musl libc is used.
So, introduce yet another wrapper, str_error_r(), that has the GNU
interface, but uses the portable XSI variant of strerror_r(), so that
users rest asured that the provided buffer is used and it is what is
returned.
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: David Ahern <dsahern@gmail.com>
Cc: Jiri Olsa <jolsa@kernel.org>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Wang Nan <wangnan0@huawei.com>
Link: http://lkml.kernel.org/n/tip-d4t42fnf48ytlk8rjxs822tf@git.kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2016-07-06 22:56:20 +08:00
|
|
|
str_error_r(errno, sbuf, sizeof(sbuf)));
|
2015-07-15 17:14:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
free(buf);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __del_trace_probe_event(int fd, struct str_node *ent)
|
|
|
|
{
|
|
|
|
char *p;
|
|
|
|
char buf[128];
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Convert from perf-probe event to trace-probe event */
|
|
|
|
ret = e_snprintf(buf, 128, "-:%s", ent->s);
|
|
|
|
if (ret < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
p = strchr(buf + 2, ':');
|
|
|
|
if (!p) {
|
|
|
|
pr_debug("Internal error: %s should have ':' but not.\n",
|
|
|
|
ent->s);
|
|
|
|
ret = -ENOTSUP;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
*p = '/';
|
|
|
|
|
|
|
|
pr_debug("Writing event: %s\n", buf);
|
|
|
|
ret = write(fd, buf, strlen(buf));
|
|
|
|
if (ret < 0) {
|
|
|
|
ret = -errno;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
error:
|
|
|
|
pr_warning("Failed to delete event: %s\n",
|
tools: Introduce str_error_r()
The tools so far have been using the strerror_r() GNU variant, that
returns a string, be it the buffer passed or something else.
But that, besides being tricky in cases where we expect that the
function using strerror_r() returns the error formatted in a provided
buffer (we have to check if it returned something else and copy that
instead), breaks the build on systems not using glibc, like Alpine
Linux, where musl libc is used.
So, introduce yet another wrapper, str_error_r(), that has the GNU
interface, but uses the portable XSI variant of strerror_r(), so that
users rest asured that the provided buffer is used and it is what is
returned.
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: David Ahern <dsahern@gmail.com>
Cc: Jiri Olsa <jolsa@kernel.org>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Wang Nan <wangnan0@huawei.com>
Link: http://lkml.kernel.org/n/tip-d4t42fnf48ytlk8rjxs822tf@git.kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2016-07-06 22:56:20 +08:00
|
|
|
str_error_r(-ret, buf, sizeof(buf)));
|
2015-07-15 17:14:07 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-09-04 20:16:03 +08:00
|
|
|
int probe_file__get_events(int fd, struct strfilter *filter,
|
|
|
|
struct strlist *plist)
|
2015-07-15 17:14:07 +08:00
|
|
|
{
|
|
|
|
struct strlist *namelist;
|
|
|
|
struct str_node *ent;
|
|
|
|
const char *p;
|
|
|
|
int ret = -ENOENT;
|
|
|
|
|
2015-11-06 17:50:15 +08:00
|
|
|
if (!plist)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2015-07-15 17:14:07 +08:00
|
|
|
namelist = __probe_file__get_namelist(fd, true);
|
|
|
|
if (!namelist)
|
|
|
|
return -ENOENT;
|
|
|
|
|
2016-06-23 22:31:20 +08:00
|
|
|
strlist__for_each_entry(ent, namelist) {
|
2015-07-15 17:14:07 +08:00
|
|
|
p = strchr(ent->s, ':');
|
|
|
|
if ((p && strfilter__compare(filter, p + 1)) ||
|
|
|
|
strfilter__compare(filter, ent->s)) {
|
2015-09-04 20:16:02 +08:00
|
|
|
strlist__add(plist, ent->s);
|
|
|
|
ret = 0;
|
2015-07-15 17:14:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
strlist__delete(namelist);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2015-09-04 20:16:02 +08:00
|
|
|
|
2015-09-04 20:16:03 +08:00
|
|
|
int probe_file__del_strlist(int fd, struct strlist *namelist)
|
2015-09-04 20:16:02 +08:00
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
struct str_node *ent;
|
|
|
|
|
2016-06-23 22:31:20 +08:00
|
|
|
strlist__for_each_entry(ent, namelist) {
|
2015-09-04 20:16:02 +08:00
|
|
|
ret = __del_trace_probe_event(fd, ent);
|
|
|
|
if (ret < 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int probe_file__del_events(int fd, struct strfilter *filter)
|
|
|
|
{
|
|
|
|
struct strlist *namelist;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
namelist = strlist__new(NULL, NULL);
|
|
|
|
if (!namelist)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
ret = probe_file__get_events(fd, filter, namelist);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = probe_file__del_strlist(fd, namelist);
|
|
|
|
strlist__delete(namelist);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2016-06-15 11:28:30 +08:00
|
|
|
|
|
|
|
/* Caller must ensure to remove this entry from list */
|
|
|
|
static void probe_cache_entry__delete(struct probe_cache_entry *entry)
|
|
|
|
{
|
|
|
|
if (entry) {
|
|
|
|
BUG_ON(!list_empty(&entry->node));
|
|
|
|
|
|
|
|
strlist__delete(entry->tevlist);
|
|
|
|
clear_perf_probe_event(&entry->pev);
|
|
|
|
zfree(&entry->spev);
|
|
|
|
free(entry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct probe_cache_entry *
|
|
|
|
probe_cache_entry__new(struct perf_probe_event *pev)
|
|
|
|
{
|
|
|
|
struct probe_cache_entry *entry = zalloc(sizeof(*entry));
|
|
|
|
|
|
|
|
if (entry) {
|
|
|
|
INIT_LIST_HEAD(&entry->node);
|
|
|
|
entry->tevlist = strlist__new(NULL, NULL);
|
|
|
|
if (!entry->tevlist)
|
|
|
|
zfree(&entry);
|
|
|
|
else if (pev) {
|
|
|
|
entry->spev = synthesize_perf_probe_command(pev);
|
|
|
|
if (!entry->spev ||
|
|
|
|
perf_probe_event__copy(&entry->pev, pev) < 0) {
|
|
|
|
probe_cache_entry__delete(entry);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
2016-07-12 18:05:18 +08:00
|
|
|
int probe_cache_entry__get_event(struct probe_cache_entry *entry,
|
|
|
|
struct probe_trace_event **tevs)
|
|
|
|
{
|
|
|
|
struct probe_trace_event *tev;
|
|
|
|
struct str_node *node;
|
|
|
|
int ret, i;
|
|
|
|
|
|
|
|
ret = strlist__nr_entries(entry->tevlist);
|
|
|
|
if (ret > probe_conf.max_probes)
|
|
|
|
return -E2BIG;
|
|
|
|
|
|
|
|
*tevs = zalloc(ret * sizeof(*tev));
|
|
|
|
if (!*tevs)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
strlist__for_each_entry(node, entry->tevlist) {
|
|
|
|
tev = &(*tevs)[i++];
|
|
|
|
ret = parse_probe_trace_command(node->s, tev);
|
|
|
|
if (ret < 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* For the kernel probe caches, pass target = NULL or DSO__NAME_KALLSYMS */
|
2016-06-15 11:28:30 +08:00
|
|
|
static int probe_cache__open(struct probe_cache *pcache, const char *target)
|
|
|
|
{
|
|
|
|
char cpath[PATH_MAX];
|
|
|
|
char sbuildid[SBUILD_ID_SIZE];
|
2016-07-01 16:03:26 +08:00
|
|
|
char *dir_name = NULL;
|
2016-07-12 18:05:18 +08:00
|
|
|
bool is_kallsyms = false;
|
2016-06-15 11:28:30 +08:00
|
|
|
int ret, fd;
|
|
|
|
|
2016-07-01 16:03:26 +08:00
|
|
|
if (target && build_id_cache__cached(target)) {
|
|
|
|
/* This is a cached buildid */
|
|
|
|
strncpy(sbuildid, target, SBUILD_ID_SIZE);
|
|
|
|
dir_name = build_id_cache__linkname(sbuildid, NULL, 0);
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
|
2016-07-12 18:05:18 +08:00
|
|
|
if (!target || !strcmp(target, DSO__NAME_KALLSYMS)) {
|
2016-06-15 11:28:30 +08:00
|
|
|
target = DSO__NAME_KALLSYMS;
|
2016-07-12 18:05:18 +08:00
|
|
|
is_kallsyms = true;
|
2016-06-15 11:28:30 +08:00
|
|
|
ret = sysfs__sprintf_build_id("/", sbuildid);
|
2016-07-12 18:05:18 +08:00
|
|
|
} else
|
|
|
|
ret = filename__sprintf_build_id(target, sbuildid);
|
|
|
|
|
2016-06-15 11:28:30 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
pr_debug("Failed to get build-id from %s.\n", target);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we have no buildid cache, make it */
|
|
|
|
if (!build_id_cache__cached(sbuildid)) {
|
|
|
|
ret = build_id_cache__add_s(sbuildid, target,
|
|
|
|
is_kallsyms, NULL);
|
|
|
|
if (ret < 0) {
|
|
|
|
pr_debug("Failed to add build-id cache: %s\n", target);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
dir_name = build_id_cache__cachedir(sbuildid, target, is_kallsyms,
|
|
|
|
false);
|
2016-07-01 16:03:26 +08:00
|
|
|
found:
|
|
|
|
if (!dir_name) {
|
|
|
|
pr_debug("Failed to get cache from %s\n", target);
|
2016-06-15 11:28:30 +08:00
|
|
|
return -ENOMEM;
|
2016-07-01 16:03:26 +08:00
|
|
|
}
|
2016-06-15 11:28:30 +08:00
|
|
|
|
|
|
|
snprintf(cpath, PATH_MAX, "%s/probes", dir_name);
|
|
|
|
fd = open(cpath, O_CREAT | O_RDWR, 0644);
|
|
|
|
if (fd < 0)
|
|
|
|
pr_debug("Failed to open cache(%d): %s\n", fd, cpath);
|
|
|
|
free(dir_name);
|
|
|
|
pcache->fd = fd;
|
|
|
|
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int probe_cache__load(struct probe_cache *pcache)
|
|
|
|
{
|
|
|
|
struct probe_cache_entry *entry = NULL;
|
|
|
|
char buf[MAX_CMDLEN], *p;
|
2016-08-13 05:44:56 +08:00
|
|
|
int ret = 0, fddup;
|
2016-06-15 11:28:30 +08:00
|
|
|
FILE *fp;
|
|
|
|
|
2016-08-13 05:44:56 +08:00
|
|
|
fddup = dup(pcache->fd);
|
|
|
|
if (fddup < 0)
|
|
|
|
return -errno;
|
|
|
|
fp = fdopen(fddup, "r");
|
2016-08-16 04:06:47 +08:00
|
|
|
if (!fp) {
|
|
|
|
close(fddup);
|
2016-06-15 11:28:30 +08:00
|
|
|
return -EINVAL;
|
2016-08-16 04:06:47 +08:00
|
|
|
}
|
2016-06-15 11:28:30 +08:00
|
|
|
|
|
|
|
while (!feof(fp)) {
|
|
|
|
if (!fgets(buf, MAX_CMDLEN, fp))
|
|
|
|
break;
|
|
|
|
p = strchr(buf, '\n');
|
|
|
|
if (p)
|
|
|
|
*p = '\0';
|
2016-07-01 16:04:10 +08:00
|
|
|
/* #perf_probe_event or %sdt_event */
|
|
|
|
if (buf[0] == '#' || buf[0] == '%') {
|
2016-06-15 11:28:30 +08:00
|
|
|
entry = probe_cache_entry__new(NULL);
|
|
|
|
if (!entry) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out;
|
|
|
|
}
|
2016-07-01 16:04:10 +08:00
|
|
|
if (buf[0] == '%')
|
|
|
|
entry->sdt = true;
|
2016-06-15 11:28:30 +08:00
|
|
|
entry->spev = strdup(buf + 1);
|
|
|
|
if (entry->spev)
|
|
|
|
ret = parse_perf_probe_command(buf + 1,
|
|
|
|
&entry->pev);
|
|
|
|
else
|
|
|
|
ret = -ENOMEM;
|
|
|
|
if (ret < 0) {
|
|
|
|
probe_cache_entry__delete(entry);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
list_add_tail(&entry->node, &pcache->entries);
|
|
|
|
} else { /* trace_probe_event */
|
|
|
|
if (!entry) {
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
strlist__add(entry->tevlist, buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
fclose(fp);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct probe_cache *probe_cache__alloc(void)
|
|
|
|
{
|
|
|
|
struct probe_cache *pcache = zalloc(sizeof(*pcache));
|
|
|
|
|
|
|
|
if (pcache) {
|
|
|
|
INIT_LIST_HEAD(&pcache->entries);
|
|
|
|
pcache->fd = -EINVAL;
|
|
|
|
}
|
|
|
|
return pcache;
|
|
|
|
}
|
|
|
|
|
|
|
|
void probe_cache__purge(struct probe_cache *pcache)
|
|
|
|
{
|
|
|
|
struct probe_cache_entry *entry, *n;
|
|
|
|
|
|
|
|
list_for_each_entry_safe(entry, n, &pcache->entries, node) {
|
|
|
|
list_del_init(&entry->node);
|
|
|
|
probe_cache_entry__delete(entry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void probe_cache__delete(struct probe_cache *pcache)
|
|
|
|
{
|
|
|
|
if (!pcache)
|
|
|
|
return;
|
|
|
|
|
|
|
|
probe_cache__purge(pcache);
|
|
|
|
if (pcache->fd > 0)
|
|
|
|
close(pcache->fd);
|
|
|
|
free(pcache);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct probe_cache *probe_cache__new(const char *target)
|
|
|
|
{
|
|
|
|
struct probe_cache *pcache = probe_cache__alloc();
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!pcache)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
ret = probe_cache__open(pcache, target);
|
|
|
|
if (ret < 0) {
|
|
|
|
pr_debug("Cache open error: %d\n", ret);
|
|
|
|
goto out_err;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = probe_cache__load(pcache);
|
|
|
|
if (ret < 0) {
|
|
|
|
pr_debug("Cache read error: %d\n", ret);
|
|
|
|
goto out_err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return pcache;
|
|
|
|
|
|
|
|
out_err:
|
|
|
|
probe_cache__delete(pcache);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool streql(const char *a, const char *b)
|
|
|
|
{
|
|
|
|
if (a == b)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (!a || !b)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return !strcmp(a, b);
|
|
|
|
}
|
|
|
|
|
2016-07-01 16:03:12 +08:00
|
|
|
struct probe_cache_entry *
|
2016-06-15 11:28:30 +08:00
|
|
|
probe_cache__find(struct probe_cache *pcache, struct perf_probe_event *pev)
|
|
|
|
{
|
|
|
|
struct probe_cache_entry *entry = NULL;
|
|
|
|
char *cmd = synthesize_perf_probe_command(pev);
|
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
return NULL;
|
|
|
|
|
2016-07-12 18:05:04 +08:00
|
|
|
for_each_probe_cache_entry(entry, pcache) {
|
2016-07-12 18:04:43 +08:00
|
|
|
if (pev->sdt) {
|
|
|
|
if (entry->pev.event &&
|
|
|
|
streql(entry->pev.event, pev->event) &&
|
|
|
|
(!pev->group ||
|
|
|
|
streql(entry->pev.group, pev->group)))
|
|
|
|
goto found;
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
2016-06-15 11:28:30 +08:00
|
|
|
/* Hit if same event name or same command-string */
|
|
|
|
if ((pev->event &&
|
|
|
|
(streql(entry->pev.group, pev->group) &&
|
|
|
|
streql(entry->pev.event, pev->event))) ||
|
|
|
|
(!strcmp(entry->spev, cmd)))
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
entry = NULL;
|
|
|
|
|
|
|
|
found:
|
|
|
|
free(cmd);
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
2016-07-01 16:03:12 +08:00
|
|
|
struct probe_cache_entry *
|
|
|
|
probe_cache__find_by_name(struct probe_cache *pcache,
|
|
|
|
const char *group, const char *event)
|
|
|
|
{
|
|
|
|
struct probe_cache_entry *entry = NULL;
|
|
|
|
|
2016-07-12 18:05:04 +08:00
|
|
|
for_each_probe_cache_entry(entry, pcache) {
|
2016-07-01 16:03:12 +08:00
|
|
|
/* Hit if same event name or same command-string */
|
|
|
|
if (streql(entry->pev.group, group) &&
|
|
|
|
streql(entry->pev.event, event))
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
entry = NULL;
|
|
|
|
|
|
|
|
found:
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
2016-06-15 11:28:30 +08:00
|
|
|
int probe_cache__add_entry(struct probe_cache *pcache,
|
|
|
|
struct perf_probe_event *pev,
|
|
|
|
struct probe_trace_event *tevs, int ntevs)
|
|
|
|
{
|
|
|
|
struct probe_cache_entry *entry = NULL;
|
|
|
|
char *command;
|
|
|
|
int i, ret = 0;
|
|
|
|
|
|
|
|
if (!pcache || !pev || !tevs || ntevs <= 0) {
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out_err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Remove old cache entry */
|
|
|
|
entry = probe_cache__find(pcache, pev);
|
|
|
|
if (entry) {
|
|
|
|
list_del_init(&entry->node);
|
|
|
|
probe_cache_entry__delete(entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = -ENOMEM;
|
|
|
|
entry = probe_cache_entry__new(pev);
|
|
|
|
if (!entry)
|
|
|
|
goto out_err;
|
|
|
|
|
|
|
|
for (i = 0; i < ntevs; i++) {
|
|
|
|
if (!tevs[i].point.symbol)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
command = synthesize_probe_trace_command(&tevs[i]);
|
|
|
|
if (!command)
|
|
|
|
goto out_err;
|
|
|
|
strlist__add(entry->tevlist, command);
|
|
|
|
free(command);
|
|
|
|
}
|
|
|
|
list_add_tail(&entry->node, &pcache->entries);
|
|
|
|
pr_debug("Added probe cache: %d\n", ntevs);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_err:
|
|
|
|
pr_debug("Failed to add probe caches\n");
|
|
|
|
probe_cache_entry__delete(entry);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-07-12 23:19:09 +08:00
|
|
|
#ifdef HAVE_GELF_GETNOTE_SUPPORT
|
2016-07-01 16:04:10 +08:00
|
|
|
static unsigned long long sdt_note__get_addr(struct sdt_note *note)
|
|
|
|
{
|
|
|
|
return note->bit32 ? (unsigned long long)note->addr.a32[0]
|
|
|
|
: (unsigned long long)note->addr.a64[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
int probe_cache__scan_sdt(struct probe_cache *pcache, const char *pathname)
|
|
|
|
{
|
|
|
|
struct probe_cache_entry *entry = NULL;
|
|
|
|
struct list_head sdtlist;
|
|
|
|
struct sdt_note *note;
|
|
|
|
char *buf;
|
|
|
|
char sdtgrp[64];
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
INIT_LIST_HEAD(&sdtlist);
|
|
|
|
ret = get_sdt_note_list(&sdtlist, pathname);
|
|
|
|
if (ret < 0) {
|
|
|
|
pr_debug("Failed to get sdt note: %d\n", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
list_for_each_entry(note, &sdtlist, note_list) {
|
|
|
|
ret = snprintf(sdtgrp, 64, "sdt_%s", note->provider);
|
|
|
|
if (ret < 0)
|
|
|
|
break;
|
|
|
|
/* Try to find same-name entry */
|
|
|
|
entry = probe_cache__find_by_name(pcache, sdtgrp, note->name);
|
|
|
|
if (!entry) {
|
|
|
|
entry = probe_cache_entry__new(NULL);
|
|
|
|
if (!entry) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
entry->sdt = true;
|
|
|
|
ret = asprintf(&entry->spev, "%s:%s=%s", sdtgrp,
|
|
|
|
note->name, note->name);
|
|
|
|
if (ret < 0)
|
|
|
|
break;
|
|
|
|
entry->pev.event = strdup(note->name);
|
|
|
|
entry->pev.group = strdup(sdtgrp);
|
|
|
|
list_add_tail(&entry->node, &pcache->entries);
|
|
|
|
}
|
|
|
|
ret = asprintf(&buf, "p:%s/%s %s:0x%llx",
|
|
|
|
sdtgrp, note->name, pathname,
|
|
|
|
sdt_note__get_addr(note));
|
|
|
|
if (ret < 0)
|
|
|
|
break;
|
|
|
|
strlist__add(entry->tevlist, buf);
|
|
|
|
free(buf);
|
|
|
|
entry = NULL;
|
|
|
|
}
|
|
|
|
if (entry) {
|
|
|
|
list_del_init(&entry->node);
|
|
|
|
probe_cache_entry__delete(entry);
|
|
|
|
}
|
|
|
|
cleanup_sdt_note_list(&sdtlist);
|
|
|
|
return ret;
|
|
|
|
}
|
2016-07-12 23:19:09 +08:00
|
|
|
#endif
|
2016-07-01 16:04:10 +08:00
|
|
|
|
2016-06-15 11:28:30 +08:00
|
|
|
static int probe_cache_entry__write(struct probe_cache_entry *entry, int fd)
|
|
|
|
{
|
|
|
|
struct str_node *snode;
|
|
|
|
struct stat st;
|
|
|
|
struct iovec iov[3];
|
2016-07-01 16:04:10 +08:00
|
|
|
const char *prefix = entry->sdt ? "%" : "#";
|
2016-06-15 11:28:30 +08:00
|
|
|
int ret;
|
|
|
|
/* Save stat for rollback */
|
|
|
|
ret = fstat(fd, &st);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2016-07-01 16:04:10 +08:00
|
|
|
pr_debug("Writing cache: %s%s\n", prefix, entry->spev);
|
|
|
|
iov[0].iov_base = (void *)prefix; iov[0].iov_len = 1;
|
2016-06-15 11:28:30 +08:00
|
|
|
iov[1].iov_base = entry->spev; iov[1].iov_len = strlen(entry->spev);
|
|
|
|
iov[2].iov_base = (void *)"\n"; iov[2].iov_len = 1;
|
|
|
|
ret = writev(fd, iov, 3);
|
|
|
|
if (ret < (int)iov[1].iov_len + 2)
|
|
|
|
goto rollback;
|
|
|
|
|
2016-06-23 22:31:20 +08:00
|
|
|
strlist__for_each_entry(snode, entry->tevlist) {
|
2016-06-15 11:28:30 +08:00
|
|
|
iov[0].iov_base = (void *)snode->s;
|
|
|
|
iov[0].iov_len = strlen(snode->s);
|
|
|
|
iov[1].iov_base = (void *)"\n"; iov[1].iov_len = 1;
|
|
|
|
ret = writev(fd, iov, 2);
|
|
|
|
if (ret < (int)iov[0].iov_len + 1)
|
|
|
|
goto rollback;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
rollback:
|
|
|
|
/* Rollback to avoid cache file corruption */
|
|
|
|
if (ret > 0)
|
|
|
|
ret = -1;
|
|
|
|
if (ftruncate(fd, st.st_size) < 0)
|
|
|
|
ret = -2;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int probe_cache__commit(struct probe_cache *pcache)
|
|
|
|
{
|
|
|
|
struct probe_cache_entry *entry;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
/* TBD: if we do not update existing entries, skip it */
|
|
|
|
ret = lseek(pcache->fd, 0, SEEK_SET);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = ftruncate(pcache->fd, 0);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
2016-07-12 18:05:04 +08:00
|
|
|
for_each_probe_cache_entry(entry, pcache) {
|
2016-06-15 11:28:30 +08:00
|
|
|
ret = probe_cache_entry__write(entry, pcache->fd);
|
|
|
|
pr_debug("Cache committed: %d\n", ret);
|
|
|
|
if (ret < 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
2016-07-01 16:03:26 +08:00
|
|
|
|
2016-07-01 16:03:36 +08:00
|
|
|
static bool probe_cache_entry__compare(struct probe_cache_entry *entry,
|
|
|
|
struct strfilter *filter)
|
|
|
|
{
|
|
|
|
char buf[128], *ptr = entry->spev;
|
|
|
|
|
|
|
|
if (entry->pev.event) {
|
|
|
|
snprintf(buf, 128, "%s:%s", entry->pev.group, entry->pev.event);
|
|
|
|
ptr = buf;
|
|
|
|
}
|
|
|
|
return strfilter__compare(filter, ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
int probe_cache__filter_purge(struct probe_cache *pcache,
|
|
|
|
struct strfilter *filter)
|
|
|
|
{
|
|
|
|
struct probe_cache_entry *entry, *tmp;
|
|
|
|
|
|
|
|
list_for_each_entry_safe(entry, tmp, &pcache->entries, node) {
|
|
|
|
if (probe_cache_entry__compare(entry, filter)) {
|
|
|
|
pr_info("Removed cached event: %s\n", entry->spev);
|
|
|
|
list_del_init(&entry->node);
|
|
|
|
probe_cache_entry__delete(entry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-07-01 16:03:26 +08:00
|
|
|
static int probe_cache__show_entries(struct probe_cache *pcache,
|
|
|
|
struct strfilter *filter)
|
|
|
|
{
|
|
|
|
struct probe_cache_entry *entry;
|
|
|
|
|
2016-07-12 18:05:04 +08:00
|
|
|
for_each_probe_cache_entry(entry, pcache) {
|
2016-07-01 16:03:36 +08:00
|
|
|
if (probe_cache_entry__compare(entry, filter))
|
2016-07-01 16:03:26 +08:00
|
|
|
printf("%s\n", entry->spev);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Show all cached probes */
|
|
|
|
int probe_cache__show_all_caches(struct strfilter *filter)
|
|
|
|
{
|
|
|
|
struct probe_cache *pcache;
|
|
|
|
struct strlist *bidlist;
|
|
|
|
struct str_node *nd;
|
|
|
|
char *buf = strfilter__string(filter);
|
|
|
|
|
|
|
|
pr_debug("list cache with filter: %s\n", buf);
|
|
|
|
free(buf);
|
|
|
|
|
2016-07-12 18:04:54 +08:00
|
|
|
bidlist = build_id_cache__list_all(true);
|
2016-07-01 16:03:26 +08:00
|
|
|
if (!bidlist) {
|
|
|
|
pr_debug("Failed to get buildids: %d\n", errno);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
strlist__for_each_entry(nd, bidlist) {
|
|
|
|
pcache = probe_cache__new(nd->s);
|
|
|
|
if (!pcache)
|
|
|
|
continue;
|
|
|
|
if (!list_empty(&pcache->entries)) {
|
|
|
|
buf = build_id_cache__origname(nd->s);
|
|
|
|
printf("%s (%s):\n", buf, nd->s);
|
|
|
|
free(buf);
|
|
|
|
probe_cache__show_entries(pcache, filter);
|
|
|
|
}
|
|
|
|
probe_cache__delete(pcache);
|
|
|
|
}
|
|
|
|
strlist__delete(bidlist);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|