diff options
Diffstat (limited to 'tools')
8 files changed, 663 insertions, 0 deletions
| diff --git a/tools/testing/selftests/bpf/bpf_testmod/bpf_testmod.c b/tools/testing/selftests/bpf/bpf_testmod/bpf_testmod.c index 987d41af71d2..cc9dde507aba 100644 --- a/tools/testing/selftests/bpf/bpf_testmod/bpf_testmod.c +++ b/tools/testing/selftests/bpf/bpf_testmod/bpf_testmod.c @@ -245,6 +245,39 @@ __bpf_kfunc void bpf_testmod_ctx_release(struct bpf_testmod_ctx *ctx)  		call_rcu(&ctx->rcu, testmod_free_cb);  } +static struct bpf_testmod_ops3 *st_ops3; + +static int bpf_testmod_test_3(void) +{ +	return 0; +} + +static int bpf_testmod_test_4(void) +{ +	return 0; +} + +static struct bpf_testmod_ops3 __bpf_testmod_ops3 = { +	.test_1 = bpf_testmod_test_3, +	.test_2 = bpf_testmod_test_4, +}; + +static void bpf_testmod_test_struct_ops3(void) +{ +	if (st_ops3) +		st_ops3->test_1(); +} + +__bpf_kfunc void bpf_testmod_ops3_call_test_1(void) +{ +	st_ops3->test_1(); +} + +__bpf_kfunc void bpf_testmod_ops3_call_test_2(void) +{ +	st_ops3->test_2(); +} +  struct bpf_testmod_btf_type_tag_1 {  	int a;  }; @@ -382,6 +415,8 @@ bpf_testmod_test_read(struct file *file, struct kobject *kobj,  	(void)trace_bpf_testmod_test_raw_tp_null(NULL); +	bpf_testmod_test_struct_ops3(); +  	struct_arg3 = kmalloc((sizeof(struct bpf_testmod_struct_arg_3) +  				sizeof(int)), GFP_KERNEL);  	if (struct_arg3 != NULL) { @@ -586,6 +621,8 @@ BTF_ID_FLAGS(func, bpf_kfunc_trusted_num_test, KF_TRUSTED_ARGS)  BTF_ID_FLAGS(func, bpf_kfunc_rcu_task_test, KF_RCU)  BTF_ID_FLAGS(func, bpf_testmod_ctx_create, KF_ACQUIRE | KF_RET_NULL)  BTF_ID_FLAGS(func, bpf_testmod_ctx_release, KF_RELEASE) +BTF_ID_FLAGS(func, bpf_testmod_ops3_call_test_1) +BTF_ID_FLAGS(func, bpf_testmod_ops3_call_test_2)  BTF_KFUNCS_END(bpf_testmod_common_kfunc_ids)  BTF_ID_LIST(bpf_testmod_dtor_ids) @@ -1096,6 +1133,10 @@ static const struct bpf_verifier_ops bpf_testmod_verifier_ops = {  	.is_valid_access = bpf_testmod_ops_is_valid_access,  }; +static const struct bpf_verifier_ops bpf_testmod_verifier_ops3 = { +	.is_valid_access = bpf_testmod_ops_is_valid_access, +}; +  static int bpf_dummy_reg(void *kdata, struct bpf_link *link)  {  	struct bpf_testmod_ops *ops = kdata; @@ -1175,6 +1216,68 @@ struct bpf_struct_ops bpf_testmod_ops2 = {  	.owner = THIS_MODULE,  }; +static int st_ops3_reg(void *kdata, struct bpf_link *link) +{ +	int err = 0; + +	mutex_lock(&st_ops_mutex); +	if (st_ops3) { +		pr_err("st_ops has already been registered\n"); +		err = -EEXIST; +		goto unlock; +	} +	st_ops3 = kdata; + +unlock: +	mutex_unlock(&st_ops_mutex); +	return err; +} + +static void st_ops3_unreg(void *kdata, struct bpf_link *link) +{ +	mutex_lock(&st_ops_mutex); +	st_ops3 = NULL; +	mutex_unlock(&st_ops_mutex); +} + +static void test_1_recursion_detected(struct bpf_prog *prog) +{ +	struct bpf_prog_stats *stats; + +	stats = this_cpu_ptr(prog->stats); +	printk("bpf_testmod: oh no, recursing into test_1, recursion_misses %llu", +	       u64_stats_read(&stats->misses)); +} + +static int st_ops3_check_member(const struct btf_type *t, +				const struct btf_member *member, +				const struct bpf_prog *prog) +{ +	u32 moff = __btf_member_bit_offset(t, member) / 8; + +	switch (moff) { +	case offsetof(struct bpf_testmod_ops3, test_1): +		prog->aux->priv_stack_requested = true; +		prog->aux->recursion_detected = test_1_recursion_detected; +		fallthrough; +	default: +		break; +	} +	return 0; +} + +struct bpf_struct_ops bpf_testmod_ops3 = { +	.verifier_ops = &bpf_testmod_verifier_ops3, +	.init = bpf_testmod_ops_init, +	.init_member = bpf_testmod_ops_init_member, +	.reg = st_ops3_reg, +	.unreg = st_ops3_unreg, +	.check_member = st_ops3_check_member, +	.cfi_stubs = &__bpf_testmod_ops3, +	.name = "bpf_testmod_ops3", +	.owner = THIS_MODULE, +}; +  static int bpf_test_mod_st_ops__test_prologue(struct st_ops_args *args)  {  	return 0; @@ -1333,6 +1436,7 @@ static int bpf_testmod_init(void)  	ret = ret ?: register_btf_kfunc_id_set(BPF_PROG_TYPE_STRUCT_OPS, &bpf_testmod_kfunc_set);  	ret = ret ?: register_bpf_struct_ops(&bpf_bpf_testmod_ops, bpf_testmod_ops);  	ret = ret ?: register_bpf_struct_ops(&bpf_testmod_ops2, bpf_testmod_ops2); +	ret = ret ?: register_bpf_struct_ops(&bpf_testmod_ops3, bpf_testmod_ops3);  	ret = ret ?: register_bpf_struct_ops(&testmod_st_ops, bpf_testmod_st_ops);  	ret = ret ?: register_btf_id_dtor_kfuncs(bpf_testmod_dtors,  						 ARRAY_SIZE(bpf_testmod_dtors), diff --git a/tools/testing/selftests/bpf/bpf_testmod/bpf_testmod.h b/tools/testing/selftests/bpf/bpf_testmod/bpf_testmod.h index fb7dff47597a..356803d1c10e 100644 --- a/tools/testing/selftests/bpf/bpf_testmod/bpf_testmod.h +++ b/tools/testing/selftests/bpf/bpf_testmod/bpf_testmod.h @@ -94,6 +94,11 @@ struct bpf_testmod_ops2 {  	int (*test_1)(void);  }; +struct bpf_testmod_ops3 { +	int (*test_1)(void); +	int (*test_2)(void); +}; +  struct st_ops_args {  	u64 a;  }; diff --git a/tools/testing/selftests/bpf/prog_tests/struct_ops_private_stack.c b/tools/testing/selftests/bpf/prog_tests/struct_ops_private_stack.c new file mode 100644 index 000000000000..4006879ca3fe --- /dev/null +++ b/tools/testing/selftests/bpf/prog_tests/struct_ops_private_stack.c @@ -0,0 +1,106 @@ +// SPDX-License-Identifier: GPL-2.0 + +#include <test_progs.h> +#include "struct_ops_private_stack.skel.h" +#include "struct_ops_private_stack_fail.skel.h" +#include "struct_ops_private_stack_recur.skel.h" + +static void test_private_stack(void) +{ +	struct struct_ops_private_stack *skel; +	struct bpf_link *link; +	int err; + +	skel = struct_ops_private_stack__open(); +	if (!ASSERT_OK_PTR(skel, "struct_ops_private_stack__open")) +		return; + +	if (skel->data->skip) { +		test__skip(); +		goto cleanup; +	} + +	err = struct_ops_private_stack__load(skel); +	if (!ASSERT_OK(err, "struct_ops_private_stack__load")) +		goto cleanup; + +	link = bpf_map__attach_struct_ops(skel->maps.testmod_1); +	if (!ASSERT_OK_PTR(link, "attach_struct_ops")) +		goto cleanup; + +	ASSERT_OK(trigger_module_test_read(256), "trigger_read"); + +	ASSERT_EQ(skel->bss->val_i, 3, "val_i"); +	ASSERT_EQ(skel->bss->val_j, 8, "val_j"); + +	bpf_link__destroy(link); + +cleanup: +	struct_ops_private_stack__destroy(skel); +} + +static void test_private_stack_fail(void) +{ +	struct struct_ops_private_stack_fail *skel; +	int err; + +	skel = struct_ops_private_stack_fail__open(); +	if (!ASSERT_OK_PTR(skel, "struct_ops_private_stack_fail__open")) +		return; + +	if (skel->data->skip) { +		test__skip(); +		goto cleanup; +	} + +	err = struct_ops_private_stack_fail__load(skel); +	if (!ASSERT_ERR(err, "struct_ops_private_stack_fail__load")) +		goto cleanup; +	return; + +cleanup: +	struct_ops_private_stack_fail__destroy(skel); +} + +static void test_private_stack_recur(void) +{ +	struct struct_ops_private_stack_recur *skel; +	struct bpf_link *link; +	int err; + +	skel = struct_ops_private_stack_recur__open(); +	if (!ASSERT_OK_PTR(skel, "struct_ops_private_stack_recur__open")) +		return; + +	if (skel->data->skip) { +		test__skip(); +		goto cleanup; +	} + +	err = struct_ops_private_stack_recur__load(skel); +	if (!ASSERT_OK(err, "struct_ops_private_stack_recur__load")) +		goto cleanup; + +	link = bpf_map__attach_struct_ops(skel->maps.testmod_1); +	if (!ASSERT_OK_PTR(link, "attach_struct_ops")) +		goto cleanup; + +	ASSERT_OK(trigger_module_test_read(256), "trigger_read"); + +	ASSERT_EQ(skel->bss->val_j, 3, "val_j"); + +	bpf_link__destroy(link); + +cleanup: +	struct_ops_private_stack_recur__destroy(skel); +} + +void test_struct_ops_private_stack(void) +{ +	if (test__start_subtest("private_stack")) +		test_private_stack(); +	if (test__start_subtest("private_stack_fail")) +		test_private_stack_fail(); +	if (test__start_subtest("private_stack_recur")) +		test_private_stack_recur(); +} diff --git a/tools/testing/selftests/bpf/prog_tests/verifier.c b/tools/testing/selftests/bpf/prog_tests/verifier.c index 75f7a2ce334b..d9f65adb456b 100644 --- a/tools/testing/selftests/bpf/prog_tests/verifier.c +++ b/tools/testing/selftests/bpf/prog_tests/verifier.c @@ -61,6 +61,7 @@  #include "verifier_or_jmp32_k.skel.h"  #include "verifier_precision.skel.h"  #include "verifier_prevent_map_lookup.skel.h" +#include "verifier_private_stack.skel.h"  #include "verifier_raw_stack.skel.h"  #include "verifier_raw_tp_writable.skel.h"  #include "verifier_reg_equal.skel.h" @@ -188,6 +189,7 @@ void test_verifier_bpf_fastcall(void)         { RUN(verifier_bpf_fastcall); }  void test_verifier_or_jmp32_k(void)           { RUN(verifier_or_jmp32_k); }  void test_verifier_precision(void)            { RUN(verifier_precision); }  void test_verifier_prevent_map_lookup(void)   { RUN(verifier_prevent_map_lookup); } +void test_verifier_private_stack(void)        { RUN(verifier_private_stack); }  void test_verifier_raw_stack(void)            { RUN(verifier_raw_stack); }  void test_verifier_raw_tp_writable(void)      { RUN(verifier_raw_tp_writable); }  void test_verifier_reg_equal(void)            { RUN(verifier_reg_equal); } diff --git a/tools/testing/selftests/bpf/progs/struct_ops_private_stack.c b/tools/testing/selftests/bpf/progs/struct_ops_private_stack.c new file mode 100644 index 000000000000..8ea57e5348ab --- /dev/null +++ b/tools/testing/selftests/bpf/progs/struct_ops_private_stack.c @@ -0,0 +1,62 @@ +// SPDX-License-Identifier: GPL-2.0 + +#include <vmlinux.h> +#include <bpf/bpf_helpers.h> +#include <bpf/bpf_tracing.h> +#include "../bpf_testmod/bpf_testmod.h" + +char _license[] SEC("license") = "GPL"; + +#if defined(__TARGET_ARCH_x86) +bool skip __attribute((__section__(".data"))) = false; +#else +bool skip = true; +#endif + +void bpf_testmod_ops3_call_test_2(void) __ksym; + +int val_i, val_j; + +__noinline static int subprog2(int *a, int *b) +{ +	return val_i + a[10] + b[20]; +} + +__noinline static int subprog1(int *a) +{ +	/* stack size 200 bytes */ +	int b[50] = {}; + +	b[20] = 2; +	return subprog2(a, b); +} + + +SEC("struct_ops") +int BPF_PROG(test_1) +{ +	/* stack size 400 bytes */ +	int a[100] = {}; + +	a[10] = 1; +	val_i = subprog1(a); +	bpf_testmod_ops3_call_test_2(); +	return 0; +} + +SEC("struct_ops") +int BPF_PROG(test_2) +{ +	/* stack size 200 bytes */ +	int a[50] = {}; + +	a[10] = 3; +	val_j = subprog1(a); +	return 0; +} + +SEC(".struct_ops") +struct bpf_testmod_ops3 testmod_1 = { +	.test_1 = (void *)test_1, +	.test_2 = (void *)test_2, +}; diff --git a/tools/testing/selftests/bpf/progs/struct_ops_private_stack_fail.c b/tools/testing/selftests/bpf/progs/struct_ops_private_stack_fail.c new file mode 100644 index 000000000000..1f55ec4cee37 --- /dev/null +++ b/tools/testing/selftests/bpf/progs/struct_ops_private_stack_fail.c @@ -0,0 +1,62 @@ +// SPDX-License-Identifier: GPL-2.0 + +#include <vmlinux.h> +#include <bpf/bpf_helpers.h> +#include <bpf/bpf_tracing.h> +#include "../bpf_testmod/bpf_testmod.h" + +char _license[] SEC("license") = "GPL"; + +#if defined(__TARGET_ARCH_x86) +bool skip __attribute((__section__(".data"))) = false; +#else +bool skip = true; +#endif + +void bpf_testmod_ops3_call_test_2(void) __ksym; + +int val_i, val_j; + +__noinline static int subprog2(int *a, int *b) +{ +	return val_i + a[10] + b[20]; +} + +__noinline static int subprog1(int *a) +{ +	/* stack size 200 bytes */ +	int b[50] = {}; + +	b[20] = 2; +	return subprog2(a, b); +} + + +SEC("struct_ops") +int BPF_PROG(test_1) +{ +	/* stack size 100 bytes */ +	int a[25] = {}; + +	a[10] = 1; +	val_i = subprog1(a); +	bpf_testmod_ops3_call_test_2(); +	return 0; +} + +SEC("struct_ops") +int BPF_PROG(test_2) +{ +	/* stack size 400 bytes */ +	int a[100] = {}; + +	a[10] = 3; +	val_j = subprog1(a); +	return 0; +} + +SEC(".struct_ops") +struct bpf_testmod_ops3 testmod_1 = { +	.test_1 = (void *)test_1, +	.test_2 = (void *)test_2, +}; diff --git a/tools/testing/selftests/bpf/progs/struct_ops_private_stack_recur.c b/tools/testing/selftests/bpf/progs/struct_ops_private_stack_recur.c new file mode 100644 index 000000000000..f2f300d50988 --- /dev/null +++ b/tools/testing/selftests/bpf/progs/struct_ops_private_stack_recur.c @@ -0,0 +1,50 @@ +// SPDX-License-Identifier: GPL-2.0 + +#include <vmlinux.h> +#include <bpf/bpf_helpers.h> +#include <bpf/bpf_tracing.h> +#include "../bpf_testmod/bpf_testmod.h" + +char _license[] SEC("license") = "GPL"; + +#if defined(__TARGET_ARCH_x86) +bool skip __attribute((__section__(".data"))) = false; +#else +bool skip = true; +#endif + +void bpf_testmod_ops3_call_test_1(void) __ksym; + +int val_i, val_j; + +__noinline static int subprog2(int *a, int *b) +{ +	return val_i + a[1] + b[20]; +} + +__noinline static int subprog1(int *a) +{ +	/* stack size 400 bytes */ +	int b[100] = {}; + +	b[20] = 2; +	return subprog2(a, b); +} + + +SEC("struct_ops") +int BPF_PROG(test_1) +{ +	/* stack size 20 bytes */ +	int a[5] = {}; + +	a[1] = 1; +	val_j += subprog1(a); +	bpf_testmod_ops3_call_test_1(); +	return 0; +} + +SEC(".struct_ops") +struct bpf_testmod_ops3 testmod_1 = { +	.test_1 = (void *)test_1, +}; diff --git a/tools/testing/selftests/bpf/progs/verifier_private_stack.c b/tools/testing/selftests/bpf/progs/verifier_private_stack.c new file mode 100644 index 000000000000..b1fbdf119553 --- /dev/null +++ b/tools/testing/selftests/bpf/progs/verifier_private_stack.c @@ -0,0 +1,272 @@ +// SPDX-License-Identifier: GPL-2.0 + +#include <vmlinux.h> +#include <bpf/bpf_helpers.h> +#include "bpf_misc.h" +#include "bpf_experimental.h" + +/* From include/linux/filter.h */ +#define MAX_BPF_STACK    512 + +#if defined(__TARGET_ARCH_x86) + +struct elem { +	struct bpf_timer t; +	char pad[256]; +}; + +struct { +	__uint(type, BPF_MAP_TYPE_ARRAY); +	__uint(max_entries, 1); +	__type(key, int); +	__type(value, struct elem); +} array SEC(".maps"); + +SEC("kprobe") +__description("Private stack, single prog") +__success +__arch_x86_64 +__jited("	movabsq	$0x{{.*}}, %r9") +__jited("	addq	%gs:0x{{.*}}, %r9") +__jited("	movl	$0x2a, %edi") +__jited("	movq	%rdi, -0x100(%r9)") +__naked void private_stack_single_prog(void) +{ +	asm volatile ("			\ +	r1 = 42;			\ +	*(u64 *)(r10 - 256) = r1;	\ +	r0 = 0;				\ +	exit;				\ +"	::: __clobber_all); +} + +SEC("raw_tp") +__description("No private stack") +__success +__arch_x86_64 +__jited("	subq	$0x8, %rsp") +__naked void no_private_stack_nested(void) +{ +	asm volatile ("			\ +	r1 = 42;			\ +	*(u64 *)(r10 - 8) = r1;		\ +	r0 = 0;				\ +	exit;				\ +"	::: __clobber_all); +} + +__used +__naked static void cumulative_stack_depth_subprog(void) +{ +	asm volatile ("				\ +	r1 = 41;				\ +	*(u64 *)(r10 - 32) = r1;		\ +	call %[bpf_get_smp_processor_id];	\ +	exit;					\ +"	: +	: __imm(bpf_get_smp_processor_id) +	: __clobber_all); +} + +SEC("kprobe") +__description("Private stack, subtree > MAX_BPF_STACK") +__success +__arch_x86_64 +/* private stack fp for the main prog */ +__jited("	movabsq	$0x{{.*}}, %r9") +__jited("	addq	%gs:0x{{.*}}, %r9") +__jited("	movl	$0x2a, %edi") +__jited("	movq	%rdi, -0x200(%r9)") +__jited("	pushq	%r9") +__jited("	callq	0x{{.*}}") +__jited("	popq	%r9") +__jited("	xorl	%eax, %eax") +__naked void private_stack_nested_1(void) +{ +	asm volatile ("				\ +	r1 = 42;				\ +	*(u64 *)(r10 - %[max_bpf_stack]) = r1;	\ +	call cumulative_stack_depth_subprog;	\ +	r0 = 0;					\ +	exit;					\ +"	: +	: __imm_const(max_bpf_stack, MAX_BPF_STACK) +	: __clobber_all); +} + +__naked __noinline __used +static unsigned long loop_callback(void) +{ +	asm volatile ("				\ +	call %[bpf_get_prandom_u32];		\ +	r1 = 42;				\ +	*(u64 *)(r10 - 512) = r1;		\ +	call cumulative_stack_depth_subprog;	\ +	r0 = 0;					\ +	exit;					\ +"	: +	: __imm(bpf_get_prandom_u32) +	: __clobber_common); +} + +SEC("raw_tp") +__description("Private stack, callback") +__success +__arch_x86_64 +/* for func loop_callback */ +__jited("func #1") +__jited("	endbr64") +__jited("	nopl	(%rax,%rax)") +__jited("	nopl	(%rax)") +__jited("	pushq	%rbp") +__jited("	movq	%rsp, %rbp") +__jited("	endbr64") +__jited("	movabsq	$0x{{.*}}, %r9") +__jited("	addq	%gs:0x{{.*}}, %r9") +__jited("	pushq	%r9") +__jited("	callq") +__jited("	popq	%r9") +__jited("	movl	$0x2a, %edi") +__jited("	movq	%rdi, -0x200(%r9)") +__jited("	pushq	%r9") +__jited("	callq") +__jited("	popq	%r9") +__naked void private_stack_callback(void) +{ +	asm volatile ("			\ +	r1 = 1;				\ +	r2 = %[loop_callback];		\ +	r3 = 0;				\ +	r4 = 0;				\ +	call %[bpf_loop];		\ +	r0 = 0;				\ +	exit;				\ +"	: +	: __imm_ptr(loop_callback), +	  __imm(bpf_loop) +	: __clobber_common); +} + +SEC("fentry/bpf_fentry_test9") +__description("Private stack, exception in main prog") +__success __retval(0) +__arch_x86_64 +__jited("	pushq	%r9") +__jited("	callq") +__jited("	popq	%r9") +int private_stack_exception_main_prog(void) +{ +	asm volatile ("			\ +	r1 = 42;			\ +	*(u64 *)(r10 - 512) = r1;	\ +"	::: __clobber_common); + +	bpf_throw(0); +	return 0; +} + +__used static int subprog_exception(void) +{ +	bpf_throw(0); +	return 0; +} + +SEC("fentry/bpf_fentry_test9") +__description("Private stack, exception in subprog") +__success __retval(0) +__arch_x86_64 +__jited("	movq	%rdi, -0x200(%r9)") +__jited("	pushq	%r9") +__jited("	callq") +__jited("	popq	%r9") +int private_stack_exception_sub_prog(void) +{ +	asm volatile ("			\ +	r1 = 42;			\ +	*(u64 *)(r10 - 512) = r1;	\ +	call subprog_exception;		\ +"	::: __clobber_common); + +	return 0; +} + +int glob; +__noinline static void subprog2(int *val) +{ +	glob += val[0] * 2; +} + +__noinline static void subprog1(int *val) +{ +	int tmp[64] = {}; + +	tmp[0] = *val; +	subprog2(tmp); +} + +__noinline static int timer_cb1(void *map, int *key, struct bpf_timer *timer) +{ +	subprog1(key); +	return 0; +} + +__noinline static int timer_cb2(void *map, int *key, struct bpf_timer *timer) +{ +	return 0; +} + +SEC("fentry/bpf_fentry_test9") +__description("Private stack, async callback, not nested") +__success __retval(0) +__arch_x86_64 +__jited("	movabsq	$0x{{.*}}, %r9") +int private_stack_async_callback_1(void) +{ +	struct bpf_timer *arr_timer; +	int array_key = 0; + +	arr_timer = bpf_map_lookup_elem(&array, &array_key); +	if (!arr_timer) +		return 0; + +	bpf_timer_init(arr_timer, &array, 1); +	bpf_timer_set_callback(arr_timer, timer_cb2); +	bpf_timer_start(arr_timer, 0, 0); +	subprog1(&array_key); +	return 0; +} + +SEC("fentry/bpf_fentry_test9") +__description("Private stack, async callback, potential nesting") +__success __retval(0) +__arch_x86_64 +__jited("	subq	$0x100, %rsp") +int private_stack_async_callback_2(void) +{ +	struct bpf_timer *arr_timer; +	int array_key = 0; + +	arr_timer = bpf_map_lookup_elem(&array, &array_key); +	if (!arr_timer) +		return 0; + +	bpf_timer_init(arr_timer, &array, 1); +	bpf_timer_set_callback(arr_timer, timer_cb1); +	bpf_timer_start(arr_timer, 0, 0); +	subprog1(&array_key); +	return 0; +} + +#else + +SEC("kprobe") +__description("private stack is not supported, use a dummy test") +__success +int dummy_test(void) +{ +	return 0; +} + +#endif + +char _license[] SEC("license") = "GPL"; | 
