Select one of the symbols to view example projects that use it.
 
Outline
#include <stdint.h>
#include <stdbool.h>
#include "soc/soc_caps.h"
#include "xtensa/config/core-isa.h"
#include "xtensa/config/core.h"
#include "xtensa/config/extreg.h"
#include "xtensa/config/specreg.h"
#include "xtensa/xtruntime.h"
#include "xt_instr_macros.h"
#include "esp_bit_defs.h"
#include "esp_attr.h"
xt_utils_get_core_id()
xt_utils_get_raw_core_id()
xt_utils_get_sp()
xt_utils_get_cycle_count()
xt_utils_set_cycle_count(uint32_t)
xt_utils_wait_for_intr()
Interrupt Configuration
xt_utils_set_vecbase(uint32_t)
Interrupt Control
xt_utils_intr_get_enabled_mask()
Breakpoints/Watchpoints
xt_utils_set_breakpoint(int, uint32_t)
xt_utils_clear_breakpoint(int)
xt_utils_set_watchpoint(int, uint32_t, size_t, bool, bool)
xt_utils_clear_watchpoint(int)
Debugger
xt_utils_dbgr_is_attached()
xt_utils_dbgr_break()
xt_utils_compare_and_set(volatile uint32_t *, uint32_t, uint32_t)
Files
loading...
SourceVuESP-IDF Framework and ExamplesESP-IDFcomponents/xtensa/include/xt_utils.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
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
/* * SPDX-FileCopyrightText: 2020-2022 Espressif Systems (Shanghai) CO LTD * * SPDX-License-Identifier: Apache-2.0 *//* ... */ #pragma once #include <stdint.h> #include <stdbool.h> #include "soc/soc_caps.h" #include "xtensa/config/core-isa.h" #include "xtensa/config/core.h" #include "xtensa/config/extreg.h" #include "xtensa/config/specreg.h" #include "xtensa/xtruntime.h" #include "xt_instr_macros.h" #include "esp_bit_defs.h" #include "esp_attr.h"11 includes #ifdef __cplusplus extern "C" { #endif /* -------------------------------------------------- CPU Registers ---------------------------------------------------- * * ------------------------------------------------------------------------------------------------------------------ *//* ... */ FORCE_INLINE_ATTR __attribute__((pure)) uint32_t xt_utils_get_core_id(void) { /* Note: We depend on SOC_CPU_CORES_NUM instead of XCHAL_HAVE_PRID as some single Xtensa targets (such as ESP32-S2) have the PRID register even though they are single core. *//* ... */ #if SOC_CPU_CORES_NUM > 1 // Read and extract bit 13 of special register PRID uint32_t id; asm volatile ( "rsr.prid %0\n" "extui %0,%0,13,1" :"=r"(id)); return id;/* ... */ #else return 0; #endif // SOC_CPU_CORES_NUM > 1 }{ ... } FORCE_INLINE_ATTR __attribute__((pure)) uint32_t xt_utils_get_raw_core_id(void) { #if XCHAL_HAVE_PRID // Read the raw value of special register PRID uint32_t id; asm volatile ( "rsr.prid %0\n" :"=r"(id)); return id;/* ... */ #else return 0; #endif // XCHAL_HAVE_PRID }{ ... } FORCE_INLINE_ATTR void *xt_utils_get_sp(void) { void *sp; asm volatile ("mov %0, sp;" : "=r" (sp)); return sp; }{ ... } FORCE_INLINE_ATTR uint32_t xt_utils_get_cycle_count(void) { uint32_t ccount; RSR(CCOUNT, ccount); return ccount; }{ ... } static inline void xt_utils_set_cycle_count(uint32_t ccount) { WSR(CCOUNT, ccount); }{ ... } FORCE_INLINE_ATTR void xt_utils_wait_for_intr(void) { asm volatile ("waiti 0\n"); }{ ... } /* ------------------------------------------------- CPU Interrupts ---------------------------------------------------- * * ------------------------------------------------------------------------------------------------------------------ *//* ... */ // ---------------- Interrupt Descriptors ------------------ // --------------- Interrupt Configuration ----------------- FORCE_INLINE_ATTR void xt_utils_set_vecbase(uint32_t vecbase) { asm volatile ("wsr %0, vecbase" :: "r" (vecbase)); }{ ... } Interrupt Configuration// ------------------ Interrupt Control -------------------- FORCE_INLINE_ATTR uint32_t xt_utils_intr_get_enabled_mask(void) { uint32_t intr_mask; RSR(INTENABLE, intr_mask); return intr_mask; }{ ... } /* -------------------------------------------------- Memory Ports ----------------------------------------------------- * * ------------------------------------------------------------------------------------------------------------------ *//* ... */ /* ---------------------------------------------------- Debugging ------------------------------------------------------ * * ------------------------------------------------------------------------------------------------------------------ *//* ... */ Interrupt Control // --------------- Breakpoints/Watchpoints ----------------- FORCE_INLINE_ATTR void xt_utils_set_breakpoint(int bp_num, uint32_t bp_addr) { //Set the breakpoint's address if (bp_num == 1) { WSR(IBREAKA_1, bp_addr); }{...} else { WSR(IBREAKA_0, bp_addr); }{...} //Enable the breakpoint uint32_t brk_ena_reg; RSR(IBREAKENABLE, brk_ena_reg); brk_ena_reg |= BIT(bp_num); WSR(IBREAKENABLE, brk_ena_reg); }{ ... } FORCE_INLINE_ATTR void xt_utils_clear_breakpoint(int bp_num) { // Disable the breakpoint using the break enable register uint32_t bp_en = 0; RSR(IBREAKENABLE, bp_en); bp_en &= ~BIT(bp_num); WSR(IBREAKENABLE, bp_en); // Zero the break address register uint32_t bp_addr = 0; if (bp_num == 1) { WSR(IBREAKA_1, bp_addr); }{...} else { WSR(IBREAKA_0, bp_addr); }{...} }{ ... } FORCE_INLINE_ATTR void xt_utils_set_watchpoint(int wp_num, uint32_t wp_addr, size_t size, bool on_read, bool on_write) { // Initialize DBREAKC bits (see Table 4–143 or isa_rm.pdf) uint32_t dbreakc_reg = 0x3F; dbreakc_reg = dbreakc_reg << (__builtin_ffsll(size) - 1); dbreakc_reg = dbreakc_reg & 0x3F; if (on_read) { dbreakc_reg |= BIT(30); }{...} if (on_write) { dbreakc_reg |= BIT(31); }{...} // Enable break address and break control register if (wp_num == 1) { WSR(DBREAKA_1, (uint32_t) wp_addr); WSR(DBREAKC_1, dbreakc_reg); }{...} else { WSR(DBREAKA_0, (uint32_t) wp_addr); WSR(DBREAKC_0, dbreakc_reg); }{...} }{ ... } FORCE_INLINE_ATTR void xt_utils_clear_watchpoint(int wp_num) { // Clear both break control and break address register if (wp_num == 1) { WSR(DBREAKC_1, 0); WSR(DBREAKA_1, 0); }{...} else { WSR(DBREAKC_0, 0); WSR(DBREAKA_0, 0); }{...} }{ ... } Breakpoints/Watchpoints// ---------------------- Debugger ------------------------- FORCE_INLINE_ATTR bool xt_utils_dbgr_is_attached(void) { uint32_t dcr = 0; uint32_t reg = DSRSET; RER(reg, dcr); return (bool)(dcr & 0x1); }{ ... } FORCE_INLINE_ATTR void xt_utils_dbgr_break(void) { __asm__ ("break 1,15"); }{ ... } /* ------------------------------------------------------ Misc --------------------------------------------------------- * * ------------------------------------------------------------------------------------------------------------------ *//* ... */ FORCE_INLINE_ATTR bool xt_utils_compare_and_set(volatile uint32_t *addr, uint32_t compare_value, uint32_t new_value) { #if XCHAL_HAVE_S32C1I #ifdef __clang_analyzer__ //Teach clang-tidy that "addr" cannot be const as it can be updated by S32C1I instruction volatile uint32_t temp; temp = *addr; *addr = temp;/* ... */ #endif // Atomic compare and set using S32C1I instruction uint32_t old_value = new_value; __asm__ __volatile__ ( "WSR %2, SCOMPARE1 \n" "S32C1I %0, %1, 0 \n" :"=r"(old_value) :"r"(addr), "r"(compare_value), "0"(old_value) ); return (old_value == compare_value);/* ... */ #else // XCHAL_HAVE_S32C1I // Single core target has no atomic CAS instruction. We can achieve atomicity by disabling interrupts uint32_t intr_level; __asm__ __volatile__ ("rsil %0, " XTSTR(XCHAL_EXCM_LEVEL) "\n" : "=r"(intr_level)); // Compare and set uint32_t old_value; old_value = *addr; if (old_value == compare_value) { *addr = new_value; }{...} // Restore interrupts __asm__ __volatile__ ("memw \n" "wsr %0, ps\n" :: "r"(intr_level)); return (old_value == compare_value);/* ... */ #endif // XCHAL_HAVE_S32C1I }{ ... } #ifdef __cplusplus }Debugger {...}#endif
Details