test_maps.c 23.4 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
/*
 * Testsuite for eBPF maps
 *
 * Copyright (c) 2014 PLUMgrid, http://plumgrid.com
 * Copyright (c) 2016 Facebook
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of version 2 of the GNU General Public
 * License as published by the Free Software Foundation.
 */

#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>

#include <sys/wait.h>
#include <sys/resource.h>

#include <linux/bpf.h>

24
#include <bpf/bpf.h>
25
#include <bpf/libbpf.h>
26
#include "bpf_util.h"
27 28 29 30 31

static int map_flags;

static void test_hashmap(int task, void *data)
{
32
	long long key, next_key, first_key, value;
33 34
	int fd;

35
	fd = bpf_create_map(BPF_MAP_TYPE_HASH, sizeof(key), sizeof(value),
36 37 38 39 40 41 42 43 44
			    2, map_flags);
	if (fd < 0) {
		printf("Failed to create hashmap '%s'!\n", strerror(errno));
		exit(1);
	}

	key = 1;
	value = 1234;
	/* Insert key=1 element. */
45
	assert(bpf_map_update_elem(fd, &key, &value, BPF_ANY) == 0);
46 47 48

	value = 0;
	/* BPF_NOEXIST means add new element if it doesn't exist. */
49
	assert(bpf_map_update_elem(fd, &key, &value, BPF_NOEXIST) == -1 &&
50 51 52 53
	       /* key=1 already exists. */
	       errno == EEXIST);

	/* -1 is an invalid flag. */
54 55
	assert(bpf_map_update_elem(fd, &key, &value, -1) == -1 &&
	       errno == EINVAL);
56 57

	/* Check that key=1 can be found. */
58
	assert(bpf_map_lookup_elem(fd, &key, &value) == 0 && value == 1234);
59 60 61

	key = 2;
	/* Check that key=2 is not found. */
62
	assert(bpf_map_lookup_elem(fd, &key, &value) == -1 && errno == ENOENT);
63 64

	/* BPF_EXIST means update existing element. */
65
	assert(bpf_map_update_elem(fd, &key, &value, BPF_EXIST) == -1 &&
66 67 68 69
	       /* key=2 is not there. */
	       errno == ENOENT);

	/* Insert key=2 element. */
70
	assert(bpf_map_update_elem(fd, &key, &value, BPF_NOEXIST) == 0);
71 72 73 74 75

	/* key=1 and key=2 were inserted, check that key=0 cannot be
	 * inserted due to max_entries limit.
	 */
	key = 0;
76
	assert(bpf_map_update_elem(fd, &key, &value, BPF_NOEXIST) == -1 &&
77 78 79 80
	       errno == E2BIG);

	/* Update existing element, though the map is full. */
	key = 1;
81
	assert(bpf_map_update_elem(fd, &key, &value, BPF_EXIST) == 0);
82
	key = 2;
83
	assert(bpf_map_update_elem(fd, &key, &value, BPF_ANY) == 0);
84 85 86
	key = 3;
	assert(bpf_map_update_elem(fd, &key, &value, BPF_NOEXIST) == -1 &&
	       errno == E2BIG);
87 88 89

	/* Check that key = 0 doesn't exist. */
	key = 0;
90
	assert(bpf_map_delete_elem(fd, &key) == -1 && errno == ENOENT);
91 92

	/* Iterate over two elements. */
93 94
	assert(bpf_map_get_next_key(fd, NULL, &first_key) == 0 &&
	       (first_key == 1 || first_key == 2));
95
	assert(bpf_map_get_next_key(fd, &key, &next_key) == 0 &&
96
	       (next_key == first_key));
97
	assert(bpf_map_get_next_key(fd, &next_key, &next_key) == 0 &&
98 99
	       (next_key == 1 || next_key == 2) &&
	       (next_key != first_key));
100
	assert(bpf_map_get_next_key(fd, &next_key, &next_key) == -1 &&
101 102 103 104
	       errno == ENOENT);

	/* Delete both elements. */
	key = 1;
105
	assert(bpf_map_delete_elem(fd, &key) == 0);
106
	key = 2;
107 108
	assert(bpf_map_delete_elem(fd, &key) == 0);
	assert(bpf_map_delete_elem(fd, &key) == -1 && errno == ENOENT);
109 110 111

	key = 0;
	/* Check that map is empty. */
112 113
	assert(bpf_map_get_next_key(fd, NULL, &next_key) == -1 &&
	       errno == ENOENT);
114
	assert(bpf_map_get_next_key(fd, &key, &next_key) == -1 &&
115 116 117 118 119
	       errno == ENOENT);

	close(fd);
}

