target/aarch64: fix watchpoint management

The early documentation for armv8a report the debug register WFAR
as containing the address of the instruction that triggered the
watchpoint. More recent documentation report the register EDWAR as
containing the data memory address that triggered the watchpoint.

The name of macros CPUV8_DBG_WFAR0 and CPUV8_DBG_WFAR1 is not
correct as they point to the debug register EDWAR, so reading such
register returns directly the data memory address that triggered
the watchpoint. The code incorrectly passes this address value to
the function armv8_dpm_report_wfar(); this function is supposed to
adjust the PC value, decrementing it to remove the effects of the
CPU pipeline. This pipeline offset, that has no meaning on the
value in EDWAR, caused commit 651b861d5d ("target/aarch64: Add
watchpoint support") to add back the offset while comparing the
address with the watchpoint enabled.

The upper 32 bits of EDWAR are not valid in aarch32 mode and have
to be ignored.

Rename CPUV8_DBG_WFAR0/1 as CPUV8_DBG_EDWAR0/1.
Remove the function armv8_dpm_report_wfar().
Remove the offset while searching the matching watchpoint.
Ignore the upper 32 bits of EDWAR in aarch32 mode.
Fix a comment and the LOG text.

Change-Id: I7cbdbeb766fa18e31cc72be098ca2bc501877ed1
Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com>
Reviewed-on: http://openocd.zylin.com/6205
Tested-by: jenkins
Reviewed-by: Liming Sun <limings@nvidia.com>
This commit is contained in:
Antonio Borneo 2021-05-05 15:05:21 +02:00
parent 510df38407
commit 15dd48119a
5 changed files with 22 additions and 56 deletions

View File

@ -988,25 +988,26 @@ static int aarch64_debug_entry(struct target *target)
/* Examine debug reason */ /* Examine debug reason */
armv8_dpm_report_dscr(dpm, dscr); armv8_dpm_report_dscr(dpm, dscr);
/* save address of instruction that triggered the watchpoint? */ /* save the memory address that triggered the watchpoint */
if (target->debug_reason == DBG_REASON_WATCHPOINT) { if (target->debug_reason == DBG_REASON_WATCHPOINT) {
uint32_t tmp; uint32_t tmp;
uint64_t wfar = 0;
retval = mem_ap_read_atomic_u32(armv8->debug_ap, retval = mem_ap_read_atomic_u32(armv8->debug_ap,
armv8->debug_base + CPUV8_DBG_WFAR1, armv8->debug_base + CPUV8_DBG_EDWAR0, &tmp);
&tmp);
if (retval != ERROR_OK) if (retval != ERROR_OK)
return retval; return retval;
wfar = tmp; target_addr_t edwar = tmp;
wfar = (wfar << 32);
retval = mem_ap_read_atomic_u32(armv8->debug_ap, /* EDWAR[63:32] has unknown content in aarch32 state */
armv8->debug_base + CPUV8_DBG_WFAR0, if (core_state == ARM_STATE_AARCH64) {
&tmp); retval = mem_ap_read_atomic_u32(armv8->debug_ap,
if (retval != ERROR_OK) armv8->debug_base + CPUV8_DBG_EDWAR1, &tmp);
return retval; if (retval != ERROR_OK)
wfar |= tmp; return retval;
armv8_dpm_report_wfar(&armv8->dpm, wfar); edwar |= ((target_addr_t)tmp) << 32;
}
armv8->dpm.wp_addr = edwar;
} }
retval = armv8_dpm_read_current_registers(&armv8->dpm); retval = armv8_dpm_read_current_registers(&armv8->dpm);
@ -1853,7 +1854,7 @@ int aarch64_hit_watchpoint(struct target *target,
struct armv8_common *armv8 = target_to_armv8(target); struct armv8_common *armv8 = target_to_armv8(target);
uint64_t exception_address; target_addr_t exception_address;
struct watchpoint *wp; struct watchpoint *wp;
exception_address = armv8->dpm.wp_addr; exception_address = armv8->dpm.wp_addr;
@ -1861,23 +1862,11 @@ int aarch64_hit_watchpoint(struct target *target,
if (exception_address == 0xFFFFFFFF) if (exception_address == 0xFFFFFFFF)
return ERROR_FAIL; return ERROR_FAIL;
/**********************************************************/ for (wp = target->watchpoints; wp; wp = wp->next)
/* see if a watchpoint address matches a value read from */ if (exception_address >= wp->address && exception_address < (wp->address + wp->length)) {
/* the EDWAR register. Testing shows that on some ARM CPUs*/ *hit_watchpoint = wp;
/* the EDWAR value needs to have 8 added to it so we add */ return ERROR_OK;
/* that check as well not sure if that is a core bug) */
/**********************************************************/
for (exception_address = armv8->dpm.wp_addr; exception_address <= (armv8->dpm.wp_addr + 8);
exception_address += 8) {
for (wp = target->watchpoints; wp; wp = wp->next) {
if ((exception_address >= wp->address) && (exception_address < (wp->address + wp->length))) {
*hit_watchpoint = wp;
if (exception_address != armv8->dpm.wp_addr)
LOG_DEBUG("watchpoint hit required EDWAR to be increased by 8");
return ERROR_OK;
}
} }
}
return ERROR_FAIL; return ERROR_FAIL;
} }

View File

@ -1169,7 +1169,7 @@ int armv8_arch_state(struct target *target)
armv8_show_fault_registers(target); armv8_show_fault_registers(target);
if (target->debug_reason == DBG_REASON_WATCHPOINT) if (target->debug_reason == DBG_REASON_WATCHPOINT)
LOG_USER("Watchpoint triggered at PC " TARGET_ADDR_FMT, armv8->dpm.wp_addr); LOG_USER("Watchpoint triggered at " TARGET_ADDR_FMT, armv8->dpm.wp_addr);
return ERROR_OK; return ERROR_OK;
} }

