Linux Audio

Check our new training course

Loading...
Note: File does not exist in v6.13.7.
  1#include <errno.h>
  2#include <stdlib.h>
  3#include <stdio.h>
  4#include <string.h>
  5#include <unistd.h>
  6#include <sys/time.h>
  7
  8#include <linux/bpf.h>
  9#include <linux/filter.h>
 10#include <linux/unistd.h>
 11
 12#include <bpf/bpf.h>
 13
 14#define LOG_SIZE (1 << 20)
 15
 16#define err(str...)	printf("ERROR: " str)
 17
 18static const struct bpf_insn code_sample[] = {
 19	/* We need a few instructions to pass the min log length */
 20	BPF_MOV64_IMM(BPF_REG_0, 0),
 21	BPF_MOV64_IMM(BPF_REG_0, 0),
 22	BPF_MOV64_IMM(BPF_REG_0, 0),
 23	BPF_MOV64_IMM(BPF_REG_0, 0),
 24	BPF_MOV64_IMM(BPF_REG_0, 0),
 25	BPF_MOV64_IMM(BPF_REG_0, 0),
 26	BPF_MOV64_IMM(BPF_REG_0, 0),
 27	BPF_MOV64_IMM(BPF_REG_0, 0),
 28	BPF_MOV64_IMM(BPF_REG_0, 0),
 29	BPF_MOV64_IMM(BPF_REG_0, 0),
 30	BPF_MOV64_IMM(BPF_REG_0, 0),
 31	BPF_MOV64_IMM(BPF_REG_0, 0),
 32	BPF_MOV64_IMM(BPF_REG_0, 0),
 33	BPF_MOV64_IMM(BPF_REG_0, 0),
 34	BPF_MOV64_IMM(BPF_REG_0, 0),
 35	BPF_MOV64_IMM(BPF_REG_0, 0),
 36	BPF_RAW_INSN(BPF_JMP | BPF_CALL, 0, 0, 0,
 37		     BPF_FUNC_map_lookup_elem),
 38	BPF_EXIT_INSN(),
 39};
 40
 41static inline __u64 ptr_to_u64(const void *ptr)
 42{
 43	return (__u64) (unsigned long) ptr;
 44}
 45
 46static int load(char *log, size_t log_len, int log_level)
 47{
 48	union bpf_attr attr;
 49
 50	bzero(&attr, sizeof(attr));
 51	attr.prog_type = BPF_PROG_TYPE_SOCKET_FILTER;
 52	attr.insn_cnt = (__u32)(sizeof(code_sample) / sizeof(struct bpf_insn));
 53	attr.insns = ptr_to_u64(code_sample);
 54	attr.license = ptr_to_u64("GPL");
 55	attr.log_buf = ptr_to_u64(log);
 56	attr.log_size = log_len;
 57	attr.log_level = log_level;
 58
 59	return syscall(__NR_bpf, BPF_PROG_LOAD, &attr, sizeof(attr));
 60}
 61
 62static void check_ret(int ret, int exp_errno)
 63{
 64	if (ret > 0) {
 65		close(ret);
 66		err("broken sample loaded successfully!?\n");
 67		exit(1);
 68	}
 69
 70	if (!ret || errno != exp_errno) {
 71		err("Program load returned: ret:%d/errno:%d, expected ret:%d/errno:%d\n",
 72		    ret, errno, -1, exp_errno);
 73		exit(1);
 74	}
 75}
 76
 77static void check_ones(const char *buf, size_t len, const char *msg)
 78{
 79	while (len--)
 80		if (buf[len] != 1) {
 81			err("%s", msg);
 82			exit(1);
 83		}
 84}
 85
 86static void test_log_good(char *log, size_t buf_len, size_t log_len,
 87			  size_t exp_len, int exp_errno, const char *full_log)
 88{
 89	size_t len;
 90	int ret;
 91
 92	memset(log, 1, buf_len);
 93
 94	ret = load(log, log_len, 1);
 95	check_ret(ret, exp_errno);
 96
 97	len = strnlen(log, buf_len);
 98	if (len == buf_len) {
 99		err("verifier did not NULL terminate the log\n");
100		exit(1);
101	}
102	if (exp_len && len != exp_len) {
103		err("incorrect log length expected:%zd have:%zd\n",
104		    exp_len, len);
105		exit(1);
106	}
107
108	if (strchr(log, 1)) {
109		err("verifier leaked a byte through\n");
110		exit(1);
111	}
112
113	check_ones(log + len + 1, buf_len - len - 1,
114		   "verifier wrote bytes past NULL termination\n");
115
116	if (memcmp(full_log, log, LOG_SIZE)) {
117		err("log did not match expected output\n");
118		exit(1);
119	}
120}
121
122static void test_log_bad(char *log, size_t log_len, int log_level)
123{
124	int ret;
125
126	ret = load(log, log_len, log_level);
127	check_ret(ret, EINVAL);
128	if (log)
129		check_ones(log, LOG_SIZE,
130			   "verifier touched log with bad parameters\n");
131}
132
133int main(int argc, char **argv)
134{
135	char full_log[LOG_SIZE];
136	char log[LOG_SIZE];
137	size_t want_len;
138	int i;
139
140	memset(log, 1, LOG_SIZE);
141
142	/* Use libbpf 1.0 API mode */
143	libbpf_set_strict_mode(LIBBPF_STRICT_ALL);
144
145	/* Test incorrect attr */
146	printf("Test log_level 0...\n");
147	test_log_bad(log, LOG_SIZE, 0);
148
149	printf("Test log_size < 128...\n");
150	test_log_bad(log, 15, 1);
151
152	printf("Test log_buff = NULL...\n");
153	test_log_bad(NULL, LOG_SIZE, 1);
154
155	/* Test with log big enough */
156	printf("Test oversized buffer...\n");
157	test_log_good(full_log, LOG_SIZE, LOG_SIZE, 0, EACCES, full_log);
158
159	want_len = strlen(full_log);
160
161	printf("Test exact buffer...\n");
162	test_log_good(log, LOG_SIZE, want_len + 2, want_len, EACCES, full_log);
163
164	printf("Test undersized buffers...\n");
165	for (i = 0; i < 64; i++) {
166		full_log[want_len - i + 1] = 1;
167		full_log[want_len - i] = 0;
168
169		test_log_good(log, LOG_SIZE, want_len + 1 - i, want_len - i,
170			      ENOSPC, full_log);
171	}
172
173	printf("test_verifier_log: OK\n");
174	return 0;
175}