Linux Audio

Check our new training course

Loading...
Note: File does not exist in v3.15.
  1// SPDX-License-Identifier: GPL-2.0
  2/* Copyright (c) 2020 Facebook */
  3
  4#define _GNU_SOURCE
  5#include <netinet/in.h>
  6#include <arpa/inet.h>
  7#include <unistd.h>
  8#include <stdlib.h>
  9#include <string.h>
 10#include <errno.h>
 11#include <sched.h>
 12#include <linux/compiler.h>
 13#include <bpf/libbpf.h>
 14
 15#include "network_helpers.h"
 16#include "test_progs.h"
 17#include "test_btf_skc_cls_ingress.skel.h"
 18
 19static struct test_btf_skc_cls_ingress *skel;
 20static struct sockaddr_in6 srv_sa6;
 21static __u32 duration;
 22
 23#define PROG_PIN_FILE "/sys/fs/bpf/btf_skc_cls_ingress"
 24
 25static int write_sysctl(const char *sysctl, const char *value)
 26{
 27	int fd, err, len;
 28
 29	fd = open(sysctl, O_WRONLY);
 30	if (CHECK(fd == -1, "open sysctl", "open(%s): %s (%d)\n",
 31		  sysctl, strerror(errno), errno))
 32		return -1;
 33
 34	len = strlen(value);
 35	err = write(fd, value, len);
 36	close(fd);
 37	if (CHECK(err != len, "write sysctl",
 38		  "write(%s, %s, %d): err:%d %s (%d)\n",
 39		  sysctl, value, len, err, strerror(errno), errno))
 40		return -1;
 41
 42	return 0;
 43}
 44
 45static int prepare_netns(void)
 46{
 47	if (CHECK(unshare(CLONE_NEWNET), "create netns",
 48		  "unshare(CLONE_NEWNET): %s (%d)",
 49		  strerror(errno), errno))
 50		return -1;
 51
 52	if (CHECK(system("ip link set dev lo up"),
 53		  "ip link set dev lo up", "failed\n"))
 54		return -1;
 55
 56	if (CHECK(system("tc qdisc add dev lo clsact"),
 57		  "tc qdisc add dev lo clsact", "failed\n"))
 58		return -1;
 59
 60	if (CHECK(system("tc filter add dev lo ingress bpf direct-action object-pinned " PROG_PIN_FILE),
 61		  "install tc cls-prog at ingress", "failed\n"))
 62		return -1;
 63
 64	/* Ensure 20 bytes options (i.e. in total 40 bytes tcp header) for the
 65	 * bpf_tcp_gen_syncookie() helper.
 66	 */
 67	if (write_sysctl("/proc/sys/net/ipv4/tcp_window_scaling", "1") ||
 68	    write_sysctl("/proc/sys/net/ipv4/tcp_timestamps", "1") ||
 69	    write_sysctl("/proc/sys/net/ipv4/tcp_sack", "1"))
 70		return -1;
 71
 72	return 0;
 73}
 74
 75static void reset_test(void)
 76{
 77	memset(&skel->bss->srv_sa6, 0, sizeof(skel->bss->srv_sa6));
 78	skel->bss->listen_tp_sport = 0;
 79	skel->bss->req_sk_sport = 0;
 80	skel->bss->recv_cookie = 0;
 81	skel->bss->gen_cookie = 0;
 82	skel->bss->linum = 0;
 83}
 84
 85static void print_err_line(void)
 86{
 87	if (skel->bss->linum)
 88		printf("bpf prog error at line %u\n", skel->bss->linum);
 89}
 90
 91static void test_conn(void)
 92{
 93	int listen_fd = -1, cli_fd = -1, err;
 94	socklen_t addrlen = sizeof(srv_sa6);
 95	int srv_port;
 96
 97	if (write_sysctl("/proc/sys/net/ipv4/tcp_syncookies", "1"))
 98		return;
 99
100	listen_fd = start_server(AF_INET6, SOCK_STREAM, "::1", 0, 0);
101	if (CHECK_FAIL(listen_fd == -1))
102		return;
103
104	err = getsockname(listen_fd, (struct sockaddr *)&srv_sa6, &addrlen);
105	if (CHECK(err, "getsockname(listen_fd)", "err:%d errno:%d\n", err,
106		  errno))
107		goto done;
108	memcpy(&skel->bss->srv_sa6, &srv_sa6, sizeof(srv_sa6));
109	srv_port = ntohs(srv_sa6.sin6_port);
110
111	cli_fd = connect_to_fd(listen_fd, 0);
112	if (CHECK_FAIL(cli_fd == -1))
113		goto done;
114
115	if (CHECK(skel->bss->listen_tp_sport != srv_port ||
116		  skel->bss->req_sk_sport != srv_port,
117		  "Unexpected sk src port",
118		  "listen_tp_sport:%u req_sk_sport:%u expected:%u\n",
119		  skel->bss->listen_tp_sport, skel->bss->req_sk_sport,
120		  srv_port))
121		goto done;
122
123	if (CHECK(skel->bss->gen_cookie || skel->bss->recv_cookie,
124		  "Unexpected syncookie states",
125		  "gen_cookie:%u recv_cookie:%u\n",
126		  skel->bss->gen_cookie, skel->bss->recv_cookie))
127		goto done;
128
129	CHECK(skel->bss->linum, "bpf prog detected error", "at line %u\n",
130	      skel->bss->linum);
131
132done:
133	if (listen_fd != -1)
134		close(listen_fd);
135	if (cli_fd != -1)
136		close(cli_fd);
137}
138
139static void test_syncookie(void)
140{
141	int listen_fd = -1, cli_fd = -1, err;
142	socklen_t addrlen = sizeof(srv_sa6);
143	int srv_port;
144
145	/* Enforce syncookie mode */
146	if (write_sysctl("/proc/sys/net/ipv4/tcp_syncookies", "2"))
147		return;
148
149	listen_fd = start_server(AF_INET6, SOCK_STREAM, "::1", 0, 0);
150	if (CHECK_FAIL(listen_fd == -1))
151		return;
152
153	err = getsockname(listen_fd, (struct sockaddr *)&srv_sa6, &addrlen);
154	if (CHECK(err, "getsockname(listen_fd)", "err:%d errno:%d\n", err,
155		  errno))
156		goto done;
157	memcpy(&skel->bss->srv_sa6, &srv_sa6, sizeof(srv_sa6));
158	srv_port = ntohs(srv_sa6.sin6_port);
159
160	cli_fd = connect_to_fd(listen_fd, 0);
161	if (CHECK_FAIL(cli_fd == -1))
162		goto done;
163
164	if (CHECK(skel->bss->listen_tp_sport != srv_port,
165		  "Unexpected tp src port",
166		  "listen_tp_sport:%u expected:%u\n",
167		  skel->bss->listen_tp_sport, srv_port))
168		goto done;
169
170	if (CHECK(skel->bss->req_sk_sport,
171		  "Unexpected req_sk src port",
172		  "req_sk_sport:%u expected:0\n",
173		   skel->bss->req_sk_sport))
174		goto done;
175
176	if (CHECK(!skel->bss->gen_cookie ||
177		  skel->bss->gen_cookie != skel->bss->recv_cookie,
178		  "Unexpected syncookie states",
179		  "gen_cookie:%u recv_cookie:%u\n",
180		  skel->bss->gen_cookie, skel->bss->recv_cookie))
181		goto done;
182
183	CHECK(skel->bss->linum, "bpf prog detected error", "at line %u\n",
184	      skel->bss->linum);
185
186done:
187	if (listen_fd != -1)
188		close(listen_fd);
189	if (cli_fd != -1)
190		close(cli_fd);
191}
192
193struct test {
194	const char *desc;
195	void (*run)(void);
196};
197
198#define DEF_TEST(name) { #name, test_##name }
199static struct test tests[] = {
200	DEF_TEST(conn),
201	DEF_TEST(syncookie),
202};
203
204void test_btf_skc_cls_ingress(void)
205{
206	int i, err;
207
208	skel = test_btf_skc_cls_ingress__open_and_load();
209	if (CHECK(!skel, "test_btf_skc_cls_ingress__open_and_load", "failed\n"))
210		return;
211
212	err = bpf_program__pin(skel->progs.cls_ingress, PROG_PIN_FILE);
213	if (CHECK(err, "bpf_program__pin",
214		  "cannot pin bpf prog to %s. err:%d\n", PROG_PIN_FILE, err)) {
215		test_btf_skc_cls_ingress__destroy(skel);
216		return;
217	}
218
219	for (i = 0; i < ARRAY_SIZE(tests); i++) {
220		if (!test__start_subtest(tests[i].desc))
221			continue;
222
223		if (prepare_netns())
224			break;
225
226		tests[i].run();
227
228		print_err_line();
229		reset_test();
230	}
231
232	bpf_program__unpin(skel->progs.cls_ingress, PROG_PIN_FILE);
233	test_btf_skc_cls_ingress__destroy(skel);
234}