120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137
static void test_hashmap_sizes(int task, void *data)
{
	int fd, i, j;

	for (i = 1; i <= 512; i <<= 1)
		for (j = 1; j <= 1 << 18; j <<= 1) {
			fd = bpf_create_map(BPF_MAP_TYPE_HASH, i, j,
					    2, map_flags);
			if (fd < 0) {
				printf("Failed to create hashmap key=%d value=%d '%s'\n",
				       i, j, strerror(errno));
				exit(1);
			}
			close(fd);
			usleep(10); /* give kernel time to destroy */
		}
}

138 139
static void test_hashmap_percpu(int task, void *data)
{
140
	unsigned int nr_cpus = bpf_num_possible_cpus();
141
	BPF_DECLARE_PERCPU(long, value);
142
	long long key, next_key, first_key;
143 144 145
	int expected_key_mask = 0;
	int fd, i;

146
	fd = bpf_create_map(BPF_MAP_TYPE_PERCPU_HASH, sizeof(key),
147
			    sizeof(bpf_percpu(value, 0)), 2, map_flags);
148 149 150 151 152 153
	if (fd < 0) {
		printf("Failed to create hashmap '%s'!\n", strerror(errno));
		exit(1);
	}

	for (i = 0; i < nr_cpus; i++)
154
		bpf_percpu(value, i) = i + 100;
155 156 157 158

	key = 1;
	/* Insert key=1 element. */
	assert(!(expected_key_mask & key));
159
	assert(bpf_map_update_elem(fd, &key, value, BPF_ANY) == 0);
160 161 162
	expected_key_mask |= key;

	/* BPF_NOEXIST means add new element if it doesn't exist. */
163
	assert(bpf_map_update_elem(fd, &key, value, BPF_NOEXIST) == -1 &&
164 165 166 167
	       /* key=1 already exists. */
	       errno == EEXIST);

	/* -1 is an invalid flag. */
168 169
	assert(bpf_map_update_elem(fd, &key, value, -1) == -1 &&
	       errno == EINVAL);
170 171 172 173

	/* Check that key=1 can be found. Value could be 0 if the lookup
	 * was run from a different CPU.
	 */
174 175 176
	bpf_percpu(value, 0) = 1;
	assert(bpf_map_lookup_elem(fd, &key, value) == 0 &&
	       bpf_percpu(value, 0) == 100);
177 178 179

	key = 2;
	/* Check that key=2 is not found. */
180
	assert(bpf_map_lookup_elem(fd, &key, value) == -1 && errno == ENOENT);
181 182

	/* BPF_EXIST means update existing element. */
183
	assert(bpf_map_update_elem(fd, &key, value, BPF_EXIST) == -1 &&
184 185 186 187 188
	       /* key=2 is not there. */
	       errno == ENOENT);

	/* Insert key=2 element. */
	assert(!(expected_key_mask & key));
189
	assert(bpf_map_update_elem(fd, &key, value, BPF_NOEXIST) == 0);
190 191 192 193 194 195
	expected_key_mask |= key;

	/* key=1 and key=2 were inserted, check that key=0 cannot be
	 * inserted due to max_entries limit.
	 */
	key = 0;
196
	assert(bpf_map_update_elem(fd, &key, value, BPF_NOEXIST) == -1 &&
197 198 199
	       errno == E2BIG);

	/* Check that key = 0 doesn't exist. */
200
	assert(bpf_map_delete_elem(fd, &key) == -1 && errno == ENOENT);
201 202

	/* Iterate over two elements. */
203 204
	assert(bpf_map_get_next_key(fd, NULL, &first_key) == 0 &&
	       ((expected_key_mask & first_key) == first_key));
205
	while (!bpf_map_get_next_key(fd, &key, &next_key)) {
206 207 208 209
		if (first_key) {
			assert(next_key == first_key);
			first_key = 0;
		}
210 211 212
		assert((expected_key_mask & next_key) == next_key);
		expected_key_mask &= ~next_key;

213
		assert(bpf_map_lookup_elem(fd, &next_key, value) == 0);
214 215

		for (i = 0; i < nr_cpus; i++)
216
			assert(bpf_percpu(value, i) == i + 100);
217 218 219 220 221 222 223

		key = next_key;
	}
	assert(errno == ENOENT);

	/* Update with BPF_EXIST. */
	key = 1;
224
	assert(bpf_map_update_elem(fd, &key, value, BPF_EXIST) == 0);
225 226 227

	/* Delete both elements. */
	key = 1;
228
	assert(bpf_map_delete_elem(fd, &key) == 0);
229
	key = 2;
230 231
	assert(bpf_map_delete_elem(fd, &key) == 0);
	assert(bpf_map_delete_elem(fd, &key) == -1 && errno == ENOENT);
232 233 234

	key = 0;
	/* Check that map is empty. */
235 236
	assert(bpf_map_get_next_key(fd, NULL, &next_key) == -1 &&
	       errno == ENOENT);
237
	assert(bpf_map_get_next_key(fd, &key, &next_key) == -1 &&
238 239 240 241 242
	       errno == ENOENT);

	close(fd);
}

