1 /***************************************************************************
2 * Copyright (C) 2005 by Dominic Rath *
3 * Dominic.Rath@gmx.de *
5 * Copyright (C) 2006 by Magnus Lundin *
8 * Copyright (C) 2008 by Spencer Oliver *
9 * spen@spen-soft.co.uk *
11 * Copyright (C) 2009 by Dirk Behme *
12 * dirk.behme@gmail.com - copy from cortex_m3 *
14 * This program is free software; you can redistribute it and/or modify *
15 * it under the terms of the GNU General Public License as published by *
16 * the Free Software Foundation; either version 2 of the License, or *
17 * (at your option) any later version. *
19 * This program is distributed in the hope that it will be useful, *
20 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
21 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
22 * GNU General Public License for more details. *
24 * You should have received a copy of the GNU General Public License *
25 * along with this program; if not, write to the *
26 * Free Software Foundation, Inc., *
27 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
29 * Cortex-A8(tm) TRM, ARM DDI 0344H *
31 ***************************************************************************/
36 #include "breakpoints.h"
37 #include "cortex_a8.h"
39 #include "target_request.h"
40 #include "target_type.h"
42 static int cortex_a8_poll(struct target
*target
);
43 static int cortex_a8_debug_entry(struct target
*target
);
44 static int cortex_a8_restore_context(struct target
*target
, bool bpwp
);
45 static int cortex_a8_set_breakpoint(struct target
*target
,
46 struct breakpoint
*breakpoint
, uint8_t matchmode
);
47 static int cortex_a8_unset_breakpoint(struct target
*target
,
48 struct breakpoint
*breakpoint
);
49 static int cortex_a8_dap_read_coreregister_u32(struct target
*target
,
50 uint32_t *value
, int regnum
);
51 static int cortex_a8_dap_write_coreregister_u32(struct target
*target
,
52 uint32_t value
, int regnum
);
54 * FIXME do topology discovery using the ROM; don't
55 * assume this is an OMAP3.
57 #define swjdp_memoryap 0
58 #define swjdp_debugap 1
59 #define OMAP3530_DEBUG_BASE 0x54011000
62 * Cortex-A8 Basic debug access, very low level assumes state is saved
64 static int cortex_a8_init_debug_access(struct target
*target
)
66 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
67 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
74 /* Unlocking the debug registers for modification */
75 /* The debugport might be uninitialised so try twice */
76 retval
= mem_ap_write_atomic_u32(swjdp
, armv7a
->debug_base
+ CPUDBG_LOCKACCESS
, 0xC5ACCE55);
77 if (retval
!= ERROR_OK
)
78 mem_ap_write_atomic_u32(swjdp
, armv7a
->debug_base
+ CPUDBG_LOCKACCESS
, 0xC5ACCE55);
79 /* Clear Sticky Power Down status Bit in PRSR to enable access to
80 the registers in the Core Power Domain */
81 retval
= mem_ap_read_atomic_u32(swjdp
, armv7a
->debug_base
+ CPUDBG_PRSR
, &dummy
);
82 /* Enabling of instruction execution in debug mode is done in debug_entry code */
84 /* Resync breakpoint registers */
86 /* Since this is likley called from init or reset, update targtet state information*/
87 cortex_a8_poll(target
);
92 /* To reduce needless round-trips, pass in a pointer to the current
93 * DSCR value. Initialize it to zero if you just need to know the
94 * value on return from this function; or DSCR_INSTR_COMP if you
95 * happen to know that no instruction is pending.
97 static int cortex_a8_exec_opcode(struct target
*target
,
98 uint32_t opcode
, uint32_t *dscr_p
)
102 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
103 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
105 dscr
= dscr_p
? *dscr_p
: 0;
107 LOG_DEBUG("exec opcode 0x%08" PRIx32
, opcode
);
109 /* Wait for InstrCompl bit to be set */
110 while ((dscr
& DSCR_INSTR_COMP
) == 0)
112 retval
= mem_ap_read_atomic_u32(swjdp
,
113 armv7a
->debug_base
+ CPUDBG_DSCR
, &dscr
);
114 if (retval
!= ERROR_OK
)
116 LOG_ERROR("Could not read DSCR register, opcode = 0x%08" PRIx32
, opcode
);
121 mem_ap_write_u32(swjdp
, armv7a
->debug_base
+ CPUDBG_ITR
, opcode
);
125 retval
= mem_ap_read_atomic_u32(swjdp
,
126 armv7a
->debug_base
+ CPUDBG_DSCR
, &dscr
);
127 if (retval
!= ERROR_OK
)
129 LOG_ERROR("Could not read DSCR register");
133 while ((dscr
& DSCR_INSTR_COMP
) == 0); /* Wait for InstrCompl bit to be set */
141 /**************************************************************************
142 Read core register with very few exec_opcode, fast but needs work_area.
143 This can cause problems with MMU active.
144 **************************************************************************/
145 static int cortex_a8_read_regs_through_mem(struct target
*target
, uint32_t address
,
148 int retval
= ERROR_OK
;
149 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
150 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
152 cortex_a8_dap_read_coreregister_u32(target
, regfile
, 0);
153 cortex_a8_dap_write_coreregister_u32(target
, address
, 0);
154 cortex_a8_exec_opcode(target
, ARMV4_5_STMIA(0, 0xFFFE, 0, 0), NULL
);
155 dap_ap_select(swjdp
, swjdp_memoryap
);
156 mem_ap_read_buf_u32(swjdp
, (uint8_t *)(®file
[1]), 4*15, address
);
157 dap_ap_select(swjdp
, swjdp_debugap
);
162 static int cortex_a8_dap_read_coreregister_u32(struct target
*target
,
163 uint32_t *value
, int regnum
)
165 int retval
= ERROR_OK
;
166 uint8_t reg
= regnum
&0xFF;
168 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
169 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
176 /* Rn to DCCTX, "MCR p14, 0, Rn, c0, c5, 0" 0xEE00nE15 */
177 cortex_a8_exec_opcode(target
,
178 ARMV4_5_MCR(14, 0, reg
, 0, 5, 0),
183 /* "MOV r0, r15"; then move r0 to DCCTX */
184 cortex_a8_exec_opcode(target
, 0xE1A0000F, &dscr
);
185 cortex_a8_exec_opcode(target
,
186 ARMV4_5_MCR(14, 0, 0, 0, 5, 0),
191 /* "MRS r0, CPSR" or "MRS r0, SPSR"
192 * then move r0 to DCCTX
194 cortex_a8_exec_opcode(target
, ARMV4_5_MRS(0, reg
& 1), &dscr
);
195 cortex_a8_exec_opcode(target
,
196 ARMV4_5_MCR(14, 0, 0, 0, 5, 0),
200 /* Wait for DTRRXfull then read DTRRTX */
201 while ((dscr
& DSCR_DTR_TX_FULL
) == 0)
203 retval
= mem_ap_read_atomic_u32(swjdp
,
204 armv7a
->debug_base
+ CPUDBG_DSCR
, &dscr
);
207 retval
= mem_ap_read_atomic_u32(swjdp
,
208 armv7a
->debug_base
+ CPUDBG_DTRTX
, value
);
209 LOG_DEBUG("read DCC 0x%08" PRIx32
, *value
);
214 static int cortex_a8_dap_write_coreregister_u32(struct target
*target
,
215 uint32_t value
, int regnum
)
217 int retval
= ERROR_OK
;
218 uint8_t Rd
= regnum
&0xFF;
220 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
221 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
223 LOG_DEBUG("register %i, value 0x%08" PRIx32
, regnum
, value
);
225 /* Check that DCCRX is not full */
226 retval
= mem_ap_read_atomic_u32(swjdp
,
227 armv7a
->debug_base
+ CPUDBG_DSCR
, &dscr
);
228 if (dscr
& DSCR_DTR_RX_FULL
)
230 LOG_ERROR("DSCR_DTR_RX_FULL, dscr 0x%08" PRIx32
, dscr
);
231 /* Clear DCCRX with MCR(p14, 0, Rd, c0, c5, 0), opcode 0xEE000E15 */
232 cortex_a8_exec_opcode(target
, ARMV4_5_MRC(14, 0, 0, 0, 5, 0),
239 /* Write DTRRX ... sets DSCR.DTRRXfull but exec_opcode() won't care */
240 LOG_DEBUG("write DCC 0x%08" PRIx32
, value
);
241 retval
= mem_ap_write_u32(swjdp
,
242 armv7a
->debug_base
+ CPUDBG_DTRRX
, value
);
246 /* DCCRX to Rn, "MCR p14, 0, Rn, c0, c5, 0", 0xEE00nE15 */
247 cortex_a8_exec_opcode(target
, ARMV4_5_MRC(14, 0, Rd
, 0, 5, 0),
252 /* DCCRX to R0, "MCR p14, 0, R0, c0, c5, 0", 0xEE000E15
255 cortex_a8_exec_opcode(target
, ARMV4_5_MRC(14, 0, 0, 0, 5, 0),
257 cortex_a8_exec_opcode(target
, 0xE1A0F000, &dscr
);
261 /* DCCRX to R0, "MCR p14, 0, R0, c0, c5, 0", 0xEE000E15
262 * then "MSR CPSR_cxsf, r0" or "MSR SPSR_cxsf, r0" (all fields)
264 cortex_a8_exec_opcode(target
, ARMV4_5_MRC(14, 0, 0, 0, 5, 0),
266 cortex_a8_exec_opcode(target
, ARMV4_5_MSR_GP(0, 0xF, Rd
& 1),
269 /* "Prefetch flush" after modifying execution status in CPSR */
271 cortex_a8_exec_opcode(target
,
272 ARMV4_5_MCR(15, 0, 0, 7, 5, 4),
279 /* Write to memory mapped registers directly with no cache or mmu handling */
280 static int cortex_a8_dap_write_memap_register_u32(struct target
*target
, uint32_t address
, uint32_t value
)
283 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
284 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
286 retval
= mem_ap_write_atomic_u32(swjdp
, address
, value
);
292 * Cortex-A8 implementation of Debug Programmer's Model
294 * NOTE the invariant: these routines return with DSCR_INSTR_COMP set,
295 * so there's no need to poll for it before executing an instruction.
297 * NOTE that in several of these cases the "stall" mode might be useful.
298 * It'd let us queue a few operations together... prepare/finish might
299 * be the places to enable/disable that mode.
302 static inline struct cortex_a8_common
*dpm_to_a8(struct arm_dpm
*dpm
)
304 return container_of(dpm
, struct cortex_a8_common
, armv7a_common
.dpm
);
307 static int cortex_a8_write_dcc(struct cortex_a8_common
*a8
, uint32_t data
)
309 LOG_DEBUG("write DCC 0x%08" PRIx32
, data
);
310 return mem_ap_write_u32(&a8
->armv7a_common
.swjdp_info
,
311 a8
->armv7a_common
.debug_base
+ CPUDBG_DTRRX
, data
);
314 static int cortex_a8_read_dcc(struct cortex_a8_common
*a8
, uint32_t *data
,
317 struct swjdp_common
*swjdp
= &a8
->armv7a_common
.swjdp_info
;
318 uint32_t dscr
= DSCR_INSTR_COMP
;
324 /* Wait for DTRRXfull */
325 while ((dscr
& DSCR_DTR_TX_FULL
) == 0) {
326 retval
= mem_ap_read_atomic_u32(swjdp
,
327 a8
->armv7a_common
.debug_base
+ CPUDBG_DSCR
,
331 retval
= mem_ap_read_atomic_u32(swjdp
,
332 a8
->armv7a_common
.debug_base
+ CPUDBG_DTRTX
, data
);
333 //LOG_DEBUG("read DCC 0x%08" PRIx32, *data);
341 static int cortex_a8_dpm_prepare(struct arm_dpm
*dpm
)
343 struct cortex_a8_common
*a8
= dpm_to_a8(dpm
);
344 struct swjdp_common
*swjdp
= &a8
->armv7a_common
.swjdp_info
;
348 /* set up invariant: INSTR_COMP is set after ever DPM operation */
350 retval
= mem_ap_read_atomic_u32(swjdp
,
351 a8
->armv7a_common
.debug_base
+ CPUDBG_DSCR
,
353 } while ((dscr
& DSCR_INSTR_COMP
) == 0);
355 /* this "should never happen" ... */
356 if (dscr
& DSCR_DTR_RX_FULL
) {
357 LOG_ERROR("DSCR_DTR_RX_FULL, dscr 0x%08" PRIx32
, dscr
);
359 retval
= cortex_a8_exec_opcode(
360 a8
->armv7a_common
.armv4_5_common
.target
,
361 ARMV4_5_MRC(14, 0, 0, 0, 5, 0),
368 static int cortex_a8_dpm_finish(struct arm_dpm
*dpm
)
370 /* REVISIT what could be done here? */
374 static int cortex_a8_instr_write_data_dcc(struct arm_dpm
*dpm
,
375 uint32_t opcode
, uint32_t data
)
377 struct cortex_a8_common
*a8
= dpm_to_a8(dpm
);
379 uint32_t dscr
= DSCR_INSTR_COMP
;
381 retval
= cortex_a8_write_dcc(a8
, data
);
383 return cortex_a8_exec_opcode(
384 a8
->armv7a_common
.armv4_5_common
.target
,
389 static int cortex_a8_instr_write_data_r0(struct arm_dpm
*dpm
,
390 uint32_t opcode
, uint32_t data
)
392 struct cortex_a8_common
*a8
= dpm_to_a8(dpm
);
393 uint32_t dscr
= DSCR_INSTR_COMP
;
396 retval
= cortex_a8_write_dcc(a8
, data
);
398 /* DCCRX to R0, "MCR p14, 0, R0, c0, c5, 0", 0xEE000E15 */
399 retval
= cortex_a8_exec_opcode(
400 a8
->armv7a_common
.armv4_5_common
.target
,
401 ARMV4_5_MRC(14, 0, 0, 0, 5, 0),
404 /* then the opcode, taking data from R0 */
405 retval
= cortex_a8_exec_opcode(
406 a8
->armv7a_common
.armv4_5_common
.target
,
413 static int cortex_a8_instr_cpsr_sync(struct arm_dpm
*dpm
)
415 struct target
*target
= dpm
->arm
->target
;
416 uint32_t dscr
= DSCR_INSTR_COMP
;
418 /* "Prefetch flush" after modifying execution status in CPSR */
419 return cortex_a8_exec_opcode(target
,
420 ARMV4_5_MCR(15, 0, 0, 7, 5, 4),
424 static int cortex_a8_instr_read_data_dcc(struct arm_dpm
*dpm
,
425 uint32_t opcode
, uint32_t *data
)
427 struct cortex_a8_common
*a8
= dpm_to_a8(dpm
);
429 uint32_t dscr
= DSCR_INSTR_COMP
;
431 /* the opcode, writing data to DCC */
432 retval
= cortex_a8_exec_opcode(
433 a8
->armv7a_common
.armv4_5_common
.target
,
437 return cortex_a8_read_dcc(a8
, data
, &dscr
);
441 static int cortex_a8_instr_read_data_r0(struct arm_dpm
*dpm
,
442 uint32_t opcode
, uint32_t *data
)
444 struct cortex_a8_common
*a8
= dpm_to_a8(dpm
);
445 uint32_t dscr
= DSCR_INSTR_COMP
;
448 /* the opcode, writing data to R0 */
449 retval
= cortex_a8_exec_opcode(
450 a8
->armv7a_common
.armv4_5_common
.target
,
454 /* write R0 to DCC */
455 retval
= cortex_a8_exec_opcode(
456 a8
->armv7a_common
.armv4_5_common
.target
,
457 ARMV4_5_MCR(14, 0, 0, 0, 5, 0),
460 return cortex_a8_read_dcc(a8
, data
, &dscr
);
463 static int cortex_a8_bpwp_enable(struct arm_dpm
*dpm
, unsigned index
,
464 uint32_t addr
, uint32_t control
)
466 struct cortex_a8_common
*a8
= dpm_to_a8(dpm
);
467 uint32_t vr
= a8
->armv7a_common
.debug_base
;
468 uint32_t cr
= a8
->armv7a_common
.debug_base
;
472 case 0 ... 15: /* breakpoints */
473 vr
+= CPUDBG_BVR_BASE
;
474 cr
+= CPUDBG_BCR_BASE
;
476 case 16 ... 31: /* watchpoints */
477 vr
+= CPUDBG_WVR_BASE
;
478 cr
+= CPUDBG_WCR_BASE
;
487 LOG_DEBUG("A8: bpwp enable, vr %08x cr %08x",
488 (unsigned) vr
, (unsigned) cr
);
490 retval
= cortex_a8_dap_write_memap_register_u32(dpm
->arm
->target
,
492 if (retval
!= ERROR_OK
)
494 retval
= cortex_a8_dap_write_memap_register_u32(dpm
->arm
->target
,
499 static int cortex_a8_bpwp_disable(struct arm_dpm
*dpm
, unsigned index
)
501 struct cortex_a8_common
*a8
= dpm_to_a8(dpm
);
506 cr
= a8
->armv7a_common
.debug_base
+ CPUDBG_BCR_BASE
;
509 cr
= a8
->armv7a_common
.debug_base
+ CPUDBG_WCR_BASE
;
517 LOG_DEBUG("A8: bpwp disable, cr %08x", (unsigned) cr
);
519 /* clear control register */
520 return cortex_a8_dap_write_memap_register_u32(dpm
->arm
->target
, cr
, 0);
523 static int cortex_a8_dpm_setup(struct cortex_a8_common
*a8
, uint32_t didr
)
525 struct arm_dpm
*dpm
= &a8
->armv7a_common
.dpm
;
528 dpm
->arm
= &a8
->armv7a_common
.armv4_5_common
;
531 dpm
->prepare
= cortex_a8_dpm_prepare
;
532 dpm
->finish
= cortex_a8_dpm_finish
;
534 dpm
->instr_write_data_dcc
= cortex_a8_instr_write_data_dcc
;
535 dpm
->instr_write_data_r0
= cortex_a8_instr_write_data_r0
;
536 dpm
->instr_cpsr_sync
= cortex_a8_instr_cpsr_sync
;
538 dpm
->instr_read_data_dcc
= cortex_a8_instr_read_data_dcc
;
539 dpm
->instr_read_data_r0
= cortex_a8_instr_read_data_r0
;
541 dpm
->bpwp_enable
= cortex_a8_bpwp_enable
;
542 dpm
->bpwp_disable
= cortex_a8_bpwp_disable
;
544 retval
= arm_dpm_setup(dpm
);
545 if (retval
== ERROR_OK
)
546 retval
= arm_dpm_initialize(dpm
);
553 * Cortex-A8 Run control
556 static int cortex_a8_poll(struct target
*target
)
558 int retval
= ERROR_OK
;
560 struct cortex_a8_common
*cortex_a8
= target_to_cortex_a8(target
);
561 struct armv7a_common
*armv7a
= &cortex_a8
->armv7a_common
;
562 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
563 enum target_state prev_target_state
= target
->state
;
564 uint8_t saved_apsel
= dap_ap_get_select(swjdp
);
566 dap_ap_select(swjdp
, swjdp_debugap
);
567 retval
= mem_ap_read_atomic_u32(swjdp
,
568 armv7a
->debug_base
+ CPUDBG_DSCR
, &dscr
);
569 if (retval
!= ERROR_OK
)
571 dap_ap_select(swjdp
, saved_apsel
);
574 cortex_a8
->cpudbg_dscr
= dscr
;
576 if ((dscr
& 0x3) == 0x3)
578 if (prev_target_state
!= TARGET_HALTED
)
580 /* We have a halting debug event */
581 LOG_DEBUG("Target halted");
582 target
->state
= TARGET_HALTED
;
583 if ((prev_target_state
== TARGET_RUNNING
)
584 || (prev_target_state
== TARGET_RESET
))
586 retval
= cortex_a8_debug_entry(target
);
587 if (retval
!= ERROR_OK
)
590 target_call_event_callbacks(target
,
591 TARGET_EVENT_HALTED
);
593 if (prev_target_state
== TARGET_DEBUG_RUNNING
)
597 retval
= cortex_a8_debug_entry(target
);
598 if (retval
!= ERROR_OK
)
601 target_call_event_callbacks(target
,
602 TARGET_EVENT_DEBUG_HALTED
);
606 else if ((dscr
& 0x3) == 0x2)
608 target
->state
= TARGET_RUNNING
;
612 LOG_DEBUG("Unknown target state dscr = 0x%08" PRIx32
, dscr
);
613 target
->state
= TARGET_UNKNOWN
;
616 dap_ap_select(swjdp
, saved_apsel
);
621 static int cortex_a8_halt(struct target
*target
)
623 int retval
= ERROR_OK
;
625 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
626 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
627 uint8_t saved_apsel
= dap_ap_get_select(swjdp
);
628 dap_ap_select(swjdp
, swjdp_debugap
);
631 * Tell the core to be halted by writing DRCR with 0x1
632 * and then wait for the core to be halted.
634 retval
= mem_ap_write_atomic_u32(swjdp
,
635 armv7a
->debug_base
+ CPUDBG_DRCR
, 0x1);
638 * enter halting debug mode
640 mem_ap_read_atomic_u32(swjdp
, armv7a
->debug_base
+ CPUDBG_DSCR
, &dscr
);
641 retval
= mem_ap_write_atomic_u32(swjdp
,
642 armv7a
->debug_base
+ CPUDBG_DSCR
, dscr
| DSCR_HALT_DBG_MODE
);
644 if (retval
!= ERROR_OK
)
648 mem_ap_read_atomic_u32(swjdp
,
649 armv7a
->debug_base
+ CPUDBG_DSCR
, &dscr
);
650 } while ((dscr
& DSCR_CORE_HALTED
) == 0);
652 target
->debug_reason
= DBG_REASON_DBGRQ
;
655 dap_ap_select(swjdp
, saved_apsel
);
659 static int cortex_a8_resume(struct target
*target
, int current
,
660 uint32_t address
, int handle_breakpoints
, int debug_execution
)
662 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
663 struct arm
*armv4_5
= &armv7a
->armv4_5_common
;
664 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
666 // struct breakpoint *breakpoint = NULL;
667 uint32_t resume_pc
, dscr
;
669 uint8_t saved_apsel
= dap_ap_get_select(swjdp
);
670 dap_ap_select(swjdp
, swjdp_debugap
);
672 if (!debug_execution
)
673 target_free_all_working_areas(target
);
678 /* Disable interrupts */
679 /* We disable interrupts in the PRIMASK register instead of
680 * masking with C_MASKINTS,
681 * This is probably the same issue as Cortex-M3 Errata 377493:
682 * C_MASKINTS in parallel with disabled interrupts can cause
683 * local faults to not be taken. */
684 buf_set_u32(armv7m
->core_cache
->reg_list
[ARMV7M_PRIMASK
].value
, 0, 32, 1);
685 armv7m
->core_cache
->reg_list
[ARMV7M_PRIMASK
].dirty
= 1;
686 armv7m
->core_cache
->reg_list
[ARMV7M_PRIMASK
].valid
= 1;
688 /* Make sure we are in Thumb mode */
689 buf_set_u32(armv7m
->core_cache
->reg_list
[ARMV7M_xPSR
].value
, 0, 32,
690 buf_get_u32(armv7m
->core_cache
->reg_list
[ARMV7M_xPSR
].value
, 0, 32) | (1 << 24));
691 armv7m
->core_cache
->reg_list
[ARMV7M_xPSR
].dirty
= 1;
692 armv7m
->core_cache
->reg_list
[ARMV7M_xPSR
].valid
= 1;
696 /* current = 1: continue on current pc, otherwise continue at <address> */
697 resume_pc
= buf_get_u32(
698 armv4_5
->core_cache
->reg_list
[15].value
,
703 /* Make sure that the Armv7 gdb thumb fixups does not
704 * kill the return address
706 switch (armv4_5
->core_state
)
708 case ARMV4_5_STATE_ARM
:
709 resume_pc
&= 0xFFFFFFFC;
711 case ARMV4_5_STATE_THUMB
:
712 case ARM_STATE_THUMB_EE
:
713 /* When the return address is loaded into PC
714 * bit 0 must be 1 to stay in Thumb state
718 case ARMV4_5_STATE_JAZELLE
:
719 LOG_ERROR("How do I resume into Jazelle state??");
722 LOG_DEBUG("resume pc = 0x%08" PRIx32
, resume_pc
);
723 buf_set_u32(armv4_5
->core_cache
->reg_list
[15].value
,
725 armv4_5
->core_cache
->reg_list
[15].dirty
= 1;
726 armv4_5
->core_cache
->reg_list
[15].valid
= 1;
728 cortex_a8_restore_context(target
, handle_breakpoints
);
731 /* the front-end may request us not to handle breakpoints */
732 if (handle_breakpoints
)
734 /* Single step past breakpoint at current address */
735 if ((breakpoint
= breakpoint_find(target
, resume_pc
)))
737 LOG_DEBUG("unset breakpoint at 0x%8.8x", breakpoint
->address
);
738 cortex_m3_unset_breakpoint(target
, breakpoint
);
739 cortex_m3_single_step_core(target
);
740 cortex_m3_set_breakpoint(target
, breakpoint
);
745 /* Restart core and wait for it to be started
746 * NOTE: this clears DSCR_ITR_EN and other bits.
748 * REVISIT: for single stepping, we probably want to
749 * disable IRQs by default, with optional override...
751 mem_ap_write_atomic_u32(swjdp
, armv7a
->debug_base
+ CPUDBG_DRCR
, 0x2);
754 mem_ap_read_atomic_u32(swjdp
,
755 armv7a
->debug_base
+ CPUDBG_DSCR
, &dscr
);
756 } while ((dscr
& DSCR_CORE_RESTARTED
) == 0);
758 target
->debug_reason
= DBG_REASON_NOTHALTED
;
759 target
->state
= TARGET_RUNNING
;
761 /* registers are now invalid */
762 register_cache_invalidate(armv4_5
->core_cache
);
764 if (!debug_execution
)
766 target
->state
= TARGET_RUNNING
;
767 target_call_event_callbacks(target
, TARGET_EVENT_RESUMED
);
768 LOG_DEBUG("target resumed at 0x%" PRIx32
, resume_pc
);
772 target
->state
= TARGET_DEBUG_RUNNING
;
773 target_call_event_callbacks(target
, TARGET_EVENT_DEBUG_RESUMED
);
774 LOG_DEBUG("target debug resumed at 0x%" PRIx32
, resume_pc
);
777 dap_ap_select(swjdp
, saved_apsel
);
782 static int cortex_a8_debug_entry(struct target
*target
)
785 uint32_t regfile
[16], wfar
, cpsr
, dscr
;
786 int retval
= ERROR_OK
;
787 struct working_area
*regfile_working_area
= NULL
;
788 struct cortex_a8_common
*cortex_a8
= target_to_cortex_a8(target
);
789 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
790 struct arm
*armv4_5
= &armv7a
->armv4_5_common
;
791 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
794 LOG_DEBUG("dscr = 0x%08" PRIx32
, cortex_a8
->cpudbg_dscr
);
796 mem_ap_read_atomic_u32(swjdp
,
797 armv7a
->debug_base
+ CPUDBG_DSCR
, &dscr
);
799 /* REVISIT see A8 TRM 12.11.4 steps 2..3 -- make sure that any
800 * imprecise data aborts get discarded by issuing a Data
801 * Synchronization Barrier: ARMV4_5_MCR(15, 0, 0, 7, 10, 4).
804 /* Enable the ITR execution once we are in debug mode */
806 retval
= mem_ap_write_atomic_u32(swjdp
,
807 armv7a
->debug_base
+ CPUDBG_DSCR
, dscr
);
809 /* Examine debug reason */
810 switch (DSCR_ENTRY(cortex_a8
->cpudbg_dscr
))
812 case 0: /* DRCR[0] write */
814 target
->debug_reason
= DBG_REASON_DBGRQ
;
816 case 1: /* HW breakpoint */
817 case 3: /* SW BKPT */
818 case 5: /* vector catch */
819 target
->debug_reason
= DBG_REASON_BREAKPOINT
;
821 case 2: /* asynch watchpoint */
822 case 10: /* precise watchpoint */
823 target
->debug_reason
= DBG_REASON_WATCHPOINT
;
825 /* save address of faulting instruction */
826 retval
= mem_ap_read_atomic_u32(swjdp
,
827 armv7a
->debug_base
+ CPUDBG_WFAR
,
829 arm_dpm_report_wfar(&armv7a
->dpm
, wfar
);
832 target
->debug_reason
= DBG_REASON_UNDEFINED
;
836 /* REVISIT fast_reg_read is never set ... */
838 /* Examine target state and mode */
839 if (cortex_a8
->fast_reg_read
)
840 target_alloc_working_area(target
, 64, ®file_working_area
);
842 /* First load register acessible through core debug port*/
843 if (!regfile_working_area
)
845 retval
= arm_dpm_read_current_registers(&armv7a
->dpm
);
849 dap_ap_select(swjdp
, swjdp_memoryap
);
850 cortex_a8_read_regs_through_mem(target
,
851 regfile_working_area
->address
, regfile
);
852 dap_ap_select(swjdp
, swjdp_memoryap
);
853 target_free_working_area(target
, regfile_working_area
);
855 /* read Current PSR */
856 cortex_a8_dap_read_coreregister_u32(target
, &cpsr
, 16);
857 dap_ap_select(swjdp
, swjdp_debugap
);
858 LOG_DEBUG("cpsr: %8.8" PRIx32
, cpsr
);
860 arm_set_cpsr(armv4_5
, cpsr
);
863 for (i
= 0; i
<= ARM_PC
; i
++)
865 reg
= arm_reg_current(armv4_5
, i
);
867 buf_set_u32(reg
->value
, 0, 32, regfile
[i
]);
872 /* Fixup PC Resume Address */
875 // T bit set for Thumb or ThumbEE state
876 regfile
[ARM_PC
] -= 4;
881 regfile
[ARM_PC
] -= 8;
884 reg
= armv4_5
->core_cache
->reg_list
+ 15;
885 buf_set_u32(reg
->value
, 0, 32, regfile
[ARM_PC
]);
886 reg
->dirty
= reg
->valid
;
890 /* TODO, Move this */
891 uint32_t cp15_control_register
, cp15_cacr
, cp15_nacr
;
892 cortex_a8_read_cp(target
, &cp15_control_register
, 15, 0, 1, 0, 0);
893 LOG_DEBUG("cp15_control_register = 0x%08x", cp15_control_register
);
895 cortex_a8_read_cp(target
, &cp15_cacr
, 15, 0, 1, 0, 2);
896 LOG_DEBUG("cp15 Coprocessor Access Control Register = 0x%08x", cp15_cacr
);
898 cortex_a8_read_cp(target
, &cp15_nacr
, 15, 0, 1, 1, 2);
899 LOG_DEBUG("cp15 Nonsecure Access Control Register = 0x%08x", cp15_nacr
);
902 /* Are we in an exception handler */
903 // armv4_5->exception_number = 0;
904 if (armv7a
->post_debug_entry
)
905 armv7a
->post_debug_entry(target
);
910 static void cortex_a8_post_debug_entry(struct target
*target
)
912 struct cortex_a8_common
*cortex_a8
= target_to_cortex_a8(target
);
913 struct armv7a_common
*armv7a
= &cortex_a8
->armv7a_common
;
916 /* MRC p15,0,<Rt>,c1,c0,0 ; Read CP15 System Control Register */
917 retval
= armv7a
->armv4_5_common
.mrc(target
, 15,
920 &cortex_a8
->cp15_control_reg
);
921 LOG_DEBUG("cp15_control_reg: %8.8" PRIx32
, cortex_a8
->cp15_control_reg
);
923 if (armv7a
->armv4_5_mmu
.armv4_5_cache
.ctype
== -1)
925 uint32_t cache_type_reg
;
927 /* MRC p15,0,<Rt>,c0,c0,1 ; Read CP15 Cache Type Register */
928 retval
= armv7a
->armv4_5_common
.mrc(target
, 15,
932 LOG_DEBUG("cp15 cache type: %8.8x", (unsigned) cache_type_reg
);
934 /* FIXME the armv4_4 cache info DOES NOT APPLY to Cortex-A8 */
935 armv4_5_identify_cache(cache_type_reg
,
936 &armv7a
->armv4_5_mmu
.armv4_5_cache
);
939 armv7a
->armv4_5_mmu
.mmu_enabled
=
940 (cortex_a8
->cp15_control_reg
& 0x1U
) ? 1 : 0;
941 armv7a
->armv4_5_mmu
.armv4_5_cache
.d_u_cache_enabled
=
942 (cortex_a8
->cp15_control_reg
& 0x4U
) ? 1 : 0;
943 armv7a
->armv4_5_mmu
.armv4_5_cache
.i_cache_enabled
=
944 (cortex_a8
->cp15_control_reg
& 0x1000U
) ? 1 : 0;
949 static int cortex_a8_step(struct target
*target
, int current
, uint32_t address
,
950 int handle_breakpoints
)
952 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
953 struct arm
*armv4_5
= &armv7a
->armv4_5_common
;
954 struct breakpoint
*breakpoint
= NULL
;
955 struct breakpoint stepbreakpoint
;
960 if (target
->state
!= TARGET_HALTED
)
962 LOG_WARNING("target not halted");
963 return ERROR_TARGET_NOT_HALTED
;
966 /* current = 1: continue on current pc, otherwise continue at <address> */
967 r
= armv4_5
->core_cache
->reg_list
+ 15;
970 buf_set_u32(r
->value
, 0, 32, address
);
974 address
= buf_get_u32(r
->value
, 0, 32);
977 /* The front-end may request us not to handle breakpoints.
978 * But since Cortex-A8 uses breakpoint for single step,
979 * we MUST handle breakpoints.
981 handle_breakpoints
= 1;
982 if (handle_breakpoints
) {
983 breakpoint
= breakpoint_find(target
, address
);
985 cortex_a8_unset_breakpoint(target
, breakpoint
);
988 /* Setup single step breakpoint */
989 stepbreakpoint
.address
= address
;
990 stepbreakpoint
.length
= (armv4_5
->core_state
== ARMV4_5_STATE_THUMB
)
992 stepbreakpoint
.type
= BKPT_HARD
;
993 stepbreakpoint
.set
= 0;
995 /* Break on IVA mismatch */
996 cortex_a8_set_breakpoint(target
, &stepbreakpoint
, 0x04);
998 target
->debug_reason
= DBG_REASON_SINGLESTEP
;
1000 cortex_a8_resume(target
, 1, address
, 0, 0);
1002 while (target
->state
!= TARGET_HALTED
)
1004 cortex_a8_poll(target
);
1007 LOG_WARNING("timeout waiting for target halt");
1012 cortex_a8_unset_breakpoint(target
, &stepbreakpoint
);
1014 target
->debug_reason
= DBG_REASON_BREAKPOINT
;
1017 cortex_a8_set_breakpoint(target
, breakpoint
, 0);
1019 if (target
->state
!= TARGET_HALTED
)
1020 LOG_DEBUG("target stepped");
1025 static int cortex_a8_restore_context(struct target
*target
, bool bpwp
)
1027 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
1031 if (armv7a
->pre_restore_context
)
1032 armv7a
->pre_restore_context(target
);
1034 arm_dpm_write_dirty_registers(&armv7a
->dpm
, bpwp
);
1036 if (armv7a
->post_restore_context
)
1037 armv7a
->post_restore_context(target
);
1044 * Cortex-A8 Breakpoint and watchpoint fuctions
1047 /* Setup hardware Breakpoint Register Pair */
1048 static int cortex_a8_set_breakpoint(struct target
*target
,
1049 struct breakpoint
*breakpoint
, uint8_t matchmode
)
1054 uint8_t byte_addr_select
= 0x0F;
1055 struct cortex_a8_common
*cortex_a8
= target_to_cortex_a8(target
);
1056 struct armv7a_common
*armv7a
= &cortex_a8
->armv7a_common
;
1057 struct cortex_a8_brp
* brp_list
= cortex_a8
->brp_list
;
1059 if (breakpoint
->set
)
1061 LOG_WARNING("breakpoint already set");
1065 if (breakpoint
->type
== BKPT_HARD
)
1067 while (brp_list
[brp_i
].used
&& (brp_i
< cortex_a8
->brp_num
))
1069 if (brp_i
>= cortex_a8
->brp_num
)
1071 LOG_ERROR("ERROR Can not find free Breakpoint Register Pair");
1074 breakpoint
->set
= brp_i
+ 1;
1075 if (breakpoint
->length
== 2)
1077 byte_addr_select
= (3 << (breakpoint
->address
& 0x02));
1079 control
= ((matchmode
& 0x7) << 20)
1080 | (byte_addr_select
<< 5)
1082 brp_list
[brp_i
].used
= 1;
1083 brp_list
[brp_i
].value
= (breakpoint
->address
& 0xFFFFFFFC);
1084 brp_list
[brp_i
].control
= control
;
1085 cortex_a8_dap_write_memap_register_u32(target
, armv7a
->debug_base
1086 + CPUDBG_BVR_BASE
+ 4 * brp_list
[brp_i
].BRPn
,
1087 brp_list
[brp_i
].value
);
1088 cortex_a8_dap_write_memap_register_u32(target
, armv7a
->debug_base
1089 + CPUDBG_BCR_BASE
+ 4 * brp_list
[brp_i
].BRPn
,
1090 brp_list
[brp_i
].control
);
1091 LOG_DEBUG("brp %i control 0x%0" PRIx32
" value 0x%0" PRIx32
, brp_i
,
1092 brp_list
[brp_i
].control
,
1093 brp_list
[brp_i
].value
);
1095 else if (breakpoint
->type
== BKPT_SOFT
)
1098 if (breakpoint
->length
== 2)
1100 buf_set_u32(code
, 0, 32, ARMV5_T_BKPT(0x11));
1104 buf_set_u32(code
, 0, 32, ARMV5_BKPT(0x11));
1106 retval
= target
->type
->read_memory(target
,
1107 breakpoint
->address
& 0xFFFFFFFE,
1108 breakpoint
->length
, 1,
1109 breakpoint
->orig_instr
);
1110 if (retval
!= ERROR_OK
)
1112 retval
= target
->type
->write_memory(target
,
1113 breakpoint
->address
& 0xFFFFFFFE,
1114 breakpoint
->length
, 1, code
);
1115 if (retval
!= ERROR_OK
)
1117 breakpoint
->set
= 0x11; /* Any nice value but 0 */
1123 static int cortex_a8_unset_breakpoint(struct target
*target
, struct breakpoint
*breakpoint
)
1126 struct cortex_a8_common
*cortex_a8
= target_to_cortex_a8(target
);
1127 struct armv7a_common
*armv7a
= &cortex_a8
->armv7a_common
;
1128 struct cortex_a8_brp
* brp_list
= cortex_a8
->brp_list
;
1130 if (!breakpoint
->set
)
1132 LOG_WARNING("breakpoint not set");
1136 if (breakpoint
->type
== BKPT_HARD
)
1138 int brp_i
= breakpoint
->set
- 1;
1139 if ((brp_i
< 0) || (brp_i
>= cortex_a8
->brp_num
))
1141 LOG_DEBUG("Invalid BRP number in breakpoint");
1144 LOG_DEBUG("rbp %i control 0x%0" PRIx32
" value 0x%0" PRIx32
, brp_i
,
1145 brp_list
[brp_i
].control
, brp_list
[brp_i
].value
);
1146 brp_list
[brp_i
].used
= 0;
1147 brp_list
[brp_i
].value
= 0;
1148 brp_list
[brp_i
].control
= 0;
1149 cortex_a8_dap_write_memap_register_u32(target
, armv7a
->debug_base
1150 + CPUDBG_BCR_BASE
+ 4 * brp_list
[brp_i
].BRPn
,
1151 brp_list
[brp_i
].control
);
1152 cortex_a8_dap_write_memap_register_u32(target
, armv7a
->debug_base
1153 + CPUDBG_BVR_BASE
+ 4 * brp_list
[brp_i
].BRPn
,
1154 brp_list
[brp_i
].value
);
1158 /* restore original instruction (kept in target endianness) */
1159 if (breakpoint
->length
== 4)
1161 retval
= target
->type
->write_memory(target
,
1162 breakpoint
->address
& 0xFFFFFFFE,
1163 4, 1, breakpoint
->orig_instr
);
1164 if (retval
!= ERROR_OK
)
1169 retval
= target
->type
->write_memory(target
,
1170 breakpoint
->address
& 0xFFFFFFFE,
1171 2, 1, breakpoint
->orig_instr
);
1172 if (retval
!= ERROR_OK
)
1176 breakpoint
->set
= 0;
1181 static int cortex_a8_add_breakpoint(struct target
*target
,
1182 struct breakpoint
*breakpoint
)
1184 struct cortex_a8_common
*cortex_a8
= target_to_cortex_a8(target
);
1186 if ((breakpoint
->type
== BKPT_HARD
) && (cortex_a8
->brp_num_available
< 1))
1188 LOG_INFO("no hardware breakpoint available");
1189 return ERROR_TARGET_RESOURCE_NOT_AVAILABLE
;
1192 if (breakpoint
->type
== BKPT_HARD
)
1193 cortex_a8
->brp_num_available
--;
1194 cortex_a8_set_breakpoint(target
, breakpoint
, 0x00); /* Exact match */
1199 static int cortex_a8_remove_breakpoint(struct target
*target
, struct breakpoint
*breakpoint
)
1201 struct cortex_a8_common
*cortex_a8
= target_to_cortex_a8(target
);
1204 /* It is perfectly possible to remove brakpoints while the taget is running */
1205 if (target
->state
!= TARGET_HALTED
)
1207 LOG_WARNING("target not halted");
1208 return ERROR_TARGET_NOT_HALTED
;
1212 if (breakpoint
->set
)
1214 cortex_a8_unset_breakpoint(target
, breakpoint
);
1215 if (breakpoint
->type
== BKPT_HARD
)
1216 cortex_a8
->brp_num_available
++ ;
1226 * Cortex-A8 Reset fuctions
1229 static int cortex_a8_assert_reset(struct target
*target
)
1231 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
1235 /* FIXME when halt is requested, make it work somehow... */
1237 /* Issue some kind of warm reset. */
1238 if (target_has_event_action(target
, TARGET_EVENT_RESET_ASSERT
)) {
1239 target_handle_event(target
, TARGET_EVENT_RESET_ASSERT
);
1240 } else if (jtag_get_reset_config() & RESET_HAS_SRST
) {
1241 /* REVISIT handle "pulls" cases, if there's
1242 * hardware that needs them to work.
1244 jtag_add_reset(0, 1);
1246 LOG_ERROR("%s: how to reset?", target_name(target
));
1250 /* registers are now invalid */
1251 register_cache_invalidate(armv7a
->armv4_5_common
.core_cache
);
1253 target
->state
= TARGET_RESET
;
1258 static int cortex_a8_deassert_reset(struct target
*target
)
1264 /* be certain SRST is off */
1265 jtag_add_reset(0, 0);
1267 retval
= cortex_a8_poll(target
);
1269 if (target
->reset_halt
) {
1270 if (target
->state
!= TARGET_HALTED
) {
1271 LOG_WARNING("%s: ran after reset and before halt ...",
1272 target_name(target
));
1273 if ((retval
= target_halt(target
)) != ERROR_OK
)
1282 * Cortex-A8 Memory access
1284 * This is same Cortex M3 but we must also use the correct
1285 * ap number for every access.
1288 static int cortex_a8_read_memory(struct target
*target
, uint32_t address
,
1289 uint32_t size
, uint32_t count
, uint8_t *buffer
)
1291 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
1292 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
1293 int retval
= ERROR_INVALID_ARGUMENTS
;
1295 /* cortex_a8 handles unaligned memory access */
1297 // ??? dap_ap_select(swjdp, swjdp_memoryap);
1299 if (count
&& buffer
) {
1302 retval
= mem_ap_read_buf_u32(swjdp
, buffer
, 4 * count
, address
);
1305 retval
= mem_ap_read_buf_u16(swjdp
, buffer
, 2 * count
, address
);
1308 retval
= mem_ap_read_buf_u8(swjdp
, buffer
, count
, address
);
1316 static int cortex_a8_write_memory(struct target
*target
, uint32_t address
,
1317 uint32_t size
, uint32_t count
, uint8_t *buffer
)
1319 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
1320 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
1321 int retval
= ERROR_INVALID_ARGUMENTS
;
1323 // ??? dap_ap_select(swjdp, swjdp_memoryap);
1325 if (count
&& buffer
) {
1328 retval
= mem_ap_write_buf_u32(swjdp
, buffer
, 4 * count
, address
);
1331 retval
= mem_ap_write_buf_u16(swjdp
, buffer
, 2 * count
, address
);
1334 retval
= mem_ap_write_buf_u8(swjdp
, buffer
, count
, address
);
1339 /* REVISIT this op is generic ARMv7-A/R stuff */
1340 if (retval
== ERROR_OK
&& target
->state
== TARGET_HALTED
)
1342 struct arm_dpm
*dpm
= armv7a
->armv4_5_common
.dpm
;
1344 retval
= dpm
->prepare(dpm
);
1345 if (retval
!= ERROR_OK
)
1348 /* The Cache handling will NOT work with MMU active, the
1349 * wrong addresses will be invalidated!
1351 * For both ICache and DCache, walk all cache lines in the
1352 * address range. Cortex-A8 has fixed 64 byte line length.
1354 * REVISIT per ARMv7, these may trigger watchpoints ...
1357 /* invalidate I-Cache */
1358 if (armv7a
->armv4_5_mmu
.armv4_5_cache
.i_cache_enabled
)
1360 /* ICIMVAU - Invalidate Cache single entry
1362 * MCR p15, 0, r0, c7, c5, 1
1364 for (uint32_t cacheline
= address
;
1365 cacheline
< address
+ size
* count
;
1367 retval
= dpm
->instr_write_data_r0(dpm
,
1368 ARMV4_5_MCR(15, 0, 0, 7, 5, 1),
1373 /* invalidate D-Cache */
1374 if (armv7a
->armv4_5_mmu
.armv4_5_cache
.d_u_cache_enabled
)
1376 /* DCIMVAC - Invalidate data Cache line
1378 * MCR p15, 0, r0, c7, c6, 1
1380 for (uint32_t cacheline
= address
;
1381 cacheline
< address
+ size
* count
;
1383 retval
= dpm
->instr_write_data_r0(dpm
,
1384 ARMV4_5_MCR(15, 0, 0, 7, 6, 1),
1389 /* (void) */ dpm
->finish(dpm
);
1395 static int cortex_a8_bulk_write_memory(struct target
*target
, uint32_t address
,
1396 uint32_t count
, uint8_t *buffer
)
1398 return cortex_a8_write_memory(target
, address
, 4, count
, buffer
);
1402 static int cortex_a8_dcc_read(struct swjdp_common
*swjdp
, uint8_t *value
, uint8_t *ctrl
)
1407 mem_ap_read_buf_u16(swjdp
, (uint8_t*)&dcrdr
, 1, DCB_DCRDR
);
1408 *ctrl
= (uint8_t)dcrdr
;
1409 *value
= (uint8_t)(dcrdr
>> 8);
1411 LOG_DEBUG("data 0x%x ctrl 0x%x", *value
, *ctrl
);
1413 /* write ack back to software dcc register
1414 * signify we have read data */
1415 if (dcrdr
& (1 << 0))
1418 mem_ap_write_buf_u16(swjdp
, (uint8_t*)&dcrdr
, 1, DCB_DCRDR
);
1425 static int cortex_a8_handle_target_request(void *priv
)
1427 struct target
*target
= priv
;
1428 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
1429 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
1431 if (!target_was_examined(target
))
1433 if (!target
->dbg_msg_enabled
)
1436 if (target
->state
== TARGET_RUNNING
)
1441 cortex_a8_dcc_read(swjdp
, &data
, &ctrl
);
1443 /* check if we have data */
1444 if (ctrl
& (1 << 0))
1448 /* we assume target is quick enough */
1450 cortex_a8_dcc_read(swjdp
, &data
, &ctrl
);
1451 request
|= (data
<< 8);
1452 cortex_a8_dcc_read(swjdp
, &data
, &ctrl
);
1453 request
|= (data
<< 16);
1454 cortex_a8_dcc_read(swjdp
, &data
, &ctrl
);
1455 request
|= (data
<< 24);
1456 target_request(target
, request
);
1464 * Cortex-A8 target information and configuration
1467 static int cortex_a8_examine_first(struct target
*target
)
1469 struct cortex_a8_common
*cortex_a8
= target_to_cortex_a8(target
);
1470 struct armv7a_common
*armv7a
= &cortex_a8
->armv7a_common
;
1471 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
1473 int retval
= ERROR_OK
;
1474 uint32_t didr
, ctypr
, ttypr
, cpuid
;
1478 /* Here we shall insert a proper ROM Table scan */
1479 armv7a
->debug_base
= OMAP3530_DEBUG_BASE
;
1481 /* We do one extra read to ensure DAP is configured,
1482 * we call ahbap_debugport_init(swjdp) instead
1484 ahbap_debugport_init(swjdp
);
1485 mem_ap_read_atomic_u32(swjdp
, armv7a
->debug_base
+ CPUDBG_CPUID
, &cpuid
);
1486 if ((retval
= mem_ap_read_atomic_u32(swjdp
,
1487 armv7a
->debug_base
+ CPUDBG_CPUID
, &cpuid
)) != ERROR_OK
)
1489 LOG_DEBUG("Examine failed");
1493 if ((retval
= mem_ap_read_atomic_u32(swjdp
,
1494 armv7a
->debug_base
+ CPUDBG_CTYPR
, &ctypr
)) != ERROR_OK
)
1496 LOG_DEBUG("Examine failed");
1500 if ((retval
= mem_ap_read_atomic_u32(swjdp
,
1501 armv7a
->debug_base
+ CPUDBG_TTYPR
, &ttypr
)) != ERROR_OK
)
1503 LOG_DEBUG("Examine failed");
1507 if ((retval
= mem_ap_read_atomic_u32(swjdp
,
1508 armv7a
->debug_base
+ CPUDBG_DIDR
, &didr
)) != ERROR_OK
)
1510 LOG_DEBUG("Examine failed");
1514 LOG_DEBUG("cpuid = 0x%08" PRIx32
, cpuid
);
1515 LOG_DEBUG("ctypr = 0x%08" PRIx32
, ctypr
);
1516 LOG_DEBUG("ttypr = 0x%08" PRIx32
, ttypr
);
1517 LOG_DEBUG("didr = 0x%08" PRIx32
, didr
);
1519 armv7a
->armv4_5_common
.core_type
= ARM_MODE_MON
;
1520 cortex_a8_dpm_setup(cortex_a8
, didr
);
1522 /* Setup Breakpoint Register Pairs */
1523 cortex_a8
->brp_num
= ((didr
>> 24) & 0x0F) + 1;
1524 cortex_a8
->brp_num_context
= ((didr
>> 20) & 0x0F) + 1;
1525 cortex_a8
->brp_num_available
= cortex_a8
->brp_num
;
1526 cortex_a8
->brp_list
= calloc(cortex_a8
->brp_num
, sizeof(struct cortex_a8_brp
));
1527 // cortex_a8->brb_enabled = ????;
1528 for (i
= 0; i
< cortex_a8
->brp_num
; i
++)
1530 cortex_a8
->brp_list
[i
].used
= 0;
1531 if (i
< (cortex_a8
->brp_num
-cortex_a8
->brp_num_context
))
1532 cortex_a8
->brp_list
[i
].type
= BRP_NORMAL
;
1534 cortex_a8
->brp_list
[i
].type
= BRP_CONTEXT
;
1535 cortex_a8
->brp_list
[i
].value
= 0;
1536 cortex_a8
->brp_list
[i
].control
= 0;
1537 cortex_a8
->brp_list
[i
].BRPn
= i
;
1540 LOG_DEBUG("Configured %i hw breakpoints", cortex_a8
->brp_num
);
1542 target_set_examined(target
);
1546 static int cortex_a8_examine(struct target
*target
)
1548 int retval
= ERROR_OK
;
1550 /* don't re-probe hardware after each reset */
1551 if (!target_was_examined(target
))
1552 retval
= cortex_a8_examine_first(target
);
1554 /* Configure core debug access */
1555 if (retval
== ERROR_OK
)
1556 retval
= cortex_a8_init_debug_access(target
);
1562 * Cortex-A8 target creation and initialization
1565 static int cortex_a8_init_target(struct command_context
*cmd_ctx
,
1566 struct target
*target
)
1568 /* examine_first() does a bunch of this */
1572 static int cortex_a8_init_arch_info(struct target
*target
,
1573 struct cortex_a8_common
*cortex_a8
, struct jtag_tap
*tap
)
1575 struct armv7a_common
*armv7a
= &cortex_a8
->armv7a_common
;
1576 struct arm
*armv4_5
= &armv7a
->armv4_5_common
;
1577 struct swjdp_common
*swjdp
= &armv7a
->swjdp_info
;
1579 /* Setup struct cortex_a8_common */
1580 cortex_a8
->common_magic
= CORTEX_A8_COMMON_MAGIC
;
1581 armv4_5
->arch_info
= armv7a
;
1583 /* prepare JTAG information for the new target */
1584 cortex_a8
->jtag_info
.tap
= tap
;
1585 cortex_a8
->jtag_info
.scann_size
= 4;
1587 swjdp
->dp_select_value
= -1;
1588 swjdp
->ap_csw_value
= -1;
1589 swjdp
->ap_tar_value
= -1;
1590 swjdp
->jtag_info
= &cortex_a8
->jtag_info
;
1591 swjdp
->memaccess_tck
= 80;
1593 /* Number of bits for tar autoincrement, impl. dep. at least 10 */
1594 swjdp
->tar_autoincr_block
= (1 << 10);
1596 cortex_a8
->fast_reg_read
= 0;
1598 /* register arch-specific functions */
1599 armv7a
->examine_debug_reason
= NULL
;
1601 armv7a
->post_debug_entry
= cortex_a8_post_debug_entry
;
1603 armv7a
->pre_restore_context
= NULL
;
1604 armv7a
->post_restore_context
= NULL
;
1605 armv7a
->armv4_5_mmu
.armv4_5_cache
.ctype
= -1;
1606 // armv7a->armv4_5_mmu.get_ttb = armv7a_get_ttb;
1607 armv7a
->armv4_5_mmu
.read_memory
= cortex_a8_read_memory
;
1608 armv7a
->armv4_5_mmu
.write_memory
= cortex_a8_write_memory
;
1609 // armv7a->armv4_5_mmu.disable_mmu_caches = armv7a_disable_mmu_caches;
1610 // armv7a->armv4_5_mmu.enable_mmu_caches = armv7a_enable_mmu_caches;
1611 armv7a
->armv4_5_mmu
.has_tiny_pages
= 1;
1612 armv7a
->armv4_5_mmu
.mmu_enabled
= 0;
1615 // arm7_9->handle_target_request = cortex_a8_handle_target_request;
1617 /* REVISIT v7a setup should be in a v7a-specific routine */
1618 armv4_5_init_arch_info(target
, armv4_5
);
1619 armv7a
->common_magic
= ARMV7_COMMON_MAGIC
;
1621 target_register_timer_callback(cortex_a8_handle_target_request
, 1, 1, target
);
1626 static int cortex_a8_target_create(struct target
*target
, Jim_Interp
*interp
)
1628 struct cortex_a8_common
*cortex_a8
= calloc(1, sizeof(struct cortex_a8_common
));
1630 cortex_a8_init_arch_info(target
, cortex_a8
, target
->tap
);
1635 COMMAND_HANDLER(cortex_a8_handle_cache_info_command
)
1637 struct target
*target
= get_current_target(CMD_CTX
);
1638 struct armv7a_common
*armv7a
= target_to_armv7a(target
);
1640 return armv4_5_handle_cache_info_command(CMD_CTX
,
1641 &armv7a
->armv4_5_mmu
.armv4_5_cache
);
1645 COMMAND_HANDLER(cortex_a8_handle_dbginit_command
)
1647 struct target
*target
= get_current_target(CMD_CTX
);
1649 cortex_a8_init_debug_access(target
);
1654 static const struct command_registration cortex_a8_exec_command_handlers
[] = {
1656 .name
= "cache_info",
1657 .handler
= &cortex_a8_handle_cache_info_command
,
1658 .mode
= COMMAND_EXEC
,
1659 .help
= "display information about target caches",
1663 .handler
= &cortex_a8_handle_dbginit_command
,
1664 .mode
= COMMAND_EXEC
,
1665 .help
= "Initialize core debug",
1667 COMMAND_REGISTRATION_DONE
1669 static const struct command_registration cortex_a8_command_handlers
[] = {
1671 .chain
= arm_command_handlers
,
1674 .chain
= armv7a_command_handlers
,
1677 .name
= "cortex_a8",
1678 .mode
= COMMAND_ANY
,
1679 .help
= "Cortex-A8 command group",
1680 .chain
= cortex_a8_exec_command_handlers
,
1682 COMMAND_REGISTRATION_DONE
1685 struct target_type cortexa8_target
= {
1686 .name
= "cortex_a8",
1688 .poll
= cortex_a8_poll
,
1689 .arch_state
= armv7a_arch_state
,
1691 .target_request_data
= NULL
,
1693 .halt
= cortex_a8_halt
,
1694 .resume
= cortex_a8_resume
,
1695 .step
= cortex_a8_step
,
1697 .assert_reset
= cortex_a8_assert_reset
,
1698 .deassert_reset
= cortex_a8_deassert_reset
,
1699 .soft_reset_halt
= NULL
,
1701 .get_gdb_reg_list
= armv4_5_get_gdb_reg_list
,
1703 .read_memory
= cortex_a8_read_memory
,
1704 .write_memory
= cortex_a8_write_memory
,
1705 .bulk_write_memory
= cortex_a8_bulk_write_memory
,
1707 .checksum_memory
= arm_checksum_memory
,
1708 .blank_check_memory
= arm_blank_check_memory
,
1710 .run_algorithm
= armv4_5_run_algorithm
,
1712 .add_breakpoint
= cortex_a8_add_breakpoint
,
1713 .remove_breakpoint
= cortex_a8_remove_breakpoint
,
1714 .add_watchpoint
= NULL
,
1715 .remove_watchpoint
= NULL
,
1717 .commands
= cortex_a8_command_handlers
,
1718 .target_create
= cortex_a8_target_create
,
1719 .init_target
= cortex_a8_init_target
,
1720 .examine
= cortex_a8_examine
,
Linking to existing account procedure
If you already have an account and want to add another login method
you
MUST first sign in with your existing account and
then change URL to read
https://review.openocd.org/login/?link
to get to this page again but this time it'll work for linking. Thank you.
SSH host keys fingerprints
1024 SHA256:YKx8b7u5ZWdcbp7/4AeXNaqElP49m6QrwfXaqQGJAOk gerrit-code-review@openocd.zylin.com (DSA)
384 SHA256:jHIbSQa4REvwCFG4cq5LBlBLxmxSqelQPem/EXIrxjk gerrit-code-review@openocd.org (ECDSA)
521 SHA256:UAOPYkU9Fjtcao0Ul/Rrlnj/OsQvt+pgdYSZ4jOYdgs gerrit-code-review@openocd.org (ECDSA)
256 SHA256:A13M5QlnozFOvTllybRZH6vm7iSt0XLxbA48yfc2yfY gerrit-code-review@openocd.org (ECDSA)
256 SHA256:spYMBqEYoAOtK7yZBrcwE8ZpYt6b68Cfh9yEVetvbXg gerrit-code-review@openocd.org (ED25519)
+--[ED25519 256]--+
|=.. |
|+o.. . |
|*.o . . |
|+B . . . |
|Bo. = o S |
|Oo.+ + = |
|oB=.* = . o |
| =+=.+ + E |
|. .=o . o |
+----[SHA256]-----+
2048 SHA256:0Onrb7/PHjpo6iVZ7xQX2riKN83FJ3KGU0TvI0TaFG4 gerrit-code-review@openocd.zylin.com (RSA)