124 lines
2.8 KiB
C
124 lines
2.8 KiB
C
|
// SPDX-License-Identifier: GPL-2.0
|
||
|
#include <linux/bpf.h>
|
||
|
#include <bpf/bpf_helpers.h>
|
||
|
#include <bpf/bpf_tracing.h>
|
||
|
#include <errno.h>
|
||
|
|
||
|
char _license[] SEC("license") = "GPL";
|
||
|
|
||
|
__u64 test1_result = 0;
|
||
|
SEC("fentry/bpf_fentry_test1")
|
||
|
int BPF_PROG(test1)
|
||
|
{
|
||
|
__u64 cnt = bpf_get_func_arg_cnt(ctx);
|
||
|
__u64 a = 0, z = 0, ret = 0;
|
||
|
__s64 err;
|
||
|
|
||
|
test1_result = cnt == 1;
|
||
|
|
||
|
/* valid arguments */
|
||
|
err = bpf_get_func_arg(ctx, 0, &a);
|
||
|
|
||
|
/* We need to cast access to traced function argument values with
|
||
|
* proper type cast, because trampoline uses type specific instruction
|
||
|
* to save it, like for 'int a' with 32-bit mov like:
|
||
|
*
|
||
|
* mov %edi,-0x8(%rbp)
|
||
|
*
|
||
|
* so the upper 4 bytes are not zeroed.
|
||
|
*/
|
||
|
test1_result &= err == 0 && ((int) a == 1);
|
||
|
|
||
|
/* not valid argument */
|
||
|
err = bpf_get_func_arg(ctx, 1, &z);
|
||
|
test1_result &= err == -EINVAL;
|
||
|
|
||
|
/* return value fails in fentry */
|
||
|
err = bpf_get_func_ret(ctx, &ret);
|
||
|
test1_result &= err == -EOPNOTSUPP;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
__u64 test2_result = 0;
|
||
|
SEC("fexit/bpf_fentry_test2")
|
||
|
int BPF_PROG(test2)
|
||
|
{
|
||
|
__u64 cnt = bpf_get_func_arg_cnt(ctx);
|
||
|
__u64 a = 0, b = 0, z = 0, ret = 0;
|
||
|
__s64 err;
|
||
|
|
||
|
test2_result = cnt == 2;
|
||
|
|
||
|
/* valid arguments */
|
||
|
err = bpf_get_func_arg(ctx, 0, &a);
|
||
|
test2_result &= err == 0 && (int) a == 2;
|
||
|
|
||
|
err = bpf_get_func_arg(ctx, 1, &b);
|
||
|
test2_result &= err == 0 && b == 3;
|
||
|
|
||
|
/* not valid argument */
|
||
|
err = bpf_get_func_arg(ctx, 2, &z);
|
||
|
test2_result &= err == -EINVAL;
|
||
|
|
||
|
/* return value */
|
||
|
err = bpf_get_func_ret(ctx, &ret);
|
||
|
test2_result &= err == 0 && ret == 5;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
__u64 test3_result = 0;
|
||
|
SEC("fmod_ret/bpf_modify_return_test")
|
||
|
int BPF_PROG(fmod_ret_test, int _a, int *_b, int _ret)
|
||
|
{
|
||
|
__u64 cnt = bpf_get_func_arg_cnt(ctx);
|
||
|
__u64 a = 0, b = 0, z = 0, ret = 0;
|
||
|
__s64 err;
|
||
|
|
||
|
test3_result = cnt == 2;
|
||
|
|
||
|
/* valid arguments */
|
||
|
err = bpf_get_func_arg(ctx, 0, &a);
|
||
|
test3_result &= err == 0 && ((int) a == 1);
|
||
|
|
||
|
err = bpf_get_func_arg(ctx, 1, &b);
|
||
|
test3_result &= err == 0 && ((int *) b == _b);
|
||
|
|
||
|
/* not valid argument */
|
||
|
err = bpf_get_func_arg(ctx, 2, &z);
|
||
|
test3_result &= err == -EINVAL;
|
||
|
|
||
|
/* return value */
|
||
|
err = bpf_get_func_ret(ctx, &ret);
|
||
|
test3_result &= err == 0 && ret == 0;
|
||
|
|
||
|
/* change return value, it's checked in fexit_test program */
|
||
|
return 1234;
|
||
|
}
|
||
|
|
||
|
__u64 test4_result = 0;
|
||
|
SEC("fexit/bpf_modify_return_test")
|
||
|
int BPF_PROG(fexit_test, int _a, int *_b, int _ret)
|
||
|
{
|
||
|
__u64 cnt = bpf_get_func_arg_cnt(ctx);
|
||
|
__u64 a = 0, b = 0, z = 0, ret = 0;
|
||
|
__s64 err;
|
||
|
|
||
|
test4_result = cnt == 2;
|
||
|
|
||
|
/* valid arguments */
|
||
|
err = bpf_get_func_arg(ctx, 0, &a);
|
||
|
test4_result &= err == 0 && ((int) a == 1);
|
||
|
|
||
|
err = bpf_get_func_arg(ctx, 1, &b);
|
||
|
test4_result &= err == 0 && ((int *) b == _b);
|
||
|
|
||
|
/* not valid argument */
|
||
|
err = bpf_get_func_arg(ctx, 2, &z);
|
||
|
test4_result &= err == -EINVAL;
|
||
|
|
||
|
/* return value */
|
||
|
err = bpf_get_func_ret(ctx, &ret);
|
||
|
test4_result &= err == 0 && ret == 1234;
|
||
|
return 0;
|
||
|
}
|