243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290
static void test_hashmap_walk(int task, void *data)
{
	int fd, i, max_entries = 100000;
	long long key, value, next_key;
	bool next_key_valid = true;

	fd = bpf_create_map(BPF_MAP_TYPE_HASH, sizeof(key), sizeof(value),
			    max_entries, map_flags);
	if (fd < 0) {
		printf("Failed to create hashmap '%s'!\n", strerror(errno));
		exit(1);
	}

	for (i = 0; i < max_entries; i++) {
		key = i; value = key;
		assert(bpf_map_update_elem(fd, &key, &value, BPF_NOEXIST) == 0);
	}

	for (i = 0; bpf_map_get_next_key(fd, !i ? NULL : &key,
					 &next_key) == 0; i++) {
		key = next_key;
		assert(bpf_map_lookup_elem(fd, &key, &value) == 0);
	}

	assert(i == max_entries);

	assert(bpf_map_get_next_key(fd, NULL, &key) == 0);
	for (i = 0; next_key_valid; i++) {
		next_key_valid = bpf_map_get_next_key(fd, &key, &next_key) == 0;
		assert(bpf_map_lookup_elem(fd, &key, &value) == 0);
		value++;
		assert(bpf_map_update_elem(fd, &key, &value, BPF_EXIST) == 0);
		key = next_key;
	}

	assert(i == max_entries);

	for (i = 0; bpf_map_get_next_key(fd, !i ? NULL : &key,
					 &next_key) == 0; i++) {
		key = next_key;
		assert(bpf_map_lookup_elem(fd, &key, &value) == 0);
		assert(value - 1 == key);
	}

	assert(i == max_entries);
	close(fd);
}

291 292 293 294 295
static void test_arraymap(int task, void *data)
{
	int key, next_key, fd;
	long long value;

296
	fd = bpf_create_map(BPF_MAP_TYPE_ARRAY, sizeof(key), sizeof(value),
297 298 299 300 301 302 303 304 305
			    2, 0);
	if (fd < 0) {
		printf("Failed to create arraymap '%s'!\n", strerror(errno));
		exit(1);
	}

	key = 1;
	value = 1234;
	/* Insert key=1 element. */
306
	assert(bpf_map_update_elem(fd, &key, &value, BPF_ANY) == 0);
307 308

	value = 0;
309
	assert(bpf_map_update_elem(fd, &key, &value, BPF_NOEXIST) == -1 &&
310 311 312
	       errno == EEXIST);

	/* Check that key=1 can be found. */
313
	assert(bpf_map_lookup_elem(fd, &key, &value) == 0 && value == 1234);
314 315 316

	key = 0;
	/* Check that key=0 is also found and zero initialized. */
317
	assert(bpf_map_lookup_elem(fd, &key, &value) == 0 && value == 0);
318 319 320 321 322

	/* key=0 and key=1 were inserted, check that key=2 cannot be inserted
	 * due to max_entries limit.
	 */
	key = 2;
323
	assert(bpf_map_update_elem(fd, &key, &value, BPF_EXIST) == -1 &&
324 325 326
	       errno == E2BIG);

	/* Check that key = 2 doesn't exist. */
327
	assert(bpf_map_lookup_elem(fd, &key, &value) == -1 && errno == ENOENT);
328 329

	/* Iterate over two elements. */
330 331
	assert(bpf_map_get_next_key(fd, NULL, &next_key) == 0 &&
	       next_key == 0);
332
	assert(bpf_map_get_next_key(fd, &key, &next_key) == 0 &&
333
	       next_key == 0);
334
	assert(bpf_map_get_next_key(fd, &next_key, &next_key) == 0 &&
335
	       next_key == 1);
336
	assert(bpf_map_get_next_key(fd, &next_key, &next_key) == -1 &&
337 338 339 340
	       errno == ENOENT);

	/* Delete shouldn't succeed. */
	key = 1;
341
	assert(bpf_map_delete_elem(fd, &key) == -1 && errno == EINVAL);
342 343 344 345 346 347

	close(fd);
}

