forked from luck/tmp_suning_uos_patched
perf probe: Support escaped character in parser
Support the special characters escaped by '\' in parser. This allows user to specify versions directly like below. ===== # ./perf probe -x /lib64/libc-2.25.so malloc_get_state\\@GLIBC_2.2.5 Added new event: probe_libc:malloc_get_state (on malloc_get_state@GLIBC_2.2.5 in /usr/lib64/libc-2.25.so) You can now use it in all perf tools, such as: perf record -e probe_libc:malloc_get_state -aR sleep 1 ===== Or, you can use separators in source filename, e.g. ===== # ./perf probe -x /opt/test/a.out foo+bar.c:3 Semantic error :There is non-digit character in offset. Error: Command Parse Error. ===== Usually "+" in source file cause parser error, but ===== # ./perf probe -x /opt/test/a.out foo\\+bar.c:4 Added new event: probe_a:main (on @foo+bar.c:4 in /opt/test/a.out) You can now use it in all perf tools, such as: perf record -e probe_a:main -aR sleep 1 ===== escaped "\+" allows you to specify that. Signed-off-by: Masami Hiramatsu <mhiramat@kernel.org> Reviewed-by: Thomas Richter <tmricht@linux.vnet.ibm.com> Acked-by: Ravi Bangoria <ravi.bangoria@linux.vnet.ibm.com> Cc: Paul Clarke <pc@us.ibm.com> Cc: bhargavb <bhargavaramudu@gmail.com> Cc: linux-rt-users@vger.kernel.org Link: http://lkml.kernel.org/r/151309111236.18107.5634753157435343410.stgit@devbox Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
This commit is contained in:
parent
1e9f9e8af0
commit
c588d15812
|
@ -182,6 +182,14 @@ Note that before using the SDT event, the target binary (on which SDT events are
|
||||||
For details of the SDT, see below.
|
For details of the SDT, see below.
|
||||||
https://sourceware.org/gdb/onlinedocs/gdb/Static-Probe-Points.html
|
https://sourceware.org/gdb/onlinedocs/gdb/Static-Probe-Points.html
|
||||||
|
|
||||||
|
ESCAPED CHARACTER
|
||||||
|
-----------------
|
||||||
|
|
||||||
|
In the probe syntax, '=', '@', '+', ':' and ';' are treated as a special character. You can use a backslash ('\') to escape the special characters.
|
||||||
|
This is useful if you need to probe on a specific versioned symbols, like @GLIBC_... suffixes, or also you need to specify a source file which includes the special characters.
|
||||||
|
Note that usually single backslash is consumed by shell, so you might need to pass double backslash (\\) or wrapping with single quotes (\'AAA\@BBB').
|
||||||
|
See EXAMPLES how it is used.
|
||||||
|
|
||||||
PROBE ARGUMENT
|
PROBE ARGUMENT
|
||||||
--------------
|
--------------
|
||||||
Each probe argument follows below syntax.
|
Each probe argument follows below syntax.
|
||||||
|
@ -277,6 +285,14 @@ Add a USDT probe to a target process running in a different mount namespace
|
||||||
|
|
||||||
./perf probe --target-ns <target pid> -x /usr/lib/jvm/java-1.8.0-openjdk-1.8.0.121-0.b13.el7_3.x86_64/jre/lib/amd64/server/libjvm.so %sdt_hotspot:thread__sleep__end
|
./perf probe --target-ns <target pid> -x /usr/lib/jvm/java-1.8.0-openjdk-1.8.0.121-0.b13.el7_3.x86_64/jre/lib/amd64/server/libjvm.so %sdt_hotspot:thread__sleep__end
|
||||||
|
|
||||||
|
Add a probe on specific versioned symbol by backslash escape
|
||||||
|
|
||||||
|
./perf probe -x /lib64/libc-2.25.so 'malloc_get_state\@GLIBC_2.2.5'
|
||||||
|
|
||||||
|
Add a probe in a source file using special characters by backslash escape
|
||||||
|
|
||||||
|
./perf probe -x /opt/test/a.out 'foo\+bar.c:4'
|
||||||
|
|
||||||
|
|
||||||
SEE ALSO
|
SEE ALSO
|
||||||
--------
|
--------
|
||||||
|
|
|
@ -1325,27 +1325,30 @@ static int parse_perf_probe_event_name(char **arg, struct perf_probe_event *pev)
|
||||||
{
|
{
|
||||||
char *ptr;
|
char *ptr;
|
||||||
|
|
||||||
ptr = strchr(*arg, ':');
|
ptr = strpbrk_esc(*arg, ":");
|
||||||
if (ptr) {
|
if (ptr) {
|
||||||
*ptr = '\0';
|
*ptr = '\0';
|
||||||
if (!pev->sdt && !is_c_func_name(*arg))
|
if (!pev->sdt && !is_c_func_name(*arg))
|
||||||
goto ng_name;
|
goto ng_name;
|
||||||
pev->group = strdup(*arg);
|
pev->group = strdup_esc(*arg);
|
||||||
if (!pev->group)
|
if (!pev->group)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
*arg = ptr + 1;
|
*arg = ptr + 1;
|
||||||
} else
|
} else
|
||||||
pev->group = NULL;
|
pev->group = NULL;
|
||||||
if (!pev->sdt && !is_c_func_name(*arg)) {
|
|
||||||
|
pev->event = strdup_esc(*arg);
|
||||||
|
if (pev->event == NULL)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
if (!pev->sdt && !is_c_func_name(pev->event)) {
|
||||||
|
zfree(&pev->event);
|
||||||
ng_name:
|
ng_name:
|
||||||
|
zfree(&pev->group);
|
||||||
semantic_error("%s is bad for event name -it must "
|
semantic_error("%s is bad for event name -it must "
|
||||||
"follow C symbol-naming rule.\n", *arg);
|
"follow C symbol-naming rule.\n", *arg);
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
pev->event = strdup(*arg);
|
|
||||||
if (pev->event == NULL)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1373,7 +1376,7 @@ static int parse_perf_probe_point(char *arg, struct perf_probe_event *pev)
|
||||||
arg++;
|
arg++;
|
||||||
}
|
}
|
||||||
|
|
||||||
ptr = strpbrk(arg, ";=@+%");
|
ptr = strpbrk_esc(arg, ";=@+%");
|
||||||
if (pev->sdt) {
|
if (pev->sdt) {
|
||||||
if (ptr) {
|
if (ptr) {
|
||||||
if (*ptr != '@') {
|
if (*ptr != '@') {
|
||||||
|
@ -1387,7 +1390,7 @@ static int parse_perf_probe_point(char *arg, struct perf_probe_event *pev)
|
||||||
pev->target = build_id_cache__origname(tmp);
|
pev->target = build_id_cache__origname(tmp);
|
||||||
free(tmp);
|
free(tmp);
|
||||||
} else
|
} else
|
||||||
pev->target = strdup(ptr + 1);
|
pev->target = strdup_esc(ptr + 1);
|
||||||
if (!pev->target)
|
if (!pev->target)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
*ptr = '\0';
|
*ptr = '\0';
|
||||||
|
@ -1421,13 +1424,14 @@ static int parse_perf_probe_point(char *arg, struct perf_probe_event *pev)
|
||||||
*
|
*
|
||||||
* Otherwise, we consider arg to be a function specification.
|
* Otherwise, we consider arg to be a function specification.
|
||||||
*/
|
*/
|
||||||
if (!strpbrk(arg, "+@%") && (ptr = strpbrk(arg, ";:")) != NULL) {
|
if (!strpbrk_esc(arg, "+@%")) {
|
||||||
|
ptr = strpbrk_esc(arg, ";:");
|
||||||
/* This is a file spec if it includes a '.' before ; or : */
|
/* This is a file spec if it includes a '.' before ; or : */
|
||||||
if (memchr(arg, '.', ptr - arg))
|
if (ptr && memchr(arg, '.', ptr - arg))
|
||||||
file_spec = true;
|
file_spec = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
ptr = strpbrk(arg, ";:+@%");
|
ptr = strpbrk_esc(arg, ";:+@%");
|
||||||
if (ptr) {
|
if (ptr) {
|
||||||
nc = *ptr;
|
nc = *ptr;
|
||||||
*ptr++ = '\0';
|
*ptr++ = '\0';
|
||||||
|
@ -1436,7 +1440,7 @@ static int parse_perf_probe_point(char *arg, struct perf_probe_event *pev)
|
||||||
if (arg[0] == '\0')
|
if (arg[0] == '\0')
|
||||||
tmp = NULL;
|
tmp = NULL;
|
||||||
else {
|
else {
|
||||||
tmp = strdup(arg);
|
tmp = strdup_esc(arg);
|
||||||
if (tmp == NULL)
|
if (tmp == NULL)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
@ -1469,12 +1473,12 @@ static int parse_perf_probe_point(char *arg, struct perf_probe_event *pev)
|
||||||
arg = ptr;
|
arg = ptr;
|
||||||
c = nc;
|
c = nc;
|
||||||
if (c == ';') { /* Lazy pattern must be the last part */
|
if (c == ';') { /* Lazy pattern must be the last part */
|
||||||
pp->lazy_line = strdup(arg);
|
pp->lazy_line = strdup(arg); /* let leave escapes */
|
||||||
if (pp->lazy_line == NULL)
|
if (pp->lazy_line == NULL)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
ptr = strpbrk(arg, ";:+@%");
|
ptr = strpbrk_esc(arg, ";:+@%");
|
||||||
if (ptr) {
|
if (ptr) {
|
||||||
nc = *ptr;
|
nc = *ptr;
|
||||||
*ptr++ = '\0';
|
*ptr++ = '\0';
|
||||||
|
@ -1501,7 +1505,7 @@ static int parse_perf_probe_point(char *arg, struct perf_probe_event *pev)
|
||||||
semantic_error("SRC@SRC is not allowed.\n");
|
semantic_error("SRC@SRC is not allowed.\n");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
pp->file = strdup(arg);
|
pp->file = strdup_esc(arg);
|
||||||
if (pp->file == NULL)
|
if (pp->file == NULL)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
break;
|
break;
|
||||||
|
@ -2803,23 +2807,31 @@ static int find_probe_functions(struct map *map, char *name,
|
||||||
struct rb_node *tmp;
|
struct rb_node *tmp;
|
||||||
const char *norm, *ver;
|
const char *norm, *ver;
|
||||||
char *buf = NULL;
|
char *buf = NULL;
|
||||||
|
bool cut_version = true;
|
||||||
|
|
||||||
if (map__load(map) < 0)
|
if (map__load(map) < 0)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
/* If user gives a version, don't cut off the version from symbols */
|
||||||
|
if (strchr(name, '@'))
|
||||||
|
cut_version = false;
|
||||||
|
|
||||||
map__for_each_symbol(map, sym, tmp) {
|
map__for_each_symbol(map, sym, tmp) {
|
||||||
norm = arch__normalize_symbol_name(sym->name);
|
norm = arch__normalize_symbol_name(sym->name);
|
||||||
if (!norm)
|
if (!norm)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
/* We don't care about default symbol or not */
|
if (cut_version) {
|
||||||
ver = strchr(norm, '@');
|
/* We don't care about default symbol or not */
|
||||||
if (ver) {
|
ver = strchr(norm, '@');
|
||||||
buf = strndup(norm, ver - norm);
|
if (ver) {
|
||||||
if (!buf)
|
buf = strndup(norm, ver - norm);
|
||||||
return -ENOMEM;
|
if (!buf)
|
||||||
norm = buf;
|
return -ENOMEM;
|
||||||
|
norm = buf;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (strglobmatch(norm, name)) {
|
if (strglobmatch(norm, name)) {
|
||||||
found++;
|
found++;
|
||||||
if (syms && found < probe_conf.max_probes)
|
if (syms && found < probe_conf.max_probes)
|
||||||
|
|
Loading…
Reference in New Issue
Block a user