context.h 10.8 KB
Newer Older
1
/*
Soby Mathew's avatar
Soby Mathew committed
2
 * Copyright (c) 2013-2016, ARM Limited and Contributors. All rights reserved.
3
 *
dp-arm's avatar
dp-arm committed
4
 * SPDX-License-Identifier: BSD-3-Clause
5
6
7
8
9
 */

#ifndef __CONTEXT_H__
#define __CONTEXT_H__

10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
/*******************************************************************************
 * Constants that allow assembler code to access members of and the 'gp_regs'
 * structure at their correct offsets.
 ******************************************************************************/
#define CTX_GPREGS_OFFSET	0x0
#define CTX_GPREG_X0		0x0
#define CTX_GPREG_X1		0x8
#define CTX_GPREG_X2		0x10
#define CTX_GPREG_X3		0x18
#define CTX_GPREG_X4		0x20
#define CTX_GPREG_X5		0x28
#define CTX_GPREG_X6		0x30
#define CTX_GPREG_X7		0x38
#define CTX_GPREG_X8		0x40
#define CTX_GPREG_X9		0x48
#define CTX_GPREG_X10		0x50
#define CTX_GPREG_X11		0x58
#define CTX_GPREG_X12		0x60
#define CTX_GPREG_X13		0x68
#define CTX_GPREG_X14		0x70
#define CTX_GPREG_X15		0x78
#define CTX_GPREG_X16		0x80
#define CTX_GPREG_X17		0x88
#define CTX_GPREG_X18		0x90
34
35
36
37
38
39
40
41
42
43
44
45
46
47
#define CTX_GPREG_X19		0x98
#define CTX_GPREG_X20		0xa0
#define CTX_GPREG_X21		0xa8
#define CTX_GPREG_X22		0xb0
#define CTX_GPREG_X23		0xb8
#define CTX_GPREG_X24		0xc0
#define CTX_GPREG_X25		0xc8
#define CTX_GPREG_X26		0xd0
#define CTX_GPREG_X27		0xd8
#define CTX_GPREG_X28		0xe0
#define CTX_GPREG_X29		0xe8
#define CTX_GPREG_LR		0xf0
#define CTX_GPREG_SP_EL0	0xf8
#define CTX_GPREGS_END		0x100
48

49
50
51
52
53
/*******************************************************************************
 * Constants that allow assembler code to access members of and the 'el3_state'
 * structure at their correct offsets. Note that some of the registers are only
 * 32-bits wide but are stored as 64-bit values for convenience
 ******************************************************************************/
54
#define CTX_EL3STATE_OFFSET	(CTX_GPREGS_OFFSET + CTX_GPREGS_END)
55
#define CTX_SCR_EL3		0x0
56
#define CTX_RUNTIME_SP		0x8
57
58
#define CTX_SPSR_EL3		0x10
#define CTX_ELR_EL3		0x18
59
#define CTX_EL3STATE_END	0x20
60
61
62
63
64
65
66
67
68
69

/*******************************************************************************
 * Constants that allow assembler code to access members of and the
 * 'el1_sys_regs' structure at their correct offsets. Note that some of the
 * registers are only 32-bits wide but are stored as 64-bit values for
 * convenience
 ******************************************************************************/
#define CTX_SYSREGS_OFFSET	(CTX_EL3STATE_OFFSET + CTX_EL3STATE_END)
#define CTX_SPSR_EL1		0x0
#define CTX_ELR_EL1		0x8
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
#define CTX_SCTLR_EL1		0x10
#define CTX_ACTLR_EL1		0x18
#define CTX_CPACR_EL1		0x20
#define CTX_CSSELR_EL1		0x28
#define CTX_SP_EL1		0x30
#define CTX_ESR_EL1		0x38
#define CTX_TTBR0_EL1		0x40
#define CTX_TTBR1_EL1		0x48
#define CTX_MAIR_EL1		0x50
#define CTX_AMAIR_EL1		0x58
#define CTX_TCR_EL1		0x60
#define CTX_TPIDR_EL1		0x68
#define CTX_TPIDR_EL0		0x70
#define CTX_TPIDRRO_EL0		0x78
#define CTX_PAR_EL1		0x80
#define CTX_FAR_EL1		0x88
#define CTX_AFSR0_EL1		0x90
#define CTX_AFSR1_EL1		0x98
#define CTX_CONTEXTIDR_EL1	0xa0
#define CTX_VBAR_EL1		0xa8

