| // SPDX-License-Identifier: GPL-2.0 |
| /* Copyright (C) 2025. Huawei Technologies Co., Ltd */ |
| #define _GNU_SOURCE |
| #include <stdbool.h> |
| #include <test_progs.h> |
| #include "fd_htab_lookup.skel.h" |
| |
| struct htab_op_ctx { |
| int fd; |
| int loop; |
| unsigned int entries; |
| bool stop; |
| }; |
| |
| #define ERR_TO_RETVAL(where, err) ((void *)(long)(((where) << 12) | (-err))) |
| |
| static void *htab_lookup_fn(void *arg) |
| { |
| struct htab_op_ctx *ctx = arg; |
| int i = 0; |
| |
| while (i++ < ctx->loop && !ctx->stop) { |
| unsigned int j; |
| |
| for (j = 0; j < ctx->entries; j++) { |
| unsigned int key = j, zero = 0, value; |
| int inner_fd, err; |
| |
| err = bpf_map_lookup_elem(ctx->fd, &key, &value); |
| if (err) { |
| ctx->stop = true; |
| return ERR_TO_RETVAL(1, err); |
| } |
| |
| inner_fd = bpf_map_get_fd_by_id(value); |
| if (inner_fd < 0) { |
| /* The old map has been freed */ |
| if (inner_fd == -ENOENT) |
| continue; |
| ctx->stop = true; |
| return ERR_TO_RETVAL(2, inner_fd); |
| } |
| |
| err = bpf_map_lookup_elem(inner_fd, &zero, &value); |
| if (err) { |
| close(inner_fd); |
| ctx->stop = true; |
| return ERR_TO_RETVAL(3, err); |
| } |
| close(inner_fd); |
| |
| if (value != key) { |
| ctx->stop = true; |
| return ERR_TO_RETVAL(4, -EINVAL); |
| } |
| } |
| } |
| |
| return NULL; |
| } |
| |
| static void *htab_update_fn(void *arg) |
| { |
| struct htab_op_ctx *ctx = arg; |
| int i = 0; |
| |
| while (i++ < ctx->loop && !ctx->stop) { |
| unsigned int j; |
| |
| for (j = 0; j < ctx->entries; j++) { |
| unsigned int key = j, zero = 0; |
| int inner_fd, err; |
| |
| inner_fd = bpf_map_create(BPF_MAP_TYPE_ARRAY, NULL, 4, 4, 1, NULL); |
| if (inner_fd < 0) { |
| ctx->stop = true; |
| return ERR_TO_RETVAL(1, inner_fd); |
| } |
| |
| err = bpf_map_update_elem(inner_fd, &zero, &key, 0); |
| if (err) { |
| close(inner_fd); |
| ctx->stop = true; |
| return ERR_TO_RETVAL(2, err); |
| } |
| |
| err = bpf_map_update_elem(ctx->fd, &key, &inner_fd, BPF_EXIST); |
| if (err) { |
| close(inner_fd); |
| ctx->stop = true; |
| return ERR_TO_RETVAL(3, err); |
| } |
| close(inner_fd); |
| } |
| } |
| |
| return NULL; |
| } |
| |
| static int setup_htab(int fd, unsigned int entries) |
| { |
| unsigned int i; |
| |
| for (i = 0; i < entries; i++) { |
| unsigned int key = i, zero = 0; |
| int inner_fd, err; |
| |
| inner_fd = bpf_map_create(BPF_MAP_TYPE_ARRAY, NULL, 4, 4, 1, NULL); |
| if (!ASSERT_OK_FD(inner_fd, "new array")) |
| return -1; |
| |
| err = bpf_map_update_elem(inner_fd, &zero, &key, 0); |
| if (!ASSERT_OK(err, "init array")) { |
| close(inner_fd); |
| return -1; |
| } |
| |
| err = bpf_map_update_elem(fd, &key, &inner_fd, 0); |
| if (!ASSERT_OK(err, "init outer")) { |
| close(inner_fd); |
| return -1; |
| } |
| close(inner_fd); |
| } |
| |
| return 0; |
| } |
| |
| static int get_int_from_env(const char *name, int dft) |
| { |
| const char *value; |
| |
| value = getenv(name); |
| if (!value) |
| return dft; |
| |
| return atoi(value); |
| } |
| |
| void test_fd_htab_lookup(void) |
| { |
| unsigned int i, wr_nr = 8, rd_nr = 16; |
| pthread_t tids[wr_nr + rd_nr]; |
| struct fd_htab_lookup *skel; |
| struct htab_op_ctx ctx; |
| int err; |
| |
| skel = fd_htab_lookup__open_and_load(); |
| if (!ASSERT_OK_PTR(skel, "fd_htab_lookup__open_and_load")) |
| return; |
| |
| ctx.fd = bpf_map__fd(skel->maps.outer_map); |
| ctx.loop = get_int_from_env("FD_HTAB_LOOP_NR", 5); |
| ctx.stop = false; |
| ctx.entries = 8; |
| |
| err = setup_htab(ctx.fd, ctx.entries); |
| if (err) |
| goto destroy; |
| |
| memset(tids, 0, sizeof(tids)); |
| for (i = 0; i < wr_nr; i++) { |
| err = pthread_create(&tids[i], NULL, htab_update_fn, &ctx); |
| if (!ASSERT_OK(err, "pthread_create")) { |
| ctx.stop = true; |
| goto reap; |
| } |
| } |
| for (i = 0; i < rd_nr; i++) { |
| err = pthread_create(&tids[i + wr_nr], NULL, htab_lookup_fn, &ctx); |
| if (!ASSERT_OK(err, "pthread_create")) { |
| ctx.stop = true; |
| goto reap; |
| } |
| } |
| |
| reap: |
| for (i = 0; i < wr_nr + rd_nr; i++) { |
| void *ret = NULL; |
| char desc[32]; |
| |
| if (!tids[i]) |
| continue; |
| |
| snprintf(desc, sizeof(desc), "thread %u", i + 1); |
| err = pthread_join(tids[i], &ret); |
| ASSERT_OK(err, desc); |
| ASSERT_EQ(ret, NULL, desc); |
| } |
| destroy: |
| fd_htab_lookup__destroy(skel); |
| } |