DPDK  22.03.0
rte_mcslock.h
Go to the documentation of this file.
1 /* SPDX-License-Identifier: BSD-3-Clause
2  * Copyright(c) 2019 Arm Limited
3  */
4 
5 #ifndef _RTE_MCSLOCK_H_
6 #define _RTE_MCSLOCK_H_
7 
22 #include <rte_lcore.h>
23 #include <rte_common.h>
24 #include <rte_pause.h>
25 #include <rte_branch_prediction.h>
26 
30 typedef struct rte_mcslock {
31  struct rte_mcslock *next;
32  int locked; /* 1 if the queue locked, 0 otherwise */
34 
46 static inline void
48 {
49  rte_mcslock_t *prev;
50 
51  /* Init me node */
52  __atomic_store_n(&me->locked, 1, __ATOMIC_RELAXED);
53  __atomic_store_n(&me->next, NULL, __ATOMIC_RELAXED);
54 
55  /* If the queue is empty, the exchange operation is enough to acquire
56  * the lock. Hence, the exchange operation requires acquire semantics.
57  * The store to me->next above should complete before the node is
58  * visible to other CPUs/threads. Hence, the exchange operation requires
59  * release semantics as well.
60  */
61  prev = __atomic_exchange_n(msl, me, __ATOMIC_ACQ_REL);
62  if (likely(prev == NULL)) {
63  /* Queue was empty, no further action required,
64  * proceed with lock taken.
65  */
66  return;
67  }
68  /* The store to me->next above should also complete before the node is
69  * visible to predecessor thread releasing the lock. Hence, the store
70  * prev->next also requires release semantics. Note that, for example,
71  * on ARM, the release semantics in the exchange operation is not
72  * strong as a release fence and is not sufficient to enforce the
73  * desired order here.
74  */
75  __atomic_store_n(&prev->next, me, __ATOMIC_RELEASE);
76 
77  /* The while-load of me->locked should not move above the previous
78  * store to prev->next. Otherwise it will cause a deadlock. Need a
79  * store-load barrier.
80  */
81  __atomic_thread_fence(__ATOMIC_ACQ_REL);
82  /* If the lock has already been acquired, it first atomically
83  * places the node at the end of the queue and then proceeds
84  * to spin on me->locked until the previous lock holder resets
85  * the me->locked using mcslock_unlock().
86  */
87  rte_wait_until_equal_32((uint32_t *)&me->locked, 0, __ATOMIC_ACQUIRE);
88 }
89 
98 static inline void
100 {
101  /* Check if there are more nodes in the queue. */
102  if (likely(__atomic_load_n(&me->next, __ATOMIC_RELAXED) == NULL)) {
103  /* No, last member in the queue. */
104  rte_mcslock_t *save_me = __atomic_load_n(&me, __ATOMIC_RELAXED);
105 
106  /* Release the lock by setting it to NULL */
107  if (likely(__atomic_compare_exchange_n(msl, &save_me, NULL, 0,
108  __ATOMIC_RELEASE, __ATOMIC_RELAXED)))
109  return;
110 
111  /* Speculative execution would be allowed to read in the
112  * while-loop first. This has the potential to cause a
113  * deadlock. Need a load barrier.
114  */
115  __atomic_thread_fence(__ATOMIC_ACQUIRE);
116  /* More nodes added to the queue by other CPUs.
117  * Wait until the next pointer is set.
118  */
119  uintptr_t *next;
120  next = (uintptr_t *)&me->next;
121  RTE_WAIT_UNTIL_MASKED(next, UINTPTR_MAX, !=, 0,
122  __ATOMIC_RELAXED);
123  }
124 
125  /* Pass lock to next waiter. */
126  __atomic_store_n(&me->next->locked, 0, __ATOMIC_RELEASE);
127 }
128 
139 static inline int
141 {
142  /* Init me node */
143  __atomic_store_n(&me->next, NULL, __ATOMIC_RELAXED);
144 
145  /* Try to lock */
146  rte_mcslock_t *expected = NULL;
147 
148  /* The lock can be taken only when the queue is empty. Hence,
149  * the compare-exchange operation requires acquire semantics.
150  * The store to me->next above should complete before the node
151  * is visible to other CPUs/threads. Hence, the compare-exchange
152  * operation requires release semantics as well.
153  */
154  return __atomic_compare_exchange_n(msl, &expected, me, 0,
155  __ATOMIC_ACQ_REL, __ATOMIC_RELAXED);
156 }
157 
166 static inline int
168 {
169  return (__atomic_load_n(&msl, __ATOMIC_RELAXED) != NULL);
170 }
171 
172 #endif /* _RTE_MCSLOCK_H_ */
static void rte_mcslock_lock(rte_mcslock_t **msl, rte_mcslock_t *me)
Definition: rte_mcslock.h:47
#define likely(x)
struct rte_mcslock rte_mcslock_t
static int rte_mcslock_trylock(rte_mcslock_t **msl, rte_mcslock_t *me)
Definition: rte_mcslock.h:140
static __rte_always_inline void rte_wait_until_equal_32(volatile uint32_t *addr, uint32_t expected, int memorder)
Definition: rte_pause.h:96
static int rte_mcslock_is_locked(rte_mcslock_t *msl)
Definition: rte_mcslock.h:167
static void rte_mcslock_unlock(rte_mcslock_t **msl, rte_mcslock_t *me)
Definition: rte_mcslock.h:99