perf: Make perf_event_output() propagate the output() return

For the original mode of operation it isn't needed, since we report back
errors via PERF_RECORD_LOST records in the ring buffer, but for use in
bpf_perf_event_output() it is convenient to return the errors, basically
-ENOSPC.

Currently bpf_perf_event_output() returns an error indication, the last
thing it does, which is to push it to the ring buffer is that can fail
and if so, this failure won't be reported back to its users, fix it.
Reported-by: NJamal Hadi Salim <jhs@mojatatu.com>
Tested-by: NJamal Hadi Salim <jhs@mojatatu.com>
Acked-by: NPeter Zijlstra (Intel) <peterz@infradead.org>
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: Alexei Starovoitov <alexei.starovoitov@gmail.com>
Cc: Daniel Borkmann <daniel@iogearbox.net>
Cc: Jiri Olsa <jolsa@kernel.org>
Cc: Namhyung Kim <namhyung@kernel.org>
Link: https://lkml.kernel.org/r/20190118150938.GN5823@kernel.orgSigned-off-by: NArnaldo Carvalho de Melo <acme@redhat.com>
上级 8dabe9c4
...@@ -978,7 +978,7 @@ extern void perf_event_output_forward(struct perf_event *event, ...@@ -978,7 +978,7 @@ extern void perf_event_output_forward(struct perf_event *event,
extern void perf_event_output_backward(struct perf_event *event, extern void perf_event_output_backward(struct perf_event *event,
struct perf_sample_data *data, struct perf_sample_data *data,
struct pt_regs *regs); struct pt_regs *regs);
extern void perf_event_output(struct perf_event *event, extern int perf_event_output(struct perf_event *event,
struct perf_sample_data *data, struct perf_sample_data *data,
struct pt_regs *regs); struct pt_regs *regs);
......
...@@ -6489,7 +6489,7 @@ void perf_prepare_sample(struct perf_event_header *header, ...@@ -6489,7 +6489,7 @@ void perf_prepare_sample(struct perf_event_header *header,
data->phys_addr = perf_virt_to_phys(data->addr); data->phys_addr = perf_virt_to_phys(data->addr);
} }
static __always_inline void static __always_inline int
__perf_event_output(struct perf_event *event, __perf_event_output(struct perf_event *event,
struct perf_sample_data *data, struct perf_sample_data *data,
struct pt_regs *regs, struct pt_regs *regs,
...@@ -6499,13 +6499,15 @@ __perf_event_output(struct perf_event *event, ...@@ -6499,13 +6499,15 @@ __perf_event_output(struct perf_event *event,
{ {
struct perf_output_handle handle; struct perf_output_handle handle;
struct perf_event_header header; struct perf_event_header header;
int err;
/* protect the callchain buffers */ /* protect the callchain buffers */
rcu_read_lock(); rcu_read_lock();
perf_prepare_sample(&header, data, event, regs); perf_prepare_sample(&header, data, event, regs);
if (output_begin(&handle, event, header.size)) err = output_begin(&handle, event, header.size);
if (err)
goto exit; goto exit;
perf_output_sample(&handle, &header, data, event); perf_output_sample(&handle, &header, data, event);
...@@ -6514,6 +6516,7 @@ __perf_event_output(struct perf_event *event, ...@@ -6514,6 +6516,7 @@ __perf_event_output(struct perf_event *event,
exit: exit:
rcu_read_unlock(); rcu_read_unlock();
return err;
} }
void void
...@@ -6532,12 +6535,12 @@ perf_event_output_backward(struct perf_event *event, ...@@ -6532,12 +6535,12 @@ perf_event_output_backward(struct perf_event *event,
__perf_event_output(event, data, regs, perf_output_begin_backward); __perf_event_output(event, data, regs, perf_output_begin_backward);
} }
void int
perf_event_output(struct perf_event *event, perf_event_output(struct perf_event *event,
struct perf_sample_data *data, struct perf_sample_data *data,
struct pt_regs *regs) struct pt_regs *regs)
{ {
__perf_event_output(event, data, regs, perf_output_begin); return __perf_event_output(event, data, regs, perf_output_begin);
} }
/* /*
......
...@@ -431,8 +431,7 @@ __bpf_perf_event_output(struct pt_regs *regs, struct bpf_map *map, ...@@ -431,8 +431,7 @@ __bpf_perf_event_output(struct pt_regs *regs, struct bpf_map *map,
if (unlikely(event->oncpu != cpu)) if (unlikely(event->oncpu != cpu))
return -EOPNOTSUPP; return -EOPNOTSUPP;
perf_event_output(event, sd, regs); return perf_event_output(event, sd, regs);
return 0;
} }
BPF_CALL_5(bpf_perf_event_output, struct pt_regs *, regs, struct bpf_map *, map, BPF_CALL_5(bpf_perf_event_output, struct pt_regs *, regs, struct bpf_map *, map,
......
...@@ -141,8 +141,8 @@ int sys_enter(struct syscall_enter_args *args) ...@@ -141,8 +141,8 @@ int sys_enter(struct syscall_enter_args *args)
len = sizeof(augmented_args.args); len = sizeof(augmented_args.args);
} }
perf_event_output(args, &__augmented_syscalls__, BPF_F_CURRENT_CPU, &augmented_args, len); /* If perf_event_output fails, return non-zero so that it gets recorded unaugmented */
return 0; return perf_event_output(args, &__augmented_syscalls__, BPF_F_CURRENT_CPU, &augmented_args, len);
} }
SEC("raw_syscalls:sys_exit") SEC("raw_syscalls:sys_exit")
......
...@@ -55,9 +55,9 @@ int syscall_enter(syscall)(struct syscall_enter_##syscall##_args *args) \ ...@@ -55,9 +55,9 @@ int syscall_enter(syscall)(struct syscall_enter_##syscall##_args *args) \
len -= sizeof(augmented_args.filename.value) - augmented_args.filename.size; \ len -= sizeof(augmented_args.filename.value) - augmented_args.filename.size; \
len &= sizeof(augmented_args.filename.value) - 1; \ len &= sizeof(augmented_args.filename.value) - 1; \
} \ } \
perf_event_output(args, &__augmented_syscalls__, BPF_F_CURRENT_CPU, \ /* If perf_event_output fails, return non-zero so that it gets recorded unaugmented */ \
return perf_event_output(args, &__augmented_syscalls__, BPF_F_CURRENT_CPU, \
&augmented_args, len); \ &augmented_args, len); \
return 0; \
} \ } \
int syscall_exit(syscall)(struct syscall_exit_args *args) \ int syscall_exit(syscall)(struct syscall_exit_args *args) \
{ \ { \
...@@ -125,10 +125,10 @@ int syscall_enter(syscall)(struct syscall_enter_##syscall##_args *args) \ ...@@ -125,10 +125,10 @@ int syscall_enter(syscall)(struct syscall_enter_##syscall##_args *args) \
/* addrlen = augmented_args.args.addrlen; */ \ /* addrlen = augmented_args.args.addrlen; */ \
/* */ \ /* */ \
probe_read(&augmented_args.addr, addrlen, args->addr_ptr); \ probe_read(&augmented_args.addr, addrlen, args->addr_ptr); \
perf_event_output(args, &__augmented_syscalls__, BPF_F_CURRENT_CPU, \ /* If perf_event_output fails, return non-zero so that it gets recorded unaugmented */ \
return perf_event_output(args, &__augmented_syscalls__, BPF_F_CURRENT_CPU, \
&augmented_args, \ &augmented_args, \
sizeof(augmented_args) - sizeof(augmented_args.addr) + addrlen); \ sizeof(augmented_args) - sizeof(augmented_args.addr) + addrlen);\
return 0; \
} \ } \
int syscall_exit(syscall)(struct syscall_exit_args *args) \ int syscall_exit(syscall)(struct syscall_exit_args *args) \
{ \ { \
......
...@@ -49,11 +49,11 @@ int syscall_enter(syscall)(struct syscall_enter_##syscall##_args *args) \ ...@@ -49,11 +49,11 @@ int syscall_enter(syscall)(struct syscall_enter_##syscall##_args *args) \
args->filename_ptr); \ args->filename_ptr); \
if (__builtin_memcmp(augmented_args.filename.value, etc, 4) != 0) \ if (__builtin_memcmp(augmented_args.filename.value, etc, 4) != 0) \
return 0; \ return 0; \
perf_event_output(args, &__augmented_syscalls__, BPF_F_CURRENT_CPU, \ /* If perf_event_output fails, return non-zero so that it gets recorded unaugmented */ \
return perf_event_output(args, &__augmented_syscalls__, BPF_F_CURRENT_CPU, \
&augmented_args, \ &augmented_args, \
(sizeof(augmented_args) - sizeof(augmented_args.filename.value) + \ (sizeof(augmented_args) - sizeof(augmented_args.filename.value) + \
augmented_args.filename.size)); \ augmented_args.filename.size)); \
return 0; \
} }
struct syscall_enter_openat_args { struct syscall_enter_openat_args {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册