7#ifndef _HARDWARE_SYNC_SPIN_LOCK_H
8#define _HARDWARE_SYNC_SPIN_LOCK_H
14#ifndef PICO_USE_SW_SPIN_LOCKS
16#define PICO_USE_SW_SPIN_LOCKS 1
21#ifndef PICO_SPINLOCK_ID_IRQ
22#define PICO_SPINLOCK_ID_IRQ 9
26#ifndef PICO_SPINLOCK_ID_TIMER
27#define PICO_SPINLOCK_ID_TIMER 10
31#ifndef PICO_SPINLOCK_ID_HARDWARE_CLAIM
32#define PICO_SPINLOCK_ID_HARDWARE_CLAIM 11
36#ifndef PICO_SPINLOCK_ID_RAND
37#define PICO_SPINLOCK_ID_RAND 12
41#ifndef PICO_SPINLOCK_ID_ATOMIC
42#define PICO_SPINLOCK_ID_ATOMIC 13
46#ifndef PICO_SPINLOCK_ID_OS1
47#define PICO_SPINLOCK_ID_OS1 14
51#ifndef PICO_SPINLOCK_ID_OS2
52#define PICO_SPINLOCK_ID_OS2 15
56#ifndef PICO_SPINLOCK_ID_STRIPED_FIRST
57#define PICO_SPINLOCK_ID_STRIPED_FIRST 16
61#ifndef PICO_SPINLOCK_ID_STRIPED_LAST
62#define PICO_SPINLOCK_ID_STRIPED_LAST 23
66#ifndef PICO_SPINLOCK_ID_CLAIM_FREE_FIRST
67#define PICO_SPINLOCK_ID_CLAIM_FREE_FIRST 24
70#ifdef PICO_SPINLOCK_ID_CLAIM_FREE_END
71#warning PICO_SPINLOCK_ID_CLAIM_FREE_END has been renamed to PICO_SPINLOCK_ID_CLAIM_FREE_LAST
75#ifndef PICO_SPINLOCK_ID_CLAIM_FREE_LAST
76#define PICO_SPINLOCK_ID_CLAIM_FREE_LAST 31
82#if !PICO_USE_SW_SPIN_LOCKS
84typedef io_rw_32 spin_lock_t;
86#ifndef SW_SPIN_LOCK_TYPE
88#define SW_SPIN_LOCK_TYPE volatile uint8_t
93#if PICO_USE_SW_SPIN_LOCKS
94#ifndef SW_SPIN_LOCK_INSTANCE
95#define SW_SPIN_LOCK_INSTANCE(lock_num) ({ \
96 extern spin_lock_t _sw_spin_locks[NUM_SPIN_LOCKS]; \
97 &_sw_spin_locks[lock_num]; \
101#ifndef SW_SPIN_LOCK_NUM
102#define SW_SPIN_LOCK_NUM(lock) ({ \
103 extern spin_lock_t _sw_spin_locks[NUM_SPIN_LOCKS]; \
104 (lock) - _sw_spin_locks; \
108#ifndef SW_SPIN_LOCK_IS_LOCKED
109#define SW_SPIN_LOCK_IS_LOCKED(lock) ((bool) *(lock))
112#ifndef SW_SPIN_LOCK_LOCK
113#if __ARM_ARCH_8M_MAIN__
114#define SW_SPIN_LOCK_LOCK(lock) ({ \
115 uint32_t _tmp0, _tmp1; \
116 pico_default_asm_volatile ( \
118 "ldaexb %1, [%2]\n" \
124 "strexb %1, %0, [%2]\n" \
128 : "=&r" (_tmp0), "=&r" (_tmp1) : "r" (lock) \
130 __mem_fence_acquire(); \
132#elif __riscv && (defined(__riscv_a) || defined(__riscv_zaamo))
133#define SW_SPIN_LOCK_LOCK(lock) ({ \
134 uint32_t _tmp0, _tmp1; \
135 pico_default_asm_volatile ( \
139 "bset %1, zero, %1\n" \
140 "andi %0, %0, -4\n" \
144 "amoor.w.aq %2, %1, (%0)\n" \
147 : "+r" (lock), "=r" (_tmp0), "=r" (_tmp1) \
149 __mem_fence_acquire(); \
152#error no SW_SPIN_TRY_LOCK available for PICO_USE_SW_SPIN_LOCK on this platform
156#ifndef SW_SPIN_TRY_LOCK
157#if __ARM_ARCH_8M_MAIN__
158#define SW_SPIN_TRY_LOCK(lock) ({ \
159 uint32_t _tmp0, _tmp1; \
160 pico_default_asm_volatile ( \
161 "ldaexb %1, [%2]\n" \
167 "strexb %1, %0, [%2]\n" \
169 : "=&r" (_tmp0), "=&r" (_tmp1) : "r" (lock) \
171 __mem_fence_acquire(); \
174#elif __riscv && (defined(__riscv_a) || defined(__riscv_zaamo))
175#define SW_SPIN_TRY_LOCK(lock) ({ \
177 pico_default_asm_volatile ( \
181 "bset %1, zero, %1\n" \
182 "andi %0, %0, -4\n" \
185 "amoor.w.aq %0, %1, (%0)\n" \
187 : "+r" (lock), "=r" (_tmp0) \
189 __mem_fence_acquire(); \
193#error no SW_SPIN_TRY_LOCK available for PICO_USE_SW_SPIN_LOCK on this platform
197#ifndef SW_SPIN_LOCK_UNLOCK
198#if __ARM_ARCH_8M_MAIN__
199#define SW_SPIN_LOCK_UNLOCK(lock) ({ \
202 pico_default_asm_volatile( \
204 : : "r" (zero), "r" (lock) \
208#define SW_SPIN_LOCK_UNLOCK(lock) ({ \
209 __mem_fence_release(); \
213#error no SW_SPIN_TRY_LOCK available for PICO_USE_SW_SPIN_LOCK on this platform
226 invalid_params_if(HARDWARE_SYNC, lock_num >= NUM_SPIN_LOCKS);
227#if PICO_USE_SW_SPIN_LOCKS
228 return SW_SPIN_LOCK_INSTANCE(lock_num);
230 return (spin_lock_t *) (SIO_BASE + SIO_SPINLOCK0_OFFSET + lock_num * 4);
241#if PICO_USE_SW_SPIN_LOCKS
242 uint lock_num = SW_SPIN_LOCK_NUM(lock);
243 invalid_params_if(HARDWARE_SYNC, lock_num >= (uint)NUM_SPIN_LOCKS);
246 invalid_params_if(HARDWARE_SYNC, (uint) lock < SIO_BASE + SIO_SPINLOCK0_OFFSET ||
247 (uint) lock >= NUM_SPIN_LOCKS *
sizeof(spin_lock_t) + SIO_BASE + SIO_SPINLOCK0_OFFSET ||
248 ((uint) lock - SIO_BASE + SIO_SPINLOCK0_OFFSET) %
sizeof(spin_lock_t) != 0);
249 return (uint) (lock - (spin_lock_t *) (SIO_BASE + SIO_SPINLOCK0_OFFSET));
262#if PICO_USE_SW_SPIN_LOCKS
263 SW_SPIN_LOCK_LOCK(lock);
265 while (__builtin_expect(!*lock, 0)) {
272__force_inline static bool spin_try_lock_unsafe(spin_lock_t *lock) {
273#if PICO_USE_SW_SPIN_LOCKS
274 return SW_SPIN_TRY_LOCK(lock);
285#if PICO_USE_SW_SPIN_LOCKS
286 SW_SPIN_LOCK_UNLOCK(lock);
313#if PICO_USE_SW_SPIN_LOCKS
314 return SW_SPIN_LOCK_IS_LOCKED(lock);
316 check_hw_size(spin_lock_t, 4);
318 return 0 != (*(io_ro_32 *) (SIO_BASE + SIO_SPINLOCK_ST_OFFSET) & (1u << lock_num));
static __force_inline uint32_t spin_lock_blocking(spin_lock_t *lock)
Acquire a spin lock safely.
Definition: spin_lock.h:301
static __force_inline uint32_t save_and_disable_interrupts(void)
Save and disable interrupts.
Definition: sync.h:206
static __force_inline void __mem_fence_release(void)
Release a memory fence.
Definition: sync.h:189
static __force_inline void spin_unlock_unsafe(spin_lock_t *lock)
Release a spin lock without re-enabling interrupts.
Definition: spin_lock.h:284
static __force_inline void spin_unlock(spin_lock_t *lock, uint32_t saved_irq)
Release a spin lock safely.
Definition: spin_lock.h:332
static __force_inline void restore_interrupts_from_disabled(uint32_t status)
Restore interrupts to a specified state with restricted transitions.
Definition: sync.h:249
void spin_locks_reset(void)
Release all spin locks.
Definition: sync_spin_lock.c:8
static __force_inline spin_lock_t * spin_lock_instance(uint lock_num)
Get HW Spinlock instance from number.
Definition: spin_lock.h:225
static __force_inline void __mem_fence_acquire(void)
Acquire a memory fence.
Definition: sync.h:173
static __force_inline void spin_lock_unsafe_blocking(spin_lock_t *lock)
Acquire a spin lock without disabling interrupts (hence unsafe)
Definition: spin_lock.h:258
spin_lock_t * spin_lock_init(uint lock_num)
Initialise a spin lock.
Definition: sync_spin_lock.c:14
static __force_inline uint spin_lock_get_num(spin_lock_t *lock)
Get HW Spinlock number from instance.
Definition: spin_lock.h:240
static bool is_spin_locked(spin_lock_t *lock)
Check to see if a spinlock is currently acquired elsewhere.
Definition: spin_lock.h:312
#define SW_SPIN_LOCK_TYPE
A spin lock identifier.
Definition: spin_lock.h:88