1*67730e6cSSean Christopherson // SPDX-License-Identifier: GPL-2.0 2*67730e6cSSean Christopherson #include <test_util.h> 3*67730e6cSSean Christopherson #include <kvm_util.h> 4*67730e6cSSean Christopherson #include <processor.h> 5*67730e6cSSean Christopherson #include <linux/bitfield.h> 6*67730e6cSSean Christopherson 7*67730e6cSSean Christopherson #define MDSCR_KDE (1 << 13) 8*67730e6cSSean Christopherson #define MDSCR_MDE (1 << 15) 9*67730e6cSSean Christopherson #define MDSCR_SS (1 << 0) 10*67730e6cSSean Christopherson 11*67730e6cSSean Christopherson #define DBGBCR_LEN8 (0xff << 5) 12*67730e6cSSean Christopherson #define DBGBCR_EXEC (0x0 << 3) 13*67730e6cSSean Christopherson #define DBGBCR_EL1 (0x1 << 1) 14*67730e6cSSean Christopherson #define DBGBCR_E (0x1 << 0) 15*67730e6cSSean Christopherson #define DBGBCR_LBN_SHIFT 16 16*67730e6cSSean Christopherson #define DBGBCR_BT_SHIFT 20 17*67730e6cSSean Christopherson #define DBGBCR_BT_ADDR_LINK_CTX (0x1 << DBGBCR_BT_SHIFT) 18*67730e6cSSean Christopherson #define DBGBCR_BT_CTX_LINK (0x3 << DBGBCR_BT_SHIFT) 19*67730e6cSSean Christopherson 20*67730e6cSSean Christopherson #define DBGWCR_LEN8 (0xff << 5) 21*67730e6cSSean Christopherson #define DBGWCR_RD (0x1 << 3) 22*67730e6cSSean Christopherson #define DBGWCR_WR (0x2 << 3) 23*67730e6cSSean Christopherson #define DBGWCR_EL1 (0x1 << 1) 24*67730e6cSSean Christopherson #define DBGWCR_E (0x1 << 0) 25*67730e6cSSean Christopherson #define DBGWCR_LBN_SHIFT 16 26*67730e6cSSean Christopherson #define DBGWCR_WT_SHIFT 20 27*67730e6cSSean Christopherson #define DBGWCR_WT_LINK (0x1 << DBGWCR_WT_SHIFT) 28*67730e6cSSean Christopherson 29*67730e6cSSean Christopherson #define SPSR_D (1 << 9) 30*67730e6cSSean Christopherson #define SPSR_SS (1 << 21) 31*67730e6cSSean Christopherson 32*67730e6cSSean Christopherson extern unsigned char sw_bp, sw_bp2, hw_bp, hw_bp2, bp_svc, bp_brk, hw_wp, ss_start, hw_bp_ctx; 33*67730e6cSSean Christopherson extern unsigned char iter_ss_begin, iter_ss_end; 34*67730e6cSSean Christopherson static volatile uint64_t sw_bp_addr, hw_bp_addr; 35*67730e6cSSean Christopherson static volatile uint64_t wp_addr, wp_data_addr; 36*67730e6cSSean Christopherson static volatile uint64_t svc_addr; 37*67730e6cSSean Christopherson static volatile uint64_t ss_addr[4], ss_idx; 38*67730e6cSSean Christopherson #define PC(v) ((uint64_t)&(v)) 39*67730e6cSSean Christopherson 40*67730e6cSSean Christopherson #define GEN_DEBUG_WRITE_REG(reg_name) \ 41*67730e6cSSean Christopherson static void write_##reg_name(int num, uint64_t val) \ 42*67730e6cSSean Christopherson { \ 43*67730e6cSSean Christopherson switch (num) { \ 44*67730e6cSSean Christopherson case 0: \ 45*67730e6cSSean Christopherson write_sysreg(val, reg_name##0_el1); \ 46*67730e6cSSean Christopherson break; \ 47*67730e6cSSean Christopherson case 1: \ 48*67730e6cSSean Christopherson write_sysreg(val, reg_name##1_el1); \ 49*67730e6cSSean Christopherson break; \ 50*67730e6cSSean Christopherson case 2: \ 51*67730e6cSSean Christopherson write_sysreg(val, reg_name##2_el1); \ 52*67730e6cSSean Christopherson break; \ 53*67730e6cSSean Christopherson case 3: \ 54*67730e6cSSean Christopherson write_sysreg(val, reg_name##3_el1); \ 55*67730e6cSSean Christopherson break; \ 56*67730e6cSSean Christopherson case 4: \ 57*67730e6cSSean Christopherson write_sysreg(val, reg_name##4_el1); \ 58*67730e6cSSean Christopherson break; \ 59*67730e6cSSean Christopherson case 5: \ 60*67730e6cSSean Christopherson write_sysreg(val, reg_name##5_el1); \ 61*67730e6cSSean Christopherson break; \ 62*67730e6cSSean Christopherson case 6: \ 63*67730e6cSSean Christopherson write_sysreg(val, reg_name##6_el1); \ 64*67730e6cSSean Christopherson break; \ 65*67730e6cSSean Christopherson case 7: \ 66*67730e6cSSean Christopherson write_sysreg(val, reg_name##7_el1); \ 67*67730e6cSSean Christopherson break; \ 68*67730e6cSSean Christopherson case 8: \ 69*67730e6cSSean Christopherson write_sysreg(val, reg_name##8_el1); \ 70*67730e6cSSean Christopherson break; \ 71*67730e6cSSean Christopherson case 9: \ 72*67730e6cSSean Christopherson write_sysreg(val, reg_name##9_el1); \ 73*67730e6cSSean Christopherson break; \ 74*67730e6cSSean Christopherson case 10: \ 75*67730e6cSSean Christopherson write_sysreg(val, reg_name##10_el1); \ 76*67730e6cSSean Christopherson break; \ 77*67730e6cSSean Christopherson case 11: \ 78*67730e6cSSean Christopherson write_sysreg(val, reg_name##11_el1); \ 79*67730e6cSSean Christopherson break; \ 80*67730e6cSSean Christopherson case 12: \ 81*67730e6cSSean Christopherson write_sysreg(val, reg_name##12_el1); \ 82*67730e6cSSean Christopherson break; \ 83*67730e6cSSean Christopherson case 13: \ 84*67730e6cSSean Christopherson write_sysreg(val, reg_name##13_el1); \ 85*67730e6cSSean Christopherson break; \ 86*67730e6cSSean Christopherson case 14: \ 87*67730e6cSSean Christopherson write_sysreg(val, reg_name##14_el1); \ 88*67730e6cSSean Christopherson break; \ 89*67730e6cSSean Christopherson case 15: \ 90*67730e6cSSean Christopherson write_sysreg(val, reg_name##15_el1); \ 91*67730e6cSSean Christopherson break; \ 92*67730e6cSSean Christopherson default: \ 93*67730e6cSSean Christopherson GUEST_ASSERT(0); \ 94*67730e6cSSean Christopherson } \ 95*67730e6cSSean Christopherson } 96*67730e6cSSean Christopherson 97*67730e6cSSean Christopherson /* Define write_dbgbcr()/write_dbgbvr()/write_dbgwcr()/write_dbgwvr() */ 98*67730e6cSSean Christopherson GEN_DEBUG_WRITE_REG(dbgbcr) 99*67730e6cSSean Christopherson GEN_DEBUG_WRITE_REG(dbgbvr) 100*67730e6cSSean Christopherson GEN_DEBUG_WRITE_REG(dbgwcr) 101*67730e6cSSean Christopherson GEN_DEBUG_WRITE_REG(dbgwvr) 102*67730e6cSSean Christopherson 103*67730e6cSSean Christopherson static void reset_debug_state(void) 104*67730e6cSSean Christopherson { 105*67730e6cSSean Christopherson uint8_t brps, wrps, i; 106*67730e6cSSean Christopherson uint64_t dfr0; 107*67730e6cSSean Christopherson 108*67730e6cSSean Christopherson asm volatile("msr daifset, #8"); 109*67730e6cSSean Christopherson 110*67730e6cSSean Christopherson write_sysreg(0, osdlr_el1); 111*67730e6cSSean Christopherson write_sysreg(0, oslar_el1); 112*67730e6cSSean Christopherson isb(); 113*67730e6cSSean Christopherson 114*67730e6cSSean Christopherson write_sysreg(0, mdscr_el1); 115*67730e6cSSean Christopherson write_sysreg(0, contextidr_el1); 116*67730e6cSSean Christopherson 117*67730e6cSSean Christopherson /* Reset all bcr/bvr/wcr/wvr registers */ 118*67730e6cSSean Christopherson dfr0 = read_sysreg(id_aa64dfr0_el1); 119*67730e6cSSean Christopherson brps = FIELD_GET(ARM64_FEATURE_MASK(ID_AA64DFR0_EL1_BRPs), dfr0); 120*67730e6cSSean Christopherson for (i = 0; i <= brps; i++) { 121*67730e6cSSean Christopherson write_dbgbcr(i, 0); 122*67730e6cSSean Christopherson write_dbgbvr(i, 0); 123*67730e6cSSean Christopherson } 124*67730e6cSSean Christopherson wrps = FIELD_GET(ARM64_FEATURE_MASK(ID_AA64DFR0_EL1_WRPs), dfr0); 125*67730e6cSSean Christopherson for (i = 0; i <= wrps; i++) { 126*67730e6cSSean Christopherson write_dbgwcr(i, 0); 127*67730e6cSSean Christopherson write_dbgwvr(i, 0); 128*67730e6cSSean Christopherson } 129*67730e6cSSean Christopherson 130*67730e6cSSean Christopherson isb(); 131*67730e6cSSean Christopherson } 132*67730e6cSSean Christopherson 133*67730e6cSSean Christopherson static void enable_os_lock(void) 134*67730e6cSSean Christopherson { 135*67730e6cSSean Christopherson write_sysreg(1, oslar_el1); 136*67730e6cSSean Christopherson isb(); 137*67730e6cSSean Christopherson 138*67730e6cSSean Christopherson GUEST_ASSERT(read_sysreg(oslsr_el1) & 2); 139*67730e6cSSean Christopherson } 140*67730e6cSSean Christopherson 141*67730e6cSSean Christopherson static void enable_monitor_debug_exceptions(void) 142*67730e6cSSean Christopherson { 143*67730e6cSSean Christopherson uint32_t mdscr; 144*67730e6cSSean Christopherson 145*67730e6cSSean Christopherson asm volatile("msr daifclr, #8"); 146*67730e6cSSean Christopherson 147*67730e6cSSean Christopherson mdscr = read_sysreg(mdscr_el1) | MDSCR_KDE | MDSCR_MDE; 148*67730e6cSSean Christopherson write_sysreg(mdscr, mdscr_el1); 149*67730e6cSSean Christopherson isb(); 150*67730e6cSSean Christopherson } 151*67730e6cSSean Christopherson 152*67730e6cSSean Christopherson static void install_wp(uint8_t wpn, uint64_t addr) 153*67730e6cSSean Christopherson { 154*67730e6cSSean Christopherson uint32_t wcr; 155*67730e6cSSean Christopherson 156*67730e6cSSean Christopherson wcr = DBGWCR_LEN8 | DBGWCR_RD | DBGWCR_WR | DBGWCR_EL1 | DBGWCR_E; 157*67730e6cSSean Christopherson write_dbgwcr(wpn, wcr); 158*67730e6cSSean Christopherson write_dbgwvr(wpn, addr); 159*67730e6cSSean Christopherson 160*67730e6cSSean Christopherson isb(); 161*67730e6cSSean Christopherson 162*67730e6cSSean Christopherson enable_monitor_debug_exceptions(); 163*67730e6cSSean Christopherson } 164*67730e6cSSean Christopherson 165*67730e6cSSean Christopherson static void install_hw_bp(uint8_t bpn, uint64_t addr) 166*67730e6cSSean Christopherson { 167*67730e6cSSean Christopherson uint32_t bcr; 168*67730e6cSSean Christopherson 169*67730e6cSSean Christopherson bcr = DBGBCR_LEN8 | DBGBCR_EXEC | DBGBCR_EL1 | DBGBCR_E; 170*67730e6cSSean Christopherson write_dbgbcr(bpn, bcr); 171*67730e6cSSean Christopherson write_dbgbvr(bpn, addr); 172*67730e6cSSean Christopherson isb(); 173*67730e6cSSean Christopherson 174*67730e6cSSean Christopherson enable_monitor_debug_exceptions(); 175*67730e6cSSean Christopherson } 176*67730e6cSSean Christopherson 177*67730e6cSSean Christopherson static void install_wp_ctx(uint8_t addr_wp, uint8_t ctx_bp, uint64_t addr, 178*67730e6cSSean Christopherson uint64_t ctx) 179*67730e6cSSean Christopherson { 180*67730e6cSSean Christopherson uint32_t wcr; 181*67730e6cSSean Christopherson uint64_t ctx_bcr; 182*67730e6cSSean Christopherson 183*67730e6cSSean Christopherson /* Setup a context-aware breakpoint for Linked Context ID Match */ 184*67730e6cSSean Christopherson ctx_bcr = DBGBCR_LEN8 | DBGBCR_EXEC | DBGBCR_EL1 | DBGBCR_E | 185*67730e6cSSean Christopherson DBGBCR_BT_CTX_LINK; 186*67730e6cSSean Christopherson write_dbgbcr(ctx_bp, ctx_bcr); 187*67730e6cSSean Christopherson write_dbgbvr(ctx_bp, ctx); 188*67730e6cSSean Christopherson 189*67730e6cSSean Christopherson /* Setup a linked watchpoint (linked to the context-aware breakpoint) */ 190*67730e6cSSean Christopherson wcr = DBGWCR_LEN8 | DBGWCR_RD | DBGWCR_WR | DBGWCR_EL1 | DBGWCR_E | 191*67730e6cSSean Christopherson DBGWCR_WT_LINK | ((uint32_t)ctx_bp << DBGWCR_LBN_SHIFT); 192*67730e6cSSean Christopherson write_dbgwcr(addr_wp, wcr); 193*67730e6cSSean Christopherson write_dbgwvr(addr_wp, addr); 194*67730e6cSSean Christopherson isb(); 195*67730e6cSSean Christopherson 196*67730e6cSSean Christopherson enable_monitor_debug_exceptions(); 197*67730e6cSSean Christopherson } 198*67730e6cSSean Christopherson 199*67730e6cSSean Christopherson void install_hw_bp_ctx(uint8_t addr_bp, uint8_t ctx_bp, uint64_t addr, 200*67730e6cSSean Christopherson uint64_t ctx) 201*67730e6cSSean Christopherson { 202*67730e6cSSean Christopherson uint32_t addr_bcr, ctx_bcr; 203*67730e6cSSean Christopherson 204*67730e6cSSean Christopherson /* Setup a context-aware breakpoint for Linked Context ID Match */ 205*67730e6cSSean Christopherson ctx_bcr = DBGBCR_LEN8 | DBGBCR_EXEC | DBGBCR_EL1 | DBGBCR_E | 206*67730e6cSSean Christopherson DBGBCR_BT_CTX_LINK; 207*67730e6cSSean Christopherson write_dbgbcr(ctx_bp, ctx_bcr); 208*67730e6cSSean Christopherson write_dbgbvr(ctx_bp, ctx); 209*67730e6cSSean Christopherson 210*67730e6cSSean Christopherson /* 211*67730e6cSSean Christopherson * Setup a normal breakpoint for Linked Address Match, and link it 212*67730e6cSSean Christopherson * to the context-aware breakpoint. 213*67730e6cSSean Christopherson */ 214*67730e6cSSean Christopherson addr_bcr = DBGBCR_LEN8 | DBGBCR_EXEC | DBGBCR_EL1 | DBGBCR_E | 215*67730e6cSSean Christopherson DBGBCR_BT_ADDR_LINK_CTX | 216*67730e6cSSean Christopherson ((uint32_t)ctx_bp << DBGBCR_LBN_SHIFT); 217*67730e6cSSean Christopherson write_dbgbcr(addr_bp, addr_bcr); 218*67730e6cSSean Christopherson write_dbgbvr(addr_bp, addr); 219*67730e6cSSean Christopherson isb(); 220*67730e6cSSean Christopherson 221*67730e6cSSean Christopherson enable_monitor_debug_exceptions(); 222*67730e6cSSean Christopherson } 223*67730e6cSSean Christopherson 224*67730e6cSSean Christopherson static void install_ss(void) 225*67730e6cSSean Christopherson { 226*67730e6cSSean Christopherson uint32_t mdscr; 227*67730e6cSSean Christopherson 228*67730e6cSSean Christopherson asm volatile("msr daifclr, #8"); 229*67730e6cSSean Christopherson 230*67730e6cSSean Christopherson mdscr = read_sysreg(mdscr_el1) | MDSCR_KDE | MDSCR_SS; 231*67730e6cSSean Christopherson write_sysreg(mdscr, mdscr_el1); 232*67730e6cSSean Christopherson isb(); 233*67730e6cSSean Christopherson } 234*67730e6cSSean Christopherson 235*67730e6cSSean Christopherson static volatile char write_data; 236*67730e6cSSean Christopherson 237*67730e6cSSean Christopherson static void guest_code(uint8_t bpn, uint8_t wpn, uint8_t ctx_bpn) 238*67730e6cSSean Christopherson { 239*67730e6cSSean Christopherson uint64_t ctx = 0xabcdef; /* a random context number */ 240*67730e6cSSean Christopherson 241*67730e6cSSean Christopherson /* Software-breakpoint */ 242*67730e6cSSean Christopherson reset_debug_state(); 243*67730e6cSSean Christopherson asm volatile("sw_bp: brk #0"); 244*67730e6cSSean Christopherson GUEST_ASSERT_EQ(sw_bp_addr, PC(sw_bp)); 245*67730e6cSSean Christopherson 246*67730e6cSSean Christopherson /* Hardware-breakpoint */ 247*67730e6cSSean Christopherson reset_debug_state(); 248*67730e6cSSean Christopherson install_hw_bp(bpn, PC(hw_bp)); 249*67730e6cSSean Christopherson asm volatile("hw_bp: nop"); 250*67730e6cSSean Christopherson GUEST_ASSERT_EQ(hw_bp_addr, PC(hw_bp)); 251*67730e6cSSean Christopherson 252*67730e6cSSean Christopherson /* Hardware-breakpoint + svc */ 253*67730e6cSSean Christopherson reset_debug_state(); 254*67730e6cSSean Christopherson install_hw_bp(bpn, PC(bp_svc)); 255*67730e6cSSean Christopherson asm volatile("bp_svc: svc #0"); 256*67730e6cSSean Christopherson GUEST_ASSERT_EQ(hw_bp_addr, PC(bp_svc)); 257*67730e6cSSean Christopherson GUEST_ASSERT_EQ(svc_addr, PC(bp_svc) + 4); 258*67730e6cSSean Christopherson 259*67730e6cSSean Christopherson /* Hardware-breakpoint + software-breakpoint */ 260*67730e6cSSean Christopherson reset_debug_state(); 261*67730e6cSSean Christopherson install_hw_bp(bpn, PC(bp_brk)); 262*67730e6cSSean Christopherson asm volatile("bp_brk: brk #0"); 263*67730e6cSSean Christopherson GUEST_ASSERT_EQ(sw_bp_addr, PC(bp_brk)); 264*67730e6cSSean Christopherson GUEST_ASSERT_EQ(hw_bp_addr, PC(bp_brk)); 265*67730e6cSSean Christopherson 266*67730e6cSSean Christopherson /* Watchpoint */ 267*67730e6cSSean Christopherson reset_debug_state(); 268*67730e6cSSean Christopherson install_wp(wpn, PC(write_data)); 269*67730e6cSSean Christopherson write_data = 'x'; 270*67730e6cSSean Christopherson GUEST_ASSERT_EQ(write_data, 'x'); 271*67730e6cSSean Christopherson GUEST_ASSERT_EQ(wp_data_addr, PC(write_data)); 272*67730e6cSSean Christopherson 273*67730e6cSSean Christopherson /* Single-step */ 274*67730e6cSSean Christopherson reset_debug_state(); 275*67730e6cSSean Christopherson install_ss(); 276*67730e6cSSean Christopherson ss_idx = 0; 277*67730e6cSSean Christopherson asm volatile("ss_start:\n" 278*67730e6cSSean Christopherson "mrs x0, esr_el1\n" 279*67730e6cSSean Christopherson "add x0, x0, #1\n" 280*67730e6cSSean Christopherson "msr daifset, #8\n" 281*67730e6cSSean Christopherson : : : "x0"); 282*67730e6cSSean Christopherson GUEST_ASSERT_EQ(ss_addr[0], PC(ss_start)); 283*67730e6cSSean Christopherson GUEST_ASSERT_EQ(ss_addr[1], PC(ss_start) + 4); 284*67730e6cSSean Christopherson GUEST_ASSERT_EQ(ss_addr[2], PC(ss_start) + 8); 285*67730e6cSSean Christopherson 286*67730e6cSSean Christopherson /* OS Lock does not block software-breakpoint */ 287*67730e6cSSean Christopherson reset_debug_state(); 288*67730e6cSSean Christopherson enable_os_lock(); 289*67730e6cSSean Christopherson sw_bp_addr = 0; 290*67730e6cSSean Christopherson asm volatile("sw_bp2: brk #0"); 291*67730e6cSSean Christopherson GUEST_ASSERT_EQ(sw_bp_addr, PC(sw_bp2)); 292*67730e6cSSean Christopherson 293*67730e6cSSean Christopherson /* OS Lock blocking hardware-breakpoint */ 294*67730e6cSSean Christopherson reset_debug_state(); 295*67730e6cSSean Christopherson enable_os_lock(); 296*67730e6cSSean Christopherson install_hw_bp(bpn, PC(hw_bp2)); 297*67730e6cSSean Christopherson hw_bp_addr = 0; 298*67730e6cSSean Christopherson asm volatile("hw_bp2: nop"); 299*67730e6cSSean Christopherson GUEST_ASSERT_EQ(hw_bp_addr, 0); 300*67730e6cSSean Christopherson 301*67730e6cSSean Christopherson /* OS Lock blocking watchpoint */ 302*67730e6cSSean Christopherson reset_debug_state(); 303*67730e6cSSean Christopherson enable_os_lock(); 304*67730e6cSSean Christopherson write_data = '\0'; 305*67730e6cSSean Christopherson wp_data_addr = 0; 306*67730e6cSSean Christopherson install_wp(wpn, PC(write_data)); 307*67730e6cSSean Christopherson write_data = 'x'; 308*67730e6cSSean Christopherson GUEST_ASSERT_EQ(write_data, 'x'); 309*67730e6cSSean Christopherson GUEST_ASSERT_EQ(wp_data_addr, 0); 310*67730e6cSSean Christopherson 311*67730e6cSSean Christopherson /* OS Lock blocking single-step */ 312*67730e6cSSean Christopherson reset_debug_state(); 313*67730e6cSSean Christopherson enable_os_lock(); 314*67730e6cSSean Christopherson ss_addr[0] = 0; 315*67730e6cSSean Christopherson install_ss(); 316*67730e6cSSean Christopherson ss_idx = 0; 317*67730e6cSSean Christopherson asm volatile("mrs x0, esr_el1\n\t" 318*67730e6cSSean Christopherson "add x0, x0, #1\n\t" 319*67730e6cSSean Christopherson "msr daifset, #8\n\t" 320*67730e6cSSean Christopherson : : : "x0"); 321*67730e6cSSean Christopherson GUEST_ASSERT_EQ(ss_addr[0], 0); 322*67730e6cSSean Christopherson 323*67730e6cSSean Christopherson /* Linked hardware-breakpoint */ 324*67730e6cSSean Christopherson hw_bp_addr = 0; 325*67730e6cSSean Christopherson reset_debug_state(); 326*67730e6cSSean Christopherson install_hw_bp_ctx(bpn, ctx_bpn, PC(hw_bp_ctx), ctx); 327*67730e6cSSean Christopherson /* Set context id */ 328*67730e6cSSean Christopherson write_sysreg(ctx, contextidr_el1); 329*67730e6cSSean Christopherson isb(); 330*67730e6cSSean Christopherson asm volatile("hw_bp_ctx: nop"); 331*67730e6cSSean Christopherson write_sysreg(0, contextidr_el1); 332*67730e6cSSean Christopherson GUEST_ASSERT_EQ(hw_bp_addr, PC(hw_bp_ctx)); 333*67730e6cSSean Christopherson 334*67730e6cSSean Christopherson /* Linked watchpoint */ 335*67730e6cSSean Christopherson reset_debug_state(); 336*67730e6cSSean Christopherson install_wp_ctx(wpn, ctx_bpn, PC(write_data), ctx); 337*67730e6cSSean Christopherson /* Set context id */ 338*67730e6cSSean Christopherson write_sysreg(ctx, contextidr_el1); 339*67730e6cSSean Christopherson isb(); 340*67730e6cSSean Christopherson write_data = 'x'; 341*67730e6cSSean Christopherson GUEST_ASSERT_EQ(write_data, 'x'); 342*67730e6cSSean Christopherson GUEST_ASSERT_EQ(wp_data_addr, PC(write_data)); 343*67730e6cSSean Christopherson 344*67730e6cSSean Christopherson GUEST_DONE(); 345*67730e6cSSean Christopherson } 346*67730e6cSSean Christopherson 347*67730e6cSSean Christopherson static void guest_sw_bp_handler(struct ex_regs *regs) 348*67730e6cSSean Christopherson { 349*67730e6cSSean Christopherson sw_bp_addr = regs->pc; 350*67730e6cSSean Christopherson regs->pc += 4; 351*67730e6cSSean Christopherson } 352*67730e6cSSean Christopherson 353*67730e6cSSean Christopherson static void guest_hw_bp_handler(struct ex_regs *regs) 354*67730e6cSSean Christopherson { 355*67730e6cSSean Christopherson hw_bp_addr = regs->pc; 356*67730e6cSSean Christopherson regs->pstate |= SPSR_D; 357*67730e6cSSean Christopherson } 358*67730e6cSSean Christopherson 359*67730e6cSSean Christopherson static void guest_wp_handler(struct ex_regs *regs) 360*67730e6cSSean Christopherson { 361*67730e6cSSean Christopherson wp_data_addr = read_sysreg(far_el1); 362*67730e6cSSean Christopherson wp_addr = regs->pc; 363*67730e6cSSean Christopherson regs->pstate |= SPSR_D; 364*67730e6cSSean Christopherson } 365*67730e6cSSean Christopherson 366*67730e6cSSean Christopherson static void guest_ss_handler(struct ex_regs *regs) 367*67730e6cSSean Christopherson { 368*67730e6cSSean Christopherson __GUEST_ASSERT(ss_idx < 4, "Expected index < 4, got '%lu'", ss_idx); 369*67730e6cSSean Christopherson ss_addr[ss_idx++] = regs->pc; 370*67730e6cSSean Christopherson regs->pstate |= SPSR_SS; 371*67730e6cSSean Christopherson } 372*67730e6cSSean Christopherson 373*67730e6cSSean Christopherson static void guest_svc_handler(struct ex_regs *regs) 374*67730e6cSSean Christopherson { 375*67730e6cSSean Christopherson svc_addr = regs->pc; 376*67730e6cSSean Christopherson } 377*67730e6cSSean Christopherson 378*67730e6cSSean Christopherson static void guest_code_ss(int test_cnt) 379*67730e6cSSean Christopherson { 380*67730e6cSSean Christopherson uint64_t i; 381*67730e6cSSean Christopherson uint64_t bvr, wvr, w_bvr, w_wvr; 382*67730e6cSSean Christopherson 383*67730e6cSSean Christopherson for (i = 0; i < test_cnt; i++) { 384*67730e6cSSean Christopherson /* Bits [1:0] of dbg{b,w}vr are RES0 */ 385*67730e6cSSean Christopherson w_bvr = i << 2; 386*67730e6cSSean Christopherson w_wvr = i << 2; 387*67730e6cSSean Christopherson 388*67730e6cSSean Christopherson /* 389*67730e6cSSean Christopherson * Enable Single Step execution. Note! This _must_ be a bare 390*67730e6cSSean Christopherson * ucall as the ucall() path uses atomic operations to manage 391*67730e6cSSean Christopherson * the ucall structures, and the built-in "atomics" are usually 392*67730e6cSSean Christopherson * implemented via exclusive access instructions. The exlusive 393*67730e6cSSean Christopherson * monitor is cleared on ERET, and so taking debug exceptions 394*67730e6cSSean Christopherson * during a LDREX=>STREX sequence will prevent forward progress 395*67730e6cSSean Christopherson * and hang the guest/test. 396*67730e6cSSean Christopherson */ 397*67730e6cSSean Christopherson GUEST_UCALL_NONE(); 398*67730e6cSSean Christopherson 399*67730e6cSSean Christopherson /* 400*67730e6cSSean Christopherson * The userspace will verify that the pc is as expected during 401*67730e6cSSean Christopherson * single step execution between iter_ss_begin and iter_ss_end. 402*67730e6cSSean Christopherson */ 403*67730e6cSSean Christopherson asm volatile("iter_ss_begin:nop\n"); 404*67730e6cSSean Christopherson 405*67730e6cSSean Christopherson write_sysreg(w_bvr, dbgbvr0_el1); 406*67730e6cSSean Christopherson write_sysreg(w_wvr, dbgwvr0_el1); 407*67730e6cSSean Christopherson bvr = read_sysreg(dbgbvr0_el1); 408*67730e6cSSean Christopherson wvr = read_sysreg(dbgwvr0_el1); 409*67730e6cSSean Christopherson 410*67730e6cSSean Christopherson /* Userspace disables Single Step when the end is nigh. */ 411*67730e6cSSean Christopherson asm volatile("iter_ss_end:\n"); 412*67730e6cSSean Christopherson 413*67730e6cSSean Christopherson GUEST_ASSERT_EQ(bvr, w_bvr); 414*67730e6cSSean Christopherson GUEST_ASSERT_EQ(wvr, w_wvr); 415*67730e6cSSean Christopherson } 416*67730e6cSSean Christopherson GUEST_DONE(); 417*67730e6cSSean Christopherson } 418*67730e6cSSean Christopherson 419*67730e6cSSean Christopherson static int debug_version(uint64_t id_aa64dfr0) 420*67730e6cSSean Christopherson { 421*67730e6cSSean Christopherson return FIELD_GET(ARM64_FEATURE_MASK(ID_AA64DFR0_EL1_DebugVer), id_aa64dfr0); 422*67730e6cSSean Christopherson } 423*67730e6cSSean Christopherson 424*67730e6cSSean Christopherson static void test_guest_debug_exceptions(uint8_t bpn, uint8_t wpn, uint8_t ctx_bpn) 425*67730e6cSSean Christopherson { 426*67730e6cSSean Christopherson struct kvm_vcpu *vcpu; 427*67730e6cSSean Christopherson struct kvm_vm *vm; 428*67730e6cSSean Christopherson struct ucall uc; 429*67730e6cSSean Christopherson 430*67730e6cSSean Christopherson vm = vm_create_with_one_vcpu(&vcpu, guest_code); 431*67730e6cSSean Christopherson 432*67730e6cSSean Christopherson vm_init_descriptor_tables(vm); 433*67730e6cSSean Christopherson vcpu_init_descriptor_tables(vcpu); 434*67730e6cSSean Christopherson 435*67730e6cSSean Christopherson vm_install_sync_handler(vm, VECTOR_SYNC_CURRENT, 436*67730e6cSSean Christopherson ESR_ELx_EC_BRK64, guest_sw_bp_handler); 437*67730e6cSSean Christopherson vm_install_sync_handler(vm, VECTOR_SYNC_CURRENT, 438*67730e6cSSean Christopherson ESR_ELx_EC_BREAKPT_CUR, guest_hw_bp_handler); 439*67730e6cSSean Christopherson vm_install_sync_handler(vm, VECTOR_SYNC_CURRENT, 440*67730e6cSSean Christopherson ESR_ELx_EC_WATCHPT_CUR, guest_wp_handler); 441*67730e6cSSean Christopherson vm_install_sync_handler(vm, VECTOR_SYNC_CURRENT, 442*67730e6cSSean Christopherson ESR_ELx_EC_SOFTSTP_CUR, guest_ss_handler); 443*67730e6cSSean Christopherson vm_install_sync_handler(vm, VECTOR_SYNC_CURRENT, 444*67730e6cSSean Christopherson ESR_ELx_EC_SVC64, guest_svc_handler); 445*67730e6cSSean Christopherson 446*67730e6cSSean Christopherson /* Specify bpn/wpn/ctx_bpn to be tested */ 447*67730e6cSSean Christopherson vcpu_args_set(vcpu, 3, bpn, wpn, ctx_bpn); 448*67730e6cSSean Christopherson pr_debug("Use bpn#%d, wpn#%d and ctx_bpn#%d\n", bpn, wpn, ctx_bpn); 449*67730e6cSSean Christopherson 450*67730e6cSSean Christopherson vcpu_run(vcpu); 451*67730e6cSSean Christopherson switch (get_ucall(vcpu, &uc)) { 452*67730e6cSSean Christopherson case UCALL_ABORT: 453*67730e6cSSean Christopherson REPORT_GUEST_ASSERT(uc); 454*67730e6cSSean Christopherson break; 455*67730e6cSSean Christopherson case UCALL_DONE: 456*67730e6cSSean Christopherson goto done; 457*67730e6cSSean Christopherson default: 458*67730e6cSSean Christopherson TEST_FAIL("Unknown ucall %lu", uc.cmd); 459*67730e6cSSean Christopherson } 460*67730e6cSSean Christopherson 461*67730e6cSSean Christopherson done: 462*67730e6cSSean Christopherson kvm_vm_free(vm); 463*67730e6cSSean Christopherson } 464*67730e6cSSean Christopherson 465*67730e6cSSean Christopherson void test_single_step_from_userspace(int test_cnt) 466*67730e6cSSean Christopherson { 467*67730e6cSSean Christopherson struct kvm_vcpu *vcpu; 468*67730e6cSSean Christopherson struct kvm_vm *vm; 469*67730e6cSSean Christopherson struct ucall uc; 470*67730e6cSSean Christopherson struct kvm_run *run; 471*67730e6cSSean Christopherson uint64_t pc, cmd; 472*67730e6cSSean Christopherson uint64_t test_pc = 0; 473*67730e6cSSean Christopherson bool ss_enable = false; 474*67730e6cSSean Christopherson struct kvm_guest_debug debug = {}; 475*67730e6cSSean Christopherson 476*67730e6cSSean Christopherson vm = vm_create_with_one_vcpu(&vcpu, guest_code_ss); 477*67730e6cSSean Christopherson run = vcpu->run; 478*67730e6cSSean Christopherson vcpu_args_set(vcpu, 1, test_cnt); 479*67730e6cSSean Christopherson 480*67730e6cSSean Christopherson while (1) { 481*67730e6cSSean Christopherson vcpu_run(vcpu); 482*67730e6cSSean Christopherson if (run->exit_reason != KVM_EXIT_DEBUG) { 483*67730e6cSSean Christopherson cmd = get_ucall(vcpu, &uc); 484*67730e6cSSean Christopherson if (cmd == UCALL_ABORT) { 485*67730e6cSSean Christopherson REPORT_GUEST_ASSERT(uc); 486*67730e6cSSean Christopherson /* NOT REACHED */ 487*67730e6cSSean Christopherson } else if (cmd == UCALL_DONE) { 488*67730e6cSSean Christopherson break; 489*67730e6cSSean Christopherson } 490*67730e6cSSean Christopherson 491*67730e6cSSean Christopherson TEST_ASSERT(cmd == UCALL_NONE, 492*67730e6cSSean Christopherson "Unexpected ucall cmd 0x%lx", cmd); 493*67730e6cSSean Christopherson 494*67730e6cSSean Christopherson debug.control = KVM_GUESTDBG_ENABLE | 495*67730e6cSSean Christopherson KVM_GUESTDBG_SINGLESTEP; 496*67730e6cSSean Christopherson ss_enable = true; 497*67730e6cSSean Christopherson vcpu_guest_debug_set(vcpu, &debug); 498*67730e6cSSean Christopherson continue; 499*67730e6cSSean Christopherson } 500*67730e6cSSean Christopherson 501*67730e6cSSean Christopherson TEST_ASSERT(ss_enable, "Unexpected KVM_EXIT_DEBUG"); 502*67730e6cSSean Christopherson 503*67730e6cSSean Christopherson /* Check if the current pc is expected. */ 504*67730e6cSSean Christopherson pc = vcpu_get_reg(vcpu, ARM64_CORE_REG(regs.pc)); 505*67730e6cSSean Christopherson TEST_ASSERT(!test_pc || pc == test_pc, 506*67730e6cSSean Christopherson "Unexpected pc 0x%lx (expected 0x%lx)", 507*67730e6cSSean Christopherson pc, test_pc); 508*67730e6cSSean Christopherson 509*67730e6cSSean Christopherson if ((pc + 4) == (uint64_t)&iter_ss_end) { 510*67730e6cSSean Christopherson test_pc = 0; 511*67730e6cSSean Christopherson debug.control = KVM_GUESTDBG_ENABLE; 512*67730e6cSSean Christopherson ss_enable = false; 513*67730e6cSSean Christopherson vcpu_guest_debug_set(vcpu, &debug); 514*67730e6cSSean Christopherson continue; 515*67730e6cSSean Christopherson } 516*67730e6cSSean Christopherson 517*67730e6cSSean Christopherson /* 518*67730e6cSSean Christopherson * If the current pc is between iter_ss_bgin and 519*67730e6cSSean Christopherson * iter_ss_end, the pc for the next KVM_EXIT_DEBUG should 520*67730e6cSSean Christopherson * be the current pc + 4. 521*67730e6cSSean Christopherson */ 522*67730e6cSSean Christopherson if ((pc >= (uint64_t)&iter_ss_begin) && 523*67730e6cSSean Christopherson (pc < (uint64_t)&iter_ss_end)) 524*67730e6cSSean Christopherson test_pc = pc + 4; 525*67730e6cSSean Christopherson else 526*67730e6cSSean Christopherson test_pc = 0; 527*67730e6cSSean Christopherson } 528*67730e6cSSean Christopherson 529*67730e6cSSean Christopherson kvm_vm_free(vm); 530*67730e6cSSean Christopherson } 531*67730e6cSSean Christopherson 532*67730e6cSSean Christopherson /* 533*67730e6cSSean Christopherson * Run debug testing using the various breakpoint#, watchpoint# and 534*67730e6cSSean Christopherson * context-aware breakpoint# with the given ID_AA64DFR0_EL1 configuration. 535*67730e6cSSean Christopherson */ 536*67730e6cSSean Christopherson void test_guest_debug_exceptions_all(uint64_t aa64dfr0) 537*67730e6cSSean Christopherson { 538*67730e6cSSean Christopherson uint8_t brp_num, wrp_num, ctx_brp_num, normal_brp_num, ctx_brp_base; 539*67730e6cSSean Christopherson int b, w, c; 540*67730e6cSSean Christopherson 541*67730e6cSSean Christopherson /* Number of breakpoints */ 542*67730e6cSSean Christopherson brp_num = FIELD_GET(ARM64_FEATURE_MASK(ID_AA64DFR0_EL1_BRPs), aa64dfr0) + 1; 543*67730e6cSSean Christopherson __TEST_REQUIRE(brp_num >= 2, "At least two breakpoints are required"); 544*67730e6cSSean Christopherson 545*67730e6cSSean Christopherson /* Number of watchpoints */ 546*67730e6cSSean Christopherson wrp_num = FIELD_GET(ARM64_FEATURE_MASK(ID_AA64DFR0_EL1_WRPs), aa64dfr0) + 1; 547*67730e6cSSean Christopherson 548*67730e6cSSean Christopherson /* Number of context aware breakpoints */ 549*67730e6cSSean Christopherson ctx_brp_num = FIELD_GET(ARM64_FEATURE_MASK(ID_AA64DFR0_EL1_CTX_CMPs), aa64dfr0) + 1; 550*67730e6cSSean Christopherson 551*67730e6cSSean Christopherson pr_debug("%s brp_num:%d, wrp_num:%d, ctx_brp_num:%d\n", __func__, 552*67730e6cSSean Christopherson brp_num, wrp_num, ctx_brp_num); 553*67730e6cSSean Christopherson 554*67730e6cSSean Christopherson /* Number of normal (non-context aware) breakpoints */ 555*67730e6cSSean Christopherson normal_brp_num = brp_num - ctx_brp_num; 556*67730e6cSSean Christopherson 557*67730e6cSSean Christopherson /* Lowest context aware breakpoint number */ 558*67730e6cSSean Christopherson ctx_brp_base = normal_brp_num; 559*67730e6cSSean Christopherson 560*67730e6cSSean Christopherson /* Run tests with all supported breakpoints/watchpoints */ 561*67730e6cSSean Christopherson for (c = ctx_brp_base; c < ctx_brp_base + ctx_brp_num; c++) { 562*67730e6cSSean Christopherson for (b = 0; b < normal_brp_num; b++) { 563*67730e6cSSean Christopherson for (w = 0; w < wrp_num; w++) 564*67730e6cSSean Christopherson test_guest_debug_exceptions(b, w, c); 565*67730e6cSSean Christopherson } 566*67730e6cSSean Christopherson } 567*67730e6cSSean Christopherson } 568*67730e6cSSean Christopherson 569*67730e6cSSean Christopherson static void help(char *name) 570*67730e6cSSean Christopherson { 571*67730e6cSSean Christopherson puts(""); 572*67730e6cSSean Christopherson printf("Usage: %s [-h] [-i iterations of the single step test]\n", name); 573*67730e6cSSean Christopherson puts(""); 574*67730e6cSSean Christopherson exit(0); 575*67730e6cSSean Christopherson } 576*67730e6cSSean Christopherson 577*67730e6cSSean Christopherson int main(int argc, char *argv[]) 578*67730e6cSSean Christopherson { 579*67730e6cSSean Christopherson struct kvm_vcpu *vcpu; 580*67730e6cSSean Christopherson struct kvm_vm *vm; 581*67730e6cSSean Christopherson int opt; 582*67730e6cSSean Christopherson int ss_iteration = 10000; 583*67730e6cSSean Christopherson uint64_t aa64dfr0; 584*67730e6cSSean Christopherson 585*67730e6cSSean Christopherson vm = vm_create_with_one_vcpu(&vcpu, guest_code); 586*67730e6cSSean Christopherson aa64dfr0 = vcpu_get_reg(vcpu, KVM_ARM64_SYS_REG(SYS_ID_AA64DFR0_EL1)); 587*67730e6cSSean Christopherson __TEST_REQUIRE(debug_version(aa64dfr0) >= 6, 588*67730e6cSSean Christopherson "Armv8 debug architecture not supported."); 589*67730e6cSSean Christopherson kvm_vm_free(vm); 590*67730e6cSSean Christopherson 591*67730e6cSSean Christopherson while ((opt = getopt(argc, argv, "i:")) != -1) { 592*67730e6cSSean Christopherson switch (opt) { 593*67730e6cSSean Christopherson case 'i': 594*67730e6cSSean Christopherson ss_iteration = atoi_positive("Number of iterations", optarg); 595*67730e6cSSean Christopherson break; 596*67730e6cSSean Christopherson case 'h': 597*67730e6cSSean Christopherson default: 598*67730e6cSSean Christopherson help(argv[0]); 599*67730e6cSSean Christopherson break; 600*67730e6cSSean Christopherson } 601*67730e6cSSean Christopherson } 602*67730e6cSSean Christopherson 603*67730e6cSSean Christopherson test_guest_debug_exceptions_all(aa64dfr0); 604*67730e6cSSean Christopherson test_single_step_from_userspace(ss_iteration); 605*67730e6cSSean Christopherson 606*67730e6cSSean Christopherson return 0; 607*67730e6cSSean Christopherson } 608