/*
 * If the platform is AArch64-only, there is no need to save and restore these
 * AArch32 registers.
 */
#if CTX_INCLUDE_AARCH32_REGS
#define CTX_SPSR_ABT		0xb0
#define CTX_SPSR_UND		0xb8
#define CTX_SPSR_IRQ		0xc0
#define CTX_SPSR_FIQ		0xc8
#define CTX_DACR32_EL2		0xd0
#define CTX_IFSR32_EL2		0xd8
#define CTX_FP_FPEXC32_EL2	0xe0
#define CTX_TIMER_SYSREGS_OFF		0xf0 /* Align to the next 16 byte boundary */
#else
#define CTX_TIMER_SYSREGS_OFF		0xb0
#endif /* __CTX_INCLUDE_AARCH32_REGS__ */

108
109
110
111
112
/*
 * If the timer registers aren't saved and restored, we don't have to reserve
 * space for them in the context
 */
#if NS_TIMER_SWITCH
113
114
115
116
117
118
#define CTX_CNTP_CTL_EL0	(CTX_TIMER_SYSREGS_OFF + 0x0)
#define CTX_CNTP_CVAL_EL0	(CTX_TIMER_SYSREGS_OFF + 0x8)
#define CTX_CNTV_CTL_EL0	(CTX_TIMER_SYSREGS_OFF + 0x10)
#define CTX_CNTV_CVAL_EL0	(CTX_TIMER_SYSREGS_OFF + 0x18)
#define CTX_CNTKCTL_EL1		(CTX_TIMER_SYSREGS_OFF + 0x20)
#define CTX_SYSREGS_END		(CTX_TIMER_SYSREGS_OFF + 0x30) /* Align to the next 16 byte boundary */
119
#else
120
121
#define CTX_SYSREGS_END		CTX_TIMER_SYSREGS_OFF
#endif /* __NS_TIMER_SWITCH__ */
122
123
124
125
126

/*******************************************************************************
 * Constants that allow assembler code to access members of and the 'fp_regs'
 * structure at their correct offsets.
 ******************************************************************************/
127
#if CTX_INCLUDE_FPREGS
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
#define CTX_FPREGS_OFFSET	(CTX_SYSREGS_OFFSET + CTX_SYSREGS_END)
#define CTX_FP_Q0		0x0
#define CTX_FP_Q1		0x10
#define CTX_FP_Q2		0x20
#define CTX_FP_Q3		0x30
#define CTX_FP_Q4		0x40
#define CTX_FP_Q5		0x50
#define CTX_FP_Q6		0x60
#define CTX_FP_Q7		0x70
#define CTX_FP_Q8		0x80
#define CTX_FP_Q9		0x90
#define CTX_FP_Q10		0xa0
#define CTX_FP_Q11		0xb0
#define CTX_FP_Q12		0xc0
#define CTX_FP_Q13		0xd0
#define CTX_FP_Q14		0xe0
#define CTX_FP_Q15		0xf0
#define CTX_FP_Q16		0x100
#define CTX_FP_Q17		0x110
#define CTX_FP_Q18		0x120
#define CTX_FP_Q19		0x130
#define CTX_FP_Q20		0x140
#define CTX_FP_Q21		0x150
#define CTX_FP_Q22		0x160
#define CTX_FP_Q23		0x170
#define CTX_FP_Q24		0x180
#define CTX_FP_Q25		0x190
#define CTX_FP_Q26		0x1a0
#define CTX_FP_Q27		0x1b0
#define CTX_FP_Q28		0x1c0
#define CTX_FP_Q29		0x1d0
#define CTX_FP_Q30		0x1e0
#define CTX_FP_Q31		0x1f0
#define CTX_FP_FPSR		0x200
#define CTX_FP_FPCR		0x208
#define CTX_FPREGS_END		0x210
164
#endif
165
166
167

#ifndef __ASSEMBLY__

168
#include <cassert.h>
169
#include <platform_def.h>	/* for CACHE_WRITEBACK_GRANULE */
170
171
#include <stdint.h>

