1
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
27
28
46
47
60
61
67
68
74
75
79
80
84
85
88
89
90
91
99
100
106
107
110
111
114
115
116
117
118
119
120
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
146
147
148
149
150
151
152
153
154
155
156
157
158
161
164
165
172
173
174
185
186
187
188
195
196
200
201
204
205
206
207
208
209
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
235
236
237
238
239
240
241
242
243
244
245
246
/* ... */
#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
/* ... */
FORCE_INLINE_ATTR __attribute__((pure)) uint32_t xt_utils_get_core_id(void)
{
/* ... */
#if SOC_CPU_CORES_NUM > 1
uint32_t id;
asm volatile (
"rsr.prid %0\n"
"extui %0,%0,13,1"
:"=r"(id));
return id;/* ... */
#else
return 0;
#endif
}{ ... }
FORCE_INLINE_ATTR __attribute__((pure)) uint32_t xt_utils_get_raw_core_id(void)
{
#if XCHAL_HAVE_PRID
uint32_t id;
asm volatile (
"rsr.prid %0\n"
:"=r"(id));
return id;/* ... */
#else
return 0;
#endif
}{ ... }
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");
}{ ... }
/* ... */
FORCE_INLINE_ATTR void xt_utils_set_vecbase(uint32_t vecbase)
{
asm volatile ("wsr %0, vecbase" :: "r" (vecbase));
}{ ... }
Interrupt Configuration
FORCE_INLINE_ATTR uint32_t xt_utils_intr_get_enabled_mask(void)
{
uint32_t intr_mask;
RSR(INTENABLE, intr_mask);
return intr_mask;
}{ ... }
/* ... */
/* ... */
Interrupt Control
FORCE_INLINE_ATTR void xt_utils_set_breakpoint(int bp_num, uint32_t bp_addr)
{
if (bp_num == 1) {
WSR(IBREAKA_1, bp_addr);
}{...} else {
WSR(IBREAKA_0, bp_addr);
}{...}
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)
{
uint32_t bp_en = 0;
RSR(IBREAKENABLE, bp_en);
bp_en &= ~BIT(bp_num);
WSR(IBREAKENABLE, bp_en);
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)
{
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);
}{...}
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)
{
if (wp_num == 1) {
WSR(DBREAKC_1, 0);
WSR(DBREAKA_1, 0);
}{...} else {
WSR(DBREAKC_0, 0);
WSR(DBREAKA_0, 0);
}{...}
}{ ... }
Breakpoints/Watchpoints
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");
}{ ... }
/* ... */
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__
volatile uint32_t temp;
temp = *addr;
*addr = temp;/* ... */
#endif
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
uint32_t intr_level;
__asm__ __volatile__ ("rsil %0, " XTSTR(XCHAL_EXCM_LEVEL) "\n"
: "=r"(intr_level));
uint32_t old_value;
old_value = *addr;
if (old_value == compare_value) {
*addr = new_value;
}{...}
__asm__ __volatile__ ("memw \n"
"wsr %0, ps\n"
:: "r"(intr_level));
return (old_value == compare_value);/* ... */
#endif
}{ ... }
#ifdef __cplusplus
}Debugger
{...}#endif