static void test_arraymap_percpu(int task, void *data)
{
348
	unsigned int nr_cpus = bpf_num_possible_cpus();
349
	BPF_DECLARE_PERCPU(long, values);
350 351
	int key, next_key, fd, i;

352
	fd = bpf_create_map(BPF_MAP_TYPE_PERCPU_ARRAY, sizeof(key),
353
			    sizeof(bpf_percpu(values, 0)), 2, 0);
354 355 356 357 358 359
	if (fd < 0) {
		printf("Failed to create arraymap '%s'!\n", strerror(errno));
		exit(1);
	}

	for (i = 0; i < nr_cpus; i++)
360
		bpf_percpu(values, i) = i + 100;
361 362 363

	key = 1;
	/* Insert key=1 element. */
364
	assert(bpf_map_update_elem(fd, &key, values, BPF_ANY) == 0);
365

366
	bpf_percpu(values, 0) = 0;
367
	assert(bpf_map_update_elem(fd, &key, values, BPF_NOEXIST) == -1 &&
368 369 370
	       errno == EEXIST);

	/* Check that key=1 can be found. */
371 372
	assert(bpf_map_lookup_elem(fd, &key, values) == 0 &&
	       bpf_percpu(values, 0) == 100);
373 374 375

	key = 0;
	/* Check that key=0 is also found and zero initialized. */
376
	assert(bpf_map_lookup_elem(fd, &key, values) == 0 &&
377 378
	       bpf_percpu(values, 0) == 0 &&
	       bpf_percpu(values, nr_cpus - 1) == 0);
379 380 381

	/* Check that key=2 cannot be inserted due to max_entries limit. */
	key = 2;
382
	assert(bpf_map_update_elem(fd, &key, values, BPF_EXIST) == -1 &&
383 384 385
	       errno == E2BIG);

	/* Check that key = 2 doesn't exist. */
386
	assert(bpf_map_lookup_elem(fd, &key, values) == -1 && errno == ENOENT);
387 388

	/* Iterate over two elements. */
389 390
	assert(bpf_map_get_next_key(fd, NULL, &next_key) == 0 &&
	       next_key == 0);
391
	assert(bpf_map_get_next_key(fd, &key, &next_key) == 0 &&
392
	       next_key == 0);
393
	assert(bpf_map_get_next_key(fd, &next_key, &next_key) == 0 &&
394
	       next_key == 1);
395
	assert(bpf_map_get_next_key(fd, &next_key, &next_key) == -1 &&
396 397 398 399
	       errno == ENOENT);

	/* Delete shouldn't succeed. */
	key = 1;
400
	assert(bpf_map_delete_elem(fd, &key) == -1 && errno == EINVAL);
401 402 403 404 405 406

	close(fd);
}

static void test_arraymap_percpu_many_keys(void)
{
407
	unsigned int nr_cpus = bpf_num_possible_cpus();
408
	BPF_DECLARE_PERCPU(long, values);
409 410 411 412
	/* nr_keys is not too large otherwise the test stresses percpu
	 * allocator more than anything else
	 */
	unsigned int nr_keys = 2000;
413 414
	int key, fd, i;

415
	fd = bpf_create_map(BPF_MAP_TYPE_PERCPU_ARRAY, sizeof(key),
416
			    sizeof(bpf_percpu(values, 0)), nr_keys, 0);
417 418 419 420 421 422 423
	if (fd < 0) {
		printf("Failed to create per-cpu arraymap '%s'!\n",
		       strerror(errno));
		exit(1);
	}

	for (i = 0; i < nr_cpus; i++)
424
		bpf_percpu(values, i) = i + 10;
425 426

	for (key = 0; key < nr_keys; key++)
427
		assert(bpf_map_update_elem(fd, &key, values, BPF_ANY) == 0);
428 429 430

	for (key = 0; key < nr_keys; key++) {
		for (i = 0; i < nr_cpus; i++)
431
			bpf_percpu(values, i) = 0;
432

433
		assert(bpf_map_lookup_elem(fd, &key, values) == 0);
434 435

		for (i = 0; i < nr_cpus; i++)
436
			assert(bpf_percpu(values, i) == i + 10);
437 438 439 440 441
	}

	close(fd);
}