172
173
174
175
176
177
/*
 * Common constants to help define the 'cpu_context' structure and its
 * members below.
 */
#define DWORD_SHIFT		3
#define DEFINE_REG_STRUCT(name, num_regs)	\
178
	typedef struct name {			\
179
		uint64_t _regs[num_regs];	\
180
	}  __aligned(16) name##_t
181
182

/* Constants to determine the size of individual context structures */
183
#define CTX_GPREG_ALL		(CTX_GPREGS_END >> DWORD_SHIFT)
184
#define CTX_SYSREG_ALL		(CTX_SYSREGS_END >> DWORD_SHIFT)
185
#if CTX_INCLUDE_FPREGS
186
#define CTX_FPREG_ALL		(CTX_FPREGS_END >> DWORD_SHIFT)
187
#endif
188
189
#define CTX_EL3STATE_ALL	(CTX_EL3STATE_END >> DWORD_SHIFT)

190
/*
191
192
 * AArch64 general purpose register context structure. Usually x0-x18,
 * lr are saved as the compiler is expected to preserve the remaining
193
 * callee saved registers if used by the C runtime and the assembler
194
195
 * does not touch the remaining. But in case of world switch during
 * exception handling, we need to save the callee registers too.
196
 */
197
DEFINE_REG_STRUCT(gp_regs, CTX_GPREG_ALL);
198

199
200
201
202
203
204
205
206
207
208
209
210
/*
 * AArch64 EL1 system register context structure for preserving the
 * architectural state during switches from one security state to
 * another in EL1.
 */
DEFINE_REG_STRUCT(el1_sys_regs, CTX_SYSREG_ALL);

/*
 * AArch64 floating point register context structure for preserving
 * the floating point state during switches from one security state to
 * another.
 */
211
#if CTX_INCLUDE_FPREGS
212
DEFINE_REG_STRUCT(fp_regs, CTX_FPREG_ALL);
213
#endif
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237

/*
 * Miscellaneous registers used by EL3 firmware to maintain its state
 * across exception entries and exits
 */
DEFINE_REG_STRUCT(el3_state, CTX_EL3STATE_ALL);

/*
 * Macros to access members of any of the above structures using their
 * offsets
 */
#define read_ctx_reg(ctx, offset)	((ctx)->_regs[offset >> DWORD_SHIFT])
#define write_ctx_reg(ctx, offset, val)	(((ctx)->_regs[offset >> DWORD_SHIFT]) \
					 = val)

/*
 * Top-level context structure which is used by EL3 firmware to
 * preserve the state of a core at EL1 in one of the two security
 * states and save enough EL3 meta data to be able to return to that
 * EL and security state. The context management library will be used
 * to ensure that SP_EL3 always points to an instance of this
 * structure at exception entry and exit. Each instance will
 * correspond to either the secure or the non-secure state.
 */
238
239
240
241
typedef struct cpu_context {
	gp_regs_t gpregs_ctx;
	el3_state_t el3state_ctx;
	el1_sys_regs_t sysregs_ctx;
242
#if CTX_INCLUDE_FPREGS
243
	fp_regs_t fpregs_ctx;
244
#endif
245
} cpu_context_t;
246

247
248
/* Macros to access members of the 'cpu_context_t' structure */
#define get_el3state_ctx(h)	(&((cpu_context_t *) h)->el3state_ctx)
249
#if CTX_INCLUDE_FPREGS
250
#define get_fpregs_ctx(h)	(&((cpu_context_t *) h)->fpregs_ctx)
251
#endif
252
253
#define get_sysregs_ctx(h)	(&((cpu_context_t *) h)->sysregs_ctx)
#define get_gpregs_ctx(h)	(&((cpu_context_t *) h)->gpregs_ctx)
254
255
256
257
258
259

/*
 * Compile time assertions related to the 'cpu_context' structure to
 * ensure that the assembler and the compiler view of the offsets of
 * the structure members is the same.
 */
260
CASSERT(CTX_GPREGS_OFFSET == __builtin_offsetof(cpu_context_t, gpregs_ctx), \
261
	assert_core_context_gp_offset_mismatch);
262
CASSERT(CTX_SYSREGS_OFFSET == __builtin_offsetof(cpu_context_t, sysregs_ctx), \
263
	assert_core_context_sys_offset_mismatch);
