| // SPDX-License-Identifier: GPL-2.0 | 
 | /* | 
 |  * Augment the filename syscalls with the contents of the filename pointer argument | 
 |  * filtering only those that do not start with /etc/. | 
 |  * | 
 |  * Test it with: | 
 |  * | 
 |  * perf trace -e tools/perf/examples/bpf/augmented_syscalls.c cat /etc/passwd > /dev/null | 
 |  * | 
 |  * It'll catch some openat syscalls related to the dynamic linked and | 
 |  * the last one should be the one for '/etc/passwd'. | 
 |  * | 
 |  * This matches what is marshalled into the raw_syscall:sys_enter payload | 
 |  * expected by the 'perf trace' beautifiers, and can be used by them unmodified, | 
 |  * which will be done as that feature is implemented in the next csets, for now | 
 |  * it will appear in a dump done by the default tracepoint handler in 'perf trace', | 
 |  * that uses bpf_output__fprintf() to just dump those contents, as done with | 
 |  * the bpf-output event associated with the __bpf_output__ map declared in | 
 |  * tools/perf/include/bpf/stdio.h. | 
 |  */ | 
 |  | 
 | #include <stdio.h> | 
 |  | 
 | /* bpf-output associated map */ | 
 | bpf_map(__augmented_syscalls__, PERF_EVENT_ARRAY, int, u32, __NR_CPUS__); | 
 |  | 
 | struct augmented_filename { | 
 | 	int	size; | 
 | 	int	reserved; | 
 | 	char	value[64]; | 
 | }; | 
 |  | 
 | #define augmented_filename_syscall_enter(syscall) 						\ | 
 | struct augmented_enter_##syscall##_args {			 				\ | 
 | 	struct syscall_enter_##syscall##_args	args;				 		\ | 
 | 	struct augmented_filename		filename;				 	\ | 
 | };												\ | 
 | int syscall_enter(syscall)(struct syscall_enter_##syscall##_args *args)				\ | 
 | {												\ | 
 | 	char etc[6] = "/etc/";									\ | 
 | 	struct augmented_enter_##syscall##_args augmented_args = { .filename.reserved = 0, }; 	\ | 
 | 	probe_read(&augmented_args.args, sizeof(augmented_args.args), args);			\ | 
 | 	augmented_args.filename.size = probe_read_str(&augmented_args.filename.value, 		\ | 
 | 						      sizeof(augmented_args.filename.value), 	\ | 
 | 						      args->filename_ptr); 			\ | 
 | 	if (__builtin_memcmp(augmented_args.filename.value, etc, 4) != 0)			\ | 
 | 		return 0;									\ | 
 | 	/* If perf_event_output fails, return non-zero so that it gets recorded unaugmented */	\ | 
 | 	return perf_event_output(args, &__augmented_syscalls__, BPF_F_CURRENT_CPU, 		\ | 
 | 				 &augmented_args,						\ | 
 | 				 (sizeof(augmented_args) - sizeof(augmented_args.filename.value) + \ | 
 | 				 augmented_args.filename.size));				\ | 
 | } | 
 |  | 
 | struct syscall_enter_openat_args { | 
 | 	unsigned long long common_tp_fields; | 
 | 	long		   syscall_nr; | 
 | 	long		   dfd; | 
 | 	char		   *filename_ptr; | 
 | 	long		   flags; | 
 | 	long		   mode; | 
 | }; | 
 |  | 
 | augmented_filename_syscall_enter(openat); | 
 |  | 
 | struct syscall_enter_open_args { | 
 | 	unsigned long long common_tp_fields; | 
 | 	long		   syscall_nr; | 
 | 	char		   *filename_ptr; | 
 | 	long		   flags; | 
 | 	long		   mode; | 
 | }; | 
 |  | 
 | augmented_filename_syscall_enter(open); | 
 |  | 
 | license(GPL); |