442 443
static void test_devmap(int task, void *data)
{
J
John Fastabend 已提交
444
	int fd;
445 446 447 448 449 450 451 452 453 454 455 456
	__u32 key, value;

	fd = bpf_create_map(BPF_MAP_TYPE_DEVMAP, sizeof(key), sizeof(value),
			    2, 0);
	if (fd < 0) {
		printf("Failed to create arraymap '%s'!\n", strerror(errno));
		exit(1);
	}

	close(fd);
}

457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549
#include <sys/socket.h>
#include <sys/ioctl.h>
#include <arpa/inet.h>
#include <sys/select.h>
#include <linux/err.h>
#define SOCKMAP_PARSE_PROG "./sockmap_parse_prog.o"
#define SOCKMAP_VERDICT_PROG "./sockmap_verdict_prog.o"
static void test_sockmap(int task, void *data)
{
	int ports[] = {50200, 50201, 50202, 50204};
	int err, i, fd, sfd[6] = {0xdeadbeef};
	char buf[] = "hello sockmap user\n";
	int one = 1, map_fd, s, sc, rc;
	int parse_prog, verdict_prog;
	struct bpf_map *bpf_map;
	struct sockaddr_in addr;
	struct bpf_object *obj;
	struct timeval to;
	__u32 key, value;
	fd_set w;

	/* Create some sockets to use with sockmap */
	for (i = 0; i < 2; i++) {
		sfd[i] = socket(AF_INET, SOCK_STREAM, 0);
		if (sfd[i] < 0)
			goto out;
		err = setsockopt(sfd[i], SOL_SOCKET, SO_REUSEADDR,
				 (char *)&one, sizeof(one));
		if (err) {
			printf("failed to setsockopt\n");
			goto out;
		}
		err = ioctl(sfd[i], FIONBIO, (char *)&one);
		if (err < 0) {
			printf("failed to ioctl\n");
			goto out;
		}
		memset(&addr, 0, sizeof(struct sockaddr_in));
		addr.sin_family = AF_INET;
		addr.sin_addr.s_addr = inet_addr("127.0.0.1");
		addr.sin_port = htons(ports[i]);
		err = bind(sfd[i], (struct sockaddr *)&addr, sizeof(addr));
		if (err < 0) {
			printf("failed to bind: err %i: %i:%i\n",
			       err, i, sfd[i]);
			goto out;
		}
		err = listen(sfd[i], 32);
		if (err < 0) {
			printf("failed to listeen\n");
			goto out;
		}
	}

	for (i = 2; i < 4; i++) {
		sfd[i] = socket(AF_INET, SOCK_STREAM, 0);
		if (sfd[i] < 0)
			goto out;
		err = setsockopt(sfd[i], SOL_SOCKET, SO_REUSEADDR,
				 (char *)&one, sizeof(one));
		if (err) {
			printf("set sock opt\n");
			goto out;
		}
		memset(&addr, 0, sizeof(struct sockaddr_in));
		addr.sin_family = AF_INET;
		addr.sin_addr.s_addr = inet_addr("127.0.0.1");
		addr.sin_port = htons(ports[i - 2]);
		err = connect(sfd[i], (struct sockaddr *)&addr, sizeof(addr));
		if (err) {
			printf("failed to conenct\n");
			goto out;
		}
	}


	for (i = 4; i < 6; i++) {
		sfd[i] = accept(sfd[i - 4], NULL, NULL);
		if (sfd[i] < 0) {
			printf("accept failed\n");
			goto out;
		}
	}

	/* Test sockmap with connected sockets */
	fd = bpf_create_map(BPF_MAP_TYPE_SOCKMAP,
			    sizeof(key), sizeof(value),
			    6, 0);
	if (fd < 0) {
		printf("Failed to create sockmap %i\n", fd);
		goto out_sockmap;
	}

550
	/* Test update without programs */
551 552
	for (i = 0; i < 6; i++) {
		err = bpf_map_update_elem(fd, &i, &sfd[i], BPF_ANY);
553 554
		if (err) {
			printf("Failed noprog update sockmap '%i:%i'\n",
555 556 557 558 559 560
			       i, sfd[i]);
			goto out_sockmap;
		}
	}

	/* Test attaching bad fds */
561
	err = bpf_prog_attach(-1, fd, BPF_SK_SKB_STREAM_PARSER, 0);
562
	if (!err) {
563 564 565 566 567 568 569
		printf("Failed invalid parser prog attach\n");
		goto out_sockmap;
	}

	err = bpf_prog_attach(-1, fd, BPF_SK_SKB_STREAM_VERDICT, 0);
	if (!err) {
		printf("Failed invalid verdict prog attach\n");
570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599
		goto out_sockmap;
	}

	/* Load SK_SKB program and Attach */
	err = bpf_prog_load(SOCKMAP_PARSE_PROG,
			    BPF_PROG_TYPE_SK_SKB, &obj, &parse_prog);
	if (err) {
		printf("Failed to load SK_SKB parse prog\n");
		goto out_sockmap;
	}

	err = bpf_prog_load(SOCKMAP_VERDICT_PROG,
			    BPF_PROG_TYPE_SK_SKB, &obj, &verdict_prog);
	if (err) {
		printf("Failed to load SK_SKB verdict prog\n");
		goto out_sockmap;
	}

	bpf_map = bpf_object__find_map_by_name(obj, "sock_map");
	if (IS_ERR(bpf_map)) {
		printf("Failed to load map from verdict prog\n");
		goto out_sockmap;
	}

	map_fd = bpf_map__fd(bpf_map);
	if (map_fd < 0) {
		printf("Failed to get map fd\n");
		goto out_sockmap;
	}

600 601 602 603 604 605 606 607 608
	err = bpf_prog_attach(parse_prog, map_fd,
		      BPF_SK_SKB_STREAM_PARSER, 0);
	if (err) {
		printf("Failed bpf prog attach\n");
		goto out_sockmap;
	}

	err = bpf_prog_attach(verdict_prog, map_fd,
			      BPF_SK_SKB_STREAM_VERDICT, 0);
609 610 611 612 613
	if (err) {
		printf("Failed bpf prog attach\n");
		goto out_sockmap;
	}

614
	/* Test map update elem afterwards fd lives in fd and map_fd */
615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664
	for (i = 0; i < 6; i++) {
		err = bpf_map_update_elem(map_fd, &i, &sfd[i], BPF_ANY);
		if (err) {
			printf("Failed map_fd update sockmap %i '%i:%i'\n",
			       err, i, sfd[i]);
			goto out_sockmap;
		}
	}

	/* Test map delete elem and remove send/recv sockets */
	for (i = 2; i < 4; i++) {
		err = bpf_map_delete_elem(map_fd, &i);
		if (err) {
			printf("Failed delete  sockmap %i '%i:%i'\n",
			       err, i, sfd[i]);
			goto out_sockmap;
		}
	}

	/* Test map send/recv */
	sc = send(sfd[2], buf, 10, 0);
	if (sc < 0) {
		printf("Failed sockmap send\n");
		goto out_sockmap;
	}

	FD_ZERO(&w);
	FD_SET(sfd[3], &w);
	to.tv_sec = 1;
	to.tv_usec = 0;
	s = select(sfd[3] + 1, &w, NULL, NULL, &to);
	if (s == -1) {
		perror("Failed sockmap select()");
		goto out_sockmap;
	} else if (!s) {
		printf("Failed sockmap unexpected timeout\n");
		goto out_sockmap;
	}

	if (!FD_ISSET(sfd[3], &w)) {
		printf("Failed sockmap select/recv\n");
		goto out_sockmap;
	}

	rc = recv(sfd[3], buf, sizeof(buf), 0);
	if (rc < 0) {
		printf("Failed sockmap recv\n");
		goto out_sockmap;
	}

665 666
	/* Push fd into same slot */
	i = 2;
667 668
	err = bpf_map_update_elem(fd, &i, &sfd[i], BPF_NOEXIST);
	if (!err) {
669
		printf("Failed allowed sockmap dup slot BPF_NOEXIST\n");
670 671 672 673 674
		goto out_sockmap;
	}

	err = bpf_map_update_elem(fd, &i, &sfd[i], BPF_ANY);
	if (err) {
675
		printf("Failed sockmap update new slot BPF_ANY\n");
676 677 678 679 680
		goto out_sockmap;
	}

	err = bpf_map_update_elem(fd, &i, &sfd[i], BPF_EXIST);
	if (err) {
681
		printf("Failed sockmap update new slot BPF_EXIST\n");
682 683 684
		goto out_sockmap;
	}

685 686 687 688 689 690 691
	/* Delete the elems without programs */
	for (i = 0; i < 6; i++) {
		err = bpf_map_delete_elem(fd, &i);
		if (err) {
			printf("Failed delete sockmap %i '%i:%i'\n",
			       err, i, sfd[i]);
		}
692 693
	}

694 695 696
	/* Test having multiple maps open and set with programs on same fds */
	err = bpf_prog_attach(parse_prog, fd,
			      BPF_SK_SKB_STREAM_PARSER, 0);
697
	if (err) {
698
		printf("Failed fd bpf parse prog attach\n");
699 700
		goto out_sockmap;
	}
701 702
	err = bpf_prog_attach(verdict_prog, fd,
			      BPF_SK_SKB_STREAM_VERDICT, 0);
703
	if (err) {
704
		printf("Failed fd bpf verdict prog attach\n");
705 706 707
		goto out_sockmap;
	}

708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726
	for (i = 4; i < 6; i++) {
		err = bpf_map_update_elem(fd, &i, &sfd[i], BPF_ANY);
		if (!err) {
			printf("Failed allowed duplicate programs in update ANY sockmap %i '%i:%i'\n",
			       err, i, sfd[i]);
			goto out_sockmap;
		}
		err = bpf_map_update_elem(fd, &i, &sfd[i], BPF_NOEXIST);
		if (!err) {
			printf("Failed allowed duplicate program in update NOEXIST sockmap  %i '%i:%i'\n",
			       err, i, sfd[i]);
			goto out_sockmap;
		}
		err = bpf_map_update_elem(fd, &i, &sfd[i], BPF_EXIST);
		if (!err) {
			printf("Failed allowed duplicate program in update EXIST sockmap  %i '%i:%i'\n",
			       err, i, sfd[i]);
			goto out_sockmap;
		}
727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747
	}

	/* Test map close sockets */
	for (i = 0; i < 6; i++)
		close(sfd[i]);
	close(fd);
	close(map_fd);
	bpf_object__close(obj);
	return;
out:
	for (i = 0; i < 6; i++)
		close(sfd[i]);
	printf("Failed to create sockmap '%i:%s'!\n", i, strerror(errno));
	exit(1);
out_sockmap:
	for (i = 0; i < 6; i++)
		close(sfd[i]);
	close(fd);
	exit(1);
}

