Commit b630092f authored by Sasha Goldshtein's avatar Sasha Goldshtein

trace: Use correct argument index and tgid for filters

`trace` would use the incorrect argument index for USDT probes when
filtering specifically, e.g. `trace u:lib:tp (arg1 != 0) ...` would
actually use the type of the 2nd argument, and not the 1st argument
for the type of the filter variable in the generated program. This
could cause compilation errors or subtle bugs where the data would
be either extended or contracted to fit the wrong argument's type.

Additionally, `trace` would use the pid (thread id, `-L`) filter
with the `attach_uprobe` API, which expects a tgid (process id).
As a result, incorrect filtering would happen.
parent b9aec34d
...@@ -354,33 +354,35 @@ BPF_PERF_OUTPUT(%s); ...@@ -354,33 +354,35 @@ BPF_PERF_OUTPUT(%s);
def _generate_usdt_filter_read(self): def _generate_usdt_filter_read(self):
text = "" text = ""
if self.probe_type == "u": if self.probe_type != "u":
for arg, _ in Probe.aliases.items(): return text
if not (arg.startswith("arg") and for arg, _ in Probe.aliases.items():
(arg in self.filter)): if not (arg.startswith("arg") and
continue (arg in self.filter)):
arg_index = int(arg.replace("arg", "")) continue
arg_ctype = self.usdt.get_probe_arg_ctype( arg_index = int(arg.replace("arg", ""))
self.usdt_name, arg_index) arg_ctype = self.usdt.get_probe_arg_ctype(
if not arg_ctype: self.usdt_name, arg_index-1)
self._bail("Unable to determine type of {} " if not arg_ctype:
"in the filter".format(arg)) self._bail("Unable to determine type of {} "
text += """ "in the filter".format(arg))
text += """
{} {}_filter; {} {}_filter;
bpf_usdt_readarg({}, ctx, &{}_filter); bpf_usdt_readarg({}, ctx, &{}_filter);
""".format(arg_ctype, arg, arg_index, arg) """.format(arg_ctype, arg, arg_index, arg)
self.filter = self.filter.replace( self.filter = self.filter.replace(
arg, "{}_filter".format(arg)) arg, "{}_filter".format(arg))
return text return text
def generate_program(self, include_self): def generate_program(self, include_self):
data_decl = self._generate_data_decl() data_decl = self._generate_data_decl()
# kprobes don't have built-in pid filters, so we have to add if Probe.pid != -1:
# it to the function body:
if len(self.library) == 0 and Probe.pid != -1:
pid_filter = """ pid_filter = """
if (__pid != %d) { return 0; } if (__pid != %d) { return 0; }
""" % Probe.pid """ % Probe.pid
# uprobes can have a built-in tgid filter passed to
# attach_uprobe, hence the check here -- for kprobes, we
# need to do the tgid test by hand:
elif len(self.library) == 0 and Probe.tgid != -1: elif len(self.library) == 0 and Probe.tgid != -1:
pid_filter = """ pid_filter = """
if (__tgid != %d) { return 0; } if (__tgid != %d) { return 0; }
...@@ -542,12 +544,12 @@ BPF_PERF_OUTPUT(%s); ...@@ -542,12 +544,12 @@ BPF_PERF_OUTPUT(%s);
bpf.attach_uretprobe(name=libpath, bpf.attach_uretprobe(name=libpath,
sym=self.function, sym=self.function,
fn_name=self.probe_name, fn_name=self.probe_name,
pid=Probe.pid) pid=Probe.tgid)
else: else:
bpf.attach_uprobe(name=libpath, bpf.attach_uprobe(name=libpath,
sym=self.function, sym=self.function,
fn_name=self.probe_name, fn_name=self.probe_name,
pid=Probe.pid) pid=Probe.tgid)
class Tool(object): class Tool(object):
examples = """ examples = """
......
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