mirror of
https://github.com/raspberrypi/linux.git
synced 2025-12-06 01:49:46 +00:00
powerpc/watchpoint: Fix quadword instruction handling on p10 predecessors
[ Upstream commit4759c11ed2] On p10 predecessors, watchpoint with quadword access is compared at quadword length. If the watch range is doubleword or less than that in a first half of quadword aligned 16 bytes, and if there is any unaligned quadword access which will access only the 2nd half, the handler should consider it as extraneous and emulate/single-step it before continuing. Fixes:74c6881019("powerpc/watchpoint: Prepare handler to handle more than one watchpoint") Reported-by: Pedro Miraglia Franco de Carvalho <pedromfc@linux.ibm.com> Signed-off-by: Ravi Bangoria <ravi.bangoria@linux.ibm.com> Signed-off-by: Michael Ellerman <mpe@ellerman.id.au> Link: https://lore.kernel.org/r/20200902042945.129369-2-ravi.bangoria@linux.ibm.com Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
committed by
Greg Kroah-Hartman
parent
810d23775d
commit
69a94e5324
@@ -42,6 +42,7 @@ struct arch_hw_breakpoint {
|
||||
#else
|
||||
#define HW_BREAKPOINT_SIZE 0x8
|
||||
#endif
|
||||
#define HW_BREAKPOINT_SIZE_QUADWORD 0x10
|
||||
|
||||
#define DABR_MAX_LEN 8
|
||||
#define DAWR_MAX_LEN 512
|
||||
|
||||
@@ -520,9 +520,17 @@ static bool ea_hw_range_overlaps(unsigned long ea, int size,
|
||||
struct arch_hw_breakpoint *info)
|
||||
{
|
||||
unsigned long hw_start_addr, hw_end_addr;
|
||||
unsigned long align_size = HW_BREAKPOINT_SIZE;
|
||||
|
||||
hw_start_addr = ALIGN_DOWN(info->address, HW_BREAKPOINT_SIZE);
|
||||
hw_end_addr = ALIGN(info->address + info->len, HW_BREAKPOINT_SIZE);
|
||||
/*
|
||||
* On p10 predecessors, quadword is handle differently then
|
||||
* other instructions.
|
||||
*/
|
||||
if (!cpu_has_feature(CPU_FTR_ARCH_31) && size == 16)
|
||||
align_size = HW_BREAKPOINT_SIZE_QUADWORD;
|
||||
|
||||
hw_start_addr = ALIGN_DOWN(info->address, align_size);
|
||||
hw_end_addr = ALIGN(info->address + info->len, align_size);
|
||||
|
||||
return ((ea < hw_end_addr) && (ea + size > hw_start_addr));
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user