// SPDX-License-Identifier: GPL-2.0 /* Copyright (c) 2026 Meta Platforms, Inc. and affiliates. */ #include #include #include #include "bpf_kfuncs.h" #define CLOCK_MONOTONIC 1 struct timer_elem { struct bpf_timer timer; }; struct { __uint(type, BPF_MAP_TYPE_ARRAY); __uint(max_entries, 1); __type(key, int); __type(value, struct timer_elem); } timer_map SEC(".maps"); int timer_result; #if (defined(__TARGET_ARCH_x86) || defined(__TARGET_ARCH_arm64)) && \ defined(__BPF_FEATURE_STACK_ARGUMENT) const volatile bool has_stack_arg = true; __noinline static int static_func_many_args(int a, int b, int c, int d, int e, int f, int g, int h, int i, int j) { return a + b + c + d + e + f + g + h + i + j; } __noinline int global_calls_many_args(int a, int b, int c) { return static_func_many_args(a, b, c, a + 3, a + 4, a + 5, a + 6, a + 7, a + 8, a + 9); } SEC("tc") int test_global_many_args(void) { return global_calls_many_args(1, 2, 3); } struct test_data { long x; long y; }; /* 1+2+3+4+5+6+7+8+9+10+20 = 75 */ __noinline static long func_with_ptr_stack_arg(long a, long b, long c, long d, long e, long f, long g, long h, long i, struct test_data *p) { return a + b + c + d + e + f + g + h + i + p->x + p->y; } __noinline long global_ptr_stack_arg(long a, long b, long c, long d, long e) { struct test_data data = { .x = 10, .y = 20 }; return func_with_ptr_stack_arg(a, b, c, d, e, a + 5, a + 6, a + 7, a + 8, &data); } SEC("tc") int test_bpf2bpf_ptr_stack_arg(void) { return global_ptr_stack_arg(1, 2, 3, 4, 5); } /* 1+2+3+4+5+6+7+10+8+20 = 66 */ __noinline static long func_with_mix_stack_args(long a, long b, long c, long d, long e, long f, long g, struct test_data *p, long h, struct test_data *q) { return a + b + c + d + e + f + g + p->x + h + q->y; } __noinline long global_mix_stack_args(long a, long b, long c, long d, long e) { struct test_data p = { .x = 10 }; struct test_data q = { .y = 20 }; return func_with_mix_stack_args(a, b, c, d, e, e + 1, e + 2, &p, e + 3, &q); } SEC("tc") int test_bpf2bpf_mix_stack_args(void) { return global_mix_stack_args(1, 2, 3, 4, 5); } /* * Nesting test: func_outer calls func_inner, both with struct pointer * as stack arg. * * func_inner: (a+1)+...+(i+1) + p->x + p->y * = 2+3+4+5+6+7+8+9+10+10+20 = 84 */ __noinline static long func_inner_ptr(long a, long b, long c, long d, long e, long f, long g, long h, long i, struct test_data *p) { return a + b + c + d + e + f + g + h + i + p->x + p->y; } __noinline static long func_outer_ptr(long a, long b, long c, long d, long e, long f, long g, long h, long i, struct test_data *p) { return func_inner_ptr(a + 1, b + 1, c + 1, d + 1, e + 1, f + 1, g + 1, h + 1, i + 1, p); } __noinline long global_nesting_ptr(long a, long b, long c, long d, long e) { struct test_data data = { .x = 10, .y = 20 }; return func_outer_ptr(a, b, c, d, e, a + 5, a + 6, a + 7, a + 8, &data); } SEC("tc") int test_bpf2bpf_nesting_stack_arg(void) { return global_nesting_ptr(1, 2, 3, 4, 5); } /* 1+2+3+4+5+6+7+8+9+sizeof(pkt_v4) = 45+54 = 99 */ __noinline static long func_with_dynptr(long a, long b, long c, long d, long e, long f, long g, long h, long i, struct bpf_dynptr *ptr) { return a + b + c + d + e + f + g + h + i + bpf_dynptr_size(ptr); } __noinline long global_dynptr_stack_arg(void *ctx __arg_ctx, long a, long b, long c, long d) { struct bpf_dynptr ptr; bpf_dynptr_from_skb(ctx, 0, &ptr); return func_with_dynptr(a, b, c, d, d + 1, d + 2, d + 3, d + 4, d + 5, &ptr); } SEC("tc") int test_bpf2bpf_dynptr_stack_arg(struct __sk_buff *skb) { return global_dynptr_stack_arg(skb, 1, 2, 3, 4); } /* foo1: a+b+c+d+e+f+g+h+i+j */ __noinline static int foo1(int a, int b, int c, int d, int e, int f, int g, int h, int i, int j) { return a + b + c + d + e + f + g + h + i + j; } /* foo2: a+b+c+d+e+f+g+h+i+j+k+l */ __noinline static int foo2(int a, int b, int c, int d, int e, int f, int g, int h, int i, int j, int k, int l) { return a + b + c + d + e + f + g + h + i + j + k + l; } /* global_two_callees calls foo1 (5 stack args) and foo2 (7 stack args). * The outgoing stack arg area is sized for foo2 (the larger callee). * Stores for foo1 are a subset of the area used by foo2. * Result: foo1(1..10) + foo2(1..12) = 55 + 78 = 133 * * Pass a-e through so the compiler can't constant-fold the stack args away. */ __noinline int global_two_callees(int a, int b, int c, int d, int e) { int ret; ret = foo1(a, b, c, d, e, a + 5, a + 6, a + 7, a + 8, a + 9); ret += foo2(a, b, c, d, e, a + 5, a + 6, a + 7, a + 8, a + 9, a + 10, a + 11); return ret; } SEC("tc") int test_two_callees(void) { return global_two_callees(1, 2, 3, 4, 5); } const volatile int timer_base = 10; static int timer_cb_many_args(void *map, int *key, struct bpf_timer *timer) { int v = timer_base; timer_result = static_func_many_args(v, v * 2, v * 3, v * 4, v * 5, v * 6, v * 7, v * 8, v * 9, v * 10); return 0; } SEC("tc") int test_async_cb_many_args(void) { struct timer_elem *elem; int key = 0; elem = bpf_map_lookup_elem(&timer_map, &key); if (!elem) return -1; bpf_timer_init(&elem->timer, &timer_map, CLOCK_MONOTONIC); bpf_timer_set_callback(&elem->timer, timer_cb_many_args); bpf_timer_start(&elem->timer, 1, 0); return 0; } #else const volatile bool has_stack_arg = false; SEC("tc") int test_global_many_args(void) { return 0; } SEC("tc") int test_bpf2bpf_ptr_stack_arg(void) { return 0; } SEC("tc") int test_bpf2bpf_mix_stack_args(void) { return 0; } SEC("tc") int test_bpf2bpf_nesting_stack_arg(void) { return 0; } SEC("tc") int test_bpf2bpf_dynptr_stack_arg(struct __sk_buff *skb) { return 0; } SEC("tc") int test_two_callees(void) { return 0; } SEC("tc") int test_async_cb_many_args(void) { return 0; } #endif char _license[] SEC("license") = "GPL";