748 749 750 751 752 753 754 755 756 757 758
#define MAP_SIZE (32 * 1024)

static void test_map_large(void)
{
	struct bigkey {
		int a;
		char b[116];
		long long c;
	} key;
	int fd, i, value;

759
	fd = bpf_create_map(BPF_MAP_TYPE_HASH, sizeof(key), sizeof(value),
760 761 762 763 764 765 766 767 768 769
			    MAP_SIZE, map_flags);
	if (fd < 0) {
		printf("Failed to create large map '%s'!\n", strerror(errno));
		exit(1);
	}

	for (i = 0; i < MAP_SIZE; i++) {
		key = (struct bigkey) { .c = i };
		value = i;

770
		assert(bpf_map_update_elem(fd, &key, &value, BPF_NOEXIST) == 0);
771 772 773
	}

	key.c = -1;
774
	assert(bpf_map_update_elem(fd, &key, &value, BPF_NOEXIST) == -1 &&
775 776 777
	       errno == E2BIG);

	/* Iterate through all elements. */
778 779
	assert(bpf_map_get_next_key(fd, NULL, &key) == 0);
	key.c = -1;
780
	for (i = 0; i < MAP_SIZE; i++)
781 782
		assert(bpf_map_get_next_key(fd, &key, &key) == 0);
	assert(bpf_map_get_next_key(fd, &key, &key) == -1 && errno == ENOENT);
783 784

	key.c = 0;
785
	assert(bpf_map_lookup_elem(fd, &key, &value) == 0 && value == 0);
786
	key.a = 1;
787
	assert(bpf_map_lookup_elem(fd, &key, &value) == -1 && errno == ENOENT);
788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820

	close(fd);
}

