xref: /rk3399_ARM-atf/bl31/aarch64/ea_delegate.S (revision b62673c645752a78f649282cfa293e8da09e3bef)
1/*
2 * Copyright (c) 2018-2022, ARM Limited and Contributors. All rights reserved.
3 * Copyright (c) 2022, NVIDIA Corporation. All rights reserved.
4 *
5 * SPDX-License-Identifier: BSD-3-Clause
6 */
7
8
9#include <assert_macros.S>
10#include <asm_macros.S>
11#include <assert_macros.S>
12#include <bl31/ea_handle.h>
13#include <context.h>
14#include <lib/extensions/ras_arch.h>
15#include <cpu_macros.S>
16#include <context.h>
17
18	.globl	handle_lower_el_sync_ea
19	.globl	handle_lower_el_async_ea
20	.globl	handle_pending_async_ea
21/*
22 * This function handles Synchronous External Aborts from lower EL.
23 *
24 * It delegates the handling of the EA to platform handler, and upon successfully
25 * handling the EA, exits EL3; otherwise panics.
26 *
27 * This function assumes x30 has been saved.
28 */
29func handle_lower_el_sync_ea
30	mrs	x30, esr_el3
31	ubfx	x30, x30, #ESR_EC_SHIFT, #ESR_EC_LENGTH
32
33	/* Check for I/D aborts from lower EL */
34	cmp	x30, #EC_IABORT_LOWER_EL
35	b.eq	1f
36
37	cmp	x30, #EC_DABORT_LOWER_EL
38	b.eq	1f
39
40	/* EA other than above are unhandled exceptions */
41	no_ret	report_unhandled_exception
421:
43	/*
44	 * Save general purpose and ARMv8.3-PAuth registers (if enabled).
45	 * Also save PMCR_EL0 and set the PSTATE to a known state.
46	 */
47	bl	prepare_el3_entry
48
49#if ENABLE_PAUTH
50	/* Load and program APIAKey firmware key */
51	bl	pauth_load_bl31_apiakey
52#endif
53
54	/* Setup exception class and syndrome arguments for platform handler */
55	mov	x0, #ERROR_EA_SYNC
56	mrs	x1, esr_el3
57	bl	delegate_sync_ea
58
59	/* el3_exit assumes SP_EL0 on entry */
60	msr	spsel, #MODE_SP_EL0
61	b	el3_exit
62endfunc handle_lower_el_sync_ea
63
64
65/*
66 * This function handles SErrors from lower ELs.
67 *
68 * It delegates the handling of the EA to platform handler, and upon successfully
69 * handling the EA, exits EL3; otherwise panics.
70 *
71 * This function assumes x30 has been saved.
72 */
73func handle_lower_el_async_ea
74
75	/*
76	 * Save general purpose and ARMv8.3-PAuth registers (if enabled).
77	 * Also save PMCR_EL0 and set the PSTATE to a known state.
78	 */
79	bl	prepare_el3_entry
80
81#if ENABLE_PAUTH
82	/* Load and program APIAKey firmware key */
83	bl	pauth_load_bl31_apiakey
84#endif
85
86	/* Setup exception class and syndrome arguments for platform handler */
87	mov	x0, #ERROR_EA_ASYNC
88	mrs	x1, esr_el3
89	bl	delegate_async_ea
90
91	/* el3_exit assumes SP_EL0 on entry */
92	msr	spsel, #MODE_SP_EL0
93	b	el3_exit
94endfunc handle_lower_el_async_ea
95
96/*
97 * Handler for async EA from lower EL synchronized at EL3 entry in FFH mode.
98 *
99 * This scenario may arise when there is an error (EA) in the system which is not
100 * yet signaled to PE while executing in lower EL. During entry into EL3, the errors
101 * are synchronized either implicitly or explicitly causing async EA to pend at EL3.
102 *
103 * On detecting the pending EA (via ISR_EL1.A), if the EA routing model is Firmware
104 * First handling (FFH, SCR_EL3.EA = 1) this handler first handles the pending EA
105 * and then handles the original exception.
106 *
107 * This function assumes x30 has been saved.
108 */
109func handle_pending_async_ea
110	/*
111	 * Prepare for nested handling of EA. Stash sysregs clobbered by nested
112	 * exception and handler
113	 */
114	str	x30, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_GPREG_LR]
115	mrs	x30, esr_el3
116	str	x30, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_ESR_EL3]
117	mrs	x30, spsr_el3
118	str	x30, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_SPSR_EL3]
119	mrs	x30, elr_el3
120	str	x30, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_ELR_EL3]
121
122	mov	x30, #1
123	str	x30, [sp, #CTX_EL3STATE_OFFSET + CTX_NESTED_EA_FLAG]
124	/*
125	 * Restore the original x30 saved as part of entering EL3. This is not
126	 * required for the current function but for EL3 SError vector entry
127	 * once PSTATE.A bit is unmasked. We restore x30 and then the same
128	 * value is stored in EL3 SError vector entry.
129	 */
130	ldr	x30, [sp, #CTX_GPREGS_OFFSET + CTX_GPREG_LR]
131
132	/*
133	 * After clearing PSTATE.A bit pending SError will trigger at current EL.
134	 * Put explicit synchronization event to ensure newly unmasked interrupt
135	 * is taken immediately.
136	 */
137	unmask_async_ea
138
139	/* Restore the original exception information along with zeroing the storage */
140	ldr	x30, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_ELR_EL3]
141	msr	elr_el3, x30
142	str	xzr, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_ELR_EL3]
143	ldr	x30, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_SPSR_EL3]
144	msr	spsr_el3, x30
145	str	xzr, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_SPSR_EL3]
146	ldr	x30, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_ESR_EL3]
147	msr	esr_el3, x30
148	str	xzr, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_ESR_EL3]
149
150	/*
151	 * If the original exception corresponds to SError from lower El, eret back
152	 * to lower EL, otherwise return to vector table for original exception handling.
153	 */
154	ubfx	x30, x30, #ESR_EC_SHIFT, #ESR_EC_LENGTH
155	cmp	x30, #EC_SERROR
156	ldr	x30, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_GPREG_LR]
157	str	xzr, [sp, #CTX_EL3STATE_OFFSET + CTX_SAVED_GPREG_LR]
158	b.eq	1f
159	ret
1601:
161	ldr	x30, [sp, #CTX_GPREGS_OFFSET + CTX_GPREG_LR]
162	str	xzr, [sp, #CTX_GPREGS_OFFSET + CTX_GPREG_LR]
163	exception_return
164endfunc handle_pending_async_ea
165
166/*
167 * Prelude for Synchronous External Abort handling. This function assumes that
168 * all GP registers have been saved by the caller.
169 *
170 * x0: EA reason
171 * x1: EA syndrome
172 */
173func delegate_sync_ea
174#if ENABLE_FEAT_RAS
175	/*
176	 * Check for Uncontainable error type. If so, route to the platform
177	 * fatal error handler rather than the generic EA one.
178	 */
179	ubfx    x2, x1, #EABORT_SET_SHIFT, #EABORT_SET_WIDTH
180	cmp     x2, #ERROR_STATUS_SET_UC
181	b.ne    1f
182
183	/* Check fault status code */
184	ubfx    x3, x1, #EABORT_DFSC_SHIFT, #EABORT_DFSC_WIDTH
185	cmp     x3, #SYNC_EA_FSC
186	b.ne    1f
187
188	no_ret  plat_handle_uncontainable_ea
1891:
190#endif
191
192	b       ea_proceed
193endfunc delegate_sync_ea
194
195
196/*
197 * Prelude for Asynchronous External Abort handling. This function assumes that
198 * all GP registers have been saved by the caller.
199 *
200 * x0: EA reason
201 * x1: EA syndrome
202 */
203func delegate_async_ea
204#if ENABLE_FEAT_RAS
205	/* Check Exception Class to ensure SError, as this function should
206	 * only be invoked for SError. If that is not the case, which implies
207	 * either an HW error or programming error, panic.
208	 */
209	ubfx	x2, x1, #ESR_EC_SHIFT, #ESR_EC_LENGTH
210	cmp	x2, EC_SERROR
211	b.ne	el3_panic
212	/*
213	 * Check for Implementation Defined Syndrome. If so, skip checking
214	 * Uncontainable error type from the syndrome as the format is unknown.
215	 */
216	tbnz	x1, #SERROR_IDS_BIT, 1f
217
218	/* AET only valid when DFSC is 0x11 */
219	ubfx	x2, x1, #EABORT_DFSC_SHIFT, #EABORT_DFSC_WIDTH
220	cmp	x2, #DFSC_SERROR
221	b.ne	1f
222
223	/*
224	 * Check for Uncontainable error type. If so, route to the platform
225	 * fatal error handler rather than the generic EA one.
226	 */
227	ubfx	x3, x1, #EABORT_AET_SHIFT, #EABORT_AET_WIDTH
228	cmp	x3, #ERROR_STATUS_UET_UC
229	b.ne	1f
230
231	no_ret	plat_handle_uncontainable_ea
2321:
233#endif
234
235	b	ea_proceed
236endfunc delegate_async_ea
237
238
239/*
240 * Delegate External Abort handling to platform's EA handler. This function
241 * assumes that all GP registers have been saved by the caller.
242 *
243 * x0: EA reason
244 * x1: EA syndrome
245 */
246func ea_proceed
247	/*
248	 * If the ESR loaded earlier is not zero, we were processing an EA
249	 * already, and this is a double fault.
250	 */
251	ldr	x5, [sp, #CTX_EL3STATE_OFFSET + CTX_ESR_EL3]
252	cbz	x5, 1f
253	no_ret	plat_handle_double_fault
254
2551:
256	/* Save EL3 state */
257	mrs	x2, spsr_el3
258	mrs	x3, elr_el3
259	stp	x2, x3, [sp, #CTX_EL3STATE_OFFSET + CTX_SPSR_EL3]
260
261	/*
262	 * Save ESR as handling might involve lower ELs, and returning back to
263	 * EL3 from there would trample the original ESR.
264	 */
265	mrs	x4, scr_el3
266	mrs	x5, esr_el3
267	stp	x4, x5, [sp, #CTX_EL3STATE_OFFSET + CTX_SCR_EL3]
268
269	/*
270	 * Setup rest of arguments, and call platform External Abort handler.
271	 *
272	 * x0: EA reason (already in place)
273	 * x1: Exception syndrome (already in place).
274	 * x2: Cookie (unused for now).
275	 * x3: Context pointer.
276	 * x4: Flags (security state from SCR for now).
277	 */
278	mov	x2, xzr
279	mov	x3, sp
280	ubfx	x4, x4, #0, #1
281
282	/* Switch to runtime stack */
283	ldr	x5, [sp, #CTX_EL3STATE_OFFSET + CTX_RUNTIME_SP]
284	msr	spsel, #MODE_SP_EL0
285	mov	sp, x5
286
287	mov	x29, x30
288#if ENABLE_ASSERTIONS
289	/* Stash the stack pointer */
290	mov	x28, sp
291#endif
292	bl	plat_ea_handler
293
294#if ENABLE_ASSERTIONS
295	/*
296	 * Error handling flows might involve long jumps; so upon returning from
297	 * the platform error handler, validate that the we've completely
298	 * unwound the stack.
299	 */
300	mov	x27, sp
301	cmp	x28, x27
302	ASM_ASSERT(eq)
303#endif
304
305	/* Make SP point to context */
306	msr	spsel, #MODE_SP_ELX
307
308	/* Restore EL3 state and ESR */
309	ldp	x1, x2, [sp, #CTX_EL3STATE_OFFSET + CTX_SPSR_EL3]
310	msr	spsr_el3, x1
311	msr	elr_el3, x2
312
313	/* Restore ESR_EL3 and SCR_EL3 */
314	ldp	x3, x4, [sp, #CTX_EL3STATE_OFFSET + CTX_SCR_EL3]
315	msr	scr_el3, x3
316	msr	esr_el3, x4
317
318#if ENABLE_ASSERTIONS
319	cmp	x4, xzr
320	ASM_ASSERT(ne)
321#endif
322
323	/* Clear ESR storage */
324	str	xzr, [sp, #CTX_EL3STATE_OFFSET + CTX_ESR_EL3]
325
326	ret	x29
327endfunc ea_proceed
328