DPDK 24.11.1
rte_seqcount.h
Go to the documentation of this file.
1/* SPDX-License-Identifier: BSD-3-Clause
2 * Copyright(c) 2022 Ericsson AB
3 */
4
5#ifndef _RTE_SEQCOUNT_H_
6#define _RTE_SEQCOUNT_H_
7
19#include <stdbool.h>
20#include <stdint.h>
21
22#include <rte_atomic.h>
24#include <rte_stdatomic.h>
25
26#ifdef __cplusplus
27extern "C" {
28#endif
29
33typedef struct {
34 RTE_ATOMIC(uint32_t) sn;
36
40#define RTE_SEQCOUNT_INITIALIZER { .sn = 0 }
41
48static inline void
50{
51 seqcount->sn = 0;
52}
53
96static inline uint32_t
98{
99 /* rte_memory_order_acquire to prevent loads after (in program order)
100 * from happening before the sn load. Synchronizes-with the
101 * store release in rte_seqcount_write_end().
102 */
103 return rte_atomic_load_explicit(&seqcount->sn, rte_memory_order_acquire);
104}
105
136static inline bool
137rte_seqcount_read_retry(const rte_seqcount_t *seqcount, uint32_t begin_sn)
138{
139 uint32_t end_sn;
140
141 /* An odd sequence number means the protected data was being
142 * modified already at the point of the rte_seqcount_read_begin()
143 * call.
144 */
145 if (unlikely(begin_sn & 1))
146 return true;
147
148 /* make sure the data loads happens before the sn load */
149 rte_atomic_thread_fence(rte_memory_order_acquire);
150
151 end_sn = rte_atomic_load_explicit(&seqcount->sn, rte_memory_order_relaxed);
152
153 /* A writer incremented the sequence number during this read
154 * critical section.
155 */
156 return begin_sn != end_sn;
157}
158
181static inline void
183{
184 uint32_t sn;
185
186 sn = seqcount->sn + 1;
187
188 rte_atomic_store_explicit(&seqcount->sn, sn, rte_memory_order_relaxed);
189
190 /* rte_memory_order_release to prevent stores after (in program order)
191 * from happening before the sn store.
192 */
193 rte_atomic_thread_fence(rte_memory_order_release);
194}
195
208static inline void
210{
211 uint32_t sn;
212
213 sn = seqcount->sn + 1;
214
215 /* Synchronizes-with the load acquire in rte_seqcount_read_begin(). */
216 rte_atomic_store_explicit(&seqcount->sn, sn, rte_memory_order_release);
217}
218
219#ifdef __cplusplus
220}
221#endif
222
223#endif /* _RTE_SEQCOUNT_H_ */
static void rte_atomic_thread_fence(rte_memory_order memorder)
#define unlikely(x)
static void rte_seqcount_init(rte_seqcount_t *seqcount)
Definition: rte_seqcount.h:49
static void rte_seqcount_write_end(rte_seqcount_t *seqcount)
Definition: rte_seqcount.h:209
static uint32_t rte_seqcount_read_begin(const rte_seqcount_t *seqcount)
Definition: rte_seqcount.h:97
static bool rte_seqcount_read_retry(const rte_seqcount_t *seqcount, uint32_t begin_sn)
Definition: rte_seqcount.h:137
static void rte_seqcount_write_begin(rte_seqcount_t *seqcount)
Definition: rte_seqcount.h:182