static void run_parallel(int tasks, void (*fn)(int task, void *data),
			 void *data)
{
	pid_t pid[tasks];
	int i;

	for (i = 0; i < tasks; i++) {
		pid[i] = fork();
		if (pid[i] == 0) {
			fn(i, data);
			exit(0);
		} else if (pid[i] == -1) {
			printf("Couldn't spawn #%d process!\n", i);
			exit(1);
		}
	}

	for (i = 0; i < tasks; i++) {
		int status;

		assert(waitpid(pid[i], &status, 0) == pid[i]);
		assert(status == 0);
	}
}

static void test_map_stress(void)
{
	run_parallel(100, test_hashmap, NULL);
	run_parallel(100, test_hashmap_percpu, NULL);
821
	run_parallel(100, test_hashmap_sizes, NULL);
822
	run_parallel(100, test_hashmap_walk, NULL);
823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842

	run_parallel(100, test_arraymap, NULL);
	run_parallel(100, test_arraymap_percpu, NULL);
}

#define TASKS 1024

#define DO_UPDATE 1
#define DO_DELETE 0

static void do_work(int fn, void *data)
{
	int do_update = ((int *)data)[1];
	int fd = ((int *)data)[0];
	int i, key, value;

	for (i = fn; i < MAP_SIZE; i += TASKS) {
		key = value = i;

		if (do_update) {
843 844 845 846
			assert(bpf_map_update_elem(fd, &key, &value,
						   BPF_NOEXIST) == 0);
			assert(bpf_map_update_elem(fd, &key, &value,
						   BPF_EXIST) == 0);
847
		} else {
848
			assert(bpf_map_delete_elem(fd, &key) == 0);
849 850 851 852 853 854 855 856 857
		}
	}
}

