f-stack/dpdk/app/test/test_seqlock.c

191 lines
3.7 KiB
C

/* SPDX-License-Identifier: BSD-3-Clause
* Copyright(c) 2022 Ericsson AB
*/
#include <rte_seqlock.h>
#include <rte_cycles.h>
#include <rte_malloc.h>
#include <rte_random.h>
#include <inttypes.h>
#include "test.h"
struct data {
rte_seqlock_t lock;
uint64_t a;
uint64_t b __rte_cache_aligned;
uint64_t c __rte_cache_aligned;
} __rte_cache_aligned;
struct reader {
struct data *data;
uint8_t stop;
};
#define WRITER_RUNTIME 2.0 /* s */
#define WRITER_MAX_DELAY 100 /* us */
#define INTERRUPTED_WRITER_FREQUENCY 1000
#define WRITER_INTERRUPT_TIME 1 /* us */
static int
writer_run(void *arg)
{
struct data *data = arg;
uint64_t deadline;
deadline = rte_get_timer_cycles() +
WRITER_RUNTIME * rte_get_timer_hz();
while (rte_get_timer_cycles() < deadline) {
bool interrupted;
uint64_t new_value;
unsigned int delay;
new_value = rte_rand();
interrupted = rte_rand_max(INTERRUPTED_WRITER_FREQUENCY) == 0;
rte_seqlock_write_lock(&data->lock);
data->c = new_value;
data->b = new_value;
if (interrupted)
rte_delay_us_block(WRITER_INTERRUPT_TIME);
data->a = new_value;
rte_seqlock_write_unlock(&data->lock);
delay = rte_rand_max(WRITER_MAX_DELAY);
rte_delay_us_block(delay);
}
return TEST_SUCCESS;
}
#define INTERRUPTED_READER_FREQUENCY 1000
#define READER_INTERRUPT_TIME 1000 /* us */
static int
reader_run(void *arg)
{
struct reader *r = arg;
int rc = TEST_SUCCESS;
while (__atomic_load_n(&r->stop, __ATOMIC_RELAXED) == 0 &&
rc == TEST_SUCCESS) {
struct data *data = r->data;
bool interrupted;
uint32_t sn;
uint64_t a;
uint64_t b;
uint64_t c;
interrupted = rte_rand_max(INTERRUPTED_READER_FREQUENCY) == 0;
do {
sn = rte_seqlock_read_begin(&data->lock);
a = data->a;
if (interrupted)
rte_delay_us_block(READER_INTERRUPT_TIME);
c = data->c;
b = data->b;
} while (rte_seqlock_read_retry(&data->lock, sn));
if (a != b || b != c) {
printf("Reader observed inconsistent data values "
"%" PRIu64 " %" PRIu64 " %" PRIu64 "\n",
a, b, c);
rc = TEST_FAILED;
}
}
return rc;
}
static void
reader_stop(struct reader *reader)
{
__atomic_store_n(&reader->stop, 1, __ATOMIC_RELAXED);
}
#define NUM_WRITERS 2 /* main lcore + one worker */
#define MIN_NUM_READERS 2
#define MIN_LCORE_COUNT (NUM_WRITERS + MIN_NUM_READERS)
/* Only a compile-time test */
static rte_seqlock_t __rte_unused static_init_lock = RTE_SEQLOCK_INITIALIZER;
static int
test_seqlock(void)
{
struct reader readers[RTE_MAX_LCORE];
unsigned int num_lcores;
unsigned int num_readers;
struct data *data;
unsigned int i;
unsigned int lcore_id;
unsigned int reader_lcore_ids[RTE_MAX_LCORE];
unsigned int worker_writer_lcore_id = 0;
int rc = TEST_SUCCESS;
num_lcores = rte_lcore_count();
if (num_lcores < MIN_LCORE_COUNT) {
printf("Too few cores to run test. Skipping.\n");
return TEST_SKIPPED;
}
num_readers = num_lcores - NUM_WRITERS;
data = rte_zmalloc(NULL, sizeof(struct data), 0);
if (data == NULL) {
printf("Failed to allocate memory for seqlock data\n");
return TEST_FAILED;
}
i = 0;
RTE_LCORE_FOREACH_WORKER(lcore_id) {
if (i == 0) {
rte_eal_remote_launch(writer_run, data, lcore_id);
worker_writer_lcore_id = lcore_id;
} else {
unsigned int reader_idx = i - 1;
struct reader *reader = &readers[reader_idx];
reader->data = data;
reader->stop = 0;
rte_eal_remote_launch(reader_run, reader, lcore_id);
reader_lcore_ids[reader_idx] = lcore_id;
}
i++;
}
if (writer_run(data) != 0 ||
rte_eal_wait_lcore(worker_writer_lcore_id) != 0)
rc = TEST_FAILED;
for (i = 0; i < num_readers; i++) {
reader_stop(&readers[i]);
if (rte_eal_wait_lcore(reader_lcore_ids[i]) != 0)
rc = TEST_FAILED;
}
rte_free(data);
return rc;
}
REGISTER_TEST_COMMAND(seqlock_autotest, test_seqlock);