aarch64: Add instruction stepping support using hardware step
Use AARCH64's hardware step event to do stepping. Change-Id: I2d029ceeadd381913d0c3355c8787b11dacff7f7 Signed-off-by: pierre Kuo <vichy.kuo@gmail.com> Signed-off-by: David Ung <david.ung.42@gmail.com> Signed-off-by: Matthias Welwarsky <matthias.welwarsky@sysgo.com>
This commit is contained in:
parent
5ee67ce024
commit
13d13b2e2a
|
@ -1150,45 +1150,23 @@ static int aarch64_step(struct target *target, int current, target_addr_t addres
|
||||||
int handle_breakpoints)
|
int handle_breakpoints)
|
||||||
{
|
{
|
||||||
struct armv8_common *armv8 = target_to_armv8(target);
|
struct armv8_common *armv8 = target_to_armv8(target);
|
||||||
struct arm *arm = &armv8->arm;
|
|
||||||
struct breakpoint *breakpoint = NULL;
|
|
||||||
struct breakpoint stepbreakpoint;
|
|
||||||
struct reg *r;
|
|
||||||
int retval;
|
int retval;
|
||||||
|
uint32_t tmp;
|
||||||
|
|
||||||
if (target->state != TARGET_HALTED) {
|
if (target->state != TARGET_HALTED) {
|
||||||
LOG_WARNING("target not halted");
|
LOG_WARNING("target not halted");
|
||||||
return ERROR_TARGET_NOT_HALTED;
|
return ERROR_TARGET_NOT_HALTED;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* current = 1: continue on current pc, otherwise continue at <address> */
|
retval = mem_ap_read_atomic_u32(armv8->debug_ap,
|
||||||
r = arm->pc;
|
armv8->debug_base + CPUDBG_DECR, &tmp);
|
||||||
if (!current)
|
if (retval != ERROR_OK)
|
||||||
buf_set_u64(r->value, 0, 64, address);
|
return retval;
|
||||||
else
|
|
||||||
address = buf_get_u64(r->value, 0, 64);
|
|
||||||
|
|
||||||
/* The front-end may request us not to handle breakpoints.
|
retval = mem_ap_write_atomic_u32(armv8->debug_ap,
|
||||||
* But since Cortex-A8 uses breakpoint for single step,
|
armv8->debug_base + CPUDBG_DECR, (tmp|0x4));
|
||||||
* we MUST handle breakpoints.
|
if (retval != ERROR_OK)
|
||||||
*/
|
return retval;
|
||||||
handle_breakpoints = 1;
|
|
||||||
if (handle_breakpoints) {
|
|
||||||
breakpoint = breakpoint_find(target, address);
|
|
||||||
if (breakpoint)
|
|
||||||
aarch64_unset_breakpoint(target, breakpoint);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Setup single step breakpoint */
|
|
||||||
stepbreakpoint.address = address;
|
|
||||||
stepbreakpoint.length = 4;
|
|
||||||
stepbreakpoint.type = BKPT_HARD;
|
|
||||||
stepbreakpoint.set = 0;
|
|
||||||
|
|
||||||
/* Break on IVA mismatch */
|
|
||||||
aarch64_set_breakpoint(target, &stepbreakpoint, 0x04);
|
|
||||||
|
|
||||||
target->debug_reason = DBG_REASON_SINGLESTEP;
|
|
||||||
|
|
||||||
retval = aarch64_resume(target, 1, address, 0, 0);
|
retval = aarch64_resume(target, 1, address, 0, 0);
|
||||||
if (retval != ERROR_OK)
|
if (retval != ERROR_OK)
|
||||||
|
@ -1205,12 +1183,11 @@ static int aarch64_step(struct target *target, int current, target_addr_t addres
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
aarch64_unset_breakpoint(target, &stepbreakpoint);
|
|
||||||
|
|
||||||
target->debug_reason = DBG_REASON_BREAKPOINT;
|
target->debug_reason = DBG_REASON_BREAKPOINT;
|
||||||
|
retval = mem_ap_write_atomic_u32(armv8->debug_ap,
|
||||||
if (breakpoint)
|
armv8->debug_base + CPUDBG_DECR, (tmp&(~0x4)));
|
||||||
aarch64_set_breakpoint(target, breakpoint, 0);
|
if (retval != ERROR_OK)
|
||||||
|
return retval;
|
||||||
|
|
||||||
if (target->state != TARGET_HALTED)
|
if (target->state != TARGET_HALTED)
|
||||||
LOG_DEBUG("target stepped");
|
LOG_DEBUG("target stepped");
|
||||||
|
|
|
@ -161,6 +161,7 @@ target_to_armv8(struct target *target)
|
||||||
/* register offsets from armv8.debug_base */
|
/* register offsets from armv8.debug_base */
|
||||||
|
|
||||||
#define CPUDBG_WFAR 0x018
|
#define CPUDBG_WFAR 0x018
|
||||||
|
#define CPUDBG_DECR 0x024
|
||||||
/* PCSR at 0x084 -or- 0x0a0 -or- both ... based on flags in DIDR */
|
/* PCSR at 0x084 -or- 0x0a0 -or- both ... based on flags in DIDR */
|
||||||
#define CPUDBG_DSCR 0x088
|
#define CPUDBG_DSCR 0x088
|
||||||
#define CPUDBG_DRCR 0x090
|
#define CPUDBG_DRCR 0x090
|
||||||
|
|
Loading…
Reference in New Issue