static void test_map_parallel(void)
{
	int i, fd, key = 0, value = 0;
	int data[2];

858
	fd = bpf_create_map(BPF_MAP_TYPE_HASH, sizeof(key), sizeof(value),
859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875
			    MAP_SIZE, map_flags);
	if (fd < 0) {
		printf("Failed to create map for parallel test '%s'!\n",
		       strerror(errno));
		exit(1);
	}

	/* Use the same fd in children to add elements to this map:
	 * child_0 adds key=0, key=1024, key=2048, ...
	 * child_1 adds key=1, key=1025, key=2049, ...
	 * child_1023 adds key=1023, ...
	 */
	data[0] = fd;
	data[1] = DO_UPDATE;
	run_parallel(TASKS, do_work, data);

	/* Check that key=0 is already there. */
876
	assert(bpf_map_update_elem(fd, &key, &value, BPF_NOEXIST) == -1 &&
877 878 879
	       errno == EEXIST);

	/* Check that all elements were inserted. */
880
	assert(bpf_map_get_next_key(fd, NULL, &key) == 0);
881 882
	key = -1;
	for (i = 0; i < MAP_SIZE; i++)
883 884
		assert(bpf_map_get_next_key(fd, &key, &key) == 0);
	assert(bpf_map_get_next_key(fd, &key, &key) == -1 && errno == ENOENT);
885 886 887 888 889

	/* Another check for all elements */
	for (i = 0; i < MAP_SIZE; i++) {
		key = MAP_SIZE - i - 1;

890
		assert(bpf_map_lookup_elem(fd, &key, &value) == 0 &&
891 892 893 894 895 896 897 898 899
		       value == key);
	}

	/* Now let's delete all elemenets in parallel. */
	data[1] = DO_DELETE;
	run_parallel(TASKS, do_work, data);

	/* Nothing should be left. */
	key = -1;
900
	assert(bpf_map_get_next_key(fd, NULL, &key) == -1 && errno == ENOENT);
901
	assert(bpf_map_get_next_key(fd, &key, &key) == -1 && errno == ENOENT);
902 903 904 905 906 907
}

static void run_all_tests(void)
{
	test_hashmap(0, NULL);
	test_hashmap_percpu(0, NULL);
908
	test_hashmap_walk(0, NULL);
909 910 911 912 913 914

	test_arraymap(0, NULL);
	test_arraymap_percpu(0, NULL);

	test_arraymap_percpu_many_keys();

J
John Fastabend 已提交
915
	test_devmap(0, NULL);
916
	test_sockmap(0, NULL);
J
John Fastabend 已提交
917

918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937
	test_map_large();
	test_map_parallel();
	test_map_stress();
}

int main(void)
{
	struct rlimit rinf = { RLIM_INFINITY, RLIM_INFINITY };

	setrlimit(RLIMIT_MEMLOCK, &rinf);

	map_flags = 0;
	run_all_tests();

	map_flags = BPF_F_NO_PREALLOC;
	run_all_tests();

	printf("test_maps: OK\n");
	return 0;
}