Select one of the symbols to view example projects that use it.
 
Outline
#include "sdkconfig.h"
#include <stdint.h>
#include <stdbool.h>
#include "esp_cpu.h"
#include "xtensa/xtruntime.h"
#include "xt_utils.h"
#include "riscv/rv_utils.h"
#define NEED_VOLATILE_MUX
#define NEED_VOLATILE_MUX
#define SPINLOCK_FREE
#define SPINLOCK_WAIT_FOREVER
#define SPINLOCK_NO_WAIT
#define SPINLOCK_INITIALIZER
#define SPINLOCK_OWNER_ID_0
#define SPINLOCK_OWNER_ID_1
#define CORE_ID_REGVAL_XOR_SWAP
#define SPINLOCK_OWNER_ID_XOR_SWAP
spinlock_t
spinlock_initialize(spinlock_t *)
spinlock_acquire(spinlock_t *, int32_t)
spinlock_release(spinlock_t *)
Files
loading...
SourceVuESP-IDF Framework and ExamplesESP-IDFcomponents/esp_hw_support/include/spinlock.h
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
/* * SPDX-FileCopyrightText: 2015-2024 Espressif Systems (Shanghai) CO LTD * * SPDX-License-Identifier: Apache-2.0 *//* ... */ #pragma once #include "sdkconfig.h" #include <stdint.h> #include <stdbool.h> #include "esp_cpu.h" #if __XTENSA__ #include "xtensa/xtruntime.h" #include "xt_utils.h"/* ... */ #else #include "riscv/rv_utils.h" #endif #ifdef __cplusplus extern "C" { #endif #ifdef CONFIG_SPIRAM_WORKAROUND_NEED_VOLATILE_SPINLOCK #define NEED_VOLATILE_MUX volatile #else #define NEED_VOLATILE_MUX #endif #define SPINLOCK_FREE 0xB33FFFFF #define SPINLOCK_WAIT_FOREVER (-1) #define SPINLOCK_NO_WAIT 0 #define SPINLOCK_INITIALIZER {.owner = SPINLOCK_FREE,.count = 0} #define SPINLOCK_OWNER_ID_0 0xCDCD /* Use these values to avoid 0 being a valid lock owner, same as CORE_ID_REGVAL_PRO on Xtensa */ #define SPINLOCK_OWNER_ID_1 0xABAB /* Same as CORE_ID_REGVAL_APP on Xtensa*/ #define CORE_ID_REGVAL_XOR_SWAP (0xCDCD ^ 0xABAB) #define SPINLOCK_OWNER_ID_XOR_SWAP CORE_ID_REGVAL_XOR_SWAP8 defines typedef struct { NEED_VOLATILE_MUX uint32_t owner; NEED_VOLATILE_MUX uint32_t count; }{ ... } spinlock_t; /** * @brief Initialize a lock to its default state - unlocked * @param lock - spinlock object to initialize *//* ... */ static inline void __attribute__((always_inline)) spinlock_initialize(spinlock_t *lock) { assert(lock); #if !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE lock->owner = SPINLOCK_FREE; lock->count = 0;/* ... */ #endif }{ ... } /** * @brief Top level spinlock acquire function, spins until get the lock * * This function will: * - Save current interrupt state, then disable interrupts * - Spin until lock is acquired or until timeout occurs * - Restore interrupt state * * @note Spinlocks alone do no constitute true critical sections (as this * function reenables interrupts once the spinlock is acquired). For critical * sections, use the interface provided by the operating system. * @param lock - target spinlock object * @param timeout - cycles to wait, passing SPINLOCK_WAIT_FOREVER blocks indefinitely *//* ... */ static inline bool __attribute__((always_inline)) spinlock_acquire(spinlock_t *lock, int32_t timeout) { #if !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE && !BOOTLOADER_BUILD uint32_t irq_status; uint32_t core_owner_id; // Unused if asserts are disabled uint32_t __attribute__((unused)) other_core_owner_id; bool lock_set; esp_cpu_cycle_count_t start_count; assert(lock); #if __XTENSA__ irq_status = XTOS_SET_INTLEVEL(XCHAL_EXCM_LEVEL); // Note: The core IDs are the full 32 bit (CORE_ID_REGVAL_PRO/CORE_ID_REGVAL_APP) values core_owner_id = xt_utils_get_raw_core_id();/* ... */ #else //__riscv irq_status = rv_utils_set_intlevel_regval(RVHAL_EXCM_LEVEL_CLIC); core_owner_id = rv_utils_get_core_id() == 0 ? SPINLOCK_OWNER_ID_0 : SPINLOCK_OWNER_ID_1;/* ... */ #endif other_core_owner_id = CORE_ID_REGVAL_XOR_SWAP ^ core_owner_id; /* lock->owner should be one of SPINLOCK_FREE, CORE_ID_REGVAL_PRO, * CORE_ID_REGVAL_APP: * - If SPINLOCK_FREE, we want to atomically set to 'core_owner_id'. * - If "our" core_owner_id, we can drop through immediately. * - If "other_core_owner_id", we spin here. *//* ... */ // The caller is already the owner of the lock. Simply increment the nesting count if (lock->owner == core_owner_id) { assert(lock->count > 0 && lock->count < 0xFF); // Bad count value implies memory corruption lock->count++; #if __XTENSA__ XTOS_RESTORE_INTLEVEL(irq_status); #else rv_utils_restore_intlevel_regval(irq_status); #endif return true; }{...} /* First attempt to take the lock. * * Note: We do a first attempt separately (instead of putting this into a loop) in order to avoid call to * esp_cpu_get_cycle_count(). This doing a first attempt separately makes acquiring a free lock quicker, which * is the case for the majority of spinlock_acquire() calls (as spinlocks are free most of the time since they * aren't meant to be held for long). *//* ... */ lock_set = esp_cpu_compare_and_set(&lock->owner, SPINLOCK_FREE, core_owner_id); if (lock_set || timeout == SPINLOCK_NO_WAIT) { // We've successfully taken the lock, or we are not retrying goto exit; }{...} // First attempt to take the lock has failed. Retry until the lock is taken, or until we timeout. start_count = esp_cpu_get_cycle_count(); do { lock_set = esp_cpu_compare_and_set(&lock->owner, SPINLOCK_FREE, core_owner_id); if (lock_set) { break; }{...} // Keep looping if we are waiting forever, or check if we have timed out }{...} while ((timeout == SPINLOCK_WAIT_FOREVER) || (esp_cpu_get_cycle_count() - start_count) <= (esp_cpu_cycle_count_t)timeout); exit: if (lock_set) { assert(lock->owner == core_owner_id); assert(lock->count == 0); // This is the first time the lock is set, so count should still be 0 lock->count++; // Finally, we increment the lock count }{...} else { // We timed out waiting for lock assert(lock->owner == SPINLOCK_FREE || lock->owner == other_core_owner_id); assert(lock->count < 0xFF); // Bad count value implies memory corruption }{...} #if __XTENSA__ XTOS_RESTORE_INTLEVEL(irq_status); #else rv_utils_restore_intlevel_regval(irq_status); #endif return lock_set; /* ... */ #else // !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE return true; #endif }{ ... } /** * @brief Top level spinlock unlock function, unlocks a previously locked spinlock * * This function will: * - Save current interrupt state, then disable interrupts * - Release the spinlock * - Restore interrupt state * * @note Spinlocks alone do no constitute true critical sections (as this * function reenables interrupts once the spinlock is acquired). For critical * sections, use the interface provided by the operating system. * @param lock - target, locked before, spinlock object *//* ... */ static inline void __attribute__((always_inline)) spinlock_release(spinlock_t *lock) { #if !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE && !BOOTLOADER_BUILD uint32_t irq_status; // Return value unused if asserts are disabled uint32_t __attribute__((unused)) core_owner_id; assert(lock); #if __XTENSA__ irq_status = XTOS_SET_INTLEVEL(XCHAL_EXCM_LEVEL); core_owner_id = xt_utils_get_raw_core_id();/* ... */ #else irq_status = rv_utils_set_intlevel_regval(RVHAL_EXCM_LEVEL_CLIC); core_owner_id = rv_utils_get_core_id() == 0 ? SPINLOCK_OWNER_ID_0 : SPINLOCK_OWNER_ID_1;/* ... */ #endif assert(core_owner_id == lock->owner); // This is a lock that we didn't acquire, or the lock is corrupt lock->count--; if (!lock->count) { // If this is the last recursive release of the lock, mark the lock as free lock->owner = SPINLOCK_FREE; }{...} else { assert(lock->count < 0x100); // Indicates memory corruption }{...} #if __XTENSA__ XTOS_RESTORE_INTLEVEL(irq_status); #else rv_utils_restore_intlevel_regval(irq_status); #endif //#if __XTENSA__/* ... */ #endif //#if !CONFIG_ESP_SYSTEM_SINGLE_CORE_MODE && !BOOTLOADER_BUILD }{ ... } #ifdef __cplusplus }{...} #endif
Details