app/test: count tests skipped at setup
[dpdk.git] / app / test / test_seqlock.c
1 /* SPDX-License-Identifier: BSD-3-Clause
2  * Copyright(c) 2022 Ericsson AB
3  */
4
5 #include <rte_seqlock.h>
6
7 #include <rte_cycles.h>
8 #include <rte_malloc.h>
9 #include <rte_random.h>
10
11 #include <inttypes.h>
12
13 #include "test.h"
14
15 struct data {
16         rte_seqlock_t lock;
17
18         uint64_t a;
19         uint64_t b __rte_cache_aligned;
20         uint64_t c __rte_cache_aligned;
21 } __rte_cache_aligned;
22
23 struct reader {
24         struct data *data;
25         uint8_t stop;
26 };
27
28 #define WRITER_RUNTIME 2.0 /* s */
29
30 #define WRITER_MAX_DELAY 100 /* us */
31
32 #define INTERRUPTED_WRITER_FREQUENCY 1000
33 #define WRITER_INTERRUPT_TIME 1 /* us */
34
35 static int
36 writer_run(void *arg)
37 {
38         struct data *data = arg;
39         uint64_t deadline;
40
41         deadline = rte_get_timer_cycles() +
42                 WRITER_RUNTIME * rte_get_timer_hz();
43
44         while (rte_get_timer_cycles() < deadline) {
45                 bool interrupted;
46                 uint64_t new_value;
47                 unsigned int delay;
48
49                 new_value = rte_rand();
50
51                 interrupted = rte_rand_max(INTERRUPTED_WRITER_FREQUENCY) == 0;
52
53                 rte_seqlock_write_lock(&data->lock);
54
55                 data->c = new_value;
56                 data->b = new_value;
57
58                 if (interrupted)
59                         rte_delay_us_block(WRITER_INTERRUPT_TIME);
60
61                 data->a = new_value;
62
63                 rte_seqlock_write_unlock(&data->lock);
64
65                 delay = rte_rand_max(WRITER_MAX_DELAY);
66
67                 rte_delay_us_block(delay);
68         }
69
70         return TEST_SUCCESS;
71 }
72
73 #define INTERRUPTED_READER_FREQUENCY 1000
74 #define READER_INTERRUPT_TIME 1000 /* us */
75
76 static int
77 reader_run(void *arg)
78 {
79         struct reader *r = arg;
80         int rc = TEST_SUCCESS;
81
82         while (__atomic_load_n(&r->stop, __ATOMIC_RELAXED) == 0 &&
83                         rc == TEST_SUCCESS) {
84                 struct data *data = r->data;
85                 bool interrupted;
86                 uint32_t sn;
87                 uint64_t a;
88                 uint64_t b;
89                 uint64_t c;
90
91                 interrupted = rte_rand_max(INTERRUPTED_READER_FREQUENCY) == 0;
92
93                 do {
94                         sn = rte_seqlock_read_begin(&data->lock);
95
96                         a = data->a;
97                         if (interrupted)
98                                 rte_delay_us_block(READER_INTERRUPT_TIME);
99                         c = data->c;
100                         b = data->b;
101
102                 } while (rte_seqlock_read_retry(&data->lock, sn));
103
104                 if (a != b || b != c) {
105                         printf("Reader observed inconsistent data values "
106                                 "%" PRIu64 " %" PRIu64 " %" PRIu64 "\n",
107                                 a, b, c);
108                         rc = TEST_FAILED;
109                 }
110         }
111
112         return rc;
113 }
114
115 static void
116 reader_stop(struct reader *reader)
117 {
118         __atomic_store_n(&reader->stop, 1, __ATOMIC_RELAXED);
119 }
120
121 #define NUM_WRITERS 2 /* main lcore + one worker */
122 #define MIN_NUM_READERS 2
123 #define MIN_LCORE_COUNT (NUM_WRITERS + MIN_NUM_READERS)
124
125 /* Only a compile-time test */
126 static rte_seqlock_t __rte_unused static_init_lock = RTE_SEQLOCK_INITIALIZER;
127
128 static int
129 test_seqlock(void)
130 {
131         struct reader readers[RTE_MAX_LCORE];
132         unsigned int num_lcores;
133         unsigned int num_readers;
134         struct data *data;
135         unsigned int i;
136         unsigned int lcore_id;
137         unsigned int reader_lcore_ids[RTE_MAX_LCORE];
138         unsigned int worker_writer_lcore_id = 0;
139         int rc = TEST_SUCCESS;
140
141         num_lcores = rte_lcore_count();
142
143         if (num_lcores < MIN_LCORE_COUNT) {
144                 printf("Too few cores to run test. Skipping.\n");
145                 return TEST_SKIPPED;
146         }
147
148         num_readers = num_lcores - NUM_WRITERS;
149
150         data = rte_zmalloc(NULL, sizeof(struct data), 0);
151
152         if (data == NULL) {
153                 printf("Failed to allocate memory for seqlock data\n");
154                 return TEST_FAILED;
155         }
156
157         i = 0;
158         RTE_LCORE_FOREACH_WORKER(lcore_id) {
159                 if (i == 0) {
160                         rte_eal_remote_launch(writer_run, data, lcore_id);
161                         worker_writer_lcore_id = lcore_id;
162                 } else {
163                         unsigned int reader_idx = i - 1;
164                         struct reader *reader = &readers[reader_idx];
165
166                         reader->data = data;
167                         reader->stop = 0;
168
169                         rte_eal_remote_launch(reader_run, reader, lcore_id);
170                         reader_lcore_ids[reader_idx] = lcore_id;
171                 }
172                 i++;
173         }
174
175         if (writer_run(data) != 0 ||
176                         rte_eal_wait_lcore(worker_writer_lcore_id) != 0)
177                 rc = TEST_FAILED;
178
179         for (i = 0; i < num_readers; i++) {
180                 reader_stop(&readers[i]);
181                 if (rte_eal_wait_lcore(reader_lcore_ids[i]) != 0)
182                         rc = TEST_FAILED;
183         }
184
185         rte_free(data);
186
187         return rc;
188 }
189
190 REGISTER_TEST_COMMAND(seqlock_autotest, test_seqlock);