Zephyr Project API 4.4.99
A Scalable Open Source RTOS
Loading...
Searching...
No Matches
spinlock.h
Go to the documentation of this file.
1/*
2 * Copyright (c) 2018 Intel Corporation.
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
11
12#ifndef ZEPHYR_INCLUDE_SPINLOCK_H_
13#define ZEPHYR_INCLUDE_SPINLOCK_H_
14
15#include <errno.h>
16#include <stdbool.h>
17
18#include <zephyr/arch/cpu.h>
19#include <zephyr/sys/atomic.h>
20#include <zephyr/sys/__assert.h>
22
23#ifdef __cplusplus
24extern "C" {
25#endif
26
33
34struct z_spinlock_key {
35 int key;
36};
37
45struct k_spinlock {
49#ifdef CONFIG_SMP
50#ifdef CONFIG_TICKET_SPINLOCKS
51 /*
52 * Ticket spinlocks are conceptually two atomic variables,
53 * one indicating the current FIFO head (spinlock owner),
54 * and the other indicating the current FIFO tail.
55 * Spinlock is acquired in the following manner:
56 * - current FIFO tail value is atomically incremented while it's
57 * original value is saved as a "ticket"
58 * - we spin until the FIFO head becomes equal to the ticket value
59 *
60 * Spinlock is released by atomic increment of the FIFO head
61 */
62 atomic_t owner;
63 atomic_t tail;
64#else
65 atomic_t locked;
66#endif /* CONFIG_TICKET_SPINLOCKS */
67#endif /* CONFIG_SMP */
68
69#ifdef CONFIG_SPIN_VALIDATE
70 /* Stores the thread that holds the lock with the locking CPU
71 * ID in the bottom bits (2 bits on 32-bit, 3 bits on 64-bit).
72 */
73 uintptr_t thread_cpu;
74#ifdef CONFIG_SPIN_LOCK_TIME_LIMIT
75 /* Stores the time (in cycles) when a lock was taken
76 */
77 uint32_t lock_time;
78#endif /* CONFIG_SPIN_LOCK_TIME_LIMIT */
79#endif /* CONFIG_SPIN_VALIDATE */
80
81#if defined(CONFIG_NONZERO_SPINLOCK_SIZE) && !defined(CONFIG_SMP) && !defined(CONFIG_SPIN_VALIDATE)
82 /* Add a dummy field to guarantee the spinlock has a non-zero
83 * size. If neither CONFIG_SMP nor CONFIG_SPIN_VALIDATE are
84 * defined then the k_spinlock struct would otherwise have no
85 * members and sizeof(k_spinlock) would be 0 in C and 1 in C++.
86 *
87 * That size difference causes problems when the k_spinlock
88 * is embedded into another struct like k_msgq, because C and
89 * C++ will have different ideas on the offsets of the members
90 * that come after the k_spinlock member.
91 */
92 char dummy;
93#endif
97};
98
99/* There's a spinlock validation framework available when asserts are
100 * enabled. It adds a relatively hefty overhead (about 3k or so) to
101 * kernel code size, don't use on platforms known to be small.
102 */
103#ifdef CONFIG_SPIN_VALIDATE
104bool z_spin_lock_valid(struct k_spinlock *l);
105bool z_spin_unlock_valid(struct k_spinlock *l);
106void z_spin_lock_set_owner(struct k_spinlock *l);
107#if defined(CONFIG_64BIT)
108BUILD_ASSERT(CONFIG_MP_MAX_NUM_CPUS <= 8, "Too many CPUs for mask");
109#else
110BUILD_ASSERT(CONFIG_MP_MAX_NUM_CPUS <= 4, "Too many CPUs for mask");
111#endif
112
113# ifdef CONFIG_KERNEL_COHERENCE
114bool z_spin_lock_mem_coherent(struct k_spinlock *l);
115# endif /* CONFIG_KERNEL_COHERENCE */
116
117#endif /* CONFIG_SPIN_VALIDATE */
118
130typedef struct z_spinlock_key k_spinlock_key_t;
131
132static ALWAYS_INLINE void z_spinlock_validate_pre(struct k_spinlock *l)
133{
134 ARG_UNUSED(l);
135#ifdef CONFIG_SPIN_VALIDATE
136 __ASSERT(z_spin_lock_valid(l), "Invalid spinlock %p", l);
137#ifdef CONFIG_KERNEL_COHERENCE
138 __ASSERT_NO_MSG(z_spin_lock_mem_coherent(l));
139#endif
140#endif
141}
142
143static ALWAYS_INLINE void z_spinlock_validate_post(struct k_spinlock *l)
144{
145 ARG_UNUSED(l);
146#ifdef CONFIG_SPIN_VALIDATE
147 z_spin_lock_set_owner(l);
148#if defined(CONFIG_SPIN_LOCK_TIME_LIMIT) && (CONFIG_SPIN_LOCK_TIME_LIMIT != 0)
149 l->lock_time = sys_clock_cycle_get_32();
150#endif /* CONFIG_SPIN_LOCK_TIME_LIMIT */
151#endif /* CONFIG_SPIN_VALIDATE */
152}
153
186{
187 ARG_UNUSED(l);
189
190 /* Note that we need to use the underlying arch-specific lock
191 * implementation. The "irq_lock()" API in SMP context is
192 * actually a wrapper for a global spinlock!
193 */
194 k.key = arch_irq_lock();
195
196 z_spinlock_validate_pre(l);
197#ifdef CONFIG_SMP
198#ifdef CONFIG_TICKET_SPINLOCKS
199 /*
200 * Enqueue ourselves to the end of a spinlock waiters queue
201 * receiving a ticket
202 */
203 atomic_val_t ticket = atomic_inc(&l->tail);
204 /* Spin until our ticket is served */
205 while (atomic_get(&l->owner) != ticket) {
207 }
208#else
209 while (!atomic_cas(&l->locked, 0, 1)) {
210 do {
212 } while (atomic_get(&l->locked) != 0);
213 }
214#endif /* CONFIG_TICKET_SPINLOCKS */
215#endif /* CONFIG_SMP */
216 z_spinlock_validate_post(l);
217
218 return k;
219}
220
236{
237 int key = arch_irq_lock();
238
239 z_spinlock_validate_pre(l);
240#ifdef CONFIG_SMP
241#ifdef CONFIG_TICKET_SPINLOCKS
242 /*
243 * atomic_get and atomic_cas operations below are not executed
244 * simultaneously.
245 * So in theory k_spin_trylock can lock an already locked spinlock.
246 * To reproduce this the following conditions should be met after we
247 * executed atomic_get and before we executed atomic_cas:
248 *
249 * - spinlock needs to be taken 0xffff_..._ffff + 1 times
250 * (which requires 0xffff_..._ffff number of CPUs, as k_spin_lock call
251 * is blocking) or
252 * - spinlock needs to be taken and released 0xffff_..._ffff times and
253 * then taken again
254 *
255 * In real-life systems this is considered non-reproducible given that
256 * required actions need to be done during this tiny window of several
257 * CPU instructions (which execute with interrupt locked,
258 * so no preemption can happen here)
259 */
260 atomic_val_t ticket_val = atomic_get(&l->owner);
261
262 if (!atomic_cas(&l->tail, ticket_val, ticket_val + 1)) {
263 goto busy;
264 }
265#else
266 if (!atomic_cas(&l->locked, 0, 1)) {
267 goto busy;
268 }
269#endif /* CONFIG_TICKET_SPINLOCKS */
270#endif /* CONFIG_SMP */
271 z_spinlock_validate_post(l);
272
273 k->key = key;
274
275 return 0;
276
277#ifdef CONFIG_SMP
278busy:
279 arch_irq_unlock(key);
280 return -EBUSY;
281#endif /* CONFIG_SMP */
282}
283
307{
308 ARG_UNUSED(l);
309#ifdef CONFIG_SPIN_VALIDATE
310 __ASSERT(z_spin_unlock_valid(l), "Not my spinlock %p", l);
311
312#if defined(CONFIG_SPIN_LOCK_TIME_LIMIT) && (CONFIG_SPIN_LOCK_TIME_LIMIT != 0)
313 uint32_t delta = sys_clock_cycle_get_32() - l->lock_time;
314
315 __ASSERT(delta < CONFIG_SPIN_LOCK_TIME_LIMIT,
316 "Spin lock %p held %u cycles, longer than limit of %u cycles",
317 l, delta, CONFIG_SPIN_LOCK_TIME_LIMIT);
318#endif /* CONFIG_SPIN_LOCK_TIME_LIMIT */
319#endif /* CONFIG_SPIN_VALIDATE */
320
321#ifdef CONFIG_SMP
322#ifdef CONFIG_TICKET_SPINLOCKS
323 /* Give the spinlock to the next CPU in a FIFO */
324 (void)atomic_inc(&l->owner);
325#else
326 /* Strictly we don't need atomic_clear() here (which is an
327 * exchange operation that returns the old value). We are always
328 * setting a zero and (because we hold the lock) know the existing
329 * state won't change due to a race. But some architectures need
330 * a memory barrier when used like this, and we don't have a
331 * Zephyr framework for that.
332 */
333 (void)atomic_clear(&l->locked);
334#endif /* CONFIG_TICKET_SPINLOCKS */
335#endif /* CONFIG_SMP */
336 arch_irq_unlock(key.key);
337}
338
342
343#if defined(CONFIG_TEST) || defined(CONFIG_ASSERT)
344/*
345 * @brief Checks if spinlock is held by some CPU, including the local CPU.
346 * This should only be used in tests or assertions, not to make
347 * runtime control flow decisions.
348 *
349 * @param l A pointer to the spinlock
350 * @retval true - if spinlock is held by some CPU; false - otherwise
351 */
352static ALWAYS_INLINE bool z_spin_is_locked(struct k_spinlock *l)
353{
354#ifdef CONFIG_SMP
355#ifdef CONFIG_TICKET_SPINLOCKS
356 atomic_val_t ticket_val = atomic_get(&l->owner);
357
358 return !atomic_cas(&l->tail, ticket_val, ticket_val);
359#else
360 return l->locked;
361#endif /* CONFIG_TICKET_SPINLOCKS */
362#else
363 ARG_UNUSED(l);
364 /* In UP builds a spinlock reduces to an IRQ lock. */
366#endif /* CONFIG_SMP */
367}
368#endif /* defined(CONFIG_TEST) || defined(CONFIG_ASSERT) */
369
370/* Internal function: releases the lock, but leaves local interrupts disabled */
371static ALWAYS_INLINE void k_spin_release(struct k_spinlock *l)
372{
373 ARG_UNUSED(l);
374#ifdef CONFIG_SPIN_VALIDATE
375 __ASSERT(z_spin_unlock_valid(l), "Not my spinlock %p", l);
376#endif
377#ifdef CONFIG_SMP
378#ifdef CONFIG_TICKET_SPINLOCKS
379 (void)atomic_inc(&l->owner);
380#else
381 (void)atomic_clear(&l->locked);
382#endif /* CONFIG_TICKET_SPINLOCKS */
383#endif /* CONFIG_SMP */
384}
385
386#if defined(CONFIG_SPIN_VALIDATE) && defined(__GNUC__)
387static ALWAYS_INLINE void z_spin_onexit(__maybe_unused k_spinlock_key_t *k)
388{
389 __ASSERT(k->key, "K_SPINLOCK exited with goto, break or return, "
390 "use K_SPINLOCK_BREAK instead.");
391}
392#define K_SPINLOCK_ONEXIT __attribute__((__cleanup__(z_spin_onexit)))
393#else
394#define K_SPINLOCK_ONEXIT
395#endif
396
400
407#define K_SPINLOCK_BREAK continue
408
450#define K_SPINLOCK(lck) \
451 for (k_spinlock_key_t __i K_SPINLOCK_ONEXIT = {}, __key = k_spin_lock(lck); !__i.key; \
452 k_spin_unlock((lck), __key), __i.key = 1)
453
455
456#ifdef __cplusplus
457}
458#endif
459
460#endif /* ZEPHYR_INCLUDE_SPINLOCK_H_ */
uint32_t sys_clock_cycle_get_32(void)
static ALWAYS_INLINE unsigned int arch_irq_lock(void)
Disable all interrupts on the local CPU.
Definition irq.h:168
static ALWAYS_INLINE void arch_irq_unlock(unsigned int key)
Definition irq.h:176
static ALWAYS_INLINE bool arch_cpu_irqs_are_enabled(void)
Implementation of arch_cpu_irqs_are_enabled.
Definition irq.h:191
void arch_spin_relax(void)
Perform architecture specific processing within spin loops.
long atomic_t
Definition atomic_types.h:15
atomic_t atomic_val_t
Definition atomic_types.h:16
System error numbers.
atomic_val_t atomic_get(const atomic_t *target)
Atomic get.
atomic_val_t atomic_clear(atomic_t *target)
Atomic clear.
atomic_val_t atomic_inc(atomic_t *target)
Atomic increment.
bool atomic_cas(atomic_t *target, atomic_val_t old_value, atomic_val_t new_value)
Atomic compare-and-set.
static ALWAYS_INLINE int k_spin_trylock(struct k_spinlock *l, k_spinlock_key_t *k)
Attempt to lock a spinlock.
Definition spinlock.h:235
static ALWAYS_INLINE void k_spin_unlock(struct k_spinlock *l, k_spinlock_key_t key)
Unlock a spin lock.
Definition spinlock.h:305
static ALWAYS_INLINE k_spinlock_key_t k_spin_lock(struct k_spinlock *l)
Lock a spinlock.
Definition spinlock.h:185
struct z_spinlock_key k_spinlock_key_t
Spinlock key type.
Definition spinlock.h:130
#define EBUSY
Mount device busy.
Definition errno.h:54
#define ALWAYS_INLINE
Definition common.h:161
#define BUILD_ASSERT(EXPR, MSG...)
Definition llvm.h:51
__UINT32_TYPE__ uint32_t
Definition stdint.h:90
__UINTPTR_TYPE__ uintptr_t
Definition stdint.h:105
Kernel Spin Lock.
Definition spinlock.h:45