xref: /openbmc/linux/tools/testing/selftests/bpf/progs/linked_funcs2.c (revision 1ac731c529cd4d6adbce134754b51ff7d822b145)
1f2644fb4SAndrii Nakryiko // SPDX-License-Identifier: GPL-2.0
2f2644fb4SAndrii Nakryiko /* Copyright (c) 2021 Facebook */
3f2644fb4SAndrii Nakryiko 
4f2644fb4SAndrii Nakryiko #include "vmlinux.h"
5f2644fb4SAndrii Nakryiko #include <bpf/bpf_helpers.h>
6f2644fb4SAndrii Nakryiko #include <bpf/bpf_tracing.h>
78555defeSAndrii Nakryiko #include <bpf/bpf_core_read.h>
8*c8ed6685SAndrii Nakryiko #include "bpf_misc.h"
9f2644fb4SAndrii Nakryiko 
10f2644fb4SAndrii Nakryiko /* weak and shared between both files */
11f2644fb4SAndrii Nakryiko const volatile int my_tid __weak;
12f2644fb4SAndrii Nakryiko long syscall_id __weak;
13f2644fb4SAndrii Nakryiko 
14f2644fb4SAndrii Nakryiko int output_val2;
15f2644fb4SAndrii Nakryiko int output_ctx2;
16f2644fb4SAndrii Nakryiko int output_weak2; /* should stay zero */
17f2644fb4SAndrii Nakryiko 
18f2644fb4SAndrii Nakryiko /* same "subprog" name in all files, but it's ok because they all are static */
subprog(int x)19f2644fb4SAndrii Nakryiko static __noinline int subprog(int x)
20f2644fb4SAndrii Nakryiko {
21f2644fb4SAndrii Nakryiko 	/* but different formula */
22f2644fb4SAndrii Nakryiko 	return x * 2;
23f2644fb4SAndrii Nakryiko }
24f2644fb4SAndrii Nakryiko 
25f2644fb4SAndrii Nakryiko /* Global functions can't be void */
set_output_val2(int x)26f2644fb4SAndrii Nakryiko int set_output_val2(int x)
27f2644fb4SAndrii Nakryiko {
28f2644fb4SAndrii Nakryiko 	output_val2 = 2 * x + 2 * subprog(x);
29f2644fb4SAndrii Nakryiko 	return 2 * x;
30f2644fb4SAndrii Nakryiko }
31f2644fb4SAndrii Nakryiko 
32f2644fb4SAndrii Nakryiko /* This function can't be verified as global, as it assumes raw_tp/sys_enter
33f2644fb4SAndrii Nakryiko  * context and accesses syscall id (second argument). So we mark it as
34f2644fb4SAndrii Nakryiko  * __hidden, so that libbpf will mark it as static in the final object file,
35f2644fb4SAndrii Nakryiko  * right before verifying it in the kernel.
36f2644fb4SAndrii Nakryiko  *
37f2644fb4SAndrii Nakryiko  * But we don't mark it as __hidden here, rather at extern site. __hidden is
38f2644fb4SAndrii Nakryiko  * "contaminating" visibility, so it will get propagated from either extern or
39f2644fb4SAndrii Nakryiko  * actual definition (including from the losing __weak definition).
40f2644fb4SAndrii Nakryiko  */
set_output_ctx2(__u64 * ctx)41f2644fb4SAndrii Nakryiko void set_output_ctx2(__u64 *ctx)
42f2644fb4SAndrii Nakryiko {
43f2644fb4SAndrii Nakryiko 	output_ctx2 = ctx[1]; /* long id, same as in BPF_PROG below */
44f2644fb4SAndrii Nakryiko }
45f2644fb4SAndrii Nakryiko 
46f2644fb4SAndrii Nakryiko /* this weak instance should lose, because it will be processed second */
set_output_weak(int x)47f2644fb4SAndrii Nakryiko __weak int set_output_weak(int x)
48f2644fb4SAndrii Nakryiko {
498555defeSAndrii Nakryiko 	static volatile int whatever;
508555defeSAndrii Nakryiko 
518555defeSAndrii Nakryiko 	/* make sure we use CO-RE relocations in a weak function, this used to
528555defeSAndrii Nakryiko 	 * cause problems for BPF static linker
538555defeSAndrii Nakryiko 	 */
548555defeSAndrii Nakryiko 	whatever = 2 * bpf_core_type_size(struct task_struct);
55*c8ed6685SAndrii Nakryiko 	__sink(whatever);
568555defeSAndrii Nakryiko 
57f2644fb4SAndrii Nakryiko 	output_weak2 = x;
58f2644fb4SAndrii Nakryiko 	return 2 * x;
59f2644fb4SAndrii Nakryiko }
60f2644fb4SAndrii Nakryiko 
61f2644fb4SAndrii Nakryiko extern int set_output_val1(int x);
62f2644fb4SAndrii Nakryiko 
63f2644fb4SAndrii Nakryiko /* here we'll force set_output_ctx1() to be __hidden in the final obj file */
64f2644fb4SAndrii Nakryiko __hidden extern void set_output_ctx1(__u64 *ctx);
65f2644fb4SAndrii Nakryiko 
66b82bb1ffSAndrii Nakryiko SEC("?raw_tp/sys_enter")
BPF_PROG(handler2,struct pt_regs * regs,long id)67f2644fb4SAndrii Nakryiko int BPF_PROG(handler2, struct pt_regs *regs, long id)
68f2644fb4SAndrii Nakryiko {
69b82bb1ffSAndrii Nakryiko 	static volatile int whatever;
70b82bb1ffSAndrii Nakryiko 
71f2644fb4SAndrii Nakryiko 	if (my_tid != (u32)bpf_get_current_pid_tgid() || id != syscall_id)
72f2644fb4SAndrii Nakryiko 		return 0;
73f2644fb4SAndrii Nakryiko 
74b82bb1ffSAndrii Nakryiko 	/* make sure we have CO-RE relocations in main program */
75b82bb1ffSAndrii Nakryiko 	whatever = bpf_core_type_size(struct task_struct);
76*c8ed6685SAndrii Nakryiko 	__sink(whatever);
77b82bb1ffSAndrii Nakryiko 
78f2644fb4SAndrii Nakryiko 	set_output_val1(2000);
79f2644fb4SAndrii Nakryiko 	set_output_ctx1(ctx); /* ctx definition is hidden in BPF_PROG macro */
80f2644fb4SAndrii Nakryiko 
81f2644fb4SAndrii Nakryiko 	/* keep input value the same across both files to avoid dependency on
82f2644fb4SAndrii Nakryiko 	 * handler call order; differentiate by output_weak1 vs output_weak2.
83f2644fb4SAndrii Nakryiko 	 */
84f2644fb4SAndrii Nakryiko 	set_output_weak(42);
85f2644fb4SAndrii Nakryiko 
86f2644fb4SAndrii Nakryiko 	return 0;
87f2644fb4SAndrii Nakryiko }
88f2644fb4SAndrii Nakryiko 
89f2644fb4SAndrii Nakryiko char LICENSE[] SEC("license") = "GPL";
90