View File

@ -257,8 +257,8 @@ static inline bool is_armv8(struct armv8_common *armv8)
#define CPUV8_DBG_EDESR 0x20 #define CPUV8_DBG_EDESR 0x20
#define CPUV8_DBG_EDECR 0x24 #define CPUV8_DBG_EDECR 0x24
#define CPUV8_DBG_WFAR0 0x30 #define CPUV8_DBG_EDWAR0 0x30
#define CPUV8_DBG_WFAR1 0x34 #define CPUV8_DBG_EDWAR1 0x34
#define CPUV8_DBG_DSCR 0x088 #define CPUV8_DBG_DSCR 0x088
#define CPUV8_DBG_DRCR 0x090 #define CPUV8_DBG_DRCR 0x090
#define CPUV8_DBG_ECCR 0x098 #define CPUV8_DBG_ECCR 0x098

View File

@ -1279,27 +1279,6 @@ static int dpmv8_remove_watchpoint(struct target *target, struct watchpoint *wp)
return retval; return retval;
} }
void armv8_dpm_report_wfar(struct arm_dpm *dpm, uint64_t addr)
{
switch (dpm->arm->core_state) {
case ARM_STATE_ARM:
case ARM_STATE_AARCH64:
addr -= 8;
break;
case ARM_STATE_THUMB:
case ARM_STATE_THUMB_EE:
addr -= 4;
break;
case ARM_STATE_JAZELLE:
/* ?? */
break;
default:
LOG_DEBUG("Unknown core_state");
break;
}
dpm->wp_addr = addr;
}
/* /*
* Handle exceptions taken in debug state. This happens mostly for memory * Handle exceptions taken in debug state. This happens mostly for memory
* accesses that violated a MMU policy. Taking an exception while in debug * accesses that violated a MMU policy. Taking an exception while in debug

View File

@ -38,8 +38,6 @@ int armv8_dpm_modeswitch(struct arm_dpm *dpm, enum arm_mode mode);
int armv8_dpm_write_dirty_registers(struct arm_dpm *dpm, bool bpwp); int armv8_dpm_write_dirty_registers(struct arm_dpm *dpm, bool bpwp);
void armv8_dpm_report_wfar(struct arm_dpm *dpm, uint64_t wfar);
/* DSCR bits; see ARMv7a arch spec section C10.3.1. /* DSCR bits; see ARMv7a arch spec section C10.3.1.
* Not all v7 bits are valid in v6. * Not all v7 bits are valid in v6.
*/ */