264
#if CTX_INCLUDE_FPREGS
265
CASSERT(CTX_FPREGS_OFFSET == __builtin_offsetof(cpu_context_t, fpregs_ctx), \
266
	assert_core_context_fp_offset_mismatch);
267
#endif
268
CASSERT(CTX_EL3STATE_OFFSET == __builtin_offsetof(cpu_context_t, el3state_ctx), \
269
270
	assert_core_context_el3state_offset_mismatch);

271
272
273
274
275
276
/*
 * Helper macro to set the general purpose registers that correspond to
 * parameters in an aapcs_64 call i.e. x0-x7
 */
#define set_aapcs_args0(ctx, x0)				do {	\
		write_ctx_reg(get_gpregs_ctx(ctx), CTX_GPREG_X0, x0);	\
Soby Mathew's avatar
Soby Mathew committed
277
	} while (0)
278
279
280
#define set_aapcs_args1(ctx, x0, x1)				do {	\
		write_ctx_reg(get_gpregs_ctx(ctx), CTX_GPREG_X1, x1);	\
		set_aapcs_args0(ctx, x0);				\
Soby Mathew's avatar
Soby Mathew committed
281
	} while (0)
282
283
284
#define set_aapcs_args2(ctx, x0, x1, x2)			do {	\
		write_ctx_reg(get_gpregs_ctx(ctx), CTX_GPREG_X2, x2);	\
		set_aapcs_args1(ctx, x0, x1);				\
Soby Mathew's avatar
Soby Mathew committed
285
	} while (0)
286
287
288
#define set_aapcs_args3(ctx, x0, x1, x2, x3)			do {	\
		write_ctx_reg(get_gpregs_ctx(ctx), CTX_GPREG_X3, x3);	\
		set_aapcs_args2(ctx, x0, x1, x2);			\
Soby Mathew's avatar
Soby Mathew committed
289
	} while (0)
290
291
292
#define set_aapcs_args4(ctx, x0, x1, x2, x3, x4)		do {	\
		write_ctx_reg(get_gpregs_ctx(ctx), CTX_GPREG_X4, x4);	\
		set_aapcs_args3(ctx, x0, x1, x2, x3);			\
Soby Mathew's avatar
Soby Mathew committed
293
	} while (0)
294
295
296
#define set_aapcs_args5(ctx, x0, x1, x2, x3, x4, x5)		do {	\
		write_ctx_reg(get_gpregs_ctx(ctx), CTX_GPREG_X5, x5);	\
		set_aapcs_args4(ctx, x0, x1, x2, x3, x4);		\
Soby Mathew's avatar
Soby Mathew committed
297
	} while (0)
298
299
300
#define set_aapcs_args6(ctx, x0, x1, x2, x3, x4, x5, x6)	do {	\
		write_ctx_reg(get_gpregs_ctx(ctx), CTX_GPREG_X6, x6);	\
		set_aapcs_args5(ctx, x0, x1, x2, x3, x4, x5);		\
Soby Mathew's avatar
Soby Mathew committed
301
	} while (0)
302
303
304
#define set_aapcs_args7(ctx, x0, x1, x2, x3, x4, x5, x6, x7)	do {	\
		write_ctx_reg(get_gpregs_ctx(ctx), CTX_GPREG_X7, x7);	\
		set_aapcs_args6(ctx, x0, x1, x2, x3, x4, x5, x6);	\
Soby Mathew's avatar
Soby Mathew committed
305
	} while (0)
306

307
308
309
/*******************************************************************************
 * Function prototypes
 ******************************************************************************/
310
311
void el1_sysregs_context_save(el1_sys_regs_t *regs);
void el1_sysregs_context_restore(el1_sys_regs_t *regs);
312
#if CTX_INCLUDE_FPREGS
313
314
void fpregs_context_save(fp_regs_t *regs);
void fpregs_context_restore(fp_regs_t *regs);
315
#endif
316

317

318
#undef CTX_SYSREG_ALL
319
320
321
#if CTX_INCLUDE_FPREGS
#undef CTX_FPREG_ALL
#endif
322
#undef CTX_GPREG_ALL
323
324
325
326
327
#undef CTX_EL3STATE_ALL

#endif /* __ASSEMBLY__ */

#endif /* __CONTEXT_H__ */