OpenOCD
xtensa.c
Go to the documentation of this file.
1 // SPDX-License-Identifier: GPL-2.0-or-later
2 
3 /***************************************************************************
4  * Generic Xtensa target API for OpenOCD *
5  * Copyright (C) 2020-2022 Cadence Design Systems, Inc. *
6  * Copyright (C) 2016-2019 Espressif Systems Ltd. *
7  * Derived from esp108.c *
8  * Author: Angus Gratton gus@projectgus.com *
9  ***************************************************************************/
10 
11 #ifdef HAVE_CONFIG_H
12 #include "config.h"
13 #endif
14 
15 #include <stdlib.h>
16 #include <helper/time_support.h>
17 #include <helper/align.h>
18 #include <target/register.h>
19 #include <target/algorithm.h>
20 
21 #include "xtensa_chip.h"
22 #include "xtensa.h"
23 
24 /* Swap 4-bit Xtensa opcodes and fields */
25 #define XT_NIBSWAP8(V) \
26  ((((V) & 0x0F) << 4) \
27  | (((V) & 0xF0) >> 4))
28 
29 #define XT_NIBSWAP16(V) \
30  ((((V) & 0x000F) << 12) \
31  | (((V) & 0x00F0) << 4) \
32  | (((V) & 0x0F00) >> 4) \
33  | (((V) & 0xF000) >> 12))
34 
35 #define XT_NIBSWAP24(V) \
36  ((((V) & 0x00000F) << 20) \
37  | (((V) & 0x0000F0) << 12) \
38  | (((V) & 0x000F00) << 4) \
39  | (((V) & 0x00F000) >> 4) \
40  | (((V) & 0x0F0000) >> 12) \
41  | (((V) & 0xF00000) >> 20))
42 
43 /* _XT_INS_FORMAT_*()
44  * Instruction formatting converted from little-endian inputs
45  * and shifted to the MSB-side of DIR for BE systems.
46  */
47 #define _XT_INS_FORMAT_RSR(X, OPCODE, SR, T) \
48  (XT_ISBE(X) ? (XT_NIBSWAP24(OPCODE) \
49  | (((T) & 0x0F) << 16) \
50  | (((SR) & 0xFF) << 8)) << 8 \
51  : (OPCODE) \
52  | (((SR) & 0xFF) << 8) \
53  | (((T) & 0x0F) << 4))
54 
55 #define _XT_INS_FORMAT_RRR(X, OPCODE, ST, R) \
56  (XT_ISBE(X) ? (XT_NIBSWAP24(OPCODE) \
57  | ((XT_NIBSWAP8((ST) & 0xFF)) << 12) \
58  | (((R) & 0x0F) << 8)) << 8 \
59  : (OPCODE) \
60  | (((ST) & 0xFF) << 4) \
61  | (((R) & 0x0F) << 12))
62 
63 #define _XT_INS_FORMAT_RRRN(X, OPCODE, S, T, IMM4) \
64  (XT_ISBE(X) ? (XT_NIBSWAP16(OPCODE) \
65  | (((T) & 0x0F) << 8) \
66  | (((S) & 0x0F) << 4) \
67  | ((IMM4) & 0x0F)) << 16 \
68  : (OPCODE) \
69  | (((T) & 0x0F) << 4) \
70  | (((S) & 0x0F) << 8) \
71  | (((IMM4) & 0x0F) << 12))
72 
73 #define _XT_INS_FORMAT_RRI8(X, OPCODE, R, S, T, IMM8) \
74  (XT_ISBE(X) ? (XT_NIBSWAP24(OPCODE) \
75  | (((T) & 0x0F) << 16) \
76  | (((S) & 0x0F) << 12) \
77  | (((R) & 0x0F) << 8) \
78  | ((IMM8) & 0xFF)) << 8 \
79  : (OPCODE) \
80  | (((IMM8) & 0xFF) << 16) \
81  | (((R) & 0x0F) << 12) \
82  | (((S) & 0x0F) << 8) \
83  | (((T) & 0x0F) << 4))
84 
85 #define _XT_INS_FORMAT_RRI4(X, OPCODE, IMM4, R, S, T) \
86  (XT_ISBE(X) ? (XT_NIBSWAP24(OPCODE) \
87  | (((T) & 0x0F) << 16) \
88  | (((S) & 0x0F) << 12) \
89  | (((R) & 0x0F) << 8)) << 8 \
90  | ((IMM4) & 0x0F) \
91  : (OPCODE) \
92  | (((IMM4) & 0x0F) << 20) \
93  | (((R) & 0x0F) << 12) \
94  | (((S) & 0x0F) << 8) \
95  | (((T) & 0x0F) << 4))
96 
97 /* Xtensa processor instruction opcodes
98 */
99 /* "Return From Debug Operation" to Normal */
100 #define XT_INS_RFDO(X) (XT_ISBE(X) ? 0x000e1f << 8 : 0xf1e000)
101 /* "Return From Debug and Dispatch" - allow sw debugging stuff to take over */
102 #define XT_INS_RFDD(X) (XT_ISBE(X) ? 0x010e1f << 8 : 0xf1e010)
103 
104 /* Load to DDR register, increase addr register */
105 #define XT_INS_LDDR32P(X, S) (XT_ISBE(X) ? (0x0E0700 | ((S) << 12)) << 8 : (0x0070E0 | ((S) << 8)))
106 /* Store from DDR register, increase addr register */
107 #define XT_INS_SDDR32P(X, S) (XT_ISBE(X) ? (0x0F0700 | ((S) << 12)) << 8 : (0x0070F0 | ((S) << 8)))
108 
109 /* Load 32-bit Indirect from A(S)+4*IMM8 to A(T) */
110 #define XT_INS_L32I(X, S, T, IMM8) _XT_INS_FORMAT_RRI8(X, 0x002002, 0, S, T, IMM8)
111 /* Load 16-bit Unsigned from A(S)+2*IMM8 to A(T) */
112 #define XT_INS_L16UI(X, S, T, IMM8) _XT_INS_FORMAT_RRI8(X, 0x001002, 0, S, T, IMM8)
113 /* Load 8-bit Unsigned from A(S)+IMM8 to A(T) */
114 #define XT_INS_L8UI(X, S, T, IMM8) _XT_INS_FORMAT_RRI8(X, 0x000002, 0, S, T, IMM8)
115 
116 /* Store 32-bit Indirect to A(S)+4*IMM8 from A(T) */
117 #define XT_INS_S32I(X, S, T, IMM8) _XT_INS_FORMAT_RRI8(X, 0x006002, 0, S, T, IMM8)
118 /* Store 16-bit to A(S)+2*IMM8 from A(T) */
119 #define XT_INS_S16I(X, S, T, IMM8) _XT_INS_FORMAT_RRI8(X, 0x005002, 0, S, T, IMM8)
120 /* Store 8-bit to A(S)+IMM8 from A(T) */
121 #define XT_INS_S8I(X, S, T, IMM8) _XT_INS_FORMAT_RRI8(X, 0x004002, 0, S, T, IMM8)
122 
123 /* Cache Instructions */
124 #define XT_INS_IHI(X, S, IMM8) _XT_INS_FORMAT_RRI8(X, 0x0070E2, 0, S, 0, IMM8)
125 #define XT_INS_DHWBI(X, S, IMM8) _XT_INS_FORMAT_RRI8(X, 0x007052, 0, S, 0, IMM8)
126 #define XT_INS_DHWB(X, S, IMM8) _XT_INS_FORMAT_RRI8(X, 0x007042, 0, S, 0, IMM8)
127 #define XT_INS_ISYNC(X) (XT_ISBE(X) ? 0x000200 << 8 : 0x002000)
128 
129 /* Control Instructions */
130 #define XT_INS_JX(X, S) (XT_ISBE(X) ? (0x050000 | ((S) << 12)) : (0x0000a0 | ((S) << 8)))
131 #define XT_INS_CALL0(X, IMM18) (XT_ISBE(X) ? (0x500000 | ((IMM18) & 0x3ffff)) : (0x000005 | (((IMM18) & 0x3ffff) << 6)))
132 
133 /* Read Special Register */
134 #define XT_INS_RSR(X, SR, T) _XT_INS_FORMAT_RSR(X, 0x030000, SR, T)
135 /* Write Special Register */
136 #define XT_INS_WSR(X, SR, T) _XT_INS_FORMAT_RSR(X, 0x130000, SR, T)
137 /* Swap Special Register */
138 #define XT_INS_XSR(X, SR, T) _XT_INS_FORMAT_RSR(X, 0x610000, SR, T)
139 
140 /* Rotate Window by (-8..7) */
141 #define XT_INS_ROTW(X, N) (XT_ISBE(X) ? ((0x000804) | (((N) & 15) << 16)) << 8 : ((0x408000) | (((N) & 15) << 4)))
142 
143 /* Read User Register */
144 #define XT_INS_RUR(X, UR, T) _XT_INS_FORMAT_RRR(X, 0xE30000, UR, T)
145 /* Write User Register */
146 #define XT_INS_WUR(X, UR, T) _XT_INS_FORMAT_RSR(X, 0xF30000, UR, T)
147 
148 /* Read Floating-Point Register */
149 #define XT_INS_RFR(X, FR, T) _XT_INS_FORMAT_RRR(X, 0xFA0000, ((FR << 4) | 0x4), T)
150 /* Write Floating-Point Register */
151 #define XT_INS_WFR(X, FR, T) _XT_INS_FORMAT_RRR(X, 0xFA0000, ((T << 4) | 0x5), FR)
152 
153 #define XT_INS_L32E(X, R, S, T) _XT_INS_FORMAT_RRI4(X, 0x090000, 0, R, S, T)
154 #define XT_INS_S32E(X, R, S, T) _XT_INS_FORMAT_RRI4(X, 0x490000, 0, R, S, T)
155 #define XT_INS_L32E_S32E_MASK(X) (XT_ISBE(X) ? 0xF000FF << 8 : 0xFF000F)
156 
157 #define XT_INS_RFWO(X) (XT_ISBE(X) ? 0x004300 << 8 : 0x003400)
158 #define XT_INS_RFWU(X) (XT_ISBE(X) ? 0x005300 << 8 : 0x003500)
159 #define XT_INS_RFWO_RFWU_MASK(X) (XT_ISBE(X) ? 0xFFFFFF << 8 : 0xFFFFFF)
160 
161 /* Read Protection TLB Entry Info */
162 #define XT_INS_PPTLB(X, S, T) _XT_INS_FORMAT_RRR(X, 0x500000, ((S) << 4) | (T), 0xD)
163 
164 #define XT_TLB1_ACC_SHIFT 8
165 #define XT_TLB1_ACC_MSK 0xF
166 
167 #define XT_WATCHPOINTS_NUM_MAX 2
168 
169 /* Special register number macro for DDR, PS, WB, A3, A4 registers.
170  * These get used a lot so making a shortcut is useful.
171  */
172 #define XT_SR_DDR (xtensa_regs[XT_REG_IDX_DDR].reg_num)
173 #define XT_SR_PS (xtensa_regs[XT_REG_IDX_PS].reg_num)
174 #define XT_SR_WB (xtensa_regs[XT_REG_IDX_WINDOWBASE].reg_num)
175 #define XT_REG_A0 (xtensa_regs[XT_REG_IDX_AR0].reg_num)
176 #define XT_REG_A3 (xtensa_regs[XT_REG_IDX_AR3].reg_num)
177 #define XT_REG_A4 (xtensa_regs[XT_REG_IDX_AR4].reg_num)
178 
179 #define XT_PS_REG_NUM (0xe6U)
180 #define XT_EPS_REG_NUM_BASE (0xc0U) /* (EPS2 - 2), for adding DBGLEVEL */
181 #define XT_EPC_REG_NUM_BASE (0xb0U) /* (EPC1 - 1), for adding DBGLEVEL */
182 #define XT_PC_REG_NUM_VIRTUAL (0xffU) /* Marker for computing PC (EPC[DBGLEVEL) */
183 #define XT_PC_DBREG_NUM_BASE (0x20U) /* External (i.e., GDB) access */
184 #define XT_NX_IBREAKC_BASE (0xc0U) /* (IBREAKC0..IBREAKC1) for NX */
185 
186 #define XT_SW_BREAKPOINTS_MAX_NUM 32
187 #define XT_HW_IBREAK_MAX_NUM 2
188 #define XT_HW_DBREAK_MAX_NUM 2
189 
192  XT_MK_REG_DESC("ar0", 0x00, XT_REG_GENERAL, 0),
193  XT_MK_REG_DESC("ar1", 0x01, XT_REG_GENERAL, 0),
194  XT_MK_REG_DESC("ar2", 0x02, XT_REG_GENERAL, 0),
195  XT_MK_REG_DESC("ar3", 0x03, XT_REG_GENERAL, 0),
196  XT_MK_REG_DESC("ar4", 0x04, XT_REG_GENERAL, 0),
197  XT_MK_REG_DESC("ar5", 0x05, XT_REG_GENERAL, 0),
198  XT_MK_REG_DESC("ar6", 0x06, XT_REG_GENERAL, 0),
199  XT_MK_REG_DESC("ar7", 0x07, XT_REG_GENERAL, 0),
200  XT_MK_REG_DESC("ar8", 0x08, XT_REG_GENERAL, 0),
201  XT_MK_REG_DESC("ar9", 0x09, XT_REG_GENERAL, 0),
202  XT_MK_REG_DESC("ar10", 0x0A, XT_REG_GENERAL, 0),
203  XT_MK_REG_DESC("ar11", 0x0B, XT_REG_GENERAL, 0),
204  XT_MK_REG_DESC("ar12", 0x0C, XT_REG_GENERAL, 0),
205  XT_MK_REG_DESC("ar13", 0x0D, XT_REG_GENERAL, 0),
206  XT_MK_REG_DESC("ar14", 0x0E, XT_REG_GENERAL, 0),
207  XT_MK_REG_DESC("ar15", 0x0F, XT_REG_GENERAL, 0),
208  XT_MK_REG_DESC("ar16", 0x10, XT_REG_GENERAL, 0),
209  XT_MK_REG_DESC("ar17", 0x11, XT_REG_GENERAL, 0),
210  XT_MK_REG_DESC("ar18", 0x12, XT_REG_GENERAL, 0),
211  XT_MK_REG_DESC("ar19", 0x13, XT_REG_GENERAL, 0),
212  XT_MK_REG_DESC("ar20", 0x14, XT_REG_GENERAL, 0),
213  XT_MK_REG_DESC("ar21", 0x15, XT_REG_GENERAL, 0),
214  XT_MK_REG_DESC("ar22", 0x16, XT_REG_GENERAL, 0),
215  XT_MK_REG_DESC("ar23", 0x17, XT_REG_GENERAL, 0),
216  XT_MK_REG_DESC("ar24", 0x18, XT_REG_GENERAL, 0),
217  XT_MK_REG_DESC("ar25", 0x19, XT_REG_GENERAL, 0),
218  XT_MK_REG_DESC("ar26", 0x1A, XT_REG_GENERAL, 0),
219  XT_MK_REG_DESC("ar27", 0x1B, XT_REG_GENERAL, 0),
220  XT_MK_REG_DESC("ar28", 0x1C, XT_REG_GENERAL, 0),
221  XT_MK_REG_DESC("ar29", 0x1D, XT_REG_GENERAL, 0),
222  XT_MK_REG_DESC("ar30", 0x1E, XT_REG_GENERAL, 0),
223  XT_MK_REG_DESC("ar31", 0x1F, XT_REG_GENERAL, 0),
224  XT_MK_REG_DESC("ar32", 0x20, XT_REG_GENERAL, 0),
225  XT_MK_REG_DESC("ar33", 0x21, XT_REG_GENERAL, 0),
226  XT_MK_REG_DESC("ar34", 0x22, XT_REG_GENERAL, 0),
227  XT_MK_REG_DESC("ar35", 0x23, XT_REG_GENERAL, 0),
228  XT_MK_REG_DESC("ar36", 0x24, XT_REG_GENERAL, 0),
229  XT_MK_REG_DESC("ar37", 0x25, XT_REG_GENERAL, 0),
230  XT_MK_REG_DESC("ar38", 0x26, XT_REG_GENERAL, 0),
231  XT_MK_REG_DESC("ar39", 0x27, XT_REG_GENERAL, 0),
232  XT_MK_REG_DESC("ar40", 0x28, XT_REG_GENERAL, 0),
233  XT_MK_REG_DESC("ar41", 0x29, XT_REG_GENERAL, 0),
234  XT_MK_REG_DESC("ar42", 0x2A, XT_REG_GENERAL, 0),
235  XT_MK_REG_DESC("ar43", 0x2B, XT_REG_GENERAL, 0),
236  XT_MK_REG_DESC("ar44", 0x2C, XT_REG_GENERAL, 0),
237  XT_MK_REG_DESC("ar45", 0x2D, XT_REG_GENERAL, 0),
238  XT_MK_REG_DESC("ar46", 0x2E, XT_REG_GENERAL, 0),
239  XT_MK_REG_DESC("ar47", 0x2F, XT_REG_GENERAL, 0),
240  XT_MK_REG_DESC("ar48", 0x30, XT_REG_GENERAL, 0),
241  XT_MK_REG_DESC("ar49", 0x31, XT_REG_GENERAL, 0),
242  XT_MK_REG_DESC("ar50", 0x32, XT_REG_GENERAL, 0),
243  XT_MK_REG_DESC("ar51", 0x33, XT_REG_GENERAL, 0),
244  XT_MK_REG_DESC("ar52", 0x34, XT_REG_GENERAL, 0),
245  XT_MK_REG_DESC("ar53", 0x35, XT_REG_GENERAL, 0),
246  XT_MK_REG_DESC("ar54", 0x36, XT_REG_GENERAL, 0),
247  XT_MK_REG_DESC("ar55", 0x37, XT_REG_GENERAL, 0),
248  XT_MK_REG_DESC("ar56", 0x38, XT_REG_GENERAL, 0),
249  XT_MK_REG_DESC("ar57", 0x39, XT_REG_GENERAL, 0),
250  XT_MK_REG_DESC("ar58", 0x3A, XT_REG_GENERAL, 0),
251  XT_MK_REG_DESC("ar59", 0x3B, XT_REG_GENERAL, 0),
252  XT_MK_REG_DESC("ar60", 0x3C, XT_REG_GENERAL, 0),
253  XT_MK_REG_DESC("ar61", 0x3D, XT_REG_GENERAL, 0),
254  XT_MK_REG_DESC("ar62", 0x3E, XT_REG_GENERAL, 0),
255  XT_MK_REG_DESC("ar63", 0x3F, XT_REG_GENERAL, 0),
256  XT_MK_REG_DESC("windowbase", 0x48, XT_REG_SPECIAL, 0),
257  XT_MK_REG_DESC("windowstart", 0x49, XT_REG_SPECIAL, 0),
258  XT_MK_REG_DESC("ps", XT_PS_REG_NUM, XT_REG_SPECIAL, 0), /* PS (not mapped through EPS[]) */
259  XT_MK_REG_DESC("ibreakenable", 0x60, XT_REG_SPECIAL, 0),
261  XT_MK_REG_DESC("ibreaka0", 0x80, XT_REG_SPECIAL, 0),
262  XT_MK_REG_DESC("ibreaka1", 0x81, XT_REG_SPECIAL, 0),
263  XT_MK_REG_DESC("dbreaka0", 0x90, XT_REG_SPECIAL, 0),
264  XT_MK_REG_DESC("dbreaka1", 0x91, XT_REG_SPECIAL, 0),
265  XT_MK_REG_DESC("dbreakc0", 0xA0, XT_REG_SPECIAL, 0),
266  XT_MK_REG_DESC("dbreakc1", 0xA1, XT_REG_SPECIAL, 0),
267  XT_MK_REG_DESC("cpenable", 0xE0, XT_REG_SPECIAL, 0),
268  XT_MK_REG_DESC("exccause", 0xE8, XT_REG_SPECIAL, 0),
269  XT_MK_REG_DESC("debugcause", 0xE9, XT_REG_SPECIAL, 0),
270  XT_MK_REG_DESC("icount", 0xEC, XT_REG_SPECIAL, 0),
271  XT_MK_REG_DESC("icountlevel", 0xED, XT_REG_SPECIAL, 0),
272 
273  /* WARNING: For these registers, regnum points to the
274  * index of the corresponding ARx registers, NOT to
275  * the processor register number! */
292 };
293 
305 };
306 
326 };
327 
328 /* Register definition as union for list allocation */
331  uint8_t buf[4];
332 };
333 
334 static const struct xtensa_keyval_info xt_qerr[XT_QERR_NUM] = {
335  { .chrval = "E00", .intval = ERROR_FAIL },
336  { .chrval = "E01", .intval = ERROR_FAIL },
337  { .chrval = "E02", .intval = ERROR_COMMAND_ARGUMENT_INVALID },
338  { .chrval = "E03", .intval = ERROR_FAIL },
339 };
340 
341 /* Set to true for extra debug logging */
342 static const bool xtensa_extra_debug_log;
343 
347 static inline const struct xtensa_local_mem_config *xtensa_get_mem_config(
348  struct xtensa *xtensa,
350 {
351  switch (type) {
352  case XTENSA_MEM_REG_IROM:
353  return &xtensa->core_config->irom;
354  case XTENSA_MEM_REG_IRAM:
355  return &xtensa->core_config->iram;
356  case XTENSA_MEM_REG_DROM:
357  return &xtensa->core_config->drom;
358  case XTENSA_MEM_REG_DRAM:
359  return &xtensa->core_config->dram;
360  case XTENSA_MEM_REG_SRAM:
361  return &xtensa->core_config->sram;
362  case XTENSA_MEM_REG_SROM:
363  return &xtensa->core_config->srom;
364  default:
365  return NULL;
366  }
367 }
368 
375  const struct xtensa_local_mem_config *mem,
376  target_addr_t address)
377 {
378  for (unsigned int i = 0; i < mem->count; i++) {
379  const struct xtensa_local_mem_region_config *region = &mem->regions[i];
380  if (address >= region->base && address < (region->base + region->size))
381  return region;
382  }
383  return NULL;
384 }
385 
392  struct xtensa *xtensa,
393  target_addr_t address)
394 {
395  const struct xtensa_local_mem_region_config *result;
396  const struct xtensa_local_mem_config *mcgf;
397  for (unsigned int mtype = 0; mtype < XTENSA_MEM_REGS_NUM; mtype++) {
398  mcgf = xtensa_get_mem_config(xtensa, mtype);
399  result = xtensa_memory_region_find(mcgf, address);
400  if (result)
401  return result;
402  }
403  return NULL;
404 }
405 
406 static inline bool xtensa_is_cacheable(const struct xtensa_cache_config *cache,
407  const struct xtensa_local_mem_config *mem,
408  target_addr_t address)
409 {
410  if (!cache->size)
411  return false;
412  return xtensa_memory_region_find(mem, address);
413 }
414 
415 static inline bool xtensa_is_icacheable(struct xtensa *xtensa, target_addr_t address)
416 {
421 }
422 
423 static inline bool xtensa_is_dcacheable(struct xtensa *xtensa, target_addr_t address)
424 {
429 }
430 
431 static int xtensa_core_reg_get(struct reg *reg)
432 {
433  /* We don't need this because we read all registers on halt anyway. */
434  struct xtensa *xtensa = (struct xtensa *)reg->arch_info;
435  struct target *target = xtensa->target;
436 
437  if (target->state != TARGET_HALTED)
439  if (!reg->exist) {
440  if (strncmp(reg->name, "?0x", 3) == 0) {
441  unsigned int regnum = strtoul(reg->name + 1, NULL, 0);
442  LOG_WARNING("Read unknown register 0x%04x ignored", regnum);
443  return ERROR_OK;
444  }
446  }
447  return ERROR_OK;
448 }
449 
450 static int xtensa_core_reg_set(struct reg *reg, uint8_t *buf)
451 {
452  struct xtensa *xtensa = (struct xtensa *)reg->arch_info;
453  struct target *target = xtensa->target;
454 
455  assert(reg->size <= 64 && "up to 64-bit regs are supported only!");
456  if (target->state != TARGET_HALTED)
458 
459  if (!reg->exist) {
460  if (strncmp(reg->name, "?0x", 3) == 0) {
461  unsigned int regnum = strtoul(reg->name + 1, NULL, 0);
462  LOG_WARNING("Write unknown register 0x%04x ignored", regnum);
463  return ERROR_OK;
464  }
466  }
467 
468  buf_cpy(buf, reg->value, reg->size);
469 
470  if (xtensa->core_config->windowed) {
471  /* If the user updates a potential scratch register, track for conflicts */
472  for (enum xtensa_ar_scratch_set_e s = 0; s < XT_AR_SCRATCH_NUM; s++) {
473  if (strcmp(reg->name, xtensa->scratch_ars[s].chrval) == 0) {
474  LOG_DEBUG("Scratch reg %s [0x%08" PRIx32 "] set from gdb", reg->name,
475  buf_get_u32(reg->value, 0, 32));
476  LOG_DEBUG("scratch_ars mapping: a3/%s, a4/%s",
479  xtensa->scratch_ars[s].intval = true;
480  break;
481  }
482  }
483  }
484  reg->dirty = true;
485  reg->valid = true;
486 
487  return ERROR_OK;
488 }
489 
490 static const struct reg_arch_type xtensa_reg_type = {
492  .set = xtensa_core_reg_set,
493 };
494 
495 /* Convert a register index that's indexed relative to windowbase, to the real address. */
497  enum xtensa_reg_id reg_idx,
498  int windowbase)
499 {
500  unsigned int idx;
501  if (reg_idx >= XT_REG_IDX_AR0 && reg_idx <= XT_REG_IDX_ARLAST) {
502  idx = reg_idx - XT_REG_IDX_AR0;
503  } else if (reg_idx >= XT_REG_IDX_A0 && reg_idx <= XT_REG_IDX_A15) {
504  idx = reg_idx - XT_REG_IDX_A0;
505  } else {
506  LOG_ERROR("Can't convert register %d to non-windowbased register", reg_idx);
507  return -1;
508  }
509  /* Each windowbase value represents 4 registers on LX and 8 on NX */
510  int base_inc = (xtensa->core_config->core_type == XT_LX) ? 4 : 8;
511  return ((idx + windowbase * base_inc) & (xtensa->core_config->aregs_num - 1)) + XT_REG_IDX_AR0;
512 }
513 
515  enum xtensa_reg_id reg_idx,
516  int windowbase)
517 {
518  return xtensa_windowbase_offset_to_canonical(xtensa, reg_idx, -windowbase);
519 }
520 
521 static void xtensa_mark_register_dirty(struct xtensa *xtensa, enum xtensa_reg_id reg_idx)
522 {
523  struct reg *reg_list = xtensa->core_cache->reg_list;
524  reg_list[reg_idx].dirty = true;
525 }
526 
527 static void xtensa_queue_exec_ins(struct xtensa *xtensa, uint32_t ins)
528 {
530 }
531 
532 static void xtensa_queue_exec_ins_wide(struct xtensa *xtensa, uint8_t *ops, uint8_t oplen)
533 {
534  const int max_oplen = 64; /* 8 DIRx regs: max width 64B */
535  if ((oplen > 0) && (oplen <= max_oplen)) {
536  uint8_t ops_padded[max_oplen];
537  memcpy(ops_padded, ops, oplen);
538  memset(ops_padded + oplen, 0, max_oplen - oplen);
539  unsigned int oplenw = DIV_ROUND_UP(oplen, sizeof(uint32_t));
540  for (int32_t i = oplenw - 1; i > 0; i--)
542  XDMREG_DIR0 + i,
543  target_buffer_get_u32(xtensa->target, &ops_padded[sizeof(uint32_t) * i]));
544  /* Write DIR0EXEC last */
547  target_buffer_get_u32(xtensa->target, &ops_padded[0]));
548  }
549 }
550 
551 /* NOTE: Assumes A3 has already been saved and marked dirty; A3 will be clobbered */
553 {
555  if (xtensa->core_config->mpu.enabled) {
556  /* For cores with the MPU option, issue PPTLB on start and end addresses.
557  * Parse access rights field, and confirm both have execute permissions.
558  */
559  for (int i = 0; i <= 1; i++) {
560  uint32_t at, acc;
561  uint8_t at_buf[4];
562  bool exec_acc;
563  target_addr_t addr = i ? end : start;
570  if (res != ERROR_OK)
571  LOG_TARGET_ERROR(target, "Error queuing PPTLB: %d", res);
573  if (res != ERROR_OK)
574  LOG_TARGET_ERROR(target, "Error issuing PPTLB: %d", res);
575  at = buf_get_u32(at_buf, 0, 32);
576  acc = (at >> XT_TLB1_ACC_SHIFT) & XT_TLB1_ACC_MSK;
577  exec_acc = ((acc == XTENSA_ACC_00X_000) || (acc == XTENSA_ACC_R0X_000) ||
578  (acc == XTENSA_ACC_RWX_000) || (acc == XTENSA_ACC_RWX_R0X) ||
579  (acc == XTENSA_ACC_R0X_R0X) || (acc == XTENSA_ACC_RWX_RWX));
580  LOG_TARGET_DEBUG(target, "PPTLB(" TARGET_ADDR_FMT ") -> 0x%08" PRIx32 " exec_acc %d",
581  addr, at, exec_acc);
582  if (!exec_acc)
583  return false;
584  }
585  }
586  return true;
587 }
588 
589 static int xtensa_queue_pwr_reg_write(struct xtensa *xtensa, unsigned int reg, uint32_t data)
590 {
591  struct xtensa_debug_module *dm = &xtensa->dbg_mod;
592  return dm->pwr_ops->queue_reg_write(dm, reg, data);
593 }
594 
595 /* NOTE: Assumes A3 has already been saved */
596 static int xtensa_window_state_save(struct target *target, uint32_t *woe)
597 {
599  unsigned int woe_sr = (xtensa->core_config->core_type == XT_LX) ? XT_SR_PS : XT_SR_WB;
600  uint32_t woe_dis;
601  uint8_t woe_buf[4];
602 
603  if (xtensa->core_config->windowed) {
604  /* Save PS (LX) or WB (NX) and disable window overflow exceptions prior to AR save */
609  if (res != ERROR_OK) {
610  LOG_TARGET_ERROR(target, "Failed to read %s (%d)!",
611  (woe_sr == XT_SR_PS) ? "PS" : "WB", res);
612  return res;
613  }
615  *woe = buf_get_u32(woe_buf, 0, 32);
616  woe_dis = *woe & ~((woe_sr == XT_SR_PS) ? XT_PS_WOE_MSK : XT_WB_S_MSK);
617  LOG_TARGET_DEBUG(target, "Clearing %s (0x%08" PRIx32 " -> 0x%08" PRIx32 ")",
618  (woe_sr == XT_SR_PS) ? "PS.WOE" : "WB.S", *woe, woe_dis);
622  }
623  return ERROR_OK;
624 }
625 
626 /* NOTE: Assumes A3 has already been saved */
627 static void xtensa_window_state_restore(struct target *target, uint32_t woe)
628 {
630  unsigned int woe_sr = (xtensa->core_config->core_type == XT_LX) ? XT_SR_PS : XT_SR_WB;
631  if (xtensa->core_config->windowed) {
632  /* Restore window overflow exception state */
636  LOG_TARGET_DEBUG(target, "Restored %s (0x%08" PRIx32 ")",
637  (woe_sr == XT_SR_PS) ? "PS.WOE" : "WB", woe);
638  }
639 }
640 
641 static bool xtensa_reg_is_readable(int flags, int cpenable)
642 {
643  if (flags & XT_REGF_NOREAD)
644  return false;
645  if ((flags & XT_REGF_COPROC0) && (cpenable & BIT(0)) == 0)
646  return false;
647  return true;
648 }
649 
650 static bool xtensa_scratch_regs_fixup(struct xtensa *xtensa, struct reg *reg_list, int i, int j, int a_idx, int ar_idx)
651 {
652  int a_name = (a_idx == XT_AR_SCRATCH_A3) ? 3 : 4;
653  if (xtensa->scratch_ars[a_idx].intval && !xtensa->scratch_ars[ar_idx].intval) {
654  LOG_DEBUG("AR conflict: a%d -> ar%d", a_name, j - XT_REG_IDX_AR0);
655  memcpy(reg_list[j].value, reg_list[i].value, sizeof(xtensa_reg_val_t));
656  } else {
657  LOG_DEBUG("AR conflict: ar%d -> a%d", j - XT_REG_IDX_AR0, a_name);
658  memcpy(reg_list[i].value, reg_list[j].value, sizeof(xtensa_reg_val_t));
659  }
660  return xtensa->scratch_ars[a_idx].intval && xtensa->scratch_ars[ar_idx].intval;
661 }
662 
664 {
666  int res;
667  xtensa_reg_val_t regval, windowbase = 0;
668  bool scratch_reg_dirty = false, delay_cpenable = false;
669  struct reg *reg_list = xtensa->core_cache->reg_list;
670  unsigned int reg_list_size = xtensa->core_cache->num_regs;
671  bool preserve_a3 = false;
672  uint8_t a3_buf[4];
673  xtensa_reg_val_t a3 = 0, woe;
674  unsigned int ms_idx = (xtensa->core_config->core_type == XT_NX) ?
675  xtensa->nx_reg_idx[XT_NX_REG_IDX_MS] : reg_list_size;
676  xtensa_reg_val_t ms = 0;
677  bool restore_ms = false;
678 
679  LOG_TARGET_DEBUG(target, "start");
680 
681  /* We need to write the dirty registers in the cache list back to the processor.
682  * Start by writing the SFR/user registers. */
683  for (unsigned int i = 0; i < reg_list_size; i++) {
684  struct xtensa_reg_desc *rlist = (i < XT_NUM_REGS) ? xtensa_regs : xtensa->optregs;
685  unsigned int ridx = (i < XT_NUM_REGS) ? i : i - XT_NUM_REGS;
686  if (reg_list[i].dirty) {
687  if (rlist[ridx].type == XT_REG_SPECIAL ||
688  rlist[ridx].type == XT_REG_USER ||
689  rlist[ridx].type == XT_REG_FR) {
690  scratch_reg_dirty = true;
691  if (i == XT_REG_IDX_CPENABLE) {
692  delay_cpenable = true;
693  continue;
694  }
695  regval = xtensa_reg_get(target, i);
696  LOG_TARGET_DEBUG(target, "Writing back reg %s (%d) val %08" PRIX32,
697  reg_list[i].name,
698  rlist[ridx].reg_num,
699  regval);
702  if (reg_list[i].exist) {
703  unsigned int reg_num = rlist[ridx].reg_num;
704  if (rlist[ridx].type == XT_REG_USER) {
706  } else if (rlist[ridx].type == XT_REG_FR) {
708  } else {/*SFR */
710  if (xtensa->core_config->core_type == XT_LX) {
711  /* reg number of PC for debug interrupt depends on NDEBUGLEVEL */
714  } else {
715  /* NX PC set through issuing a jump instruction */
717  }
718  } else if (i == ms_idx) {
719  /* MS must be restored after ARs. This ensures ARs remain in correct
720  * order even for reversed register groups (overflow/underflow).
721  */
722  ms = regval;
723  restore_ms = true;
724  LOG_TARGET_DEBUG(target, "Delaying MS write: 0x%x", ms);
725  } else {
727  }
728  }
729  }
730  reg_list[i].dirty = false;
731  }
732  }
733  }
734  if (scratch_reg_dirty)
736  if (delay_cpenable) {
738  LOG_TARGET_DEBUG(target, "Writing back reg cpenable (224) val %08" PRIX32, regval);
743  XT_REG_A3));
744  reg_list[XT_REG_IDX_CPENABLE].dirty = false;
745  }
746 
747  preserve_a3 = (xtensa->core_config->windowed) || (xtensa->core_config->core_type == XT_NX);
748  if (preserve_a3) {
749  /* Save (windowed) A3 for scratch use */
753  if (res != ERROR_OK)
754  return res;
756  a3 = buf_get_u32(a3_buf, 0, 32);
757  }
758 
759  if (xtensa->core_config->windowed) {
760  res = xtensa_window_state_save(target, &woe);
761  if (res != ERROR_OK)
762  return res;
763  /* Grab the windowbase, we need it. */
764  uint32_t wb_idx = (xtensa->core_config->core_type == XT_LX) ?
766  windowbase = xtensa_reg_get(target, wb_idx);
768  windowbase = (windowbase & XT_WB_P_MSK) >> XT_WB_P_SHIFT;
769 
770  /* Check if there are mismatches between the ARx and corresponding Ax registers.
771  * When the user sets a register on a windowed config, xt-gdb may set the ARx
772  * register directly. Thus we take ARx as priority over Ax if both are dirty
773  * and it's unclear if the user set one over the other explicitly.
774  */
775  for (unsigned int i = XT_REG_IDX_A0; i <= XT_REG_IDX_A15; i++) {
776  unsigned int j = xtensa_windowbase_offset_to_canonical(xtensa, i, windowbase);
777  if (reg_list[i].dirty && reg_list[j].dirty) {
778  if (memcmp(reg_list[i].value, reg_list[j].value, sizeof(xtensa_reg_val_t)) != 0) {
779  bool show_warning = true;
780  if (i == XT_REG_IDX_A3)
781  show_warning = xtensa_scratch_regs_fixup(xtensa,
782  reg_list, i, j, XT_AR_SCRATCH_A3, XT_AR_SCRATCH_AR3);
783  else if (i == XT_REG_IDX_A4)
784  show_warning = xtensa_scratch_regs_fixup(xtensa,
785  reg_list, i, j, XT_AR_SCRATCH_A4, XT_AR_SCRATCH_AR4);
786  if (show_warning)
787  LOG_WARNING(
788  "Warning: Both A%d [0x%08" PRIx32
789  "] as well as its underlying physical register "
790  "(AR%d) [0x%08" PRIx32 "] are dirty and differ in value",
791  i - XT_REG_IDX_A0,
792  buf_get_u32(reg_list[i].value, 0, 32),
793  j - XT_REG_IDX_AR0,
794  buf_get_u32(reg_list[j].value, 0, 32));
795  }
796  }
797  }
798  }
799 
800  /* Write A0-A16. */
801  for (unsigned int i = 0; i < 16; i++) {
802  if (reg_list[XT_REG_IDX_A0 + i].dirty) {
803  regval = xtensa_reg_get(target, XT_REG_IDX_A0 + i);
804  LOG_TARGET_DEBUG(target, "Writing back reg %s value %08" PRIX32 ", num =%i",
806  regval,
810  reg_list[XT_REG_IDX_A0 + i].dirty = false;
811  if (i == 3) {
812  /* Avoid stomping A3 during restore at end of function */
813  a3 = regval;
814  }
815  }
816  }
817 
818  if (xtensa->core_config->windowed) {
819  /* Now write AR registers */
820  for (unsigned int j = 0; j < XT_REG_IDX_ARLAST; j += 16) {
821  /* Write the 16 registers we can see */
822  for (unsigned int i = 0; i < 16; i++) {
823  if (i + j < xtensa->core_config->aregs_num) {
824  enum xtensa_reg_id realadr =
826  windowbase);
827  /* Write back any dirty un-windowed registers */
828  if (reg_list[realadr].dirty) {
829  regval = xtensa_reg_get(target, realadr);
831  target,
832  "Writing back reg %s value %08" PRIX32 ", num =%i",
833  xtensa_regs[realadr].name,
834  regval,
835  xtensa_regs[realadr].reg_num);
840  reg_list[realadr].dirty = false;
841  if ((i + j) == 3)
842  /* Avoid stomping AR during A3 restore at end of function */
843  a3 = regval;
844  }
845  }
846  }
847 
848  /* Now rotate the window so we'll see the next 16 registers. The final rotate
849  * will wraparound, leaving us in the state we were.
850  * Each ROTW rotates 4 registers on LX and 8 on NX */
851  int rotw_arg = (xtensa->core_config->core_type == XT_LX) ? 4 : 2;
853  }
854 
856 
857  for (enum xtensa_ar_scratch_set_e s = 0; s < XT_AR_SCRATCH_NUM; s++)
858  xtensa->scratch_ars[s].intval = false;
859  }
860 
861  if (restore_ms) {
862  uint32_t ms_regno = xtensa->optregs[ms_idx - XT_NUM_REGS].reg_num;
866  LOG_TARGET_DEBUG(target, "Delayed MS (0x%x) write complete: 0x%x", ms_regno, ms);
867  }
868 
869  if (preserve_a3) {
872  }
873 
876 
877  return res;
878 }
879 
880 static inline bool xtensa_is_stopped(struct target *target)
881 {
884 }
885 
887 {
890 
892 
894  LOG_ERROR("XTensa core not configured; is xtensa-core-openocd.cfg missing?");
895  return ERROR_FAIL;
896  }
897 
903  if (res != ERROR_OK)
904  return res;
906  LOG_ERROR("Unexpected OCD_ID = %08" PRIx32, xtensa->dbg_mod.device_id);
907  return ERROR_TARGET_FAILURE;
908  }
909  LOG_DEBUG("OCD_ID = %08" PRIx32, xtensa->dbg_mod.device_id);
912  return ERROR_OK;
913 }
914 
916 {
919 
920  if (xtensa->reset_asserted)
923  /* TODO: can we join this with the write above? */
927 }
928 
929 int xtensa_smpbreak_write(struct xtensa *xtensa, uint32_t set)
930 {
931  uint32_t dsr_data = 0x00110000;
932  uint32_t clear = (set | OCDDCR_ENABLEOCD) ^
935 
936  LOG_TARGET_DEBUG(xtensa->target, "write smpbreak set=0x%" PRIx32 " clear=0x%" PRIx32, set, clear);
942 }
943 
944 int xtensa_smpbreak_set(struct target *target, uint32_t set)
945 {
947  int res = ERROR_OK;
948 
949  xtensa->smp_break = set;
952  LOG_TARGET_DEBUG(target, "set smpbreak=%" PRIx32 ", state=%i", set, target->state);
953  return res;
954 }
955 
956 int xtensa_smpbreak_read(struct xtensa *xtensa, uint32_t *val)
957 {
958  uint8_t dcr_buf[sizeof(uint32_t)];
959 
963  *val = buf_get_u32(dcr_buf, 0, 32);
964 
965  return res;
966 }
967 
968 int xtensa_smpbreak_get(struct target *target, uint32_t *val)
969 {
971  *val = xtensa->smp_break;
972  return ERROR_OK;
973 }
974 
976 {
977  return buf_get_u32(reg->value, 0, 32);
978 }
979 
980 static inline void xtensa_reg_set_value(struct reg *reg, xtensa_reg_val_t value)
981 {
982  buf_set_u32(reg->value, 0, 32, value);
983  reg->dirty = true;
984 }
985 
987 {
989  for (enum xtensa_nx_reg_idx idx = XT_NX_REG_IDX_IEVEC; idx <= XT_NX_REG_IDX_MESR; idx++) {
990  enum xtensa_reg_id ridx = xtensa->nx_reg_idx[idx];
991  if (xtensa->nx_reg_idx[idx]) {
993  if (reg & XT_IMPR_EXC_MSK) {
994  LOG_TARGET_DEBUG(target, "Imprecise exception: %s: 0x%x",
995  xtensa->core_cache->reg_list[ridx].name, reg);
996  return true;
997  }
998  }
999  }
1000  return false;
1001 }
1002 
1004 {
1005  struct xtensa *xtensa = target_to_xtensa(target);
1006  for (enum xtensa_nx_reg_idx idx = XT_NX_REG_IDX_IEVEC; idx <= XT_NX_REG_IDX_MESRCLR; idx++) {
1007  enum xtensa_reg_id ridx = xtensa->nx_reg_idx[idx];
1008  if (ridx && idx != XT_NX_REG_IDX_MESR) {
1010  xtensa_reg_set(target, ridx, value);
1011  LOG_TARGET_DEBUG(target, "Imprecise exception: clearing %s (0x%x)",
1012  xtensa->core_cache->reg_list[ridx].name, value);
1013  }
1014  }
1015 }
1016 
1018 {
1019  struct xtensa *xtensa = target_to_xtensa(target);
1020  int res, needclear = 0, needimprclear = 0;
1021 
1024  LOG_TARGET_DEBUG(target, "DSR (%08" PRIX32 ")", dsr);
1025  if (dsr & OCDDSR_EXECBUSY) {
1027  LOG_TARGET_ERROR(target, "DSR (%08" PRIX32 ") indicates target still busy!", dsr);
1028  needclear = 1;
1029  }
1030  if (dsr & OCDDSR_EXECEXCEPTION) {
1033  "DSR (%08" PRIX32 ") indicates DIR instruction generated an exception!",
1034  dsr);
1035  needclear = 1;
1036  }
1037  if (dsr & OCDDSR_EXECOVERRUN) {
1040  "DSR (%08" PRIX32 ") indicates DIR instruction generated an overrun!",
1041  dsr);
1042  needclear = 1;
1043  }
1047  "%s: Imprecise exception occurred!", target_name(target));
1048  needclear = 1;
1049  needimprclear = 1;
1050  }
1051  if (needclear) {
1054  if (res != ERROR_OK && !xtensa->suppress_dsr_errors)
1055  LOG_TARGET_ERROR(target, "clearing DSR failed!");
1056  if (xtensa->core_config->core_type == XT_NX && needimprclear)
1058  return ERROR_FAIL;
1059  }
1060  return ERROR_OK;
1061 }
1062 
1064 {
1065  struct xtensa *xtensa = target_to_xtensa(target);
1066  struct reg *reg = &xtensa->core_cache->reg_list[reg_id];
1067  return xtensa_reg_get_value(reg);
1068 }
1069 
1071 {
1072  struct xtensa *xtensa = target_to_xtensa(target);
1073  struct reg *reg = &xtensa->core_cache->reg_list[reg_id];
1074  if (xtensa_reg_get_value(reg) == value)
1075  return;
1077 }
1078 
1079 /* Set Ax (XT_REG_RELGEN) register along with its underlying ARx (XT_REG_GENERAL) */
1081 {
1082  struct xtensa *xtensa = target_to_xtensa(target);
1083  uint32_t wb_idx = (xtensa->core_config->core_type == XT_LX) ?
1085  uint32_t windowbase = (xtensa->core_config->windowed ?
1086  xtensa_reg_get(target, wb_idx) : 0);
1087  if (xtensa->core_config->core_type == XT_NX)
1088  windowbase = (windowbase & XT_WB_P_MSK) >> XT_WB_P_SHIFT;
1089  int ar_idx = xtensa_windowbase_offset_to_canonical(xtensa, a_idx, windowbase);
1090  xtensa_reg_set(target, a_idx, value);
1091  xtensa_reg_set(target, ar_idx, value);
1092 }
1093 
1094 /* Read cause for entering halted state; return bitmask in DEBUGCAUSE_* format */
1095 uint32_t xtensa_cause_get(struct target *target)
1096 {
1097  struct xtensa *xtensa = target_to_xtensa(target);
1098  if (xtensa->core_config->core_type == XT_LX) {
1099  /* LX cause in DEBUGCAUSE */
1101  }
1103  return xtensa->nx_stop_cause;
1104 
1105  /* NX cause determined from DSR.StopCause */
1107  LOG_TARGET_ERROR(target, "Read DSR error");
1108  } else {
1109  uint32_t dsr = xtensa_dm_core_status_get(&xtensa->dbg_mod);
1110  /* NX causes are prioritized; only 1 bit can be set */
1111  switch ((dsr & OCDDSR_STOPCAUSE) >> OCDDSR_STOPCAUSE_SHIFT) {
1112  case OCDDSR_STOPCAUSE_DI:
1114  break;
1115  case OCDDSR_STOPCAUSE_SS:
1117  break;
1118  case OCDDSR_STOPCAUSE_IB:
1120  break;
1121  case OCDDSR_STOPCAUSE_B:
1122  case OCDDSR_STOPCAUSE_B1:
1124  break;
1125  case OCDDSR_STOPCAUSE_BN:
1127  break;
1128  case OCDDSR_STOPCAUSE_DB0:
1129  case OCDDSR_STOPCAUSE_DB1:
1131  break;
1132  default:
1133  LOG_TARGET_ERROR(target, "Unknown stop cause (DSR: 0x%08x)", dsr);
1134  break;
1135  }
1136  if (xtensa->nx_stop_cause)
1138  }
1139  return xtensa->nx_stop_cause;
1140 }
1141 
1143 {
1144  struct xtensa *xtensa = target_to_xtensa(target);
1145  if (xtensa->core_config->core_type == XT_LX) {
1148  } else {
1149  /* NX DSR.STOPCAUSE is not writeable; clear cached copy but leave it valid */
1151  }
1152 }
1153 
1155 {
1156  /* Clear DEBUGCAUSE_VALID to trigger re-read (on NX) */
1157  struct xtensa *xtensa = target_to_xtensa(target);
1158  xtensa->nx_stop_cause = 0;
1159 }
1160 
1162 {
1163  struct xtensa *xtensa = target_to_xtensa(target);
1164 
1165  LOG_TARGET_DEBUG(target, " begin");
1167  XDMREG_PWRCTL,
1171  int res = xtensa_dm_queue_execute(&xtensa->dbg_mod);
1172  if (res != ERROR_OK)
1173  return res;
1174 
1175  /* registers are now invalid */
1176  xtensa->reset_asserted = true;
1179  return ERROR_OK;
1180 }
1181 
1183 {
1184  struct xtensa *xtensa = target_to_xtensa(target);
1185 
1186  LOG_TARGET_DEBUG(target, "halt=%d", target->reset_halt);
1187  if (target->reset_halt)
1189  XDMREG_DCRSET,
1192  XDMREG_PWRCTL,
1196  int res = xtensa_dm_queue_execute(&xtensa->dbg_mod);
1197  if (res != ERROR_OK)
1198  return res;
1200  xtensa->reset_asserted = false;
1201  return res;
1202 }
1203 
1205 {
1206  LOG_TARGET_DEBUG(target, "begin");
1207  return xtensa_assert_reset(target);
1208 }
1209 
1211 {
1212  struct xtensa *xtensa = target_to_xtensa(target);
1213  struct reg *reg_list = xtensa->core_cache->reg_list;
1214  unsigned int reg_list_size = xtensa->core_cache->num_regs;
1215  xtensa_reg_val_t cpenable = 0, windowbase = 0, a0 = 0, a3;
1216  unsigned int ms_idx = reg_list_size;
1217  uint32_t ms = 0;
1218  uint32_t woe;
1219  uint8_t a0_buf[4], a3_buf[4], ms_buf[4];
1220  bool debug_dsrs = !xtensa->regs_fetched || LOG_LEVEL_IS(LOG_LVL_DEBUG);
1221 
1222  union xtensa_reg_val_u *regvals = calloc(reg_list_size, sizeof(*regvals));
1223  if (!regvals) {
1224  LOG_TARGET_ERROR(target, "unable to allocate memory for regvals!");
1225  return ERROR_FAIL;
1226  }
1227  union xtensa_reg_val_u *dsrs = calloc(reg_list_size, sizeof(*dsrs));
1228  if (!dsrs) {
1229  LOG_TARGET_ERROR(target, "unable to allocate memory for dsrs!");
1230  free(regvals);
1231  return ERROR_FAIL;
1232  }
1233 
1234  LOG_TARGET_DEBUG(target, "start");
1235 
1236  /* Save (windowed) A3 so cache matches physical AR3; A3 usable as scratch */
1239  if (xtensa->core_config->core_type == XT_NX) {
1240  /* Save (windowed) A0 as well--it will be required for reading PC */
1243 
1244  /* Set MS.DispSt, clear MS.DE prior to accessing ARs. This ensures ARs remain
1245  * in correct order even for reversed register groups (overflow/underflow).
1246  */
1247  ms_idx = xtensa->nx_reg_idx[XT_NX_REG_IDX_MS];
1248  uint32_t ms_regno = xtensa->optregs[ms_idx - XT_NUM_REGS].reg_num;
1252  LOG_TARGET_DEBUG(target, "Overriding MS (0x%x): 0x%x", ms_regno, XT_MS_DISPST_DBG);
1256  }
1257 
1258  int res = xtensa_window_state_save(target, &woe);
1259  if (res != ERROR_OK)
1260  goto xtensa_fetch_all_regs_done;
1261 
1262  /* Assume the CPU has just halted. We now want to fill the register cache with all the
1263  * register contents GDB needs. For speed, we pipeline all the read operations, execute them
1264  * in one go, then sort everything out from the regvals variable. */
1265 
1266  /* Start out with AREGS; we can reach those immediately. Grab them per 16 registers. */
1267  for (unsigned int j = 0; j < XT_AREGS_NUM_MAX; j += 16) {
1268  /*Grab the 16 registers we can see */
1269  for (unsigned int i = 0; i < 16; i++) {
1270  if (i + j < xtensa->core_config->aregs_num) {
1274  regvals[XT_REG_IDX_AR0 + i + j].buf);
1275  if (debug_dsrs)
1277  dsrs[XT_REG_IDX_AR0 + i + j].buf);
1278  }
1279  }
1280  if (xtensa->core_config->windowed) {
1281  /* Now rotate the window so we'll see the next 16 registers. The final rotate
1282  * will wraparound, leaving us in the state we were.
1283  * Each ROTW rotates 4 registers on LX and 8 on NX */
1284  int rotw_arg = (xtensa->core_config->core_type == XT_LX) ? 4 : 2;
1286  }
1287  }
1289 
1290  if (xtensa->core_config->coproc) {
1291  /* As the very first thing after AREGS, go grab CPENABLE */
1295  }
1297  if (res != ERROR_OK) {
1298  LOG_ERROR("Failed to read ARs (%d)!", res);
1299  goto xtensa_fetch_all_regs_done;
1300  }
1302 
1303  a3 = buf_get_u32(a3_buf, 0, 32);
1304  if (xtensa->core_config->core_type == XT_NX) {
1305  a0 = buf_get_u32(a0_buf, 0, 32);
1306  ms = buf_get_u32(ms_buf, 0, 32);
1307  }
1308 
1309  if (xtensa->core_config->coproc) {
1310  cpenable = buf_get_u32(regvals[XT_REG_IDX_CPENABLE].buf, 0, 32);
1311 
1312  /* Enable all coprocessors (by setting all bits in CPENABLE) so we can read FP and user registers. */
1316 
1317  /* Save CPENABLE; flag dirty later (when regcache updated) so original value is always restored */
1318  LOG_TARGET_DEBUG(target, "CPENABLE: was 0x%" PRIx32 ", all enabled", cpenable);
1320  }
1321  /* We're now free to use any of A0-A15 as scratch registers
1322  * Grab the SFRs and user registers first. We use A3 as a scratch register. */
1323  for (unsigned int i = 0; i < reg_list_size; i++) {
1324  struct xtensa_reg_desc *rlist = (i < XT_NUM_REGS) ? xtensa_regs : xtensa->optregs;
1325  unsigned int ridx = (i < XT_NUM_REGS) ? i : i - XT_NUM_REGS;
1326  if (xtensa_reg_is_readable(rlist[ridx].flags, cpenable) && rlist[ridx].exist) {
1327  bool reg_fetched = true;
1328  unsigned int reg_num = rlist[ridx].reg_num;
1329  switch (rlist[ridx].type) {
1330  case XT_REG_USER:
1332  break;
1333  case XT_REG_FR:
1335  break;
1336  case XT_REG_SPECIAL:
1337  if (reg_num == XT_PC_REG_NUM_VIRTUAL) {
1338  if (xtensa->core_config->core_type == XT_LX) {
1339  /* reg number of PC for debug interrupt depends on NDEBUGLEVEL */
1342  } else {
1343  /* NX PC read through CALL0(0) and reading A0 */
1346  xtensa_queue_dbg_reg_read(xtensa, XDMREG_DDR, regvals[i].buf);
1348  reg_fetched = false;
1349  }
1350  } else if ((xtensa->core_config->core_type == XT_LX)
1352  /* reg number of PS for debug interrupt depends on NDEBUGLEVEL */
1356  /* CPENABLE already read/updated; don't re-read */
1357  reg_fetched = false;
1358  break;
1359  } else {
1361  }
1362  break;
1363  default:
1364  reg_fetched = false;
1365  }
1366  if (reg_fetched) {
1368  xtensa_queue_dbg_reg_read(xtensa, XDMREG_DDR, regvals[i].buf);
1369  if (debug_dsrs)
1371  }
1372  }
1373  }
1374  /* Ok, send the whole mess to the CPU. */
1376  if (res != ERROR_OK) {
1377  LOG_ERROR("Failed to fetch AR regs!");
1378  goto xtensa_fetch_all_regs_done;
1379  }
1381 
1382  if (debug_dsrs) {
1383  /* DSR checking: follows order in which registers are requested. */
1384  for (unsigned int i = 0; i < reg_list_size; i++) {
1385  struct xtensa_reg_desc *rlist = (i < XT_NUM_REGS) ? xtensa_regs : xtensa->optregs;
1386  unsigned int ridx = (i < XT_NUM_REGS) ? i : i - XT_NUM_REGS;
1387  if (xtensa_reg_is_readable(rlist[ridx].flags, cpenable) && rlist[ridx].exist &&
1388  (rlist[ridx].type != XT_REG_DEBUG) &&
1389  (rlist[ridx].type != XT_REG_RELGEN) &&
1390  (rlist[ridx].type != XT_REG_TIE) &&
1391  (rlist[ridx].type != XT_REG_OTHER)) {
1392  if (buf_get_u32(dsrs[i].buf, 0, 32) & OCDDSR_EXECEXCEPTION) {
1393  LOG_ERROR("Exception reading %s!", reg_list[i].name);
1394  res = ERROR_FAIL;
1395  goto xtensa_fetch_all_regs_done;
1396  }
1397  }
1398  }
1399  }
1400 
1401  if (xtensa->core_config->windowed) {
1402  /* We need the windowbase to decode the general addresses. */
1403  uint32_t wb_idx = (xtensa->core_config->core_type == XT_LX) ?
1405  windowbase = buf_get_u32(regvals[wb_idx].buf, 0, 32);
1406  if (xtensa->core_config->core_type == XT_NX)
1407  windowbase = (windowbase & XT_WB_P_MSK) >> XT_WB_P_SHIFT;
1408  }
1409 
1410  /* Decode the result and update the cache. */
1411  for (unsigned int i = 0; i < reg_list_size; i++) {
1412  struct xtensa_reg_desc *rlist = (i < XT_NUM_REGS) ? xtensa_regs : xtensa->optregs;
1413  unsigned int ridx = (i < XT_NUM_REGS) ? i : i - XT_NUM_REGS;
1414  if (xtensa_reg_is_readable(rlist[ridx].flags, cpenable) && rlist[ridx].exist) {
1415  if ((xtensa->core_config->windowed) && (rlist[ridx].type == XT_REG_GENERAL)) {
1416  /* The 64-value general register set is read from (windowbase) on down.
1417  * We need to get the real register address by subtracting windowbase and
1418  * wrapping around. */
1420  windowbase);
1421  buf_cpy(regvals[realadr].buf, reg_list[i].value, reg_list[i].size);
1422  } else if (rlist[ridx].type == XT_REG_RELGEN) {
1423  buf_cpy(regvals[rlist[ridx].reg_num].buf, reg_list[i].value, reg_list[i].size);
1424  if (xtensa_extra_debug_log) {
1425  xtensa_reg_val_t regval = buf_get_u32(regvals[rlist[ridx].reg_num].buf, 0, 32);
1426  LOG_DEBUG("%s = 0x%x", rlist[ridx].name, regval);
1427  }
1428  } else {
1429  xtensa_reg_val_t regval = buf_get_u32(regvals[i].buf, 0, 32);
1430  bool is_dirty = (i == XT_REG_IDX_CPENABLE);
1432  LOG_INFO("Register %s: 0x%X", reg_list[i].name, regval);
1433  if (rlist[ridx].reg_num == XT_PC_REG_NUM_VIRTUAL &&
1435  /* A0 from prior CALL0 points to next instruction; decrement it */
1436  regval -= 3;
1437  is_dirty = 1;
1438  } else if (i == ms_idx) {
1439  LOG_TARGET_DEBUG(target, "Caching MS: 0x%x", ms);
1440  regval = ms;
1441  is_dirty = 1;
1442  }
1443  xtensa_reg_set(target, i, regval);
1444  reg_list[i].dirty = is_dirty; /*always do this _after_ xtensa_reg_set! */
1445  }
1446  reg_list[i].valid = true;
1447  } else {
1448  if ((rlist[ridx].flags & XT_REGF_MASK) == XT_REGF_NOREAD) {
1449  /* Report read-only registers all-zero but valid */
1450  reg_list[i].valid = true;
1451  xtensa_reg_set(target, i, 0);
1452  } else {
1453  reg_list[i].valid = false;
1454  }
1455  }
1456  }
1457 
1458  if (xtensa->core_config->windowed) {
1459  /* We have used A3 as a scratch register.
1460  * Windowed configs: restore A3's AR (XT_REG_GENERAL) and and flag for write-back.
1461  */
1463  xtensa_reg_set(target, ar3_idx, a3);
1465 
1466  /* Reset scratch_ars[] on fetch. .chrval tracks AR mapping and changes w/ window */
1467  sprintf(xtensa->scratch_ars[XT_AR_SCRATCH_AR3].chrval, "ar%d", ar3_idx - XT_REG_IDX_AR0);
1469  sprintf(xtensa->scratch_ars[XT_AR_SCRATCH_AR4].chrval, "ar%d", ar4_idx - XT_REG_IDX_AR0);
1470  for (enum xtensa_ar_scratch_set_e s = 0; s < XT_AR_SCRATCH_NUM; s++)
1471  xtensa->scratch_ars[s].intval = false;
1472  }
1473 
1474  /* We have used A3 (XT_REG_RELGEN) as a scratch register. Restore and flag for write-back. */
1477  if (xtensa->core_config->core_type == XT_NX) {
1480  }
1481 
1482  xtensa->regs_fetched = true;
1483 xtensa_fetch_all_regs_done:
1484  free(regvals);
1485  free(dsrs);
1486  return res;
1487 }
1488 
1490  struct reg **reg_list[],
1491  int *reg_list_size,
1492  enum target_register_class reg_class)
1493 {
1494  struct xtensa *xtensa = target_to_xtensa(target);
1495  unsigned int num_regs;
1496 
1497  if (reg_class == REG_CLASS_GENERAL) {
1499  LOG_ERROR("reg_class %d unhandled; 'xtgregs' not found", reg_class);
1500  return ERROR_FAIL;
1501  }
1502  num_regs = xtensa->genpkt_regs_num;
1503  } else {
1504  /* Determine whether to return a contiguous or sparse register map */
1506  }
1507 
1508  LOG_DEBUG("reg_class=%i, num_regs=%d", (int)reg_class, num_regs);
1509 
1510  *reg_list = calloc(num_regs, sizeof(struct reg *));
1511  if (!*reg_list)
1512  return ERROR_FAIL;
1513 
1514  *reg_list_size = num_regs;
1515  if (xtensa->regmap_contiguous) {
1516  assert((num_regs <= xtensa->total_regs_num) && "contiguous regmap size internal error!");
1517  for (unsigned int i = 0; i < num_regs; i++)
1518  (*reg_list)[i] = xtensa->contiguous_regs_list[i];
1519  return ERROR_OK;
1520  }
1521 
1522  for (unsigned int i = 0; i < num_regs; i++)
1523  (*reg_list)[i] = (struct reg *)&xtensa->empty_regs[i];
1524  unsigned int k = 0;
1525  for (unsigned int i = 0; i < xtensa->core_cache->num_regs && k < num_regs; i++) {
1526  if (xtensa->core_cache->reg_list[i].exist) {
1527  struct xtensa_reg_desc *rlist = (i < XT_NUM_REGS) ? xtensa_regs : xtensa->optregs;
1528  unsigned int ridx = (i < XT_NUM_REGS) ? i : i - XT_NUM_REGS;
1529  int sparse_idx = rlist[ridx].dbreg_num;
1530  if (i == XT_REG_IDX_PS && xtensa->core_config->core_type == XT_LX) {
1531  if (xtensa->eps_dbglevel_idx == 0) {
1532  LOG_ERROR("eps_dbglevel_idx not set\n");
1533  return ERROR_FAIL;
1534  }
1535  (*reg_list)[sparse_idx] = &xtensa->core_cache->reg_list[xtensa->eps_dbglevel_idx];
1537  LOG_DEBUG("SPARSE GDB reg 0x%x getting EPS%d 0x%x",
1538  sparse_idx, xtensa->core_config->debug.irq_level,
1539  xtensa_reg_get_value((*reg_list)[sparse_idx]));
1540  } else if (rlist[ridx].type == XT_REG_RELGEN) {
1541  (*reg_list)[sparse_idx - XT_REG_IDX_ARFIRST] = &xtensa->core_cache->reg_list[i];
1542  } else {
1543  (*reg_list)[sparse_idx] = &xtensa->core_cache->reg_list[i];
1544  }
1545  if (i == XT_REG_IDX_PC)
1546  /* Make a duplicate copy of PC for external access */
1547  (*reg_list)[XT_PC_DBREG_NUM_BASE] = &xtensa->core_cache->reg_list[i];
1548  k++;
1549  }
1550  }
1551 
1552  if (k == num_regs)
1553  LOG_ERROR("SPARSE GDB reg list full (size %d)", k);
1554 
1555  return ERROR_OK;
1556 }
1557 
1558 int xtensa_mmu_is_enabled(struct target *target, int *enabled)
1559 {
1560  struct xtensa *xtensa = target_to_xtensa(target);
1561  *enabled = xtensa->core_config->mmu.itlb_entries_count > 0 ||
1563  return ERROR_OK;
1564 }
1565 
1567 {
1568  struct xtensa *xtensa = target_to_xtensa(target);
1569 
1570  LOG_TARGET_DEBUG(target, "start");
1571  if (target->state == TARGET_HALTED) {
1572  LOG_TARGET_DEBUG(target, "target was already halted");
1573  return ERROR_OK;
1574  }
1575  /* First we have to read dsr and check if the target stopped */
1577  if (res != ERROR_OK) {
1578  LOG_TARGET_ERROR(target, "Failed to read core status!");
1579  return res;
1580  }
1581  LOG_TARGET_DEBUG(target, "Core status 0x%" PRIx32, xtensa_dm_core_status_get(&xtensa->dbg_mod));
1582  if (!xtensa_is_stopped(target)) {
1586  if (res != ERROR_OK)
1587  LOG_TARGET_ERROR(target, "Failed to set OCDDCR_DEBUGINTERRUPT. Can't halt.");
1588  }
1589 
1590  return res;
1591 }
1592 
1594  int current,
1595  target_addr_t address,
1596  int handle_breakpoints,
1597  int debug_execution)
1598 {
1599  struct xtensa *xtensa = target_to_xtensa(target);
1600  uint32_t bpena = 0;
1601 
1603  "current=%d address=" TARGET_ADDR_FMT ", handle_breakpoints=%i, debug_execution=%i)",
1604  current,
1605  address,
1606  handle_breakpoints,
1607  debug_execution);
1608 
1609  if (target->state != TARGET_HALTED) {
1610  LOG_TARGET_ERROR(target, "not halted");
1611  return ERROR_TARGET_NOT_HALTED;
1612  }
1613  xtensa->halt_request = false;
1614 
1615  if (address && !current) {
1616  xtensa_reg_set(target, XT_REG_IDX_PC, address);
1617  } else {
1618  uint32_t cause = xtensa_cause_get(target);
1619  LOG_TARGET_DEBUG(target, "DEBUGCAUSE 0x%x (watchpoint %lu) (break %lu)",
1620  cause, (cause & DEBUGCAUSE_DB), (cause & (DEBUGCAUSE_BI | DEBUGCAUSE_BN)));
1621  if (cause & DEBUGCAUSE_DB)
1622  /* We stopped due to a watchpoint. We can't just resume executing the
1623  * instruction again because */
1624  /* that would trigger the watchpoint again. To fix this, we single-step,
1625  * which ignores watchpoints. */
1626  xtensa_do_step(target, current, address, handle_breakpoints);
1627  if (cause & (DEBUGCAUSE_BI | DEBUGCAUSE_BN))
1628  /* We stopped due to a break instruction. We can't just resume executing the
1629  * instruction again because */
1630  /* that would trigger the break again. To fix this, we single-step, which
1631  * ignores break. */
1632  xtensa_do_step(target, current, address, handle_breakpoints);
1633  }
1634 
1635  /* Write back hw breakpoints. Current FreeRTOS SMP code can set a hw breakpoint on an
1636  * exception; we need to clear that and return to the breakpoints gdb has set on resume. */
1637  for (unsigned int slot = 0; slot < xtensa->core_config->debug.ibreaks_num; slot++) {
1638  if (xtensa->hw_brps[slot]) {
1639  /* Write IBREAKA[slot] and set bit #slot in IBREAKENABLE */
1641  if (xtensa->core_config->core_type == XT_NX)
1643  bpena |= BIT(slot);
1644  }
1645  }
1646  if (xtensa->core_config->core_type == XT_LX)
1648 
1649  /* Here we write all registers to the targets */
1651  if (res != ERROR_OK)
1652  LOG_TARGET_ERROR(target, "Failed to write back register cache.");
1653  return res;
1654 }
1655 
1657 {
1658  struct xtensa *xtensa = target_to_xtensa(target);
1659 
1660  LOG_TARGET_DEBUG(target, "start");
1661 
1664  int res = xtensa_dm_queue_execute(&xtensa->dbg_mod);
1665  if (res != ERROR_OK) {
1666  LOG_TARGET_ERROR(target, "Failed to exec RFDO %d!", res);
1667  return res;
1668  }
1670  return ERROR_OK;
1671 }
1672 
1674  int current,
1675  target_addr_t address,
1676  int handle_breakpoints,
1677  int debug_execution)
1678 {
1679  LOG_TARGET_DEBUG(target, "start");
1680  int res = xtensa_prepare_resume(target, current, address, handle_breakpoints, debug_execution);
1681  if (res != ERROR_OK) {
1682  LOG_TARGET_ERROR(target, "Failed to prepare for resume!");
1683  return res;
1684  }
1685  res = xtensa_do_resume(target);
1686  if (res != ERROR_OK) {
1687  LOG_TARGET_ERROR(target, "Failed to resume!");
1688  return res;
1689  }
1690 
1692  if (!debug_execution)
1694  else
1696 
1698 
1699  return ERROR_OK;
1700 }
1701 
1703 {
1704  struct xtensa *xtensa = target_to_xtensa(target);
1705  uint8_t insn_buf[XT_ISNS_SZ_MAX];
1706  int err = xtensa_read_buffer(target, pc, sizeof(insn_buf), insn_buf);
1707  if (err != ERROR_OK)
1708  return false;
1709 
1710  xtensa_insn_t insn = buf_get_u32(insn_buf, 0, 24);
1711  xtensa_insn_t masked = insn & XT_INS_L32E_S32E_MASK(xtensa);
1712  if (masked == XT_INS_L32E(xtensa, 0, 0, 0) || masked == XT_INS_S32E(xtensa, 0, 0, 0))
1713  return true;
1714 
1715  masked = insn & XT_INS_RFWO_RFWU_MASK(xtensa);
1716  if (masked == XT_INS_RFWO(xtensa) || masked == XT_INS_RFWU(xtensa))
1717  return true;
1718 
1719  return false;
1720 }
1721 
1722 int xtensa_do_step(struct target *target, int current, target_addr_t address, int handle_breakpoints)
1723 {
1724  struct xtensa *xtensa = target_to_xtensa(target);
1725  int res;
1726  const uint32_t icount_val = -2; /* ICOUNT value to load for 1 step */
1728  xtensa_reg_val_t icountlvl, cause;
1729  xtensa_reg_val_t oldps, oldpc, cur_pc;
1730  bool ps_modified = false;
1731 
1732  LOG_TARGET_DEBUG(target, "current=%d, address=" TARGET_ADDR_FMT ", handle_breakpoints=%i",
1733  current, address, handle_breakpoints);
1734 
1735  if (target->state != TARGET_HALTED) {
1736  LOG_TARGET_ERROR(target, "not halted");
1737  return ERROR_TARGET_NOT_HALTED;
1738  }
1739 
1741  LOG_TARGET_ERROR(target, "eps_dbglevel_idx not set\n");
1742  return ERROR_FAIL;
1743  }
1744 
1745  /* Save old ps (EPS[dbglvl] on LX), pc */
1749 
1750  cause = xtensa_cause_get(target);
1751  LOG_TARGET_DEBUG(target, "oldps=%" PRIx32 ", oldpc=%" PRIx32 " dbg_cause=%" PRIx32 " exc_cause=%" PRIx32,
1752  oldps,
1753  oldpc,
1754  cause,
1756  if (handle_breakpoints && (cause & (DEBUGCAUSE_BI | DEBUGCAUSE_BN))) {
1757  /* handle hard-coded SW breakpoints (e.g. syscalls) */
1758  LOG_TARGET_DEBUG(target, "Increment PC to pass break instruction...");
1759  xtensa_cause_clear(target); /* so we don't recurse into the same routine */
1760  /* pretend that we have stepped */
1761  if (cause & DEBUGCAUSE_BI)
1762  xtensa_reg_set(target, XT_REG_IDX_PC, oldpc + 3); /* PC = PC+3 */
1763  else
1764  xtensa_reg_set(target, XT_REG_IDX_PC, oldpc + 2); /* PC = PC+2 */
1765  return ERROR_OK;
1766  }
1767 
1768  /* Xtensa LX has an ICOUNTLEVEL register which sets the maximum interrupt level
1769  * at which the instructions are to be counted while stepping.
1770  *
1771  * For example, if we need to step by 2 instructions, and an interrupt occurs
1772  * in between, the processor will trigger the interrupt and halt after the 2nd
1773  * instruction within the interrupt vector and/or handler.
1774  *
1775  * However, sometimes we don't want the interrupt handlers to be executed at all
1776  * while stepping through the code. In this case (XT_STEPPING_ISR_OFF),
1777  * ICOUNTLEVEL can be lowered to the executing code's (level + 1) to prevent ISR
1778  * code from being counted during stepping. Note that C exception handlers must
1779  * run at level 0 and hence will be counted and stepped into, should one occur.
1780  *
1781  * TODO: Certain instructions should never be single-stepped and should instead
1782  * be emulated (per DUG): RSIL >= DBGLEVEL, RSR/WSR [ICOUNT|ICOUNTLEVEL], and
1783  * RFI >= DBGLEVEL.
1784  */
1786  if (xtensa->core_config->core_type == XT_LX) {
1789  "disabling IRQs while stepping is not implemented w/o high prio IRQs option!");
1790  return ERROR_FAIL;
1791  }
1792  /* Update ICOUNTLEVEL accordingly */
1793  icountlvl = MIN((oldps & 0xF) + 1, xtensa->core_config->debug.irq_level);
1794  } else {
1795  /* Xtensa NX does not have the ICOUNTLEVEL feature present in Xtensa LX
1796  * and instead disable interrupts while stepping. This could change
1797  * the timing of the system while under debug */
1798  xtensa_reg_val_t newps = oldps | XT_PS_DI_MSK;
1800  icountlvl = xtensa->core_config->debug.irq_level;
1801  ps_modified = true;
1802  }
1803  } else {
1804  icountlvl = xtensa->core_config->debug.irq_level;
1805  }
1806 
1807  if (cause & DEBUGCAUSE_DB) {
1808  /* We stopped due to a watchpoint. We can't just resume executing the instruction again because
1809  * that would trigger the watchpoint again. To fix this, we remove watchpoints,single-step and
1810  * re-enable the watchpoint. */
1812  target,
1813  "Single-stepping to get past instruction that triggered the watchpoint...");
1814  xtensa_cause_clear(target); /* so we don't recurse into the same routine */
1815  /* Save all DBREAKCx registers and set to 0 to disable watchpoints */
1816  for (unsigned int slot = 0; slot < xtensa->core_config->debug.dbreaks_num; slot++) {
1819  }
1820  }
1821 
1822  if (!handle_breakpoints && (cause & (DEBUGCAUSE_BI | DEBUGCAUSE_BN)))
1823  /* handle normal SW breakpoint */
1824  xtensa_cause_clear(target); /* so we don't recurse into the same routine */
1825  if (xtensa->core_config->core_type == XT_LX && ((oldps & 0xf) >= icountlvl)) {
1826  /* Lower interrupt level to allow stepping, but flag eps[dbglvl] to be restored */
1827  ps_modified = true;
1828  uint32_t newps = (oldps & ~0xf) | (icountlvl - 1);
1831  "Lowering PS.INTLEVEL to allow stepping: %s <- 0x%08" PRIx32 " (was 0x%08" PRIx32 ")",
1833  newps,
1834  oldps);
1835  }
1836  do {
1837  if (xtensa->core_config->core_type == XT_LX) {
1839  xtensa_reg_set(target, XT_REG_IDX_ICOUNT, icount_val);
1840  } else {
1842  }
1843 
1844  /* Now that ICOUNT (LX) or DCR.StepRequest (NX) is set,
1845  * we can resume as if we were going to run
1846  */
1847  res = xtensa_prepare_resume(target, current, address, 0, 0);
1848  if (res != ERROR_OK) {
1849  LOG_TARGET_ERROR(target, "Failed to prepare resume for single step");
1850  return res;
1851  }
1852  res = xtensa_do_resume(target);
1853  if (res != ERROR_OK) {
1854  LOG_TARGET_ERROR(target, "Failed to resume after setting up single step");
1855  return res;
1856  }
1857 
1858  /* Wait for stepping to complete */
1859  long long start = timeval_ms();
1860  while (timeval_ms() < start + 500) {
1861  /* Do not use target_poll here, it also triggers other things... just manually read the DSR
1862  *until stepping is complete. */
1863  usleep(1000);
1865  if (res != ERROR_OK) {
1866  LOG_TARGET_ERROR(target, "Failed to read core status!");
1867  return res;
1868  }
1870  break;
1871  usleep(1000);
1872  }
1873  LOG_TARGET_DEBUG(target, "Finish stepping. dsr=0x%08" PRIx32,
1875  if (!xtensa_is_stopped(target)) {
1877  target,
1878  "Timed out waiting for target to finish stepping. dsr=0x%08" PRIx32,
1882  return ERROR_FAIL;
1883  }
1884 
1886  cur_pc = xtensa_reg_get(target, XT_REG_IDX_PC);
1887 
1889  "cur_ps=%" PRIx32 ", cur_pc=%" PRIx32 " dbg_cause=%" PRIx32 " exc_cause=%" PRIx32,
1891  cur_pc,
1894 
1895  /* Do not step into WindowOverflow if ISRs are masked.
1896  If we stop in WindowOverflow at breakpoint with masked ISRs and
1897  try to do a step it will get us out of that handler */
1898  if (xtensa->core_config->windowed &&
1900  xtensa_pc_in_winexc(target, cur_pc)) {
1901  /* isrmask = on, need to step out of the window exception handler */
1902  LOG_DEBUG("Stepping out of window exception, PC=%" PRIX32, cur_pc);
1903  oldpc = cur_pc;
1904  address = oldpc + 3;
1905  continue;
1906  }
1907 
1908  if (oldpc == cur_pc)
1909  LOG_TARGET_WARNING(target, "Stepping doesn't seem to change PC! dsr=0x%08" PRIx32,
1911  else
1912  LOG_DEBUG("Stepped from %" PRIX32 " to %" PRIX32, oldpc, cur_pc);
1913  break;
1914  } while (true);
1915 
1918  LOG_DEBUG("Done stepping, PC=%" PRIX32, cur_pc);
1919 
1920  if (cause & DEBUGCAUSE_DB) {
1921  LOG_TARGET_DEBUG(target, "...Done, re-installing watchpoints.");
1922  /* Restore the DBREAKCx registers */
1923  for (unsigned int slot = 0; slot < xtensa->core_config->debug.dbreaks_num; slot++)
1925  }
1926 
1927  /* Restore int level */
1928  if (ps_modified) {
1929  LOG_DEBUG("Restoring %s after stepping: 0x%08" PRIx32,
1934  }
1935 
1936  /* write ICOUNTLEVEL back to zero */
1938  /* TODO: can we skip writing dirty registers and re-fetching them? */
1941  return res;
1942 }
1943 
1944 int xtensa_step(struct target *target, int current, target_addr_t address, int handle_breakpoints)
1945 {
1946  int retval = xtensa_do_step(target, current, address, handle_breakpoints);
1947  if (retval != ERROR_OK)
1948  return retval;
1950 
1951  return ERROR_OK;
1952 }
1953 
1957 static inline bool xtensa_memory_regions_overlap(target_addr_t r1_start,
1958  target_addr_t r1_end,
1959  target_addr_t r2_start,
1960  target_addr_t r2_end)
1961 {
1962  if ((r2_start >= r1_start) && (r2_start < r1_end))
1963  return true; /* r2_start is in r1 region */
1964  if ((r2_end > r1_start) && (r2_end <= r1_end))
1965  return true; /* r2_end is in r1 region */
1966  return false;
1967 }
1968 
1973  target_addr_t r1_end,
1974  target_addr_t r2_start,
1975  target_addr_t r2_end)
1976 {
1977  if (xtensa_memory_regions_overlap(r1_start, r1_end, r2_start, r2_end)) {
1978  target_addr_t ov_start = r1_start < r2_start ? r2_start : r1_start;
1979  target_addr_t ov_end = r1_end > r2_end ? r2_end : r1_end;
1980  return ov_end - ov_start;
1981  }
1982  return 0;
1983 }
1984 
1988 static bool xtensa_memory_op_validate_range(struct xtensa *xtensa, target_addr_t address, size_t size, int access)
1989 {
1990  target_addr_t adr_pos = address; /* address cursor set to the beginning start */
1991  target_addr_t adr_end = address + size; /* region end */
1992  target_addr_t overlap_size;
1993  const struct xtensa_local_mem_region_config *cm; /* current mem region */
1994 
1995  while (adr_pos < adr_end) {
1997  if (!cm) /* address is not belong to anything */
1998  return false;
1999  if ((cm->access & access) != access) /* access check */
2000  return false;
2001  overlap_size = xtensa_get_overlap_size(cm->base, (cm->base + cm->size), adr_pos, adr_end);
2002  assert(overlap_size != 0);
2003  adr_pos += overlap_size;
2004  }
2005  return true;
2006 }
2007 
2008 int xtensa_read_memory(struct target *target, target_addr_t address, uint32_t size, uint32_t count, uint8_t *buffer)
2009 {
2010  struct xtensa *xtensa = target_to_xtensa(target);
2011  /* We are going to read memory in 32-bit increments. This may not be what the calling
2012  * function expects, so we may need to allocate a temp buffer and read into that first. */
2013  target_addr_t addrstart_al = ALIGN_DOWN(address, 4);
2014  target_addr_t addrend_al = ALIGN_UP(address + size * count, 4);
2015  target_addr_t adr = addrstart_al;
2016  uint8_t *albuff;
2017  bool bswap = xtensa->target->endianness == TARGET_BIG_ENDIAN;
2018 
2019  if (target->state != TARGET_HALTED) {
2020  LOG_TARGET_ERROR(target, "not halted");
2021  return ERROR_TARGET_NOT_HALTED;
2022  }
2023 
2024  if (!xtensa->permissive_mode) {
2025  if (!xtensa_memory_op_validate_range(xtensa, address, (size * count),
2026  XT_MEM_ACCESS_READ)) {
2027  LOG_DEBUG("address " TARGET_ADDR_FMT " not readable", address);
2028  return ERROR_FAIL;
2029  }
2030  }
2031 
2032  unsigned int alloc_bytes = ALIGN_UP(addrend_al - addrstart_al, sizeof(uint32_t));
2033  albuff = calloc(alloc_bytes, 1);
2034  if (!albuff) {
2035  LOG_TARGET_ERROR(target, "Out of memory allocating %" PRId64 " bytes!",
2036  addrend_al - addrstart_al);
2038  }
2039 
2040  /* We're going to use A3 here */
2042  /* Write start address to A3 */
2045  /* Now we can safely read data from addrstart_al up to addrend_al into albuff */
2046  if (xtensa->probe_lsddr32p != 0) {
2048  for (unsigned int i = 0; adr != addrend_al; i += sizeof(uint32_t), adr += sizeof(uint32_t))
2050  (adr + sizeof(uint32_t) == addrend_al) ? XDMREG_DDR : XDMREG_DDREXEC,
2051  &albuff[i]);
2052  } else {
2054  for (unsigned int i = 0; adr != addrend_al; i += sizeof(uint32_t), adr += sizeof(uint32_t)) {
2058  xtensa_queue_dbg_reg_write(xtensa, XDMREG_DDR, adr + sizeof(uint32_t));
2060  }
2061  }
2062  int res = xtensa_dm_queue_execute(&xtensa->dbg_mod);
2063  if (res == ERROR_OK) {
2064  bool prev_suppress = xtensa->suppress_dsr_errors;
2065  xtensa->suppress_dsr_errors = true;
2067  if (xtensa->probe_lsddr32p == -1)
2068  xtensa->probe_lsddr32p = 1;
2069  xtensa->suppress_dsr_errors = prev_suppress;
2070  }
2071  if (res != ERROR_OK) {
2072  if (xtensa->probe_lsddr32p != 0) {
2073  /* Disable fast memory access instructions and retry before reporting an error */
2074  LOG_TARGET_DEBUG(target, "Disabling LDDR32.P/SDDR32.P");
2075  xtensa->probe_lsddr32p = 0;
2076  res = xtensa_read_memory(target, address, size, count, albuff);
2077  bswap = false;
2078  } else {
2079  LOG_TARGET_WARNING(target, "Failed reading %d bytes at address "TARGET_ADDR_FMT,
2080  count * size, address);
2081  }
2082  }
2083 
2084  if (bswap)
2085  buf_bswap32(albuff, albuff, addrend_al - addrstart_al);
2086  memcpy(buffer, albuff + (address & 3), (size * count));
2087  free(albuff);
2088  return res;
2089 }
2090 
2091 int xtensa_read_buffer(struct target *target, target_addr_t address, uint32_t count, uint8_t *buffer)
2092 {
2093  /* xtensa_read_memory can also read unaligned stuff. Just pass through to that routine. */
2094  return xtensa_read_memory(target, address, 1, count, buffer);
2095 }
2096 
2098  target_addr_t address,
2099  uint32_t size,
2100  uint32_t count,
2101  const uint8_t *buffer)
2102 {
2103  /* This memory write function can get thrown nigh everything into it, from
2104  * aligned uint32 writes to unaligned uint8ths. The Xtensa memory doesn't always
2105  * accept anything but aligned uint32 writes, though. That is why we convert
2106  * everything into that. */
2107  struct xtensa *xtensa = target_to_xtensa(target);
2108  target_addr_t addrstart_al = ALIGN_DOWN(address, 4);
2109  target_addr_t addrend_al = ALIGN_UP(address + size * count, 4);
2110  target_addr_t adr = addrstart_al;
2111  int res;
2112  uint8_t *albuff;
2113  bool fill_head_tail = false;
2114 
2115  if (target->state != TARGET_HALTED) {
2116  LOG_TARGET_ERROR(target, "not halted");
2117  return ERROR_TARGET_NOT_HALTED;
2118  }
2119 
2120  if (!xtensa->permissive_mode) {
2122  LOG_WARNING("address " TARGET_ADDR_FMT " not writable", address);
2123  return ERROR_FAIL;
2124  }
2125  }
2126 
2127  if (size == 0 || count == 0 || !buffer)
2129 
2130  /* Allocate a temporary buffer to put the aligned bytes in, if needed. */
2131  if (addrstart_al == address && addrend_al == address + (size * count)) {
2133  /* Need a buffer for byte-swapping */
2134  albuff = malloc(addrend_al - addrstart_al);
2135  else
2136  /* We discard the const here because albuff can also be non-const */
2137  albuff = (uint8_t *)buffer;
2138  } else {
2139  fill_head_tail = true;
2140  albuff = malloc(addrend_al - addrstart_al);
2141  }
2142  if (!albuff) {
2143  LOG_TARGET_ERROR(target, "Out of memory allocating %" PRId64 " bytes!",
2144  addrend_al - addrstart_al);
2146  }
2147 
2148  /* We're going to use A3 here */
2150 
2151  /* If we're using a temp aligned buffer, we need to fill the head and/or tail bit of it. */
2152  if (fill_head_tail) {
2153  /* See if we need to read the first and/or last word. */
2154  if (address & 3) {
2157  if (xtensa->probe_lsddr32p == 1) {
2159  } else {
2162  }
2164  }
2165  if ((address + (size * count)) & 3) {
2166  xtensa_queue_dbg_reg_write(xtensa, XDMREG_DDR, addrend_al - 4);
2168  if (xtensa->probe_lsddr32p == 1) {
2170  } else {
2173  }
2175  &albuff[addrend_al - addrstart_al - 4]);
2176  }
2177  /* Grab bytes */
2179  if (res != ERROR_OK) {
2180  LOG_ERROR("Error issuing unaligned memory write context instruction(s): %d", res);
2181  if (albuff != buffer)
2182  free(albuff);
2183  return res;
2184  }
2187  bool swapped_w0 = false;
2188  if (address & 3) {
2189  buf_bswap32(&albuff[0], &albuff[0], 4);
2190  swapped_w0 = true;
2191  }
2192  if ((address + (size * count)) & 3) {
2193  if ((addrend_al - addrstart_al - 4 == 0) && swapped_w0) {
2194  /* Don't double-swap if buffer start/end are within the same word */
2195  } else {
2196  buf_bswap32(&albuff[addrend_al - addrstart_al - 4],
2197  &albuff[addrend_al - addrstart_al - 4], 4);
2198  }
2199  }
2200  }
2201  /* Copy data to be written into the aligned buffer (in host-endianness) */
2202  memcpy(&albuff[address & 3], buffer, size * count);
2203  /* Now we can write albuff in aligned uint32s. */
2204  }
2205 
2207  buf_bswap32(albuff, fill_head_tail ? albuff : buffer, addrend_al - addrstart_al);
2208 
2209  /* Write start address to A3 */
2212  /* Write the aligned buffer */
2213  if (xtensa->probe_lsddr32p != 0) {
2214  for (unsigned int i = 0; adr != addrend_al; i += sizeof(uint32_t), adr += sizeof(uint32_t)) {
2215  if (i == 0) {
2218  } else {
2220  }
2221  }
2222  } else {
2224  for (unsigned int i = 0; adr != addrend_al; i += sizeof(uint32_t), adr += sizeof(uint32_t)) {
2228  xtensa_queue_dbg_reg_write(xtensa, XDMREG_DDR, adr + sizeof(uint32_t));
2230  }
2231  }
2232 
2234  if (res == ERROR_OK) {
2235  bool prev_suppress = xtensa->suppress_dsr_errors;
2236  xtensa->suppress_dsr_errors = true;
2238  if (xtensa->probe_lsddr32p == -1)
2239  xtensa->probe_lsddr32p = 1;
2240  xtensa->suppress_dsr_errors = prev_suppress;
2241  }
2242  if (res != ERROR_OK) {
2243  if (xtensa->probe_lsddr32p != 0) {
2244  /* Disable fast memory access instructions and retry before reporting an error */
2245  LOG_TARGET_INFO(target, "Disabling LDDR32.P/SDDR32.P");
2246  xtensa->probe_lsddr32p = 0;
2247  res = xtensa_write_memory(target, address, size, count, buffer);
2248  } else {
2249  LOG_TARGET_WARNING(target, "Failed writing %d bytes at address "TARGET_ADDR_FMT,
2250  count * size, address);
2251  }
2252  } else {
2253  /* Invalidate ICACHE, writeback DCACHE if present */
2254  bool issue_ihi = xtensa_is_icacheable(xtensa, address) &&
2255  xtensa_region_ar_exec(target, addrstart_al, addrend_al);
2256  bool issue_dhwbi = xtensa_is_dcacheable(xtensa, address);
2257  LOG_TARGET_DEBUG(target, "Cache OPs: IHI %d, DHWBI %d", issue_ihi, issue_dhwbi);
2258  if (issue_ihi || issue_dhwbi) {
2259  uint32_t ilinesize = issue_ihi ? xtensa->core_config->icache.line_size : UINT32_MAX;
2260  uint32_t dlinesize = issue_dhwbi ? xtensa->core_config->dcache.line_size : UINT32_MAX;
2261  uint32_t linesize = MIN(ilinesize, dlinesize);
2262  uint32_t off = 0;
2263  adr = addrstart_al;
2264 
2265  while ((adr + off) < addrend_al) {
2266  if (off == 0) {
2267  /* Write start address to A3 */
2270  }
2271  if (issue_ihi)
2273  if (issue_dhwbi)
2275  off += linesize;
2276  if (off > 1020) {
2277  /* IHI, DHWB have 8-bit immediate operands (0..1020) */
2278  adr += off;
2279  off = 0;
2280  }
2281  }
2282 
2283  /* Execute cache WB/INV instructions */
2285  if (res != ERROR_OK)
2287  "Error queuing cache writeback/invaldate instruction(s): %d",
2288  res);
2290  if (res != ERROR_OK)
2292  "Error issuing cache writeback/invaldate instruction(s): %d",
2293  res);
2294  }
2295  }
2296  if (albuff != buffer)
2297  free(albuff);
2298 
2299  return res;
2300 }
2301 
2302 int xtensa_write_buffer(struct target *target, target_addr_t address, uint32_t count, const uint8_t *buffer)
2303 {
2304  /* xtensa_write_memory can handle everything. Just pass on to that. */
2305  return xtensa_write_memory(target, address, 1, count, buffer);
2306 }
2307 
2308 int xtensa_checksum_memory(struct target *target, target_addr_t address, uint32_t count, uint32_t *checksum)
2309 {
2310  LOG_WARNING("not implemented yet");
2311  return ERROR_FAIL;
2312 }
2313 
2315 {
2316  struct xtensa *xtensa = target_to_xtensa(target);
2317  if (xtensa_dm_poll(&xtensa->dbg_mod) != ERROR_OK) {
2320  }
2321 
2325  LOG_TARGET_DEBUG(target, "PWRSTAT: read 0x%08" PRIx32 ", clear 0x%08lx, reread 0x%08" PRIx32,
2329  if (res != ERROR_OK)
2330  return res;
2331 
2333  LOG_TARGET_INFO(target, "Debug controller was reset.");
2335  if (res != ERROR_OK)
2336  return res;
2337  }
2339  LOG_TARGET_INFO(target, "Core was reset.");
2341  /* Enable JTAG, set reset if needed */
2342  res = xtensa_wakeup(target);
2343  if (res != ERROR_OK)
2344  return res;
2345 
2346  uint32_t prev_dsr = xtensa->dbg_mod.core_status.dsr;
2348  if (res != ERROR_OK)
2349  return res;
2350  if (prev_dsr != xtensa->dbg_mod.core_status.dsr)
2352  "DSR has changed: was 0x%08" PRIx32 " now 0x%08" PRIx32,
2353  prev_dsr,
2356  /* if RESET state is persitent */
2358  } else if (!xtensa_dm_is_powered(&xtensa->dbg_mod)) {
2359  LOG_TARGET_DEBUG(target, "not powered 0x%" PRIX32 "%ld",
2363  if (xtensa->come_online_probes_num == 0)
2364  target->examined = false;
2365  else
2367  } else if (xtensa_is_stopped(target)) {
2368  if (target->state != TARGET_HALTED) {
2369  enum target_state oldstate = target->state;
2371  /* Examine why the target has been halted */
2374  /* When setting debug reason DEBUGCAUSE events have the following
2375  * priorities: watchpoint == breakpoint > single step > debug interrupt. */
2376  /* Watchpoint and breakpoint events at the same time results in special
2377  * debug reason: DBG_REASON_WPTANDBKPT. */
2378  uint32_t halt_cause = xtensa_cause_get(target);
2379  /* TODO: Add handling of DBG_REASON_EXC_CATCH */
2380  if (halt_cause & DEBUGCAUSE_IC)
2382  if (halt_cause & (DEBUGCAUSE_IB | DEBUGCAUSE_BN | DEBUGCAUSE_BI)) {
2383  if (halt_cause & DEBUGCAUSE_DB)
2385  else
2387  } else if (halt_cause & DEBUGCAUSE_DB) {
2389  }
2390  LOG_TARGET_DEBUG(target, "Target halted, pc=0x%08" PRIx32
2391  ", debug_reason=%08" PRIx32 ", oldstate=%08" PRIx32,
2394  oldstate);
2395  LOG_TARGET_DEBUG(target, "Halt reason=0x%08" PRIX32 ", exc_cause=%" PRId32 ", dsr=0x%08" PRIx32,
2396  halt_cause,
2400  &xtensa->dbg_mod,
2404  if (xtensa->core_config->core_type == XT_NX) {
2405  /* Enable imprecise exceptions while in halted state */
2407  xtensa_reg_val_t newps = ps & ~(XT_PS_DIEXC_MSK);
2409  LOG_TARGET_DEBUG(target, "Enabling PS.DIEXC: 0x%08x -> 0x%08x", ps, newps);
2414  if (res != ERROR_OK) {
2415  LOG_TARGET_ERROR(target, "Failed to write PS.DIEXC (%d)!", res);
2416  return res;
2417  }
2419  }
2420  }
2421  } else {
2426  }
2427  }
2428  if (xtensa->trace_active) {
2429  /* Detect if tracing was active but has stopped. */
2432  if (res == ERROR_OK) {
2433  if (!(trace_status.stat & TRAXSTAT_TRACT)) {
2434  LOG_INFO("Detected end of trace.");
2435  if (trace_status.stat & TRAXSTAT_PCMTG)
2436  LOG_TARGET_INFO(target, "Trace stop triggered by PC match");
2437  if (trace_status.stat & TRAXSTAT_PTITG)
2438  LOG_TARGET_INFO(target, "Trace stop triggered by Processor Trigger Input");
2439  if (trace_status.stat & TRAXSTAT_CTITG)
2440  LOG_TARGET_INFO(target, "Trace stop triggered by Cross-trigger Input");
2441  xtensa->trace_active = false;
2442  }
2443  }
2444  }
2445  return ERROR_OK;
2446 }
2447 
2448 static int xtensa_update_instruction(struct target *target, target_addr_t address, uint32_t size, const uint8_t *buffer)
2449 {
2450  struct xtensa *xtensa = target_to_xtensa(target);
2451  unsigned int issue_ihi = xtensa_is_icacheable(xtensa, address) &&
2452  xtensa_region_ar_exec(target, address, address + size);
2453  unsigned int issue_dhwbi = xtensa_is_dcacheable(xtensa, address);
2454  uint32_t icache_line_size = issue_ihi ? xtensa->core_config->icache.line_size : UINT32_MAX;
2455  uint32_t dcache_line_size = issue_dhwbi ? xtensa->core_config->dcache.line_size : UINT32_MAX;
2456  unsigned int same_ic_line = ((address & (icache_line_size - 1)) + size) <= icache_line_size;
2457  unsigned int same_dc_line = ((address & (dcache_line_size - 1)) + size) <= dcache_line_size;
2458  int ret;
2459 
2460  if (size > icache_line_size)
2461  return ERROR_FAIL;
2462 
2463  if (issue_ihi || issue_dhwbi) {
2464  /* We're going to use A3 here */
2466 
2467  /* Write start address to A3 and invalidate */
2470  LOG_TARGET_DEBUG(target, "IHI %d, DHWBI %d for address " TARGET_ADDR_FMT,
2471  issue_ihi, issue_dhwbi, address);
2472  if (issue_dhwbi) {
2474  if (!same_dc_line) {
2476  "DHWBI second dcache line for address "TARGET_ADDR_FMT,
2477  address + 4);
2479  }
2480  }
2481  if (issue_ihi) {
2483  if (!same_ic_line) {
2485  "IHI second icache line for address "TARGET_ADDR_FMT,
2486  address + 4);
2488  }
2489  }
2490 
2491  /* Execute invalidate instructions */
2494  if (ret != ERROR_OK) {
2495  LOG_ERROR("Error issuing cache invaldate instruction(s): %d", ret);
2496  return ret;
2497  }
2498  }
2499 
2500  /* Write new instructions to memory */
2501  ret = target_write_buffer(target, address, size, buffer);
2502  if (ret != ERROR_OK) {
2503  LOG_TARGET_ERROR(target, "Error writing instruction to memory: %d", ret);
2504  return ret;
2505  }
2506 
2507  if (issue_dhwbi) {
2508  /* Flush dcache so instruction propagates. A3 may be corrupted during memory write */
2512  LOG_DEBUG("DHWB dcache line for address "TARGET_ADDR_FMT, address);
2513  if (!same_dc_line) {
2514  LOG_TARGET_DEBUG(target, "DHWB second dcache line for address "TARGET_ADDR_FMT, address + 4);
2516  }
2517 
2518  /* Execute invalidate instructions */
2521  }
2522 
2523  /* TODO: Handle L2 cache if present */
2524  return ret;
2525 }
2526 
2528  struct breakpoint *breakpoint,
2529  struct xtensa_sw_breakpoint *sw_bp)
2530 {
2531  struct xtensa *xtensa = target_to_xtensa(target);
2533  if (ret != ERROR_OK) {
2534  LOG_TARGET_ERROR(target, "Failed to read original instruction (%d)!", ret);
2535  return ret;
2536  }
2537 
2539  sw_bp->oocd_bp = breakpoint;
2540 
2541  uint32_t break_insn = sw_bp->insn_sz == XT_ISNS_SZ_MAX ? XT_INS_BREAK(xtensa, 0, 0) : XT_INS_BREAKN(xtensa, 0);
2542 
2543  /* Underlying memory write will convert instruction endianness, don't do that here */
2544  ret = xtensa_update_instruction(target, breakpoint->address, sw_bp->insn_sz, (uint8_t *)&break_insn);
2545  if (ret != ERROR_OK) {
2546  LOG_TARGET_ERROR(target, "Failed to write breakpoint instruction (%d)!", ret);
2547  return ret;
2548  }
2549 
2550  return ERROR_OK;
2551 }
2552 
2554 {
2555  int ret = xtensa_update_instruction(target, sw_bp->oocd_bp->address, sw_bp->insn_sz, sw_bp->insn);
2556  if (ret != ERROR_OK) {
2557  LOG_TARGET_ERROR(target, "Failed to write insn (%d)!", ret);
2558  return ret;
2559  }
2560  sw_bp->oocd_bp = NULL;
2561  return ERROR_OK;
2562 }
2563 
2565 {
2566  struct xtensa *xtensa = target_to_xtensa(target);
2567  unsigned int slot;
2568 
2569  if (breakpoint->type == BKPT_SOFT) {
2570  for (slot = 0; slot < XT_SW_BREAKPOINTS_MAX_NUM; slot++) {
2571  if (!xtensa->sw_brps[slot].oocd_bp ||
2573  break;
2574  }
2576  LOG_TARGET_WARNING(target, "No free slots to add SW breakpoint!");
2578  }
2580  if (ret != ERROR_OK) {
2581  LOG_TARGET_ERROR(target, "Failed to add SW breakpoint!");
2582  return ret;
2583  }
2584  LOG_TARGET_DEBUG(target, "placed SW breakpoint %u @ " TARGET_ADDR_FMT,
2585  slot,
2586  breakpoint->address);
2587  return ERROR_OK;
2588  }
2589 
2590  for (slot = 0; slot < xtensa->core_config->debug.ibreaks_num; slot++) {
2591  if (!xtensa->hw_brps[slot] || xtensa->hw_brps[slot] == breakpoint)
2592  break;
2593  }
2595  LOG_TARGET_ERROR(target, "No free slots to add HW breakpoint!");
2597  }
2598 
2600  /* We will actually write the breakpoints when we resume the target. */
2601  LOG_TARGET_DEBUG(target, "placed HW breakpoint %u @ " TARGET_ADDR_FMT,
2602  slot,
2603  breakpoint->address);
2604 
2605  return ERROR_OK;
2606 }
2607 
2609 {
2610  struct xtensa *xtensa = target_to_xtensa(target);
2611  unsigned int slot;
2612 
2613  if (breakpoint->type == BKPT_SOFT) {
2614  for (slot = 0; slot < XT_SW_BREAKPOINTS_MAX_NUM; slot++) {
2616  break;
2617  }
2619  LOG_TARGET_WARNING(target, "Max SW breakpoints slot reached, slot=%u!", slot);
2621  }
2623  if (ret != ERROR_OK) {
2624  LOG_TARGET_ERROR(target, "Failed to remove SW breakpoint (%d)!", ret);
2625  return ret;
2626  }
2627  LOG_TARGET_DEBUG(target, "cleared SW breakpoint %u @ " TARGET_ADDR_FMT, slot, breakpoint->address);
2628  return ERROR_OK;
2629  }
2630 
2631  for (slot = 0; slot < xtensa->core_config->debug.ibreaks_num; slot++) {
2632  if (xtensa->hw_brps[slot] == breakpoint)
2633  break;
2634  }
2636  LOG_TARGET_ERROR(target, "HW breakpoint not found!");
2638  }
2639  xtensa->hw_brps[slot] = NULL;
2640  if (xtensa->core_config->core_type == XT_NX)
2642  LOG_TARGET_DEBUG(target, "cleared HW breakpoint %u @ " TARGET_ADDR_FMT, slot, breakpoint->address);
2643  return ERROR_OK;
2644 }
2645 
2647 {
2648  struct xtensa *xtensa = target_to_xtensa(target);
2649  unsigned int slot;
2650  xtensa_reg_val_t dbreakcval;
2651 
2652  if (target->state != TARGET_HALTED) {
2653  LOG_TARGET_ERROR(target, "not halted");
2654  return ERROR_TARGET_NOT_HALTED;
2655  }
2656 
2658  LOG_TARGET_ERROR(target, "watchpoint value masks not supported");
2660  }
2661 
2662  for (slot = 0; slot < xtensa->core_config->debug.dbreaks_num; slot++) {
2663  if (!xtensa->hw_wps[slot] || xtensa->hw_wps[slot] == watchpoint)
2664  break;
2665  }
2667  LOG_TARGET_WARNING(target, "No free slots to add HW watchpoint!");
2669  }
2670 
2671  /* Figure out value for dbreakc5..0
2672  * It's basically 0x3F with an incremental bit removed from the LSB for each extra length power of 2. */
2673  if (watchpoint->length < 1 || watchpoint->length > 64 ||
2677  target,
2678  "Watchpoint with length %d on address " TARGET_ADDR_FMT
2679  " not supported by hardware.",
2680  watchpoint->length,
2681  watchpoint->address);
2683  }
2684  dbreakcval = ALIGN_DOWN(0x3F, watchpoint->length);
2685 
2686  if (watchpoint->rw == WPT_READ)
2687  dbreakcval |= BIT(30);
2688  if (watchpoint->rw == WPT_WRITE)
2689  dbreakcval |= BIT(31);
2690  if (watchpoint->rw == WPT_ACCESS)
2691  dbreakcval |= BIT(30) | BIT(31);
2692 
2693  /* Write DBREAKA[slot] and DBCREAKC[slot] */
2697  LOG_TARGET_DEBUG(target, "placed HW watchpoint @ " TARGET_ADDR_FMT,
2698  watchpoint->address);
2699  return ERROR_OK;
2700 }
2701 
2703 {
2704  struct xtensa *xtensa = target_to_xtensa(target);
2705  unsigned int slot;
2706 
2707  for (slot = 0; slot < xtensa->core_config->debug.dbreaks_num; slot++) {
2708  if (xtensa->hw_wps[slot] == watchpoint)
2709  break;
2710  }
2712  LOG_TARGET_WARNING(target, "HW watchpoint " TARGET_ADDR_FMT " not found!", watchpoint->address);
2714  }
2716  xtensa->hw_wps[slot] = NULL;
2717  LOG_TARGET_DEBUG(target, "cleared HW watchpoint @ " TARGET_ADDR_FMT,
2718  watchpoint->address);
2719  return ERROR_OK;
2720 }
2721 
2723  int num_mem_params, struct mem_param *mem_params,
2724  int num_reg_params, struct reg_param *reg_params,
2725  target_addr_t entry_point, target_addr_t exit_point,
2726  void *arch_info)
2727 {
2728  struct xtensa *xtensa = target_to_xtensa(target);
2729  struct xtensa_algorithm *algorithm_info = arch_info;
2730  int retval = ERROR_OK;
2731  bool usr_ps = false;
2732  uint32_t newps;
2733 
2734  /* NOTE: xtensa_run_algorithm requires that each algorithm uses a software breakpoint
2735  * at the exit point */
2736 
2737  if (target->state != TARGET_HALTED) {
2738  LOG_WARNING("Target not halted!");
2739  return ERROR_TARGET_NOT_HALTED;
2740  }
2741 
2742  for (unsigned int i = 0; i < xtensa->core_cache->num_regs; i++) {
2743  struct reg *reg = &xtensa->core_cache->reg_list[i];
2745  }
2746  /* save debug reason, it will be changed */
2747  if (!algorithm_info) {
2748  LOG_ERROR("BUG: arch_info not specified");
2749  return ERROR_FAIL;
2750  }
2751  algorithm_info->ctx_debug_reason = target->debug_reason;
2752  if (xtensa->core_config->core_type == XT_LX) {
2753  /* save PS and set to debug_level - 1 */
2754  algorithm_info->ctx_ps = xtensa_reg_get(target, xtensa->eps_dbglevel_idx);
2755  newps = (algorithm_info->ctx_ps & ~0xf) | (xtensa->core_config->debug.irq_level - 1);
2757  }
2758  /* write mem params */
2759  for (int i = 0; i < num_mem_params; i++) {
2760  if (mem_params[i].direction != PARAM_IN) {
2761  retval = target_write_buffer(target, mem_params[i].address,
2762  mem_params[i].size,
2763  mem_params[i].value);
2764  if (retval != ERROR_OK)
2765  return retval;
2766  }
2767  }
2768  /* write reg params */
2769  for (int i = 0; i < num_reg_params; i++) {
2770  if (reg_params[i].size > 32) {
2771  LOG_ERROR("BUG: not supported register size (%d)", reg_params[i].size);
2772  return ERROR_FAIL;
2773  }
2774  struct reg *reg = register_get_by_name(xtensa->core_cache, reg_params[i].reg_name, 0);
2775  if (!reg) {
2776  LOG_ERROR("BUG: register '%s' not found", reg_params[i].reg_name);
2777  return ERROR_FAIL;
2778  }
2779  if (reg->size != reg_params[i].size) {
2780  LOG_ERROR("BUG: register '%s' size doesn't match reg_params[i].size", reg_params[i].reg_name);
2781  return ERROR_FAIL;
2782  }
2783  if (memcmp(reg_params[i].reg_name, "ps", 3)) {
2784  usr_ps = true;
2785  } else if (xtensa->core_config->core_type == XT_LX) {
2786  unsigned int reg_id = xtensa->eps_dbglevel_idx;
2787  assert(reg_id < xtensa->core_cache->num_regs && "Attempt to access non-existing reg!");
2788  reg = &xtensa->core_cache->reg_list[reg_id];
2789  }
2790  xtensa_reg_set_value(reg, buf_get_u32(reg_params[i].value, 0, reg->size));
2791  reg->valid = 1;
2792  }
2793  /* ignore custom core mode if custom PS value is specified */
2794  if (!usr_ps && xtensa->core_config->core_type == XT_LX) {
2795  unsigned int eps_reg_idx = xtensa->eps_dbglevel_idx;
2796  xtensa_reg_val_t ps = xtensa_reg_get(target, eps_reg_idx);
2797  enum xtensa_mode core_mode = XT_PS_RING_GET(ps);
2798  if (algorithm_info->core_mode != XT_MODE_ANY && algorithm_info->core_mode != core_mode) {
2799  LOG_DEBUG("setting core_mode: 0x%x", algorithm_info->core_mode);
2800  xtensa_reg_val_t new_ps = (ps & ~XT_PS_RING_MSK) | XT_PS_RING(algorithm_info->core_mode);
2801  /* save previous core mode */
2802  /* TODO: core_mode is not restored for now. Can be added to the end of wait_algorithm */
2803  algorithm_info->core_mode = core_mode;
2804  xtensa_reg_set(target, eps_reg_idx, new_ps);
2805  xtensa->core_cache->reg_list[eps_reg_idx].valid = 1;
2806  }
2807  }
2808 
2809  return xtensa_resume(target, 0, entry_point, 1, 1);
2810 }
2811 
2814  int num_mem_params, struct mem_param *mem_params,
2815  int num_reg_params, struct reg_param *reg_params,
2816  target_addr_t exit_point, unsigned int timeout_ms,
2817  void *arch_info)
2818 {
2819  struct xtensa *xtensa = target_to_xtensa(target);
2820  struct xtensa_algorithm *algorithm_info = arch_info;
2821  int retval = ERROR_OK;
2822  xtensa_reg_val_t pc;
2823 
2824  /* NOTE: xtensa_run_algorithm requires that each algorithm uses a software breakpoint
2825  * at the exit point */
2826 
2827  retval = target_wait_state(target, TARGET_HALTED, timeout_ms);
2828  /* If the target fails to halt due to the breakpoint, force a halt */
2829  if (retval != ERROR_OK || target->state != TARGET_HALTED) {
2830  retval = target_halt(target);
2831  if (retval != ERROR_OK)
2832  return retval;
2833  retval = target_wait_state(target, TARGET_HALTED, 500);
2834  if (retval != ERROR_OK)
2835  return retval;
2836  LOG_TARGET_ERROR(target, "not halted %d, pc 0x%" PRIx32 ", ps 0x%" PRIx32, retval,
2840  return ERROR_TARGET_TIMEOUT;
2841  }
2843  if (exit_point && pc != exit_point) {
2844  LOG_ERROR("failed algorithm halted at 0x%" PRIx32 ", expected " TARGET_ADDR_FMT, pc, exit_point);
2845  return ERROR_TARGET_TIMEOUT;
2846  }
2847  /* Copy core register values to reg_params[] */
2848  for (int i = 0; i < num_reg_params; i++) {
2849  if (reg_params[i].direction != PARAM_OUT) {
2850  struct reg *reg = register_get_by_name(xtensa->core_cache, reg_params[i].reg_name, 0);
2851  if (!reg) {
2852  LOG_ERROR("BUG: register '%s' not found", reg_params[i].reg_name);
2853  return ERROR_FAIL;
2854  }
2855  if (reg->size != reg_params[i].size) {
2856  LOG_ERROR("BUG: register '%s' size doesn't match reg_params[i].size", reg_params[i].reg_name);
2857  return ERROR_FAIL;
2858  }
2859  buf_set_u32(reg_params[i].value, 0, 32, xtensa_reg_get_value(reg));
2860  }
2861  }
2862  /* Read memory values to mem_params */
2863  LOG_DEBUG("Read mem params");
2864  for (int i = 0; i < num_mem_params; i++) {
2865  LOG_DEBUG("Check mem param @ " TARGET_ADDR_FMT, mem_params[i].address);
2866  if (mem_params[i].direction != PARAM_OUT) {
2867  LOG_DEBUG("Read mem param @ " TARGET_ADDR_FMT, mem_params[i].address);
2868  retval = target_read_buffer(target, mem_params[i].address, mem_params[i].size, mem_params[i].value);
2869  if (retval != ERROR_OK)
2870  return retval;
2871  }
2872  }
2873 
2874  /* avoid gdb keep_alive warning */
2875  keep_alive();
2876 
2877  for (int i = xtensa->core_cache->num_regs - 1; i >= 0; i--) {
2878  struct reg *reg = &xtensa->core_cache->reg_list[i];
2879  if (i == XT_REG_IDX_PS) {
2880  continue; /* restore mapped reg number of PS depends on NDEBUGLEVEL */
2881  } else if (i == XT_REG_IDX_DEBUGCAUSE) {
2882  /*FIXME: restoring DEBUGCAUSE causes exception when executing corresponding
2883  * instruction in DIR */
2884  LOG_DEBUG("Skip restoring register %s: 0x%8.8" PRIx32 " -> 0x%8.8" PRIx32,
2886  buf_get_u32(reg->value, 0, 32),
2887  buf_get_u32(xtensa->algo_context_backup[i], 0, 32));
2889  xtensa->core_cache->reg_list[i].dirty = 0;
2890  xtensa->core_cache->reg_list[i].valid = 0;
2891  } else if (memcmp(xtensa->algo_context_backup[i], reg->value, reg->size / 8)) {
2892  if (reg->size <= 32) {
2893  LOG_DEBUG("restoring register %s: 0x%8.8" PRIx32 " -> 0x%8.8" PRIx32,
2895  buf_get_u32(reg->value, 0, reg->size),
2897  } else if (reg->size <= 64) {
2898  LOG_DEBUG("restoring register %s: 0x%8.8" PRIx64 " -> 0x%8.8" PRIx64,
2900  buf_get_u64(reg->value, 0, reg->size),
2902  } else {
2903  LOG_DEBUG("restoring register %s %u-bits", xtensa->core_cache->reg_list[i].name, reg->size);
2904  }
2906  xtensa->core_cache->reg_list[i].dirty = 1;
2907  xtensa->core_cache->reg_list[i].valid = 1;
2908  }
2909  }
2910  target->debug_reason = algorithm_info->ctx_debug_reason;
2911  if (xtensa->core_config->core_type == XT_LX)
2912  xtensa_reg_set(target, xtensa->eps_dbglevel_idx, algorithm_info->ctx_ps);
2913 
2915  if (retval != ERROR_OK)
2916  LOG_ERROR("Failed to write dirty regs (%d)!", retval);
2917 
2918  return retval;
2919 }
2920 
2922  int num_mem_params, struct mem_param *mem_params,
2923  int num_reg_params, struct reg_param *reg_params,
2924  target_addr_t entry_point, target_addr_t exit_point,
2925  unsigned int timeout_ms, void *arch_info)
2926 {
2927  int retval = xtensa_start_algorithm(target,
2928  num_mem_params, mem_params,
2929  num_reg_params, reg_params,
2930  entry_point, exit_point,
2931  arch_info);
2932 
2933  if (retval == ERROR_OK) {
2934  retval = xtensa_wait_algorithm(target,
2935  num_mem_params, mem_params,
2936  num_reg_params, reg_params,
2937  exit_point, timeout_ms,
2938  arch_info);
2939  }
2940 
2941  return retval;
2942 }
2943 
2945 {
2946  struct xtensa *xtensa = target_to_xtensa(target);
2947  struct reg_cache **cache_p = register_get_last_cache_p(&target->reg_cache);
2948  unsigned int last_dbreg_num = 0;
2949 
2951  LOG_TARGET_WARNING(target, "Register count MISMATCH: %d core regs, %d extended regs; %d expected",
2953 
2954  struct reg_cache *reg_cache = calloc(1, sizeof(struct reg_cache));
2955 
2956  if (!reg_cache) {
2957  LOG_ERROR("Failed to alloc reg cache!");
2958  return ERROR_FAIL;
2959  }
2960  reg_cache->name = "Xtensa registers";
2961  reg_cache->next = NULL;
2962  /* Init reglist */
2963  unsigned int reg_list_size = XT_NUM_REGS + xtensa->num_optregs;
2964  struct reg *reg_list = calloc(reg_list_size, sizeof(struct reg));
2965  if (!reg_list) {
2966  LOG_ERROR("Failed to alloc reg list!");
2967  goto fail;
2968  }
2969  xtensa->dbregs_num = 0;
2970  unsigned int didx = 0;
2971  for (unsigned int whichlist = 0; whichlist < 2; whichlist++) {
2972  struct xtensa_reg_desc *rlist = (whichlist == 0) ? xtensa_regs : xtensa->optregs;
2973  unsigned int listsize = (whichlist == 0) ? XT_NUM_REGS : xtensa->num_optregs;
2974  for (unsigned int i = 0; i < listsize; i++, didx++) {
2975  reg_list[didx].exist = rlist[i].exist;
2976  reg_list[didx].name = rlist[i].name;
2977  reg_list[didx].size = 32;
2978  reg_list[didx].value = calloc(1, 4 /*XT_REG_LEN*/); /* make Clang Static Analyzer happy */
2979  if (!reg_list[didx].value) {
2980  LOG_ERROR("Failed to alloc reg list value!");
2981  goto fail;
2982  }
2983  reg_list[didx].dirty = false;
2984  reg_list[didx].valid = false;
2985  reg_list[didx].type = &xtensa_reg_type;
2986  reg_list[didx].arch_info = xtensa;
2987  if (rlist[i].exist && (rlist[i].dbreg_num > last_dbreg_num))
2988  last_dbreg_num = rlist[i].dbreg_num;
2989 
2990  if (xtensa_extra_debug_log) {
2992  "POPULATE %-16s list %d exist %d, idx %d, type %d, dbreg_num 0x%04x",
2993  reg_list[didx].name,
2994  whichlist,
2995  reg_list[didx].exist,
2996  didx,
2997  rlist[i].type,
2998  rlist[i].dbreg_num);
2999  }
3000  }
3001  }
3002 
3003  xtensa->dbregs_num = last_dbreg_num + 1;
3004  reg_cache->reg_list = reg_list;
3005  reg_cache->num_regs = reg_list_size;
3006 
3007  LOG_TARGET_DEBUG(target, "xtensa->total_regs_num %d reg_list_size %d xtensa->dbregs_num %d",
3008  xtensa->total_regs_num, reg_list_size, xtensa->dbregs_num);
3009 
3010  /* Construct empty-register list for handling unknown register requests */
3011  xtensa->empty_regs = calloc(xtensa->dbregs_num, sizeof(struct reg));
3012  if (!xtensa->empty_regs) {
3013  LOG_TARGET_ERROR(target, "Out of memory");
3014  goto fail;
3015  }
3016  for (unsigned int i = 0; i < xtensa->dbregs_num; i++) {
3017  xtensa->empty_regs[i].name = calloc(8, sizeof(char));
3018  if (!xtensa->empty_regs[i].name) {
3019  LOG_TARGET_ERROR(target, "Out of memory");
3020  goto fail;
3021  }
3022  sprintf((char *)xtensa->empty_regs[i].name, "?0x%04x", i & 0x0000FFFF);
3023  xtensa->empty_regs[i].size = 32;
3025  xtensa->empty_regs[i].value = calloc(1, 4 /*XT_REG_LEN*/); /* make Clang Static Analyzer happy */
3026  if (!xtensa->empty_regs[i].value) {
3027  LOG_ERROR("Failed to alloc empty reg list value!");
3028  goto fail;
3029  }
3031  }
3032 
3033  /* Construct contiguous register list from contiguous descriptor list */
3035  xtensa->contiguous_regs_list = calloc(xtensa->total_regs_num, sizeof(struct reg *));
3036  if (!xtensa->contiguous_regs_list) {
3037  LOG_TARGET_ERROR(target, "Out of memory");
3038  goto fail;
3039  }
3040  for (unsigned int i = 0; i < xtensa->total_regs_num; i++) {
3041  unsigned int j;
3042  for (j = 0; j < reg_cache->num_regs; j++) {
3043  if (!strcmp(reg_cache->reg_list[j].name, xtensa->contiguous_regs_desc[i]->name)) {
3044  /* Register number field is not filled above.
3045  Here we are assigning the corresponding index from the contiguous reg list.
3046  These indexes are in the same order with gdb g-packet request/response.
3047  Some more changes may be required for sparse reg lists.
3048  */
3049  reg_cache->reg_list[j].number = i;
3052  "POPULATE contiguous regs list: %-16s, dbreg_num 0x%04x",
3055  break;
3056  }
3057  }
3058  if (j == reg_cache->num_regs)
3059  LOG_TARGET_WARNING(target, "contiguous register %s not found",
3061  }
3062  }
3063 
3064  xtensa->algo_context_backup = calloc(reg_cache->num_regs, sizeof(void *));
3065  if (!xtensa->algo_context_backup) {
3066  LOG_ERROR("Failed to alloc mem for algorithm context backup!");
3067  goto fail;
3068  }
3069  for (unsigned int i = 0; i < reg_cache->num_regs; i++) {
3070  struct reg *reg = &reg_cache->reg_list[i];
3071  xtensa->algo_context_backup[i] = calloc(1, reg->size / 8);
3072  if (!xtensa->algo_context_backup[i]) {
3073  LOG_ERROR("Failed to alloc mem for algorithm context!");
3074  goto fail;
3075  }
3076  }
3078  if (cache_p)
3079  *cache_p = reg_cache;
3080  return ERROR_OK;
3081 
3082 fail:
3083  if (reg_list) {
3084  for (unsigned int i = 0; i < reg_list_size; i++)
3085  free(reg_list[i].value);
3086  free(reg_list);
3087  }
3088  if (xtensa->empty_regs) {
3089  for (unsigned int i = 0; i < xtensa->dbregs_num; i++) {
3090  free((void *)xtensa->empty_regs[i].name);
3091  free(xtensa->empty_regs[i].value);
3092  }
3093  free(xtensa->empty_regs);
3094  }
3095  if (xtensa->algo_context_backup) {
3096  for (unsigned int i = 0; i < reg_cache->num_regs; i++)
3097  free(xtensa->algo_context_backup[i]);
3098  free(xtensa->algo_context_backup);
3099  }
3100  free(reg_cache);
3101 
3102  return ERROR_FAIL;
3103 }
3104 
3105 static int32_t xtensa_gdbqc_parse_exec_tie_ops(struct target *target, char *opstr)
3106 {
3107  struct xtensa *xtensa = target_to_xtensa(target);
3109  /* Process op[] list */
3110  while (opstr && (*opstr == ':')) {
3111  uint8_t ops[32];
3112  unsigned int oplen = strtoul(opstr + 1, &opstr, 16);
3113  if (oplen > 32) {
3114  LOG_TARGET_ERROR(target, "TIE access instruction too long (%d)\n", oplen);
3115  break;
3116  }
3117  unsigned int i = 0;
3118  while ((i < oplen) && opstr && (*opstr == ':'))
3119  ops[i++] = strtoul(opstr + 1, &opstr, 16);
3120  if (i != oplen) {
3121  LOG_TARGET_ERROR(target, "TIE access instruction malformed (%d)\n", i);
3122  break;
3123  }
3124 
3125  char insn_buf[128];
3126  sprintf(insn_buf, "Exec %d-byte TIE sequence: ", oplen);
3127  for (i = 0; i < oplen; i++)
3128  sprintf(insn_buf + strlen(insn_buf), "%02x:", ops[i]);
3129  LOG_TARGET_DEBUG(target, "%s", insn_buf);
3130  xtensa_queue_exec_ins_wide(xtensa, ops, oplen); /* Handles endian-swap */
3131  status = ERROR_OK;
3132  }
3133  return status;
3134 }
3135 
3136 static int xtensa_gdbqc_qxtreg(struct target *target, const char *packet, char **response_p)
3137 {
3138  struct xtensa *xtensa = target_to_xtensa(target);
3139  bool iswrite = (packet[0] == 'Q');
3140  enum xtensa_qerr_e error;
3141 
3142  /* Read/write TIE register. Requires spill location.
3143  * qxtreg<num>:<len>:<oplen>:<op[0]>:<...>[:<oplen>:<op[0]>:<...>]
3144  * Qxtreg<num>:<len>:<oplen>:<op[0]>:<...>[:<oplen>:<op[0]>:<...>]=<value>
3145  */
3146  if (!(xtensa->spill_buf)) {
3147  LOG_ERROR("Spill location not specified. Try 'target remote <host>:3333 &spill_location0'");
3148  error = XT_QERR_FAIL;
3149  goto xtensa_gdbqc_qxtreg_fail;
3150  }
3151 
3152  char *delim;
3153  uint32_t regnum = strtoul(packet + 6, &delim, 16);
3154  if (*delim != ':') {
3155  LOG_ERROR("Malformed qxtreg packet");
3156  error = XT_QERR_INVAL;
3157  goto xtensa_gdbqc_qxtreg_fail;
3158  }
3159  uint32_t reglen = strtoul(delim + 1, &delim, 16);
3160  if (*delim != ':') {
3161  LOG_ERROR("Malformed qxtreg packet");
3162  error = XT_QERR_INVAL;
3163  goto xtensa_gdbqc_qxtreg_fail;
3164  }
3165  uint8_t regbuf[XT_QUERYPKT_RESP_MAX];
3166  memset(regbuf, 0, XT_QUERYPKT_RESP_MAX);
3167  LOG_DEBUG("TIE reg 0x%08" PRIx32 " %s (%d bytes)", regnum, iswrite ? "write" : "read", reglen);
3168  if (reglen * 2 + 1 > XT_QUERYPKT_RESP_MAX) {
3169  LOG_ERROR("TIE register too large");
3170  error = XT_QERR_MEM;
3171  goto xtensa_gdbqc_qxtreg_fail;
3172  }
3173 
3174  /* (1) Save spill memory, (1.5) [if write then store value to spill location],
3175  * (2) read old a4, (3) write spill address to a4.
3176  * NOTE: ensure a4 is restored properly by all error handling logic
3177  */
3178  unsigned int memop_size = (xtensa->spill_loc & 3) ? 1 : 4;
3179  int status = xtensa_read_memory(target, xtensa->spill_loc, memop_size,
3180  xtensa->spill_bytes / memop_size, xtensa->spill_buf);
3181  if (status != ERROR_OK) {
3182  LOG_ERROR("Spill memory save");
3183  error = XT_QERR_MEM;
3184  goto xtensa_gdbqc_qxtreg_fail;
3185  }
3186  if (iswrite) {
3187  /* Extract value and store in spill memory */
3188  unsigned int b = 0;
3189  char *valbuf = strchr(delim, '=');
3190  if (!(valbuf && (*valbuf == '='))) {
3191  LOG_ERROR("Malformed Qxtreg packet");
3192  error = XT_QERR_INVAL;
3193  goto xtensa_gdbqc_qxtreg_fail;
3194  }
3195  valbuf++;
3196  while (*valbuf && *(valbuf + 1)) {
3197  char bytestr[3] = { 0, 0, 0 };
3198  strncpy(bytestr, valbuf, 2);
3199  regbuf[b++] = strtoul(bytestr, NULL, 16);
3200  valbuf += 2;
3201  }
3202  if (b != reglen) {
3203  LOG_ERROR("Malformed Qxtreg packet");
3204  error = XT_QERR_INVAL;
3205  goto xtensa_gdbqc_qxtreg_fail;
3206  }
3208  reglen / memop_size, regbuf);
3209  if (status != ERROR_OK) {
3210  LOG_ERROR("TIE value store");
3211  error = XT_QERR_MEM;
3212  goto xtensa_gdbqc_qxtreg_fail;
3213  }
3214  }
3218 
3219  int32_t tieop_status = xtensa_gdbqc_parse_exec_tie_ops(target, delim);
3220 
3221  /* Restore a4 but not yet spill memory. Execute it all... */
3225  if (status != ERROR_OK) {
3226  LOG_TARGET_ERROR(target, "TIE queue execute: %d\n", status);
3227  tieop_status = status;
3228  }
3230  if (status != ERROR_OK) {
3231  LOG_TARGET_ERROR(target, "TIE instr execute: %d\n", status);
3232  tieop_status = status;
3233  }
3234 
3235  if (tieop_status == ERROR_OK) {
3236  if (iswrite) {
3237  /* TIE write succeeded; send OK */
3238  strcpy(*response_p, "OK");
3239  } else {
3240  /* TIE read succeeded; copy result from spill memory */
3241  status = xtensa_read_memory(target, xtensa->spill_loc, memop_size, reglen, regbuf);
3242  if (status != ERROR_OK) {
3243  LOG_TARGET_ERROR(target, "TIE result read");
3244  tieop_status = status;
3245  }
3246  unsigned int i;
3247  for (i = 0; i < reglen; i++)
3248  sprintf(*response_p + 2 * i, "%02x", regbuf[i]);
3249  *(*response_p + 2 * i) = '\0';
3250  LOG_TARGET_DEBUG(target, "TIE response: %s", *response_p);
3251  }
3252  }
3253 
3254  /* Restore spill memory first, then report any previous errors */
3256  xtensa->spill_bytes / memop_size, xtensa->spill_buf);
3257  if (status != ERROR_OK) {
3258  LOG_ERROR("Spill memory restore");
3259  error = XT_QERR_MEM;
3260  goto xtensa_gdbqc_qxtreg_fail;
3261  }
3262  if (tieop_status != ERROR_OK) {
3263  LOG_ERROR("TIE execution");
3264  error = XT_QERR_FAIL;
3265  goto xtensa_gdbqc_qxtreg_fail;
3266  }
3267  return ERROR_OK;
3268 
3269 xtensa_gdbqc_qxtreg_fail:
3270  strcpy(*response_p, xt_qerr[error].chrval);
3271  return xt_qerr[error].intval;
3272 }
3273 
3274 int xtensa_gdb_query_custom(struct target *target, const char *packet, char **response_p)
3275 {
3276  struct xtensa *xtensa = target_to_xtensa(target);
3277  enum xtensa_qerr_e error;
3278  if (!packet || !response_p) {
3279  LOG_TARGET_ERROR(target, "invalid parameter: packet %p response_p %p", packet, response_p);
3280  return ERROR_FAIL;
3281  }
3282 
3283  *response_p = xtensa->qpkt_resp;
3284  if (strncmp(packet, "qxtn", 4) == 0) {
3285  strcpy(*response_p, "OpenOCD");
3286  return ERROR_OK;
3287  } else if (strncasecmp(packet, "qxtgdbversion=", 14) == 0) {
3288  return ERROR_OK;
3289  } else if ((strncmp(packet, "Qxtsis=", 7) == 0) || (strncmp(packet, "Qxtsds=", 7) == 0)) {
3290  /* Confirm host cache params match core .cfg file */
3291  struct xtensa_cache_config *cachep = (packet[4] == 'i') ?
3293  unsigned int line_size = 0, size = 0, way_count = 0;
3294  sscanf(&packet[7], "%x,%x,%x", &line_size, &size, &way_count);
3295  if ((cachep->line_size != line_size) ||
3296  (cachep->size != size) ||
3297  (cachep->way_count != way_count)) {
3298  LOG_TARGET_WARNING(target, "%cCache mismatch; check xtensa-core-XXX.cfg file",
3299  cachep == &xtensa->core_config->icache ? 'I' : 'D');
3300  }
3301  strcpy(*response_p, "OK");
3302  return ERROR_OK;
3303  } else if ((strncmp(packet, "Qxtiram=", 8) == 0) || (strncmp(packet, "Qxtirom=", 8) == 0)) {
3304  /* Confirm host IRAM/IROM params match core .cfg file */
3305  struct xtensa_local_mem_config *memp = (packet[5] == 'a') ?
3307  unsigned int base = 0, size = 0, i;
3308  char *pkt = (char *)&packet[7];
3309  do {
3310  pkt++;
3311  size = strtoul(pkt, &pkt, 16);
3312  pkt++;
3313  base = strtoul(pkt, &pkt, 16);
3314  LOG_TARGET_DEBUG(target, "memcheck: %dB @ 0x%08x", size, base);
3315  for (i = 0; i < memp->count; i++) {
3316  if ((memp->regions[i].base == base) && (memp->regions[i].size == size))
3317  break;
3318  }
3319  if (i == memp->count) {
3320  LOG_TARGET_WARNING(target, "%s mismatch; check xtensa-core-XXX.cfg file",
3321  memp == &xtensa->core_config->iram ? "IRAM" : "IROM");
3322  break;
3323  }
3324  for (i = 0; i < 11; i++) {
3325  pkt++;
3326  strtoul(pkt, &pkt, 16);
3327  }
3328  } while (pkt && (pkt[0] == ','));
3329  strcpy(*response_p, "OK");
3330  return ERROR_OK;
3331  } else if (strncmp(packet, "Qxtexcmlvl=", 11) == 0) {
3332  /* Confirm host EXCM_LEVEL matches core .cfg file */
3333  unsigned int excm_level = strtoul(&packet[11], NULL, 0);
3335  (excm_level != xtensa->core_config->high_irq.excm_level))
3336  LOG_TARGET_WARNING(target, "EXCM_LEVEL mismatch; check xtensa-core-XXX.cfg file");
3337  strcpy(*response_p, "OK");
3338  return ERROR_OK;
3339  } else if ((strncmp(packet, "Qxtl2cs=", 8) == 0) ||
3340  (strncmp(packet, "Qxtl2ca=", 8) == 0) ||
3341  (strncmp(packet, "Qxtdensity=", 11) == 0)) {
3342  strcpy(*response_p, "OK");
3343  return ERROR_OK;
3344  } else if (strncmp(packet, "Qxtspill=", 9) == 0) {
3345  char *delim;
3346  uint32_t spill_loc = strtoul(packet + 9, &delim, 16);
3347  if (*delim != ':') {
3348  LOG_ERROR("Malformed Qxtspill packet");
3349  error = XT_QERR_INVAL;
3350  goto xtensa_gdb_query_custom_fail;
3351  }
3352  xtensa->spill_loc = spill_loc;
3353  xtensa->spill_bytes = strtoul(delim + 1, NULL, 16);
3354  if (xtensa->spill_buf)
3355  free(xtensa->spill_buf);
3356  xtensa->spill_buf = calloc(1, xtensa->spill_bytes);
3357  if (!xtensa->spill_buf) {
3358  LOG_ERROR("Spill buf alloc");
3359  error = XT_QERR_MEM;
3360  goto xtensa_gdb_query_custom_fail;
3361  }
3362  LOG_TARGET_DEBUG(target, "Set spill 0x%08" PRIx32 " (%d)", xtensa->spill_loc, xtensa->spill_bytes);
3363  strcpy(*response_p, "OK");
3364  return ERROR_OK;
3365  } else if (strncasecmp(packet, "qxtreg", 6) == 0) {
3366  return xtensa_gdbqc_qxtreg(target, packet, response_p);
3367  } else if ((strncmp(packet, "qTStatus", 8) == 0) ||
3368  (strncmp(packet, "qxtftie", 7) == 0) ||
3369  (strncmp(packet, "qxtstie", 7) == 0)) {
3370  /* Return empty string to indicate trace, TIE wire debug are unsupported */
3371  strcpy(*response_p, "");
3372  return ERROR_OK;
3373  }
3374 
3375  /* Warn for all other queries, but do not return errors */
3376  LOG_TARGET_WARNING(target, "Unknown target-specific query packet: %s", packet);
3377  strcpy(*response_p, "");
3378  return ERROR_OK;
3379 
3380 xtensa_gdb_query_custom_fail:
3381  strcpy(*response_p, xt_qerr[error].chrval);
3382  return xt_qerr[error].intval;
3383 }
3384 
3386  const struct xtensa_debug_module_config *dm_cfg)
3387 {
3388  target->arch_info = xtensa;
3390  xtensa->target = target;
3392 
3393  xtensa->core_config = calloc(1, sizeof(struct xtensa_config));
3394  if (!xtensa->core_config) {
3395  LOG_ERROR("Xtensa configuration alloc failed\n");
3396  return ERROR_FAIL;
3397  }
3398 
3399  /* Default cache settings are disabled with 1 way */
3402 
3403  /* chrval: AR3/AR4 register names will change with window mapping.
3404  * intval: tracks whether scratch register was set through gdb P packet.
3405  */
3406  for (enum xtensa_ar_scratch_set_e s = 0; s < XT_AR_SCRATCH_NUM; s++) {
3407  xtensa->scratch_ars[s].chrval = calloc(8, sizeof(char));
3408  if (!xtensa->scratch_ars[s].chrval) {
3409  for (enum xtensa_ar_scratch_set_e f = 0; f < s; f++)
3410  free(xtensa->scratch_ars[f].chrval);
3411  free(xtensa->core_config);
3412  LOG_ERROR("Xtensa scratch AR alloc failed\n");
3413  return ERROR_FAIL;
3414  }
3415  xtensa->scratch_ars[s].intval = false;
3416  sprintf(xtensa->scratch_ars[s].chrval, "%s%d",
3417  ((s == XT_AR_SCRATCH_A3) || (s == XT_AR_SCRATCH_A4)) ? "a" : "ar",
3418  ((s == XT_AR_SCRATCH_A3) || (s == XT_AR_SCRATCH_AR3)) ? 3 : 4);
3419  }
3420 
3421  return xtensa_dm_init(&xtensa->dbg_mod, dm_cfg);
3422 }
3423 
3425 {
3427 }
3428 
3429 int xtensa_target_init(struct command_context *cmd_ctx, struct target *target)
3430 {
3431  struct xtensa *xtensa = target_to_xtensa(target);
3432 
3434  xtensa->hw_brps = calloc(XT_HW_IBREAK_MAX_NUM, sizeof(struct breakpoint *));
3435  if (!xtensa->hw_brps) {
3436  LOG_ERROR("Failed to alloc memory for HW breakpoints!");
3437  return ERROR_FAIL;
3438  }
3439  xtensa->hw_wps = calloc(XT_HW_DBREAK_MAX_NUM, sizeof(struct watchpoint *));
3440  if (!xtensa->hw_wps) {
3441  free(xtensa->hw_brps);
3442  LOG_ERROR("Failed to alloc memory for HW watchpoints!");
3443  return ERROR_FAIL;
3444  }
3445  xtensa->sw_brps = calloc(XT_SW_BREAKPOINTS_MAX_NUM, sizeof(struct xtensa_sw_breakpoint));
3446  if (!xtensa->sw_brps) {
3447  free(xtensa->hw_brps);
3448  free(xtensa->hw_wps);
3449  LOG_ERROR("Failed to alloc memory for SW breakpoints!");
3450  return ERROR_FAIL;
3451  }
3452 
3453  xtensa->spill_loc = 0xffffffff;
3454  xtensa->spill_bytes = 0;
3455  xtensa->spill_buf = NULL;
3456  xtensa->probe_lsddr32p = -1; /* Probe for fast load/store operations */
3457 
3459 }
3460 
3462 {
3463  struct xtensa *xtensa = target_to_xtensa(target);
3464  struct reg_cache *cache = xtensa->core_cache;
3465 
3466  if (cache) {
3468  for (unsigned int i = 0; i < cache->num_regs; i++) {
3469  free(xtensa->algo_context_backup[i]);
3470  free(cache->reg_list[i].value);
3471  }
3472  free(xtensa->algo_context_backup);
3473  free(cache->reg_list);
3474  free(cache);
3475  }
3476  xtensa->core_cache = NULL;
3478 
3479  if (xtensa->empty_regs) {
3480  for (unsigned int i = 0; i < xtensa->dbregs_num; i++) {
3481  free((void *)xtensa->empty_regs[i].name);
3482  free(xtensa->empty_regs[i].value);
3483  }
3484  free(xtensa->empty_regs);
3485  }
3486  xtensa->empty_regs = NULL;
3487  if (xtensa->optregs) {
3488  for (unsigned int i = 0; i < xtensa->num_optregs; i++)
3489  free((void *)xtensa->optregs[i].name);
3490  free(xtensa->optregs);
3491  }
3492  xtensa->optregs = NULL;
3493 }
3494 
3496 {
3497  struct xtensa *xtensa = target_to_xtensa(target);
3498 
3499  LOG_DEBUG("start");
3500 
3501  if (target_was_examined(target)) {
3503  if (ret != ERROR_OK) {
3504  LOG_ERROR("Failed to queue OCDDCR_ENABLEOCD clear operation!");
3505  return;
3506  }
3509  if (ret != ERROR_OK) {
3510  LOG_ERROR("Failed to clear OCDDCR_ENABLEOCD!");
3511  return;
3512  }
3514  }
3516  free(xtensa->hw_brps);
3517  free(xtensa->hw_wps);
3518  free(xtensa->sw_brps);
3519  if (xtensa->spill_buf) {
3520  free(xtensa->spill_buf);
3521  xtensa->spill_buf = NULL;
3522  }
3523  for (enum xtensa_ar_scratch_set_e s = 0; s < XT_AR_SCRATCH_NUM; s++)
3524  free(xtensa->scratch_ars[s].chrval);
3525  free(xtensa->core_config);
3526 }
3527 
3528 const char *xtensa_get_gdb_arch(const struct target *target)
3529 {
3530  return "xtensa";
3531 }
3532 
3533 /* exe <ascii-encoded hexadecimal instruction bytes> */
3534 static COMMAND_HELPER(xtensa_cmd_exe_do, struct target *target)
3535 {
3536  struct xtensa *xtensa = target_to_xtensa(target);
3537 
3538  if (CMD_ARGC != 1)
3540 
3541  /* Process ascii-encoded hex byte string */
3542  const char *parm = CMD_ARGV[0];
3543  unsigned int parm_len = strlen(parm);
3544  if ((parm_len >= 64) || (parm_len & 1)) {
3545  command_print(CMD, "Invalid parameter length (%d): must be even, < 64 characters", parm_len);
3547  }
3548 
3549  uint8_t ops[32];
3550  memset(ops, 0, 32);
3551  unsigned int oplen = parm_len / 2;
3552  char encoded_byte[3] = { 0, 0, 0 };
3553  for (unsigned int i = 0; i < oplen; i++) {
3554  encoded_byte[0] = *parm++;
3555  encoded_byte[1] = *parm++;
3556  ops[i] = strtoul(encoded_byte, NULL, 16);
3557  }
3558 
3559  /* GDB must handle state save/restore.
3560  * Flush reg cache in case spill location is in an AR
3561  * Update CPENABLE only for this execution; later restore cached copy
3562  * Keep a copy of exccause in case executed code triggers an exception
3563  */
3565  if (status != ERROR_OK) {
3566  command_print(CMD, "%s: Failed to write back register cache.", target_name(target));
3567  return ERROR_FAIL;
3568  }
3578 
3579  /* Queue instruction list and execute everything */
3580  LOG_TARGET_DEBUG(target, "execute stub: %s", CMD_ARGV[0]);
3581  xtensa_queue_exec_ins_wide(xtensa, ops, oplen); /* Handles endian-swap */
3583  if (status != ERROR_OK) {
3584  command_print(CMD, "exec: queue error %d", status);
3585  } else {
3587  if (status != ERROR_OK)
3588  command_print(CMD, "exec: status error %d", status);
3589  }
3590 
3591  /* Reread register cache and restore saved regs after instruction execution */
3593  command_print(CMD, "post-exec: register fetch error");
3594  if (status != ERROR_OK) {
3595  command_print(CMD, "post-exec: EXCCAUSE 0x%02" PRIx32,
3597  }
3600  return status;
3601 }
3602 
3603 COMMAND_HANDLER(xtensa_cmd_exe)
3604 {
3605  return CALL_COMMAND_HANDLER(xtensa_cmd_exe_do, get_current_target(CMD_CTX));
3606 }
3607 
3608 /* xtdef <name> */
3609 COMMAND_HELPER(xtensa_cmd_xtdef_do, struct xtensa *xtensa)
3610 {
3611  if (CMD_ARGC != 1)
3613 
3614  const char *core_name = CMD_ARGV[0];
3615  if (strcasecmp(core_name, "LX") == 0) {
3617  } else if (strcasecmp(core_name, "NX") == 0) {
3619  } else {
3620  command_print(CMD, "xtdef [LX|NX]\n");
3622  }
3623  return ERROR_OK;
3624 }
3625 
3626 COMMAND_HANDLER(xtensa_cmd_xtdef)
3627 {
3628  return CALL_COMMAND_HANDLER(xtensa_cmd_xtdef_do,
3630 }
3631 
3632 static inline bool xtensa_cmd_xtopt_legal_val(char *opt, int val, int min, int max)
3633 {
3634  if ((val < min) || (val > max)) {
3635  LOG_ERROR("xtopt %s (%d) out of range [%d..%d]\n", opt, val, min, max);
3636  return false;
3637  }
3638  return true;
3639 }
3640 
3641 /* xtopt <name> <value> */
3642 COMMAND_HELPER(xtensa_cmd_xtopt_do, struct xtensa *xtensa)
3643 {
3644  if (CMD_ARGC != 2)
3646 
3647  const char *opt_name = CMD_ARGV[0];
3648  int opt_val = strtol(CMD_ARGV[1], NULL, 0);
3649  if (strcasecmp(opt_name, "arnum") == 0) {
3650  if (!xtensa_cmd_xtopt_legal_val("arnum", opt_val, 0, 64))
3652  xtensa->core_config->aregs_num = opt_val;
3653  } else if (strcasecmp(opt_name, "windowed") == 0) {
3654  if (!xtensa_cmd_xtopt_legal_val("windowed", opt_val, 0, 1))
3656  xtensa->core_config->windowed = opt_val;
3657  } else if (strcasecmp(opt_name, "cpenable") == 0) {
3658  if (!xtensa_cmd_xtopt_legal_val("cpenable", opt_val, 0, 1))
3660  xtensa->core_config->coproc = opt_val;
3661  } else if (strcasecmp(opt_name, "exceptions") == 0) {
3662  if (!xtensa_cmd_xtopt_legal_val("exceptions", opt_val, 0, 1))
3664  xtensa->core_config->exceptions = opt_val;
3665  } else if (strcasecmp(opt_name, "intnum") == 0) {
3666  if (!xtensa_cmd_xtopt_legal_val("intnum", opt_val, 0, 32))
3668  xtensa->core_config->irq.enabled = (opt_val > 0);
3669  xtensa->core_config->irq.irq_num = opt_val;
3670  } else if (strcasecmp(opt_name, "hipriints") == 0) {
3671  if (!xtensa_cmd_xtopt_legal_val("hipriints", opt_val, 0, 1))
3673  xtensa->core_config->high_irq.enabled = opt_val;
3674  } else if (strcasecmp(opt_name, "excmlevel") == 0) {
3675  if (!xtensa_cmd_xtopt_legal_val("excmlevel", opt_val, 1, 6))
3678  command_print(CMD, "xtopt excmlevel requires hipriints\n");
3680  }
3681  xtensa->core_config->high_irq.excm_level = opt_val;
3682  } else if (strcasecmp(opt_name, "intlevels") == 0) {
3683  if (xtensa->core_config->core_type == XT_LX) {
3684  if (!xtensa_cmd_xtopt_legal_val("intlevels", opt_val, 2, 6))
3686  } else {
3687  if (!xtensa_cmd_xtopt_legal_val("intlevels", opt_val, 1, 255))
3689  }
3691  command_print(CMD, "xtopt intlevels requires hipriints\n");
3693  }
3694  xtensa->core_config->high_irq.level_num = opt_val;
3695  } else if (strcasecmp(opt_name, "debuglevel") == 0) {
3696  if (xtensa->core_config->core_type == XT_LX) {
3697  if (!xtensa_cmd_xtopt_legal_val("debuglevel", opt_val, 2, 6))
3699  } else {
3700  if (!xtensa_cmd_xtopt_legal_val("debuglevel", opt_val, 0, 0))
3702  }
3704  xtensa->core_config->debug.irq_level = opt_val;
3705  } else if (strcasecmp(opt_name, "ibreaknum") == 0) {
3706  if (!xtensa_cmd_xtopt_legal_val("ibreaknum", opt_val, 0, 2))
3708  xtensa->core_config->debug.ibreaks_num = opt_val;
3709  } else if (strcasecmp(opt_name, "dbreaknum") == 0) {
3710  if (!xtensa_cmd_xtopt_legal_val("dbreaknum", opt_val, 0, 2))
3712  xtensa->core_config->debug.dbreaks_num = opt_val;
3713  } else if (strcasecmp(opt_name, "tracemem") == 0) {
3714  if (!xtensa_cmd_xtopt_legal_val("tracemem", opt_val, 0, 256 * 1024))
3716  xtensa->core_config->trace.mem_sz = opt_val;
3717  xtensa->core_config->trace.enabled = (opt_val > 0);
3718  } else if (strcasecmp(opt_name, "tracememrev") == 0) {
3719  if (!xtensa_cmd_xtopt_legal_val("tracememrev", opt_val, 0, 1))
3722  } else if (strcasecmp(opt_name, "perfcount") == 0) {
3723  if (!xtensa_cmd_xtopt_legal_val("perfcount", opt_val, 0, 8))
3725  xtensa->core_config->debug.perfcount_num = opt_val;
3726  } else {
3727  LOG_WARNING("Unknown xtensa command ignored: \"xtopt %s %s\"", CMD_ARGV[0], CMD_ARGV[1]);
3728  return ERROR_OK;
3729  }
3730 
3731  return ERROR_OK;
3732 }
3733 
3734 COMMAND_HANDLER(xtensa_cmd_xtopt)
3735 {
3736  return CALL_COMMAND_HANDLER(xtensa_cmd_xtopt_do,
3738 }
3739 
3740 /* xtmem <type> [parameters] */
3741 COMMAND_HELPER(xtensa_cmd_xtmem_do, struct xtensa *xtensa)
3742 {
3743  struct xtensa_cache_config *cachep = NULL;
3744  struct xtensa_local_mem_config *memp = NULL;
3745  int mem_access = 0;
3746  bool is_dcache = false;
3747 
3748  if (CMD_ARGC == 0)
3750 
3751  const char *mem_name = CMD_ARGV[0];
3752  if (strcasecmp(mem_name, "icache") == 0) {
3753  cachep = &xtensa->core_config->icache;
3754  } else if (strcasecmp(mem_name, "dcache") == 0) {
3755  cachep = &xtensa->core_config->dcache;
3756  is_dcache = true;
3757  } else if (strcasecmp(mem_name, "l2cache") == 0) {
3758  /* TODO: support L2 cache */
3759  } else if (strcasecmp(mem_name, "l2addr") == 0) {
3760  /* TODO: support L2 cache */
3761  } else if (strcasecmp(mem_name, "iram") == 0) {
3762  memp = &xtensa->core_config->iram;
3763  mem_access = XT_MEM_ACCESS_READ | XT_MEM_ACCESS_WRITE;
3764  } else if (strcasecmp(mem_name, "dram") == 0) {
3765  memp = &xtensa->core_config->dram;
3766  mem_access = XT_MEM_ACCESS_READ | XT_MEM_ACCESS_WRITE;
3767  } else if (strcasecmp(mem_name, "sram") == 0) {
3768  memp = &xtensa->core_config->sram;
3769  mem_access = XT_MEM_ACCESS_READ | XT_MEM_ACCESS_WRITE;
3770  } else if (strcasecmp(mem_name, "irom") == 0) {
3771  memp = &xtensa->core_config->irom;
3772  mem_access = XT_MEM_ACCESS_READ;
3773  } else if (strcasecmp(mem_name, "drom") == 0) {
3774  memp = &xtensa->core_config->drom;
3775  mem_access = XT_MEM_ACCESS_READ;
3776  } else if (strcasecmp(mem_name, "srom") == 0) {
3777  memp = &xtensa->core_config->srom;
3778  mem_access = XT_MEM_ACCESS_READ;
3779  } else {
3780  command_print(CMD, "xtmem types: <icache|dcache|l2cache|l2addr|iram|irom|dram|drom|sram|srom>\n");
3782  }
3783 
3784  if (cachep) {
3785  if (CMD_ARGC != 4 && CMD_ARGC != 5)
3787  cachep->line_size = strtoul(CMD_ARGV[1], NULL, 0);
3788  cachep->size = strtoul(CMD_ARGV[2], NULL, 0);
3789  cachep->way_count = strtoul(CMD_ARGV[3], NULL, 0);
3790  cachep->writeback = ((CMD_ARGC == 5) && is_dcache) ?
3791  strtoul(CMD_ARGV[4], NULL, 0) : 0;
3792  } else if (memp) {
3793  if (CMD_ARGC != 3)
3795  struct xtensa_local_mem_region_config *memcfgp = &memp->regions[memp->count];
3796  memcfgp->base = strtoul(CMD_ARGV[1], NULL, 0);
3797  memcfgp->size = strtoul(CMD_ARGV[2], NULL, 0);
3798  memcfgp->access = mem_access;
3799  memp->count++;
3800  }
3801 
3802  return ERROR_OK;
3803 }
3804 
3805 COMMAND_HANDLER(xtensa_cmd_xtmem)
3806 {
3807  return CALL_COMMAND_HANDLER(xtensa_cmd_xtmem_do,
3809 }
3810 
3811 /* xtmpu <num FG seg> <min seg size> <lockable> <executeonly> */
3812 COMMAND_HELPER(xtensa_cmd_xtmpu_do, struct xtensa *xtensa)
3813 {
3814  if (CMD_ARGC != 4)
3816 
3817  unsigned int nfgseg = strtoul(CMD_ARGV[0], NULL, 0);
3818  unsigned int minsegsize = strtoul(CMD_ARGV[1], NULL, 0);
3819  unsigned int lockable = strtoul(CMD_ARGV[2], NULL, 0);
3820  unsigned int execonly = strtoul(CMD_ARGV[3], NULL, 0);
3821 
3822  if ((nfgseg > 32)) {
3823  command_print(CMD, "<nfgseg> must be within [0..32]\n");
3825  } else if (minsegsize & (minsegsize - 1)) {
3826  command_print(CMD, "<minsegsize> must be a power of 2 >= 32\n");
3828  } else if (lockable > 1) {
3829  command_print(CMD, "<lockable> must be 0 or 1\n");
3831  } else if (execonly > 1) {
3832  command_print(CMD, "<execonly> must be 0 or 1\n");
3834  }
3835 
3836  xtensa->core_config->mpu.enabled = true;
3837  xtensa->core_config->mpu.nfgseg = nfgseg;
3838  xtensa->core_config->mpu.minsegsize = minsegsize;
3839  xtensa->core_config->mpu.lockable = lockable;
3840  xtensa->core_config->mpu.execonly = execonly;
3841  return ERROR_OK;
3842 }
3843 
3844 COMMAND_HANDLER(xtensa_cmd_xtmpu)
3845 {
3846  return CALL_COMMAND_HANDLER(xtensa_cmd_xtmpu_do,
3848 }
3849 
3850 /* xtmmu <NIREFILLENTRIES> <NDREFILLENTRIES> <IVARWAY56> <DVARWAY56> */
3851 COMMAND_HELPER(xtensa_cmd_xtmmu_do, struct xtensa *xtensa)
3852 {
3853  if (CMD_ARGC != 2)
3855 
3856  unsigned int nirefillentries = strtoul(CMD_ARGV[0], NULL, 0);
3857  unsigned int ndrefillentries = strtoul(CMD_ARGV[1], NULL, 0);
3858  if ((nirefillentries != 16) && (nirefillentries != 32)) {
3859  command_print(CMD, "<nirefillentries> must be 16 or 32\n");
3861  } else if ((ndrefillentries != 16) && (ndrefillentries != 32)) {
3862  command_print(CMD, "<ndrefillentries> must be 16 or 32\n");
3864  }
3865 
3866  xtensa->core_config->mmu.enabled = true;
3867  xtensa->core_config->mmu.itlb_entries_count = nirefillentries;
3868  xtensa->core_config->mmu.dtlb_entries_count = ndrefillentries;
3869  return ERROR_OK;
3870 }
3871 
3872 COMMAND_HANDLER(xtensa_cmd_xtmmu)
3873 {
3874  return CALL_COMMAND_HANDLER(xtensa_cmd_xtmmu_do,
3876 }
3877 
3878 /* xtregs <numregs>
3879  * xtreg <regname> <regnum> */
3880 COMMAND_HELPER(xtensa_cmd_xtreg_do, struct xtensa *xtensa)
3881 {
3882  if (CMD_ARGC == 1) {
3883  int32_t numregs = strtoul(CMD_ARGV[0], NULL, 0);
3884  if ((numregs <= 0) || (numregs > UINT16_MAX)) {
3885  command_print(CMD, "xtreg <numregs>: Invalid 'numregs' (%d)", numregs);
3887  }
3888  if ((xtensa->genpkt_regs_num > 0) && (numregs < (int32_t)xtensa->genpkt_regs_num)) {
3889  command_print(CMD, "xtregs (%d) must be larger than numgenregs (%d) (if xtregfmt specified)",
3890  numregs, xtensa->genpkt_regs_num);
3892  }
3893  xtensa->total_regs_num = numregs;
3894  xtensa->core_regs_num = 0;
3895  xtensa->num_optregs = 0;
3896  /* A little more memory than required, but saves a second initialization pass */
3897  xtensa->optregs = calloc(xtensa->total_regs_num, sizeof(struct xtensa_reg_desc));
3898  if (!xtensa->optregs) {
3899  LOG_ERROR("Failed to allocate xtensa->optregs!");
3900  return ERROR_FAIL;
3901  }
3902  return ERROR_OK;
3903  } else if (CMD_ARGC != 2) {
3905  }
3906 
3907  /* "xtregfmt contiguous" must be specified prior to the first "xtreg" definition
3908  * if general register (g-packet) requests or contiguous register maps are supported */
3910  xtensa->contiguous_regs_desc = calloc(xtensa->total_regs_num, sizeof(struct xtensa_reg_desc *));
3911  if (!xtensa->contiguous_regs_desc) {
3912  LOG_ERROR("Failed to allocate xtensa->contiguous_regs_desc!");
3913  return ERROR_FAIL;
3914  }
3915  }
3916 
3917  const char *regname = CMD_ARGV[0];
3918  unsigned int regnum = strtoul(CMD_ARGV[1], NULL, 0);
3919  if (regnum > UINT16_MAX) {
3920  command_print(CMD, "<regnum> must be a 16-bit number");
3922  }
3923 
3925  if (xtensa->total_regs_num)
3926  command_print(CMD, "'xtreg %s 0x%04x': Too many registers (%d expected, %d core %d extended)",
3927  regname, regnum,
3929  else
3930  command_print(CMD, "'xtreg %s 0x%04x': Number of registers unspecified",
3931  regname, regnum);
3932  return ERROR_FAIL;
3933  }
3934 
3935  /* Determine whether register belongs in xtensa_regs[] or xtensa->xtensa_spec_regs[] */
3936  struct xtensa_reg_desc *rptr = &xtensa->optregs[xtensa->num_optregs];
3937  bool is_extended_reg = true;
3938  unsigned int ridx;
3939  for (ridx = 0; ridx < XT_NUM_REGS; ridx++) {
3940  if (strcmp(CMD_ARGV[0], xtensa_regs[ridx].name) == 0) {
3941  /* Flag core register as defined */
3942  rptr = &xtensa_regs[ridx];
3943  xtensa->core_regs_num++;
3944  is_extended_reg = false;
3945  break;
3946  }
3947  }
3948 
3949  rptr->exist = true;
3950  if (is_extended_reg) {
3951  /* Register ID, debugger-visible register ID */
3952  rptr->name = strdup(CMD_ARGV[0]);
3953  rptr->dbreg_num = regnum;
3954  rptr->reg_num = (regnum & XT_REG_INDEX_MASK);
3955  xtensa->num_optregs++;
3956 
3957  /* Register type */
3958  if ((regnum & XT_REG_GENERAL_MASK) == XT_REG_GENERAL_VAL) {
3959  rptr->type = XT_REG_GENERAL;
3960  } else if ((regnum & XT_REG_USER_MASK) == XT_REG_USER_VAL) {
3961  rptr->type = XT_REG_USER;
3962  } else if ((regnum & XT_REG_FR_MASK) == XT_REG_FR_VAL) {
3963  rptr->type = XT_REG_FR;
3964  } else if ((regnum & XT_REG_SPECIAL_MASK) == XT_REG_SPECIAL_VAL) {
3965  rptr->type = XT_REG_SPECIAL;
3966  } else if ((regnum & XT_REG_RELGEN_MASK) == XT_REG_RELGEN_VAL) {
3967  /* WARNING: For these registers, regnum points to the
3968  * index of the corresponding ARx registers, NOT to
3969  * the processor register number! */
3970  rptr->type = XT_REG_RELGEN;
3971  rptr->reg_num += XT_REG_IDX_ARFIRST;
3972  rptr->dbreg_num += XT_REG_IDX_ARFIRST;
3973  } else if ((regnum & XT_REG_TIE_MASK) != 0) {
3974  rptr->type = XT_REG_TIE;
3975  } else {
3976  rptr->type = XT_REG_OTHER;
3977  }
3978 
3979  /* Register flags: includes intsetN, intclearN for LX8 */
3980  if ((strcmp(rptr->name, "mmid") == 0) || (strcmp(rptr->name, "eraccess") == 0) ||
3981  (strcmp(rptr->name, "ddr") == 0) || (strncmp(rptr->name, "intset", 6) == 0) ||
3982  (strncmp(rptr->name, "intclear", 8) == 0) || (strcmp(rptr->name, "mesrclr") == 0))
3983  rptr->flags = XT_REGF_NOREAD;
3984  else
3985  rptr->flags = 0;
3986 
3988  xtensa->core_config->core_type == XT_LX && rptr->type == XT_REG_SPECIAL) {
3990  LOG_DEBUG("Setting PS (%s) index to %d", rptr->name, xtensa->eps_dbglevel_idx);
3991  }
3992  if (xtensa->core_config->core_type == XT_NX) {
3994  if (strcmp(rptr->name, "ibreakc0") == 0)
3995  idx = XT_NX_REG_IDX_IBREAKC0;
3996  else if (strcmp(rptr->name, "wb") == 0)
3997  idx = XT_NX_REG_IDX_WB;
3998  else if (strcmp(rptr->name, "ms") == 0)
3999  idx = XT_NX_REG_IDX_MS;
4000  else if (strcmp(rptr->name, "ievec") == 0)
4001  idx = XT_NX_REG_IDX_IEVEC;
4002  else if (strcmp(rptr->name, "ieextern") == 0)
4003  idx = XT_NX_REG_IDX_IEEXTERN;
4004  else if (strcmp(rptr->name, "mesr") == 0)
4005  idx = XT_NX_REG_IDX_MESR;
4006  else if (strcmp(rptr->name, "mesrclr") == 0)
4007  idx = XT_NX_REG_IDX_MESRCLR;
4008  if (idx < XT_NX_REG_IDX_NUM) {
4009  if (xtensa->nx_reg_idx[idx] != 0) {
4010  command_print(CMD, "nx_reg_idx[%d] previously set to %d",
4011  idx, xtensa->nx_reg_idx[idx]);
4012  return ERROR_FAIL;
4013  }
4015  LOG_DEBUG("NX reg %s: index %d (%d)",
4016  rptr->name, xtensa->nx_reg_idx[idx], idx);
4017  }
4018  }
4019  } else if (strcmp(rptr->name, "cpenable") == 0) {
4020  xtensa->core_config->coproc = true;
4021  }
4022 
4023  /* Build out list of contiguous registers in specified order */
4024  unsigned int running_reg_count = xtensa->num_optregs + xtensa->core_regs_num;
4026  assert((running_reg_count <= xtensa->total_regs_num) && "contiguous register address internal error!");
4027  xtensa->contiguous_regs_desc[running_reg_count - 1] = rptr;
4028  }
4030  LOG_DEBUG("Added %s register %-16s: 0x%04x/0x%02x t%d (%d of %d)",
4031  is_extended_reg ? "config-specific" : "core",
4032  rptr->name, rptr->dbreg_num, rptr->reg_num, rptr->type,
4033  is_extended_reg ? xtensa->num_optregs : ridx,
4034  is_extended_reg ? xtensa->total_regs_num : XT_NUM_REGS);
4035  return ERROR_OK;
4036 }
4037 
4038 COMMAND_HANDLER(xtensa_cmd_xtreg)
4039 {
4040  return CALL_COMMAND_HANDLER(xtensa_cmd_xtreg_do,
4042 }
4043 
4044 /* xtregfmt <contiguous|sparse> [numgregs] */
4045 COMMAND_HELPER(xtensa_cmd_xtregfmt_do, struct xtensa *xtensa)
4046 {
4047  if ((CMD_ARGC == 1) || (CMD_ARGC == 2)) {
4048  if (!strcasecmp(CMD_ARGV[0], "sparse")) {
4049  return ERROR_OK;
4050  } else if (!strcasecmp(CMD_ARGV[0], "contiguous")) {
4051  xtensa->regmap_contiguous = true;
4052  if (CMD_ARGC == 2) {
4053  unsigned int numgregs = strtoul(CMD_ARGV[1], NULL, 0);
4054  if ((numgregs <= 0) ||
4055  ((numgregs > xtensa->total_regs_num) &&
4056  (xtensa->total_regs_num > 0))) {
4057  command_print(CMD, "xtregfmt: if specified, numgregs (%d) must be <= numregs (%d)",
4058  numgregs, xtensa->total_regs_num);
4060  }
4061  xtensa->genpkt_regs_num = numgregs;
4062  }
4063  return ERROR_OK;
4064  }
4065  }
4067 }
4068 
4069 COMMAND_HANDLER(xtensa_cmd_xtregfmt)
4070 {
4071  return CALL_COMMAND_HANDLER(xtensa_cmd_xtregfmt_do,
4073 }
4074 
4075 COMMAND_HELPER(xtensa_cmd_permissive_mode_do, struct xtensa *xtensa)
4076 {
4077  return CALL_COMMAND_HANDLER(handle_command_parse_bool,
4078  &xtensa->permissive_mode, "xtensa permissive mode");
4079 }
4080 
4081 COMMAND_HANDLER(xtensa_cmd_permissive_mode)
4082 {
4083  return CALL_COMMAND_HANDLER(xtensa_cmd_permissive_mode_do,
4085 }
4086 
4087 /* perfmon_enable <counter_id> <select> [mask] [kernelcnt] [tracelevel] */
4088 COMMAND_HELPER(xtensa_cmd_perfmon_enable_do, struct xtensa *xtensa)
4089 {
4090  struct xtensa_perfmon_config config = {
4091  .mask = 0xffff,
4092  .kernelcnt = 0,
4093  .tracelevel = -1 /* use DEBUGLEVEL by default */
4094  };
4095 
4096  if (CMD_ARGC < 2 || CMD_ARGC > 6)
4098 
4099  unsigned int counter_id = strtoul(CMD_ARGV[0], NULL, 0);
4100  if (counter_id >= XTENSA_MAX_PERF_COUNTERS) {
4101  command_print(CMD, "counter_id should be < %d", XTENSA_MAX_PERF_COUNTERS);
4103  }
4104 
4105  config.select = strtoul(CMD_ARGV[1], NULL, 0);
4106  if (config.select > XTENSA_MAX_PERF_SELECT) {
4107  command_print(CMD, "select should be < %d", XTENSA_MAX_PERF_SELECT);
4109  }
4110 
4111  if (CMD_ARGC >= 3) {
4112  config.mask = strtoul(CMD_ARGV[2], NULL, 0);
4113  if (config.mask > XTENSA_MAX_PERF_MASK) {
4114  command_print(CMD, "mask should be < %d", XTENSA_MAX_PERF_MASK);
4116  }
4117  }
4118 
4119  if (CMD_ARGC >= 4) {
4120  config.kernelcnt = strtoul(CMD_ARGV[3], NULL, 0);
4121  if (config.kernelcnt > 1) {
4122  command_print(CMD, "kernelcnt should be 0 or 1");
4124  }
4125  }
4126 
4127  if (CMD_ARGC >= 5) {
4128  config.tracelevel = strtoul(CMD_ARGV[4], NULL, 0);
4129  if (config.tracelevel > 7) {
4130  command_print(CMD, "tracelevel should be <=7");
4132  }
4133  }
4134 
4135  if (config.tracelevel == -1)
4136  config.tracelevel = xtensa->core_config->debug.irq_level;
4137 
4138  return xtensa_dm_perfmon_enable(&xtensa->dbg_mod, counter_id, &config);
4139 }
4140 
4141 COMMAND_HANDLER(xtensa_cmd_perfmon_enable)
4142 {
4143  return CALL_COMMAND_HANDLER(xtensa_cmd_perfmon_enable_do,
4145 }
4146 
4147 /* perfmon_dump [counter_id] */
4148 COMMAND_HELPER(xtensa_cmd_perfmon_dump_do, struct xtensa *xtensa)
4149 {
4150  if (CMD_ARGC > 1)
4152 
4153  int counter_id = -1;
4154  if (CMD_ARGC == 1) {
4155  counter_id = strtol(CMD_ARGV[0], NULL, 0);
4156  if (counter_id > XTENSA_MAX_PERF_COUNTERS) {
4157  command_print(CMD, "counter_id should be < %d", XTENSA_MAX_PERF_COUNTERS);
4159  }
4160  }
4161 
4162  unsigned int counter_start = (counter_id < 0) ? 0 : counter_id;
4163  unsigned int counter_end = (counter_id < 0) ? XTENSA_MAX_PERF_COUNTERS : counter_id + 1;
4164  for (unsigned int counter = counter_start; counter < counter_end; ++counter) {
4165  char result_buf[128] = { 0 };
4166  size_t result_pos = snprintf(result_buf, sizeof(result_buf), "Counter %d: ", counter);
4167  struct xtensa_perfmon_result result;
4168  int res = xtensa_dm_perfmon_dump(&xtensa->dbg_mod, counter, &result);
4169  if (res != ERROR_OK)
4170  return res;
4171  snprintf(result_buf + result_pos, sizeof(result_buf) - result_pos,
4172  "%-12" PRIu64 "%s",
4173  result.value,
4174  result.overflow ? " (overflow)" : "");
4175  command_print(CMD, "%s", result_buf);
4176  }
4177 
4178  return ERROR_OK;
4179 }
4180 
4181 COMMAND_HANDLER(xtensa_cmd_perfmon_dump)
4182 {
4183  return CALL_COMMAND_HANDLER(xtensa_cmd_perfmon_dump_do,
4185 }
4186 
4187 COMMAND_HELPER(xtensa_cmd_mask_interrupts_do, struct xtensa *xtensa)
4188 {
4189  int state = -1;
4190 
4191  if (CMD_ARGC < 1) {
4192  const char *st;
4194  if (state == XT_STEPPING_ISR_ON)
4195  st = "OFF";
4196  else if (state == XT_STEPPING_ISR_OFF)
4197  st = "ON";
4198  else
4199  st = "UNKNOWN";
4200  command_print(CMD, "Current ISR step mode: %s", st);
4201  return ERROR_OK;
4202  }
4203 
4204  /* Masking is ON -> interrupts during stepping are OFF, and vice versa */
4205  if (!strcasecmp(CMD_ARGV[0], "off"))
4207  else if (!strcasecmp(CMD_ARGV[0], "on"))
4209 
4210  if (state == -1) {
4211  command_print(CMD, "Argument unknown. Please pick one of ON, OFF");
4212  return ERROR_FAIL;
4213  }
4215  return ERROR_OK;
4216 }
4217 
4218 COMMAND_HANDLER(xtensa_cmd_mask_interrupts)
4219 {
4220  return CALL_COMMAND_HANDLER(xtensa_cmd_mask_interrupts_do,
4222 }
4223 
4224 COMMAND_HELPER(xtensa_cmd_smpbreak_do, struct target *target)
4225 {
4226  int res;
4227  uint32_t val = 0;
4228 
4229  if (CMD_ARGC >= 1) {
4230  for (unsigned int i = 0; i < CMD_ARGC; i++) {
4231  if (!strcasecmp(CMD_ARGV[0], "none")) {
4232  val = 0;
4233  } else if (!strcasecmp(CMD_ARGV[i], "BreakIn")) {
4234  val |= OCDDCR_BREAKINEN;
4235  } else if (!strcasecmp(CMD_ARGV[i], "BreakOut")) {
4236  val |= OCDDCR_BREAKOUTEN;
4237  } else if (!strcasecmp(CMD_ARGV[i], "RunStallIn")) {
4238  val |= OCDDCR_RUNSTALLINEN;
4239  } else if (!strcasecmp(CMD_ARGV[i], "DebugModeOut")) {
4240  val |= OCDDCR_DEBUGMODEOUTEN;
4241  } else if (!strcasecmp(CMD_ARGV[i], "BreakInOut")) {
4243  } else if (!strcasecmp(CMD_ARGV[i], "RunStall")) {
4245  } else {
4246  command_print(CMD, "Unknown arg %s", CMD_ARGV[i]);
4247  command_print(
4248  CMD,
4249  "use either BreakInOut, None or RunStall as arguments, or any combination of BreakIn, BreakOut, RunStallIn and DebugModeOut.");
4250  return ERROR_OK;
4251  }
4252  }
4253  res = xtensa_smpbreak_set(target, val);
4254  if (res != ERROR_OK)
4255  command_print(CMD, "Failed to set smpbreak config %d", res);
4256  } else {
4257  struct xtensa *xtensa = target_to_xtensa(target);
4258  res = xtensa_smpbreak_read(xtensa, &val);
4259  if (res == ERROR_OK)
4260  command_print(CMD, "Current bits set:%s%s%s%s",
4261  (val & OCDDCR_BREAKINEN) ? " BreakIn" : "",
4262  (val & OCDDCR_BREAKOUTEN) ? " BreakOut" : "",
4263  (val & OCDDCR_RUNSTALLINEN) ? " RunStallIn" : "",
4264  (val & OCDDCR_DEBUGMODEOUTEN) ? " DebugModeOut" : ""
4265  );
4266  else
4267  command_print(CMD, "Failed to get smpbreak config %d", res);
4268  }
4269  return res;
4270 }
4271 
4272 COMMAND_HANDLER(xtensa_cmd_smpbreak)
4273 {
4274  return CALL_COMMAND_HANDLER(xtensa_cmd_smpbreak_do,
4276 }
4277 
4278 COMMAND_HELPER(xtensa_cmd_dm_rw_do, struct xtensa *xtensa)
4279 {
4280  if (CMD_ARGC == 1) {
4281  // read: xtensa dm addr
4282  uint32_t addr = strtoul(CMD_ARGV[0], NULL, 0);
4283  uint32_t val;
4284  int res = xtensa_dm_read(&xtensa->dbg_mod, addr, &val);
4285  if (res == ERROR_OK)
4286  command_print(CMD, "xtensa DM(0x%08" PRIx32 ") -> 0x%08" PRIx32, addr, val);
4287  else
4288  command_print(CMD, "xtensa DM(0x%08" PRIx32 ") : read ERROR %" PRId32, addr, res);
4289  return res;
4290  } else if (CMD_ARGC == 2) {
4291  // write: xtensa dm addr value
4292  uint32_t addr = strtoul(CMD_ARGV[0], NULL, 0);
4293  uint32_t val = strtoul(CMD_ARGV[1], NULL, 0);
4294  int res = xtensa_dm_write(&xtensa->dbg_mod, addr, val);
4295  if (res == ERROR_OK)
4296  command_print(CMD, "xtensa DM(0x%08" PRIx32 ") <- 0x%08" PRIx32, addr, val);
4297  else
4298  command_print(CMD, "xtensa DM(0x%08" PRIx32 ") : write ERROR %" PRId32, addr, res);
4299  return res;
4300  }
4302 }
4303 
4304 COMMAND_HANDLER(xtensa_cmd_dm_rw)
4305 {
4306  return CALL_COMMAND_HANDLER(xtensa_cmd_dm_rw_do,
4308 }
4309 
4310 COMMAND_HELPER(xtensa_cmd_tracestart_do, struct xtensa *xtensa)
4311 {
4313  struct xtensa_trace_start_config cfg = {
4314  .stoppc = 0,
4315  .stopmask = XTENSA_STOPMASK_DISABLED,
4316  .after = 0,
4317  .after_is_words = false
4318  };
4319 
4320  /* Parse arguments */
4321  for (unsigned int i = 0; i < CMD_ARGC; i++) {
4322  if ((!strcasecmp(CMD_ARGV[i], "pc")) && CMD_ARGC > i) {
4323  char *e;
4324  i++;
4325  cfg.stoppc = strtol(CMD_ARGV[i], &e, 0);
4326  cfg.stopmask = 0;
4327  if (*e == '/')
4328  cfg.stopmask = strtol(e, NULL, 0);
4329  } else if ((!strcasecmp(CMD_ARGV[i], "after")) && CMD_ARGC > i) {
4330  i++;
4331  cfg.after = strtol(CMD_ARGV[i], NULL, 0);
4332  } else if (!strcasecmp(CMD_ARGV[i], "ins")) {
4333  cfg.after_is_words = 0;
4334  } else if (!strcasecmp(CMD_ARGV[i], "words")) {
4335  cfg.after_is_words = 1;
4336  } else {
4337  command_print(CMD, "Did not understand %s", CMD_ARGV[i]);
4338  return ERROR_FAIL;
4339  }
4340  }
4341 
4343  if (res != ERROR_OK)
4344  return res;
4345  if (trace_status.stat & TRAXSTAT_TRACT) {
4346  LOG_WARNING("Silently stop active tracing!");
4347  res = xtensa_dm_trace_stop(&xtensa->dbg_mod, false);
4348  if (res != ERROR_OK)
4349  return res;
4350  }
4351 
4352  res = xtensa_dm_trace_start(&xtensa->dbg_mod, &cfg);
4353  if (res != ERROR_OK)
4354  return res;
4355 
4356  xtensa->trace_active = true;
4357  command_print(CMD, "Trace started.");
4358  return ERROR_OK;
4359 }
4360 
4361 COMMAND_HANDLER(xtensa_cmd_tracestart)
4362 {
4363  return CALL_COMMAND_HANDLER(xtensa_cmd_tracestart_do,
4365 }
4366 
4367 COMMAND_HELPER(xtensa_cmd_tracestop_do, struct xtensa *xtensa)
4368 {
4370 
4372  if (res != ERROR_OK)
4373  return res;
4374 
4375  if (!(trace_status.stat & TRAXSTAT_TRACT)) {
4376  command_print(CMD, "No trace is currently active.");
4377  return ERROR_FAIL;
4378  }
4379 
4380  res = xtensa_dm_trace_stop(&xtensa->dbg_mod, true);
4381  if (res != ERROR_OK)
4382  return res;
4383 
4384  xtensa->trace_active = false;
4385  command_print(CMD, "Trace stop triggered.");
4386  return ERROR_OK;
4387 }
4388 
4389 COMMAND_HANDLER(xtensa_cmd_tracestop)
4390 {
4391  return CALL_COMMAND_HANDLER(xtensa_cmd_tracestop_do,
4393 }
4394 
4395 COMMAND_HELPER(xtensa_cmd_tracedump_do, struct xtensa *xtensa, const char *fname)
4396 {
4397  struct xtensa_trace_config trace_config;
4399  uint32_t memsz, wmem;
4400 
4402  if (res != ERROR_OK)
4403  return res;
4404 
4405  if (trace_status.stat & TRAXSTAT_TRACT) {
4406  command_print(CMD, "Tracing is still active. Please stop it first.");
4407  return ERROR_FAIL;
4408  }
4409 
4410  res = xtensa_dm_trace_config_read(&xtensa->dbg_mod, &trace_config);
4411  if (res != ERROR_OK)
4412  return res;
4413 
4414  if (!(trace_config.ctrl & TRAXCTRL_TREN)) {
4415  command_print(CMD, "No active trace found; nothing to dump.");
4416  return ERROR_FAIL;
4417  }
4418 
4419  memsz = trace_config.memaddr_end - trace_config.memaddr_start + 1;
4420  command_print(CMD, "Total trace memory: %d words", memsz);
4421  if ((trace_config.addr &
4423  /*Memory hasn't overwritten itself yet. */
4424  wmem = trace_config.addr & TRAXADDR_TADDR_MASK;
4425  command_print(CMD, "...but trace is only %d words", wmem);
4426  if (wmem < memsz)
4427  memsz = wmem;
4428  } else {
4429  if (trace_config.addr & TRAXADDR_TWSAT) {
4430  command_print(CMD, "Real trace is many times longer than that (overflow)");
4431  } else {
4432  uint32_t trc_sz = (trace_config.addr >> TRAXADDR_TWRAP_SHIFT) & TRAXADDR_TWRAP_MASK;
4433  trc_sz = (trc_sz * memsz) + (trace_config.addr & TRAXADDR_TADDR_MASK);
4434  command_print(CMD, "Real trace is %d words, but the start has been truncated.", trc_sz);
4435  }
4436  }
4437 
4438  uint8_t *tracemem = malloc(memsz * 4);
4439  if (!tracemem) {
4440  command_print(CMD, "Failed to alloc memory for trace data!");
4441  return ERROR_FAIL;
4442  }
4443  res = xtensa_dm_trace_data_read(&xtensa->dbg_mod, tracemem, memsz * 4);
4444  if (res != ERROR_OK) {
4445  free(tracemem);
4446  return res;
4447  }
4448 
4449  int f = open(fname, O_WRONLY | O_CREAT | O_TRUNC, 0666);
4450  if (f <= 0) {
4451  free(tracemem);
4452  command_print(CMD, "Unable to open file %s", fname);
4453  return ERROR_FAIL;
4454  }
4455  if (write(f, tracemem, memsz * 4) != (int)memsz * 4)
4456  command_print(CMD, "Unable to write to file %s", fname);
4457  else
4458  command_print(CMD, "Written %d bytes of trace data to %s", memsz * 4, fname);
4459  close(f);
4460 
4461  bool is_all_zeroes = true;
4462  for (unsigned int i = 0; i < memsz * 4; i++) {
4463  if (tracemem[i] != 0) {
4464  is_all_zeroes = false;
4465  break;
4466  }
4467  }
4468  free(tracemem);
4469  if (is_all_zeroes)
4470  command_print(
4471  CMD,
4472  "WARNING: File written is all zeroes. Are you sure you enabled trace memory?");
4473 
4474  return ERROR_OK;
4475 }
4476 
4477 COMMAND_HANDLER(xtensa_cmd_tracedump)
4478 {
4479  if (CMD_ARGC != 1) {
4480  command_print(CMD, "Command takes exactly 1 parameter.Need filename to dump to as output!");
4481  return ERROR_FAIL;
4482  }
4483 
4484  return CALL_COMMAND_HANDLER(xtensa_cmd_tracedump_do,
4486 }
4487 
4488 static const struct command_registration xtensa_any_command_handlers[] = {
4489  {
4490  .name = "xtdef",
4491  .handler = xtensa_cmd_xtdef,
4492  .mode = COMMAND_CONFIG,
4493  .help = "Configure Xtensa core type",
4494  .usage = "<type>",
4495  },
4496  {
4497  .name = "xtopt",
4498  .handler = xtensa_cmd_xtopt,
4499  .mode = COMMAND_CONFIG,
4500  .help = "Configure Xtensa core option",
4501  .usage = "<name> <value>",
4502  },
4503  {
4504  .name = "xtmem",
4505  .handler = xtensa_cmd_xtmem,
4506  .mode = COMMAND_CONFIG,
4507  .help = "Configure Xtensa memory/cache option",
4508  .usage = "<type> [parameters]",
4509  },
4510  {
4511  .name = "xtmmu",
4512  .handler = xtensa_cmd_xtmmu,
4513  .mode = COMMAND_CONFIG,
4514  .help = "Configure Xtensa MMU option",
4515  .usage = "<NIREFILLENTRIES> <NDREFILLENTRIES> <IVARWAY56> <DVARWAY56>",
4516  },
4517  {
4518  .name = "xtmpu",
4519  .handler = xtensa_cmd_xtmpu,
4520  .mode = COMMAND_CONFIG,
4521  .help = "Configure Xtensa MPU option",
4522  .usage = "<num FG seg> <min seg size> <lockable> <executeonly>",
4523  },
4524  {
4525  .name = "xtreg",
4526  .handler = xtensa_cmd_xtreg,
4527  .mode = COMMAND_CONFIG,
4528  .help = "Configure Xtensa register",
4529  .usage = "<regname> <regnum>",
4530  },
4531  {
4532  .name = "xtregs",
4533  .handler = xtensa_cmd_xtreg,
4534  .mode = COMMAND_CONFIG,
4535  .help = "Configure number of Xtensa registers",
4536  .usage = "<numregs>",
4537  },
4538  {
4539  .name = "xtregfmt",
4540  .handler = xtensa_cmd_xtregfmt,
4541  .mode = COMMAND_CONFIG,
4542  .help = "Configure format of Xtensa register map",
4543  .usage = "<contiguous|sparse> [numgregs]",
4544  },
4545  {
4546  .name = "set_permissive",
4547  .handler = xtensa_cmd_permissive_mode,
4548  .mode = COMMAND_ANY,
4549  .help = "When set to 1, enable Xtensa permissive mode (fewer client-side checks)",
4550  .usage = "[0|1]",
4551  },
4552  {
4553  .name = "maskisr",
4554  .handler = xtensa_cmd_mask_interrupts,
4555  .mode = COMMAND_ANY,
4556  .help = "mask Xtensa interrupts at step",
4557  .usage = "['on'|'off']",
4558  },
4559  {
4560  .name = "smpbreak",
4561  .handler = xtensa_cmd_smpbreak,
4562  .mode = COMMAND_ANY,
4563  .help = "Set the way the CPU chains OCD breaks",
4564  .usage = "[none|breakinout|runstall] | [BreakIn] [BreakOut] [RunStallIn] [DebugModeOut]",
4565  },
4566  {
4567  .name = "dm",
4568  .handler = xtensa_cmd_dm_rw,
4569  .mode = COMMAND_ANY,
4570  .help = "Xtensa DM read/write",
4571  .usage = "addr [value]"
4572  },
4573  {
4574  .name = "perfmon_enable",
4575  .handler = xtensa_cmd_perfmon_enable,
4576  .mode = COMMAND_EXEC,
4577  .help = "Enable and start performance counter",
4578  .usage = "<counter_id> <select> [mask] [kernelcnt] [tracelevel]",
4579  },
4580  {
4581  .name = "perfmon_dump",
4582  .handler = xtensa_cmd_perfmon_dump,
4583  .mode = COMMAND_EXEC,
4584  .help = "Dump performance counter value. If no argument specified, dumps all counters.",
4585  .usage = "[counter_id]",
4586  },
4587  {
4588  .name = "tracestart",
4589  .handler = xtensa_cmd_tracestart,
4590  .mode = COMMAND_EXEC,
4591  .help =
4592  "Tracing: Set up and start a trace. Optionally set stop trigger address and amount of data captured after.",
4593  .usage = "[pc <pcval>/[maskbitcount]] [after <n> [ins|words]]",
4594  },
4595  {
4596  .name = "tracestop",
4597  .handler = xtensa_cmd_tracestop,
4598  .mode = COMMAND_EXEC,
4599  .help = "Tracing: Stop current trace as started by the tracestart command",
4600  .usage = "",
4601  },
4602  {
4603  .name = "tracedump",
4604  .handler = xtensa_cmd_tracedump,
4605  .mode = COMMAND_EXEC,
4606  .help = "Tracing: Dump trace memory to a files. One file per core.",
4607  .usage = "<outfile>",
4608  },
4609  {
4610  .name = "exe",
4611  .handler = xtensa_cmd_exe,
4612  .mode = COMMAND_ANY,
4613  .help = "Xtensa stub execution",
4614  .usage = "<ascii-encoded hexadecimal instruction bytes>",
4615  },
4617 };
4618 
4620  {
4621  .name = "xtensa",
4622  .mode = COMMAND_ANY,
4623  .help = "Xtensa command group",
4624  .usage = "",
4625  .chain = xtensa_any_command_handlers,
4626  },
4628 };
@ PARAM_OUT
Definition: algorithm.h:16
@ PARAM_IN
Definition: algorithm.h:15
#define IS_ALIGNED(x, a)
Definition: align.h:22
#define IS_PWR_OF_2(x)
Definition: align.h:24
#define ALIGN_DOWN(x, a)
Definition: align.h:21
#define ALIGN_UP(x, a)
Definition: align.h:20
const char * name
Definition: armv4_5.c:76
void * buf_cpy(const void *from, void *_to, unsigned int size)
Copies size bits out of from and into to.
Definition: binarybuffer.c:43
static uint32_t buf_get_u32(const uint8_t *_buffer, unsigned int first, unsigned int num)
Retrieves num bits from _buffer, starting at the first bit, returning the bits in a 32-bit word.
Definition: binarybuffer.h:104
static void buf_set_u32(uint8_t *_buffer, unsigned int first, unsigned int num, uint32_t value)
Sets num bits in _buffer, starting at the first bit, using the bits in value.
Definition: binarybuffer.h:34
static uint64_t buf_get_u64(const uint8_t *_buffer, unsigned int first, unsigned int num)
Retrieves num bits from _buffer, starting at the first bit, returning the bits in a 64-bit word.
Definition: binarybuffer.h:134
@ BKPT_SOFT
Definition: breakpoints.h:19
#define WATCHPOINT_IGNORE_DATA_VALUE_MASK
Definition: breakpoints.h:39
@ WPT_ACCESS
Definition: breakpoints.h:23
@ WPT_READ
Definition: breakpoints.h:23
@ WPT_WRITE
Definition: breakpoints.h:23
void command_print(struct command_invocation *cmd, const char *format,...)
Definition: command.c:443
#define CMD
Use this macro to access the command being handled, rather than accessing the variable directly.
Definition: command.h:141
#define CALL_COMMAND_HANDLER(name, extra ...)
Use this to macro to call a command helper (or a nested handler).
Definition: command.h:118
#define CMD_ARGV
Use this macro to access the arguments for the command being handled, rather than accessing the varia...
Definition: command.h:156
#define ERROR_COMMAND_SYNTAX_ERROR
Definition: command.h:402
#define CMD_ARGC
Use this macro to access the number of arguments for the command being handled, rather than accessing...
Definition: command.h:151
#define CMD_CTX
Use this macro to access the context of the command being handled, rather than accessing the variable...
Definition: command.h:146
#define COMMAND_REGISTRATION_DONE
Use this as the last entry in an array of command_registration records.
Definition: command.h:253
#define ERROR_COMMAND_ARGUMENT_INVALID
Definition: command.h:404
@ COMMAND_CONFIG
Definition: command.h:41
@ COMMAND_ANY
Definition: command.h:42
@ COMMAND_EXEC
Definition: command.h:40
uint8_t type
Definition: esp_usb_jtag.c:0
static uint16_t direction
Definition: ftdi.c:120
void keep_alive(void)
Definition: log.c:415
static int64_t start
Definition: log.c:42
#define LOG_TARGET_INFO(target, fmt_str,...)
Definition: log.h:152
#define LOG_TARGET_WARNING(target, fmt_str,...)
Definition: log.h:158
#define LOG_WARNING(expr ...)
Definition: log.h:129
#define ERROR_FAIL
Definition: log.h:173
#define LOG_TARGET_ERROR(target, fmt_str,...)
Definition: log.h:161
#define LOG_TARGET_DEBUG(target, fmt_str,...)
Definition: log.h:149
#define LOG_ERROR(expr ...)
Definition: log.h:132
#define LOG_LEVEL_IS(FOO)
Definition: log.h:99
#define LOG_INFO(expr ...)
Definition: log.h:126
#define LOG_DEBUG(expr ...)
Definition: log.h:109
#define ERROR_OK
Definition: log.h:167
@ LOG_LVL_DEBUG
Definition: log.h:47
#define a3
Definition: mips32.c:191
#define a0
Definition: mips32.c:188
struct reg * register_get_by_name(struct reg_cache *first, const char *name, bool search_all)
Definition: register.c:50
struct reg_cache ** register_get_last_cache_p(struct reg_cache **first)
Definition: register.c:72
void register_unlink_cache(struct reg_cache **cache_p, const struct reg_cache *cache)
Definition: register.c:85
void register_cache_invalidate(struct reg_cache *cache)
Marks the contents of the register cache as invalid (and clean).
Definition: register.c:94
#define MIN(a, b)
Definition: replacements.h:22
slot
Definition: riscv-011.c:122
target_addr_t addr
Start address to search for the control block.
Definition: rtt/rtt.c:28
struct target * target
Definition: rtt/rtt.c:26
size_t size
Size of the control block search area.
Definition: rtt/rtt.c:30
#define BIT(nr)
Definition: stm32l4x.h:18
unsigned int length
Definition: breakpoints.h:29
enum breakpoint_type type
Definition: breakpoints.h:30
target_addr_t address
Definition: breakpoints.h:27
const char * name
Definition: command.h:235
int(* get)(struct reg *reg)
Definition: register.h:152
const char * name
Definition: register.h:145
unsigned int num_regs
Definition: register.h:148
struct reg * reg_list
Definition: register.h:147
struct reg_cache * next
Definition: register.h:146
uint32_t size
Definition: algorithm.h:29
const char * reg_name
Definition: algorithm.h:28
Definition: register.h:111
bool valid
Definition: register.h:126
bool exist
Definition: register.h:128
uint32_t size
Definition: register.h:132
uint8_t * value
Definition: register.h:122
uint32_t number
Definition: register.h:115
void * arch_info
Definition: register.h:140
bool dirty
Definition: register.h:124
const struct reg_arch_type * type
Definition: register.h:141
const char * name
Definition: register.h:113
Definition: target.h:116
enum target_debug_reason debug_reason
Definition: target.h:154
enum target_state state
Definition: target.h:157
enum target_endianness endianness
Definition: target.h:155
struct reg_cache * reg_cache
Definition: target.h:158
void * arch_info
Definition: target.h:164
bool reset_halt
Definition: target.h:144
bool examined
Indicates whether this target has been examined.
Definition: target.h:131
uint64_t mask
Definition: breakpoints.h:44
enum watchpoint_rw rw
Definition: breakpoints.h:46
unsigned int length
Definition: breakpoints.h:43
target_addr_t address
Definition: breakpoints.h:42
Xtensa algorithm data.
Definition: xtensa.h:229
xtensa_reg_val_t ctx_ps
Definition: xtensa.h:234
enum target_debug_reason ctx_debug_reason
Used internally to backup and restore core state.
Definition: xtensa.h:233
enum xtensa_mode core_mode
User can set this to specify which core mode algorithm should be run in.
Definition: xtensa.h:231
uint8_t way_count
Definition: xtensa.h:113
uint32_t size
Definition: xtensa.h:115
uint32_t line_size
Definition: xtensa.h:114
struct xtensa_cache_config dcache
Definition: xtensa.h:182
struct xtensa_debug_config debug
Definition: xtensa.h:179
struct xtensa_tracing_config trace
Definition: xtensa.h:180
struct xtensa_local_mem_config irom
Definition: xtensa.h:183
struct xtensa_local_mem_config drom
Definition: xtensa.h:185
struct xtensa_mpu_config mpu
Definition: xtensa.h:178
enum xtensa_type core_type
Definition: xtensa.h:170
struct xtensa_cache_config icache
Definition: xtensa.h:181
struct xtensa_local_mem_config iram
Definition: xtensa.h:184
struct xtensa_high_prio_irq_config high_irq
Definition: xtensa.h:176
struct xtensa_mmu_config mmu
Definition: xtensa.h:177
uint8_t aregs_num
Definition: xtensa.h:171
struct xtensa_irq_config irq
Definition: xtensa.h:175
struct xtensa_local_mem_config dram
Definition: xtensa.h:186
struct xtensa_local_mem_config sram
Definition: xtensa.h:187
bool windowed
Definition: xtensa.h:172
struct xtensa_local_mem_config srom
Definition: xtensa.h:188
bool coproc
Definition: xtensa.h:173
bool exceptions
Definition: xtensa.h:174
uint8_t irq_level
Definition: xtensa.h:157
uint8_t ibreaks_num
Definition: xtensa.h:158
uint8_t dbreaks_num
Definition: xtensa.h:159
uint8_t perfcount_num
Definition: xtensa.h:160
struct xtensa_power_status power_status
const struct xtensa_power_ops * pwr_ops
struct xtensa_core_status core_status
uint8_t irq_num
Definition: xtensa.h:146
struct xtensa_local_mem_region_config regions[XT_LOCAL_MEM_REGIONS_NUM_MAX]
Definition: xtensa.h:127
uint8_t itlb_entries_count
Definition: xtensa.h:132
uint8_t dtlb_entries_count
Definition: xtensa.h:133
uint8_t nfgseg
Definition: xtensa.h:138
uint32_t minsegsize
Definition: xtensa.h:139
int(* queue_reg_write)(struct xtensa_debug_module *dm, enum xtensa_dm_pwr_reg reg, uint32_t data)
register write.
xtensa_pwrstat_t stath
unsigned int reg_num
Definition: xtensa_regs.h:116
enum xtensa_reg_flags flags
Definition: xtensa_regs.h:119
const char * name
Definition: xtensa_regs.h:114
unsigned int dbreg_num
Definition: xtensa_regs.h:117
enum xtensa_reg_type type
Definition: xtensa_regs.h:118
uint8_t insn[XT_ISNS_SZ_MAX]
Definition: xtensa.h:221
struct breakpoint * oocd_bp
Definition: xtensa.h:219
bool reversed_mem_access
Definition: xtensa.h:166
Represents a generic Xtensa core.
Definition: xtensa.h:242
struct watchpoint ** hw_wps
Definition: xtensa.h:268
uint8_t come_online_probes_num
Definition: xtensa.h:282
unsigned int dbregs_num
Definition: xtensa.h:263
struct xtensa_reg_desc ** contiguous_regs_desc
Definition: xtensa.h:252
unsigned int total_regs_num
Definition: xtensa.h:248
struct reg * empty_regs
Definition: xtensa.h:257
struct xtensa_debug_module dbg_mod
Definition: xtensa.h:246
char qpkt_resp[XT_QUERYPKT_RESP_MAX]
Definition: xtensa.h:258
bool permissive_mode
Definition: xtensa.h:271
uint32_t smp_break
Definition: xtensa.h:273
bool suppress_dsr_errors
Definition: xtensa.h:272
struct reg ** contiguous_regs_list
Definition: xtensa.h:253
bool trace_active
Definition: xtensa.h:270
uint32_t spill_loc
Definition: xtensa.h:274
struct target * target
Definition: xtensa.h:264
int8_t probe_lsddr32p
Definition: xtensa.h:277
unsigned int eps_dbglevel_idx
Definition: xtensa.h:262
void ** algo_context_backup
Definition: xtensa.h:261
bool reset_asserted
Definition: xtensa.h:265
uint8_t * spill_buf
Definition: xtensa.h:276
struct xtensa_sw_breakpoint * sw_brps
Definition: xtensa.h:269
uint32_t nx_stop_cause
Definition: xtensa.h:285
unsigned int genpkt_regs_num
Definition: xtensa.h:251
enum xtensa_stepping_isr_mode stepping_isr_mode
Definition: xtensa.h:266
bool regmap_contiguous
Definition: xtensa.h:250
bool halt_request
Definition: xtensa.h:284
struct reg_cache * core_cache
Definition: xtensa.h:247
bool regs_fetched
Definition: xtensa.h:288
unsigned int num_optregs
Definition: xtensa.h:256
unsigned int core_regs_num
Definition: xtensa.h:249
struct xtensa_keyval_info scratch_ars[XT_AR_SCRATCH_NUM]
Definition: xtensa.h:287
struct xtensa_reg_desc * optregs
Definition: xtensa.h:255
uint32_t nx_reg_idx[XT_NX_REG_IDX_NUM]
Definition: xtensa.h:286
struct breakpoint ** hw_brps
Definition: xtensa.h:267
unsigned int common_magic
Definition: xtensa.h:243
struct xtensa_config * core_config
Definition: xtensa.h:245
unsigned int spill_bytes
Definition: xtensa.h:275
int target_call_event_callbacks(struct target *target, enum target_event event)
Definition: target.c:1764
int target_halt(struct target *target)
Definition: target.c:507
int target_write_buffer(struct target *target, target_addr_t address, uint32_t size, const uint8_t *buffer)
Definition: target.c:2342
int target_read_buffer(struct target *target, target_addr_t address, uint32_t size, uint8_t *buffer)
Definition: target.c:2407
int target_wait_state(struct target *target, enum target_state state, unsigned int ms)
Definition: target.c:3214
struct target * get_current_target(struct command_context *cmd_ctx)
Definition: target.c:458
uint32_t target_buffer_get_u32(struct target *target, const uint8_t *buffer)
Definition: target.c:316
@ DBG_REASON_WPTANDBKPT
Definition: target.h:72
@ DBG_REASON_NOTHALTED
Definition: target.h:74
@ DBG_REASON_DBGRQ
Definition: target.h:69
@ DBG_REASON_SINGLESTEP
Definition: target.h:73
@ DBG_REASON_WATCHPOINT
Definition: target.h:71
@ DBG_REASON_BREAKPOINT
Definition: target.h:70
target_register_class
Definition: target.h:110
@ REG_CLASS_GENERAL
Definition: target.h:112
#define ERROR_TARGET_NOT_HALTED
Definition: target.h:790
static bool target_was_examined(const struct target *target)
Definition: target.h:436
@ TARGET_EVENT_HALTED
Definition: target.h:252
@ TARGET_EVENT_RESUMED
Definition: target.h:253
static const char * target_name(const struct target *target)
Returns the instance-specific name of the specified target.
Definition: target.h:233
target_state
Definition: target.h:53
@ TARGET_RESET
Definition: target.h:57
@ TARGET_DEBUG_RUNNING
Definition: target.h:58
@ TARGET_UNKNOWN
Definition: target.h:54
@ TARGET_HALTED
Definition: target.h:56
@ TARGET_RUNNING
Definition: target.h:55
#define ERROR_TARGET_NOT_EXAMINED
Definition: target.h:797
@ TARGET_BIG_ENDIAN
Definition: target.h:82
#define ERROR_TARGET_TIMEOUT
Definition: target.h:789
#define ERROR_TARGET_RESOURCE_NOT_AVAILABLE
Definition: target.h:794
static void target_set_examined(struct target *target)
Sets the examined flag for the given target.
Definition: target.h:443
#define ERROR_TARGET_FAILURE
Definition: target.h:791
int64_t timeval_ms(void)
trace_status
Definition: trace.h:36
#define TARGET_ADDR_FMT
Definition: types.h:342
#define DIV_ROUND_UP(m, n)
Rounds m up to the nearest multiple of n using division.
Definition: types.h:79
uint64_t target_addr_t
Definition: types.h:335
static void buf_bswap32(uint8_t *dst, const uint8_t *src, size_t len)
Byte-swap buffer 32-bit.
Definition: types.h:249
xtensa_reg_val_t val
Definition: xtensa.c:330
uint8_t buf[4]
Definition: xtensa.c:331
#define NULL
Definition: usb.h:16
uint8_t status[4]
Definition: vdebug.c:17
uint8_t cmd
Definition: vdebug.c:1
uint8_t state[4]
Definition: vdebug.c:21
uint8_t count[4]
Definition: vdebug.c:22
int xtensa_gdb_query_custom(struct target *target, const char *packet, char **response_p)
Definition: xtensa.c:3274
#define XT_INS_RSR(X, SR, T)
Definition: xtensa.c:134
static int xtensa_core_reg_set(struct reg *reg, uint8_t *buf)
Definition: xtensa.c:450
static bool xtensa_memory_op_validate_range(struct xtensa *xtensa, target_addr_t address, size_t size, int access)
Check if the address gets to memory regions, and its access mode.
Definition: xtensa.c:1988
void xtensa_reg_set_deep_relgen(struct target *target, enum xtensa_reg_id a_idx, xtensa_reg_val_t value)
Definition: xtensa.c:1080
static COMMAND_HELPER(xtensa_cmd_exe_do, struct target *target)
Definition: xtensa.c:3534
#define XT_INS_L32E(X, R, S, T)
Definition: xtensa.c:153
static void xtensa_mark_register_dirty(struct xtensa *xtensa, enum xtensa_reg_id reg_idx)
Definition: xtensa.c:521
#define XT_INS_SDDR32P(X, S)
Definition: xtensa.c:107
static bool xtensa_reg_is_readable(int flags, int cpenable)
Definition: xtensa.c:641
static enum xtensa_reg_id xtensa_canonical_to_windowbase_offset(struct xtensa *xtensa, enum xtensa_reg_id reg_idx, int windowbase)
Definition: xtensa.c:514
#define XT_INS_IHI(X, S, IMM8)
Definition: xtensa.c:124
int xtensa_breakpoint_add(struct target *target, struct breakpoint *breakpoint)
Definition: xtensa.c:2564
#define XT_HW_DBREAK_MAX_NUM
Definition: xtensa.c:188
#define XT_WATCHPOINTS_NUM_MAX
Definition: xtensa.c:167
void xtensa_target_deinit(struct target *target)
Definition: xtensa.c:3495
static const bool xtensa_extra_debug_log
Definition: xtensa.c:342
int xtensa_watchpoint_add(struct target *target, struct watchpoint *watchpoint)
Definition: xtensa.c:2646
static int xtensa_queue_pwr_reg_write(struct xtensa *xtensa, unsigned int reg, uint32_t data)
Definition: xtensa.c:589
static enum xtensa_reg_id xtensa_windowbase_offset_to_canonical(struct xtensa *xtensa, enum xtensa_reg_id reg_idx, int windowbase)
Definition: xtensa.c:496
static bool xtensa_cmd_xtopt_legal_val(char *opt, int val, int min, int max)
Definition: xtensa.c:3632
#define XT_INS_WFR(X, FR, T)
Definition: xtensa.c:151
const char * xtensa_get_gdb_arch(const struct target *target)
Definition: xtensa.c:3528
uint32_t xtensa_cause_get(struct target *target)
Definition: xtensa.c:1095
#define XT_INS_RUR(X, UR, T)
Definition: xtensa.c:144
xtensa_mem_region_type
Types of memory used at xtensa target.
Definition: xtensa.c:297
@ XTENSA_MEM_REG_IRAM
Definition: xtensa.c:299
@ XTENSA_MEM_REGS_NUM
Definition: xtensa.c:304
@ XTENSA_MEM_REG_IROM
Definition: xtensa.c:298
@ XTENSA_MEM_REG_DRAM
Definition: xtensa.c:301
@ XTENSA_MEM_REG_SRAM
Definition: xtensa.c:302
@ XTENSA_MEM_REG_SROM
Definition: xtensa.c:303
@ XTENSA_MEM_REG_DROM
Definition: xtensa.c:300
int xtensa_do_step(struct target *target, int current, target_addr_t address, int handle_breakpoints)
Definition: xtensa.c:1722
#define XT_INS_ROTW(X, N)
Definition: xtensa.c:141
static bool xtensa_pc_in_winexc(struct target *target, target_addr_t pc)
Definition: xtensa.c:1702
int xtensa_smpbreak_read(struct xtensa *xtensa, uint32_t *val)
Definition: xtensa.c:956
int xtensa_poll(struct target *target)
Definition: xtensa.c:2314
#define XT_SR_WB
Definition: xtensa.c:174
int xtensa_prepare_resume(struct target *target, int current, target_addr_t address, int handle_breakpoints, int debug_execution)
Definition: xtensa.c:1593
#define XT_HW_IBREAK_MAX_NUM
Definition: xtensa.c:187
#define XT_REG_A3
Definition: xtensa.c:176
int xtensa_halt(struct target *target)
Definition: xtensa.c:1566
static const struct command_registration xtensa_any_command_handlers[]
Definition: xtensa.c:4488
static void xtensa_reg_set_value(struct reg *reg, xtensa_reg_val_t value)
Definition: xtensa.c:980
int xtensa_breakpoint_remove(struct target *target, struct breakpoint *breakpoint)
Definition: xtensa.c:2608
static bool xtensa_scratch_regs_fixup(struct xtensa *xtensa, struct reg *reg_list, int i, int j, int a_idx, int ar_idx)
Definition: xtensa.c:650
int xtensa_read_buffer(struct target *target, target_addr_t address, uint32_t count, uint8_t *buffer)
Definition: xtensa.c:2091
int xtensa_get_gdb_reg_list(struct target *target, struct reg **reg_list[], int *reg_list_size, enum target_register_class reg_class)
Definition: xtensa.c:1489
int xtensa_step(struct target *target, int current, target_addr_t address, int handle_breakpoints)
Definition: xtensa.c:1944
int xtensa_target_init(struct command_context *cmd_ctx, struct target *target)
Definition: xtensa.c:3429
static bool xtensa_region_ar_exec(struct target *target, target_addr_t start, target_addr_t end)
Definition: xtensa.c:552
int xtensa_checksum_memory(struct target *target, target_addr_t address, uint32_t count, uint32_t *checksum)
Definition: xtensa.c:2308
#define XT_TLB1_ACC_SHIFT
Definition: xtensa.c:164
#define XT_SW_BREAKPOINTS_MAX_NUM
Definition: xtensa.c:186
const struct command_registration xtensa_command_handlers[]
Definition: xtensa.c:4619
int xtensa_smpbreak_set(struct target *target, uint32_t set)
Definition: xtensa.c:944
static bool xtensa_memory_regions_overlap(target_addr_t r1_start, target_addr_t r1_end, target_addr_t r2_start, target_addr_t r2_end)
Returns true if two ranges are overlapping.
Definition: xtensa.c:1957
int xtensa_examine(struct target *target)
Definition: xtensa.c:886
static void xtensa_free_reg_cache(struct target *target)
Definition: xtensa.c:3461
int xtensa_start_algorithm(struct target *target, int num_mem_params, struct mem_param *mem_params, int num_reg_params, struct reg_param *reg_params, target_addr_t entry_point, target_addr_t exit_point, void *arch_info)
Definition: xtensa.c:2722
int xtensa_init_arch_info(struct target *target, struct xtensa *xtensa, const struct xtensa_debug_module_config *dm_cfg)
Definition: xtensa.c:3385
int xtensa_fetch_all_regs(struct target *target)
Definition: xtensa.c:1210
#define XT_SR_DDR
Definition: xtensa.c:172
#define XT_SR_PS
Definition: xtensa.c:173
#define XT_INS_CALL0(X, IMM18)
Definition: xtensa.c:131
int xtensa_resume(struct target *target, int current, target_addr_t address, int handle_breakpoints, int debug_execution)
Definition: xtensa.c:1673
#define XT_INS_L32E_S32E_MASK(X)
Definition: xtensa.c:155
#define XT_REG_A0
Definition: xtensa.c:175
int xtensa_watchpoint_remove(struct target *target, struct watchpoint *watchpoint)
Definition: xtensa.c:2702
void xtensa_cause_reset(struct target *target)
Definition: xtensa.c:1154
int xtensa_write_buffer(struct target *target, target_addr_t address, uint32_t count, const uint8_t *buffer)
Definition: xtensa.c:2302
static void xtensa_window_state_restore(struct target *target, uint32_t woe)
Definition: xtensa.c:627
xtensa_mpu_access_type
Types of access rights for MPU option The first block is kernel RWX ARs; the second block is user rwx...
Definition: xtensa.c:311
@ XTENSA_ACC_RWX_000
Definition: xtensa.c:317
@ XTENSA_ACC_RW0_RWX
Definition: xtensa.c:319
@ XTENSA_ACC_RW0_R00
Definition: xtensa.c:320
@ XTENSA_ACC_RW0_000
Definition: xtensa.c:316
@ XTENSA_ACC_R00_R00
Definition: xtensa.c:322
@ XTENSA_ACC_R0X_R0X
Definition: xtensa.c:323
@ XTENSA_ACC_RW0_RW0
Definition: xtensa.c:324
@ XTENSA_ACC_00X_000
Definition: xtensa.c:312
@ XTENSA_ACC_R00_000
Definition: xtensa.c:314
@ XTENSA_ACC_RWX_R0X
Definition: xtensa.c:321
@ XTENSA_ACC_R0X_000
Definition: xtensa.c:315
@ XTENSA_ACC_0W0_0W0
Definition: xtensa.c:318
@ XTENSA_ACC_000_00X
Definition: xtensa.c:313
@ XTENSA_ACC_RWX_RWX
Definition: xtensa.c:325
static void xtensa_queue_exec_ins(struct xtensa *xtensa, uint32_t ins)
Definition: xtensa.c:527
static bool xtensa_is_icacheable(struct xtensa *xtensa, target_addr_t address)
Definition: xtensa.c:415
static int xtensa_window_state_save(struct target *target, uint32_t *woe)
Definition: xtensa.c:596
static bool xtensa_is_cacheable(const struct xtensa_cache_config *cache, const struct xtensa_local_mem_config *mem, target_addr_t address)
Definition: xtensa.c:406
int xtensa_smpbreak_write(struct xtensa *xtensa, uint32_t set)
Definition: xtensa.c:929
int xtensa_write_memory(struct target *target, target_addr_t address, uint32_t size, uint32_t count, const uint8_t *buffer)
Definition: xtensa.c:2097
static const struct xtensa_keyval_info xt_qerr[XT_QERR_NUM]
Definition: xtensa.c:334
static int xtensa_imprecise_exception_occurred(struct target *target)
Definition: xtensa.c:986
void xtensa_reg_set(struct target *target, enum xtensa_reg_id reg_id, xtensa_reg_val_t value)
Definition: xtensa.c:1070
void xtensa_cause_clear(struct target *target)
Definition: xtensa.c:1142
#define XT_INS_L32I(X, S, T, IMM8)
Definition: xtensa.c:110
COMMAND_HANDLER(xtensa_cmd_exe)
Definition: xtensa.c:3603
int xtensa_smpbreak_get(struct target *target, uint32_t *val)
Definition: xtensa.c:968
struct xtensa_reg_desc xtensa_regs[XT_NUM_REGS]
Definition: xtensa.c:190
static int xtensa_core_reg_get(struct reg *reg)
Definition: xtensa.c:431
#define XT_INS_PPTLB(X, S, T)
Definition: xtensa.c:162
int xtensa_core_status_check(struct target *target)
Definition: xtensa.c:1017
#define XT_INS_RFR(X, FR, T)
Definition: xtensa.c:149
static int xtensa_update_instruction(struct target *target, target_addr_t address, uint32_t size, const uint8_t *buffer)
Definition: xtensa.c:2448
static int32_t xtensa_gdbqc_parse_exec_tie_ops(struct target *target, char *opstr)
Definition: xtensa.c:3105
#define XT_INS_S32E(X, R, S, T)
Definition: xtensa.c:154
int xtensa_do_resume(struct target *target)
Definition: xtensa.c:1656
#define XT_PC_REG_NUM_VIRTUAL
Definition: xtensa.c:182
int xtensa_wakeup(struct target *target)
Definition: xtensa.c:915
static xtensa_reg_val_t xtensa_reg_get_value(struct reg *reg)
Definition: xtensa.c:975
int xtensa_mmu_is_enabled(struct target *target, int *enabled)
Definition: xtensa.c:1558
static void xtensa_imprecise_exception_clear(struct target *target)
Definition: xtensa.c:1003
#define XT_PS_REG_NUM
Definition: xtensa.c:179
#define XT_INS_DHWBI(X, S, IMM8)
Definition: xtensa.c:125
static const struct reg_arch_type xtensa_reg_type
Definition: xtensa.c:490
#define XT_INS_RFDO(X)
Definition: xtensa.c:100
static bool xtensa_is_stopped(struct target *target)
Definition: xtensa.c:880
static int xtensa_gdbqc_qxtreg(struct target *target, const char *packet, char **response_p)
Definition: xtensa.c:3136
static int xtensa_write_dirty_registers(struct target *target)
Definition: xtensa.c:663
void xtensa_set_permissive_mode(struct target *target, bool state)
Definition: xtensa.c:3424
#define XT_PC_DBREG_NUM_BASE
Definition: xtensa.c:183
#define XT_INS_WUR(X, UR, T)
Definition: xtensa.c:146
#define XT_INS_JX(X, S)
Definition: xtensa.c:130
int xtensa_deassert_reset(struct target *target)
Definition: xtensa.c:1182
#define XT_INS_RFWU(X)
Definition: xtensa.c:158
int xtensa_read_memory(struct target *target, target_addr_t address, uint32_t size, uint32_t count, uint8_t *buffer)
Definition: xtensa.c:2008
static const struct xtensa_local_mem_config * xtensa_get_mem_config(struct xtensa *xtensa, enum xtensa_mem_region_type type)
Gets a config for the specific mem type.
Definition: xtensa.c:347
static int xtensa_sw_breakpoint_add(struct target *target, struct breakpoint *breakpoint, struct xtensa_sw_breakpoint *sw_bp)
Definition: xtensa.c:2527
static int xtensa_sw_breakpoint_remove(struct target *target, struct xtensa_sw_breakpoint *sw_bp)
Definition: xtensa.c:2553
static const struct xtensa_local_mem_region_config * xtensa_target_memory_region_find(struct xtensa *xtensa, target_addr_t address)
Returns a corresponding xtensa_local_mem_region_config from the xtensa target for a given address Ret...
Definition: xtensa.c:391
int xtensa_soft_reset_halt(struct target *target)
Definition: xtensa.c:1204
#define XT_EPS_REG_NUM_BASE
Definition: xtensa.c:180
static bool xtensa_is_dcacheable(struct xtensa *xtensa, target_addr_t address)
Definition: xtensa.c:423
int xtensa_assert_reset(struct target *target)
Definition: xtensa.c:1161
#define XT_INS_S32I(X, S, T, IMM8)
Definition: xtensa.c:117
#define XT_TLB1_ACC_MSK
Definition: xtensa.c:165
#define XT_INS_LDDR32P(X, S)
Definition: xtensa.c:105
#define XT_EPC_REG_NUM_BASE
Definition: xtensa.c:181
static void xtensa_queue_exec_ins_wide(struct xtensa *xtensa, uint8_t *ops, uint8_t oplen)
Definition: xtensa.c:532
static target_addr_t xtensa_get_overlap_size(target_addr_t r1_start, target_addr_t r1_end, target_addr_t r2_start, target_addr_t r2_end)
Returns a size of overlapped region of two ranges.
Definition: xtensa.c:1972
#define XT_INS_RFWO(X)
Definition: xtensa.c:157
#define XT_REG_A4
Definition: xtensa.c:177
#define XT_INS_DHWB(X, S, IMM8)
Definition: xtensa.c:126
int xtensa_run_algorithm(struct target *target, int num_mem_params, struct mem_param *mem_params, int num_reg_params, struct reg_param *reg_params, target_addr_t entry_point, target_addr_t exit_point, unsigned int timeout_ms, void *arch_info)
Definition: xtensa.c:2921
static const struct xtensa_local_mem_region_config * xtensa_memory_region_find(const struct xtensa_local_mem_config *mem, target_addr_t address)
Extracts an exact xtensa_local_mem_region_config from xtensa_local_mem_config for a given address Ret...
Definition: xtensa.c:374
static int xtensa_build_reg_cache(struct target *target)
Definition: xtensa.c:2944
#define XT_INS_WSR(X, SR, T)
Definition: xtensa.c:136
#define XT_INS_RFWO_RFWU_MASK(X)
Definition: xtensa.c:159
xtensa_reg_val_t xtensa_reg_get(struct target *target, enum xtensa_reg_id reg_id)
Definition: xtensa.c:1063
int xtensa_wait_algorithm(struct target *target, int num_mem_params, struct mem_param *mem_params, int num_reg_params, struct reg_param *reg_params, target_addr_t exit_point, unsigned int timeout_ms, void *arch_info)
Waits for an algorithm in the target.
Definition: xtensa.c:2813
Holds the interface to Xtensa cores.
#define XT_MEM_ACCESS_READ
Definition: xtensa.h:78
xtensa_qerr_e
Definition: xtensa.h:84
@ XT_QERR_FAIL
Definition: xtensa.h:86
@ XT_QERR_INVAL
Definition: xtensa.h:87
@ XT_QERR_MEM
Definition: xtensa.h:88
@ XT_QERR_NUM
Definition: xtensa.h:89
#define XT_PS_WOE_MSK
Definition: xtensa.h:44
#define XT_PS_RING_GET(_v_)
Definition: xtensa.h:41
static struct xtensa * target_to_xtensa(struct target *target)
Definition: xtensa.h:291
static int xtensa_queue_dbg_reg_write(struct xtensa *xtensa, enum xtensa_dm_reg reg, uint32_t data)
Definition: xtensa.h:340
#define XT_AREGS_NUM_MAX
Definition: xtensa.h:74
@ XT_STEPPING_ISR_OFF
Definition: xtensa.h:194
@ XT_STEPPING_ISR_ON
Definition: xtensa.h:195
#define XT_ISNS_SZ_MAX
Definition: xtensa.h:36
#define XT_PS_RING(_v_)
Definition: xtensa.h:39
#define XT_PS_DI_MSK
Definition: xtensa.h:48
@ XT_LX
Definition: xtensa.h:108
@ XT_UNDEF
Definition: xtensa.h:107
@ XT_NX
Definition: xtensa.h:109
#define XT_MEM_ACCESS_WRITE
Definition: xtensa.h:79
#define XT_MESRCLR_IMPR_EXC_MSK
Definition: xtensa.h:70
xtensa_nx_reg_idx
Definition: xtensa.h:198
@ XT_NX_REG_IDX_IEVEC
Definition: xtensa.h:202
@ XT_NX_REG_IDX_MS
Definition: xtensa.h:201
@ XT_NX_REG_IDX_NUM
Definition: xtensa.h:206
@ XT_NX_REG_IDX_MESR
Definition: xtensa.h:204
@ XT_NX_REG_IDX_IBREAKC0
Definition: xtensa.h:199
@ XT_NX_REG_IDX_MESRCLR
Definition: xtensa.h:205
@ XT_NX_REG_IDX_IEEXTERN
Definition: xtensa.h:203
@ XT_NX_REG_IDX_WB
Definition: xtensa.h:200
#define XT_PS_RING_MSK
Definition: xtensa.h:40
#define XT_INS_BREAK(X, S, T)
Definition: xtensa.h:29
xtensa_ar_scratch_set_e
Definition: xtensa.h:93
@ XT_AR_SCRATCH_A3
Definition: xtensa.h:94
@ XT_AR_SCRATCH_AR4
Definition: xtensa.h:97
@ XT_AR_SCRATCH_NUM
Definition: xtensa.h:98
@ XT_AR_SCRATCH_A4
Definition: xtensa.h:96
@ XT_AR_SCRATCH_AR3
Definition: xtensa.h:95
#define XT_INS_BREAKN(X, IMM4)
Definition: xtensa.h:34
xtensa_mode
Definition: xtensa.h:210
@ XT_MODE_ANY
Definition: xtensa.h:215
#define XT_QUERYPKT_RESP_MAX
Definition: xtensa.h:82
#define XTENSA_COMMON_MAGIC
Definition: xtensa.h:237
#define XT_IMPR_EXC_MSK
Definition: xtensa.h:69
#define XT_WB_P_SHIFT
Definition: xtensa.h:56
#define XT_PS_DIEXC_MSK
Definition: xtensa.h:47
#define XT_MS_DISPST_DBG
Definition: xtensa.h:53
#define XT_IBREAKC_FB
Definition: xtensa.h:66
#define XT_WB_P_MSK
Definition: xtensa.h:57
#define XT_WB_S_MSK
Definition: xtensa.h:63
uint32_t xtensa_insn_t
Definition: xtensa.h:191
static int xtensa_queue_dbg_reg_read(struct xtensa *xtensa, enum xtensa_dm_reg reg, uint8_t *data)
Definition: xtensa.h:328
int xtensa_dm_trace_status_read(struct xtensa_debug_module *dm, struct xtensa_trace_status *status)
int xtensa_dm_trace_start(struct xtensa_debug_module *dm, struct xtensa_trace_start_config *cfg)
int xtensa_dm_trace_stop(struct xtensa_debug_module *dm, bool pto_enable)
int xtensa_dm_write(struct xtensa_debug_module *dm, uint32_t addr, uint32_t val)
int xtensa_dm_power_status_read(struct xtensa_debug_module *dm, uint32_t clear)
int xtensa_dm_poll(struct xtensa_debug_module *dm)
int xtensa_dm_perfmon_enable(struct xtensa_debug_module *dm, int counter_id, const struct xtensa_perfmon_config *config)
void xtensa_dm_deinit(struct xtensa_debug_module *dm)
int xtensa_dm_trace_config_read(struct xtensa_debug_module *dm, struct xtensa_trace_config *config)
int xtensa_dm_trace_data_read(struct xtensa_debug_module *dm, uint8_t *dest, uint32_t size)
int xtensa_dm_core_status_clear(struct xtensa_debug_module *dm, xtensa_dsr_t bits)
int xtensa_dm_core_status_read(struct xtensa_debug_module *dm)
int xtensa_dm_queue_enable(struct xtensa_debug_module *dm)
int xtensa_dm_init(struct xtensa_debug_module *dm, const struct xtensa_debug_module_config *cfg)
int xtensa_dm_read(struct xtensa_debug_module *dm, uint32_t addr, uint32_t *val)
int xtensa_dm_perfmon_dump(struct xtensa_debug_module *dm, int counter_id, struct xtensa_perfmon_result *out_result)
#define PWRSTAT_DEBUGWASRESET(x)
#define TRAXADDR_TWRAP_SHIFT
#define OCDDCR_DEBUGMODEOUTEN
static void xtensa_dm_power_status_cache(struct xtensa_debug_module *dm)
#define XTENSA_MAX_PERF_COUNTERS
#define DEBUGCAUSE_DI
#define OCDDSR_DEBUGPENDTRAX
#define TRAXCTRL_TREN
#define OCDDSR_STOPCAUSE_IB
#define OCDDSR_EXECBUSY
#define OCDDCR_BREAKOUTEN
#define DEBUGCAUSE_IB
#define TRAXADDR_TWSAT
#define OCDDCR_ENABLEOCD
#define OCDDCR_STEPREQUEST
#define OCDDSR_DEBUGPENDHOST
#define OCDDSR_STOPCAUSE_DB1
#define OCDDSR_STOPCAUSE_BN
#define DEBUGCAUSE_BI
#define DEBUGCAUSE_IC
uint32_t xtensa_dsr_t
static void xtensa_dm_queue_tdi_idle(struct xtensa_debug_module *dm)
static bool xtensa_dm_core_was_reset(struct xtensa_debug_module *dm)
#define OCDDSR_DEBUGINTTRAX
static xtensa_dsr_t xtensa_dm_core_status_get(struct xtensa_debug_module *dm)
@ XDMREG_PWRCTL
#define TRAXSTAT_CTITG
#define OCDDSR_EXECEXCEPTION
#define TRAXSTAT_PCMTG
#define OCDDSR_STOPCAUSE
#define OCDDSR_STOPCAUSE_B1
static bool xtensa_dm_is_powered(struct xtensa_debug_module *dm)
#define PWRCTL_CORERESET(x)
#define TRAXADDR_TWRAP_MASK
#define OCDDSR_STOPCAUSE_SHIFT
#define OCDDSR_STOPCAUSE_DB0
#define TRAXSTAT_TRACT
#define DEBUGCAUSE_BN
#define XTENSA_MAX_PERF_SELECT
#define OCDDSR_DEBUGINTBREAK
static bool xtensa_dm_tap_was_reset(struct xtensa_debug_module *dm)
#define PWRCTL_MEMWAKEUP(x)
#define TRAXSTAT_PTITG
#define OCDDSR_STOPCAUSE_B
#define PWRCTL_JTAGDEBUGUSE(x)
static int xtensa_dm_queue_execute(struct xtensa_debug_module *dm)
#define OCDDCR_BREAKINEN
@ XDMREG_DCRSET
@ XDMREG_DDREXEC
@ XDMREG_DSR
@ XDMREG_DIR0
@ XDMREG_DDR
@ XDMREG_DCRCLR
@ XDMREG_DIR0EXEC
#define PWRCTL_COREWAKEUP(x)
#define OCDDSR_DEBUGPENDBREAK
static bool xtensa_dm_is_online(struct xtensa_debug_module *dm)
#define OCDDSR_STOPCAUSE_DI
#define OCDDSR_DEBUGINTHOST
#define PWRSTAT_COREWASRESET(x)
#define OCDDCR_DEBUGINTERRUPT
#define PWRCTL_DEBUGWAKEUP(x)
#define DEBUGCAUSE_VALID
#define OCDDSR_EXECOVERRUN
#define XTENSA_STOPMASK_DISABLED
#define OCDDCR_RUNSTALLINEN
#define XTENSA_MAX_PERF_MASK
#define OCDDSR_STOPCAUSE_SS
#define OCDDSR_STOPPED
#define TRAXADDR_TADDR_MASK
#define DEBUGCAUSE_DB
xtensa_reg_id
Definition: xtensa_regs.h:15
@ XT_REG_IDX_AR12
Definition: xtensa_regs.h:30
@ XT_REG_IDX_AR10
Definition: xtensa_regs.h:28
@ XT_REG_IDX_A15
Definition: xtensa_regs.h:66
@ XT_REG_IDX_A0
Definition: xtensa_regs.h:51
@ XT_REG_IDX_AR5
Definition: xtensa_regs.h:23
@ XT_REG_IDX_AR14
Definition: xtensa_regs.h:32
@ XT_REG_IDX_PS
Definition: xtensa_regs.h:37
@ XT_REG_IDX_ARFIRST
Definition: xtensa_regs.h:18
@ XT_REG_IDX_ARLAST
Definition: xtensa_regs.h:34
@ XT_REG_IDX_AR6
Definition: xtensa_regs.h:24
@ XT_REG_IDX_PC
Definition: xtensa_regs.h:16
@ XT_REG_IDX_DEBUGCAUSE
Definition: xtensa_regs.h:48
@ XT_REG_IDX_AR1
Definition: xtensa_regs.h:19
@ XT_REG_IDX_AR15
Definition: xtensa_regs.h:33
@ XT_REG_IDX_A3
Definition: xtensa_regs.h:54
@ XT_REG_IDX_AR0
Definition: xtensa_regs.h:17
@ XT_REG_IDX_ICOUNT
Definition: xtensa_regs.h:49
@ XT_REG_IDX_AR9
Definition: xtensa_regs.h:27
@ XT_REG_IDX_ICOUNTLEVEL
Definition: xtensa_regs.h:50
@ XT_REG_IDX_AR8
Definition: xtensa_regs.h:26
@ XT_REG_IDX_AR2
Definition: xtensa_regs.h:20
@ XT_REG_IDX_AR11
Definition: xtensa_regs.h:29
@ XT_REG_IDX_DBREAKC0
Definition: xtensa_regs.h:44
@ XT_NUM_REGS
Definition: xtensa_regs.h:67
@ XT_REG_IDX_A4
Definition: xtensa_regs.h:55
@ XT_REG_IDX_EXCCAUSE
Definition: xtensa_regs.h:47
@ XT_REG_IDX_AR4
Definition: xtensa_regs.h:22
@ XT_REG_IDX_DBREAKA0
Definition: xtensa_regs.h:42
@ XT_REG_IDX_AR7
Definition: xtensa_regs.h:25
@ XT_REG_IDX_IBREAKENABLE
Definition: xtensa_regs.h:38
@ XT_REG_IDX_WINDOWBASE
Definition: xtensa_regs.h:35
@ XT_REG_IDX_CPENABLE
Definition: xtensa_regs.h:46
@ XT_REG_IDX_AR3
Definition: xtensa_regs.h:21
@ XT_REG_IDX_AR13
Definition: xtensa_regs.h:31
@ XT_REG_IDX_IBREAKA0
Definition: xtensa_regs.h:40
xtensa_reg_type
Definition: xtensa_regs.h:74
@ XT_REG_GENERAL_VAL
Definition: xtensa_regs.h:88
@ XT_REG_RELGEN_MASK
Definition: xtensa_regs.h:95
@ XT_REG_USER
Definition: xtensa_regs.h:76
@ XT_REG_INDEX_MASK
Definition: xtensa_regs.h:104
@ XT_REG_DEBUG
Definition: xtensa_regs.h:78
@ XT_REG_RELGEN
Definition: xtensa_regs.h:79
@ XT_REG_SPECIAL_MASK
Definition: xtensa_regs.h:91
@ XT_REG_SPECIAL_VAL
Definition: xtensa_regs.h:92
@ XT_REG_USER_VAL
Definition: xtensa_regs.h:90
@ XT_REG_FR_VAL
Definition: xtensa_regs.h:98
@ XT_REG_USER_MASK
Definition: xtensa_regs.h:89
@ XT_REG_RELGEN_VAL
Definition: xtensa_regs.h:96
@ XT_REG_GENERAL
Definition: xtensa_regs.h:75
@ XT_REG_GENERAL_MASK
Definition: xtensa_regs.h:87
@ XT_REG_OTHER
Definition: xtensa_regs.h:83
@ XT_REG_SPECIAL
Definition: xtensa_regs.h:77
@ XT_REG_TIE
Definition: xtensa_regs.h:82
@ XT_REG_FR
Definition: xtensa_regs.h:81
@ XT_REG_TIE_MASK
Definition: xtensa_regs.h:99
@ XT_REG_FR_MASK
Definition: xtensa_regs.h:97
@ XT_REGF_COPROC0
Definition: xtensa_regs.h:109
@ XT_REGF_MASK
Definition: xtensa_regs.h:110
@ XT_REGF_NOREAD
Definition: xtensa_regs.h:108
uint32_t xtensa_reg_val_t
Definition: xtensa_regs.h:70
#define XT_MK_REG_DESC(n, r, t, f)
Definition: xtensa_regs.h:128