xref: /linux/tools/testing/selftests/kvm/arm64/debug-exceptions.c (revision 9e676a024fa1fa2bd8150c2d2ba85478280353bc)
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