// 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);
}
Messung V0.5 in Prozent C=91 H=89 G=89
¤ Dauer der Verarbeitung: 0.9 Sekunden
(vorverarbeitet am 2026-06-06)
¤
*© Formatika GbR, Deutschland