|
@@ -225,11 +225,12 @@ static const struct bpf_func_proto bpf_perf_event_read_proto = {
|
|
|
.arg2_type = ARG_ANYTHING,
|
|
|
};
|
|
|
|
|
|
-static u64 bpf_perf_event_output(u64 r1, u64 r2, u64 index, u64 r4, u64 size)
|
|
|
+static u64 bpf_perf_event_output(u64 r1, u64 r2, u64 flags, u64 r4, u64 size)
|
|
|
{
|
|
|
struct pt_regs *regs = (struct pt_regs *) (long) r1;
|
|
|
struct bpf_map *map = (struct bpf_map *) (long) r2;
|
|
|
struct bpf_array *array = container_of(map, struct bpf_array, map);
|
|
|
+ u64 index = flags & BPF_F_INDEX_MASK;
|
|
|
void *data = (void *) (long) r4;
|
|
|
struct perf_sample_data sample_data;
|
|
|
struct perf_event *event;
|
|
@@ -239,6 +240,10 @@ static u64 bpf_perf_event_output(u64 r1, u64 r2, u64 index, u64 r4, u64 size)
|
|
|
.data = data,
|
|
|
};
|
|
|
|
|
|
+ if (unlikely(flags & ~(BPF_F_INDEX_MASK)))
|
|
|
+ return -EINVAL;
|
|
|
+ if (index == BPF_F_CURRENT_CPU)
|
|
|
+ index = raw_smp_processor_id();
|
|
|
if (unlikely(index >= array->map.max_entries))
|
|
|
return -E2BIG;
|
|
|
|
|
@@ -272,6 +277,33 @@ static const struct bpf_func_proto bpf_perf_event_output_proto = {
|
|
|
.arg5_type = ARG_CONST_STACK_SIZE,
|
|
|
};
|
|
|
|
|
|
+static DEFINE_PER_CPU(struct pt_regs, bpf_pt_regs);
|
|
|
+
|
|
|
+static u64 bpf_event_output(u64 r1, u64 r2, u64 flags, u64 r4, u64 size)
|
|
|
+{
|
|
|
+ struct pt_regs *regs = this_cpu_ptr(&bpf_pt_regs);
|
|
|
+
|
|
|
+ perf_fetch_caller_regs(regs);
|
|
|
+
|
|
|
+ return bpf_perf_event_output((long)regs, r2, flags, r4, size);
|
|
|
+}
|
|
|
+
|
|
|
+static const struct bpf_func_proto bpf_event_output_proto = {
|
|
|
+ .func = bpf_event_output,
|
|
|
+ .gpl_only = true,
|
|
|
+ .ret_type = RET_INTEGER,
|
|
|
+ .arg1_type = ARG_PTR_TO_CTX,
|
|
|
+ .arg2_type = ARG_CONST_MAP_PTR,
|
|
|
+ .arg3_type = ARG_ANYTHING,
|
|
|
+ .arg4_type = ARG_PTR_TO_STACK,
|
|
|
+ .arg5_type = ARG_CONST_STACK_SIZE,
|
|
|
+};
|
|
|
+
|
|
|
+const struct bpf_func_proto *bpf_get_event_output_proto(void)
|
|
|
+{
|
|
|
+ return &bpf_event_output_proto;
|
|
|
+}
|
|
|
+
|
|
|
static const struct bpf_func_proto *tracing_func_proto(enum bpf_func_id func_id)
|
|
|
{
|
|
|
switch (func_id) {
|