arm/arm64: KVM: Fix BE accesses to GICv2 EISR and ELRSR regs
[linux/fpc-iii.git] / arch / arm / kvm / interrupts_head.S
blob14d488388480ea50a80d24b18bbed9636c8d1c25
1 #include <linux/irqchip/arm-gic.h>
2 #include <asm/assembler.h>
4 #define VCPU_USR_REG(_reg_nr)   (VCPU_USR_REGS + (_reg_nr * 4))
5 #define VCPU_USR_SP             (VCPU_USR_REG(13))
6 #define VCPU_USR_LR             (VCPU_USR_REG(14))
7 #define CP15_OFFSET(_cp15_reg_idx) (VCPU_CP15 + (_cp15_reg_idx * 4))
9 /*
10  * Many of these macros need to access the VCPU structure, which is always
11  * held in r0. These macros should never clobber r1, as it is used to hold the
12  * exception code on the return path (except of course the macro that switches
13  * all the registers before the final jump to the VM).
14  */
15 vcpu    .req    r0              @ vcpu pointer always in r0
17 /* Clobbers {r2-r6} */
18 .macro store_vfp_state vfp_base
19         @ The VFPFMRX and VFPFMXR macros are the VMRS and VMSR instructions
20         VFPFMRX r2, FPEXC
21         @ Make sure VFP is enabled so we can touch the registers.
22         orr     r6, r2, #FPEXC_EN
23         VFPFMXR FPEXC, r6
25         VFPFMRX r3, FPSCR
26         tst     r2, #FPEXC_EX           @ Check for VFP Subarchitecture
27         beq     1f
28         @ If FPEXC_EX is 0, then FPINST/FPINST2 reads are upredictable, so
29         @ we only need to save them if FPEXC_EX is set.
30         VFPFMRX r4, FPINST
31         tst     r2, #FPEXC_FP2V
32         VFPFMRX r5, FPINST2, ne         @ vmrsne
33         bic     r6, r2, #FPEXC_EX       @ FPEXC_EX disable
34         VFPFMXR FPEXC, r6
36         VFPFSTMIA \vfp_base, r6         @ Save VFP registers
37         stm     \vfp_base, {r2-r5}      @ Save FPEXC, FPSCR, FPINST, FPINST2
38 .endm
40 /* Assume FPEXC_EN is on and FPEXC_EX is off, clobbers {r2-r6} */
41 .macro restore_vfp_state vfp_base
42         VFPFLDMIA \vfp_base, r6         @ Load VFP registers
43         ldm     \vfp_base, {r2-r5}      @ Load FPEXC, FPSCR, FPINST, FPINST2
45         VFPFMXR FPSCR, r3
46         tst     r2, #FPEXC_EX           @ Check for VFP Subarchitecture
47         beq     1f
48         VFPFMXR FPINST, r4
49         tst     r2, #FPEXC_FP2V
50         VFPFMXR FPINST2, r5, ne
52         VFPFMXR FPEXC, r2       @ FPEXC (last, in case !EN)
53 .endm
55 /* These are simply for the macros to work - value don't have meaning */
56 .equ usr, 0
57 .equ svc, 1
58 .equ abt, 2
59 .equ und, 3
60 .equ irq, 4
61 .equ fiq, 5
63 .macro push_host_regs_mode mode
64         mrs     r2, SP_\mode
65         mrs     r3, LR_\mode
66         mrs     r4, SPSR_\mode
67         push    {r2, r3, r4}
68 .endm
71  * Store all host persistent registers on the stack.
72  * Clobbers all registers, in all modes, except r0 and r1.
73  */
74 .macro save_host_regs
75         /* Hyp regs. Only ELR_hyp (SPSR_hyp already saved) */
76         mrs     r2, ELR_hyp
77         push    {r2}
79         /* usr regs */
80         push    {r4-r12}        @ r0-r3 are always clobbered
81         mrs     r2, SP_usr
82         mov     r3, lr
83         push    {r2, r3}
85         push_host_regs_mode svc
86         push_host_regs_mode abt
87         push_host_regs_mode und
88         push_host_regs_mode irq
90         /* fiq regs */
91         mrs     r2, r8_fiq
92         mrs     r3, r9_fiq
93         mrs     r4, r10_fiq
94         mrs     r5, r11_fiq
95         mrs     r6, r12_fiq
96         mrs     r7, SP_fiq
97         mrs     r8, LR_fiq
98         mrs     r9, SPSR_fiq
99         push    {r2-r9}
100 .endm
102 .macro pop_host_regs_mode mode
103         pop     {r2, r3, r4}
104         msr     SP_\mode, r2
105         msr     LR_\mode, r3
106         msr     SPSR_\mode, r4
107 .endm
110  * Restore all host registers from the stack.
111  * Clobbers all registers, in all modes, except r0 and r1.
112  */
113 .macro restore_host_regs
114         pop     {r2-r9}
115         msr     r8_fiq, r2
116         msr     r9_fiq, r3
117         msr     r10_fiq, r4
118         msr     r11_fiq, r5
119         msr     r12_fiq, r6
120         msr     SP_fiq, r7
121         msr     LR_fiq, r8
122         msr     SPSR_fiq, r9
124         pop_host_regs_mode irq
125         pop_host_regs_mode und
126         pop_host_regs_mode abt
127         pop_host_regs_mode svc
129         pop     {r2, r3}
130         msr     SP_usr, r2
131         mov     lr, r3
132         pop     {r4-r12}
134         pop     {r2}
135         msr     ELR_hyp, r2
136 .endm
139  * Restore SP, LR and SPSR for a given mode. offset is the offset of
140  * this mode's registers from the VCPU base.
142  * Assumes vcpu pointer in vcpu reg
144  * Clobbers r1, r2, r3, r4.
145  */
146 .macro restore_guest_regs_mode mode, offset
147         add     r1, vcpu, \offset
148         ldm     r1, {r2, r3, r4}
149         msr     SP_\mode, r2
150         msr     LR_\mode, r3
151         msr     SPSR_\mode, r4
152 .endm
155  * Restore all guest registers from the vcpu struct.
157  * Assumes vcpu pointer in vcpu reg
159  * Clobbers *all* registers.
160  */
161 .macro restore_guest_regs
162         restore_guest_regs_mode svc, #VCPU_SVC_REGS
163         restore_guest_regs_mode abt, #VCPU_ABT_REGS
164         restore_guest_regs_mode und, #VCPU_UND_REGS
165         restore_guest_regs_mode irq, #VCPU_IRQ_REGS
167         add     r1, vcpu, #VCPU_FIQ_REGS
168         ldm     r1, {r2-r9}
169         msr     r8_fiq, r2
170         msr     r9_fiq, r3
171         msr     r10_fiq, r4
172         msr     r11_fiq, r5
173         msr     r12_fiq, r6
174         msr     SP_fiq, r7
175         msr     LR_fiq, r8
176         msr     SPSR_fiq, r9
178         @ Load return state
179         ldr     r2, [vcpu, #VCPU_PC]
180         ldr     r3, [vcpu, #VCPU_CPSR]
181         msr     ELR_hyp, r2
182         msr     SPSR_cxsf, r3
184         @ Load user registers
185         ldr     r2, [vcpu, #VCPU_USR_SP]
186         ldr     r3, [vcpu, #VCPU_USR_LR]
187         msr     SP_usr, r2
188         mov     lr, r3
189         add     vcpu, vcpu, #(VCPU_USR_REGS)
190         ldm     vcpu, {r0-r12}
191 .endm
194  * Save SP, LR and SPSR for a given mode. offset is the offset of
195  * this mode's registers from the VCPU base.
197  * Assumes vcpu pointer in vcpu reg
199  * Clobbers r2, r3, r4, r5.
200  */
201 .macro save_guest_regs_mode mode, offset
202         add     r2, vcpu, \offset
203         mrs     r3, SP_\mode
204         mrs     r4, LR_\mode
205         mrs     r5, SPSR_\mode
206         stm     r2, {r3, r4, r5}
207 .endm
210  * Save all guest registers to the vcpu struct
211  * Expects guest's r0, r1, r2 on the stack.
213  * Assumes vcpu pointer in vcpu reg
215  * Clobbers r2, r3, r4, r5.
216  */
217 .macro save_guest_regs
218         @ Store usr registers
219         add     r2, vcpu, #VCPU_USR_REG(3)
220         stm     r2, {r3-r12}
221         add     r2, vcpu, #VCPU_USR_REG(0)
222         pop     {r3, r4, r5}            @ r0, r1, r2
223         stm     r2, {r3, r4, r5}
224         mrs     r2, SP_usr
225         mov     r3, lr
226         str     r2, [vcpu, #VCPU_USR_SP]
227         str     r3, [vcpu, #VCPU_USR_LR]
229         @ Store return state
230         mrs     r2, ELR_hyp
231         mrs     r3, spsr
232         str     r2, [vcpu, #VCPU_PC]
233         str     r3, [vcpu, #VCPU_CPSR]
235         @ Store other guest registers
236         save_guest_regs_mode svc, #VCPU_SVC_REGS
237         save_guest_regs_mode abt, #VCPU_ABT_REGS
238         save_guest_regs_mode und, #VCPU_UND_REGS
239         save_guest_regs_mode irq, #VCPU_IRQ_REGS
240 .endm
242 /* Reads cp15 registers from hardware and stores them in memory
243  * @store_to_vcpu: If 0, registers are written in-order to the stack,
244  *                 otherwise to the VCPU struct pointed to by vcpup
246  * Assumes vcpu pointer in vcpu reg
248  * Clobbers r2 - r12
249  */
250 .macro read_cp15_state store_to_vcpu
251         mrc     p15, 0, r2, c1, c0, 0   @ SCTLR
252         mrc     p15, 0, r3, c1, c0, 2   @ CPACR
253         mrc     p15, 0, r4, c2, c0, 2   @ TTBCR
254         mrc     p15, 0, r5, c3, c0, 0   @ DACR
255         mrrc    p15, 0, r6, r7, c2      @ TTBR 0
256         mrrc    p15, 1, r8, r9, c2      @ TTBR 1
257         mrc     p15, 0, r10, c10, c2, 0 @ PRRR
258         mrc     p15, 0, r11, c10, c2, 1 @ NMRR
259         mrc     p15, 2, r12, c0, c0, 0  @ CSSELR
261         .if \store_to_vcpu == 0
262         push    {r2-r12}                @ Push CP15 registers
263         .else
264         str     r2, [vcpu, #CP15_OFFSET(c1_SCTLR)]
265         str     r3, [vcpu, #CP15_OFFSET(c1_CPACR)]
266         str     r4, [vcpu, #CP15_OFFSET(c2_TTBCR)]
267         str     r5, [vcpu, #CP15_OFFSET(c3_DACR)]
268         add     r2, vcpu, #CP15_OFFSET(c2_TTBR0)
269         strd    r6, r7, [r2]
270         add     r2, vcpu, #CP15_OFFSET(c2_TTBR1)
271         strd    r8, r9, [r2]
272         str     r10, [vcpu, #CP15_OFFSET(c10_PRRR)]
273         str     r11, [vcpu, #CP15_OFFSET(c10_NMRR)]
274         str     r12, [vcpu, #CP15_OFFSET(c0_CSSELR)]
275         .endif
277         mrc     p15, 0, r2, c13, c0, 1  @ CID
278         mrc     p15, 0, r3, c13, c0, 2  @ TID_URW
279         mrc     p15, 0, r4, c13, c0, 3  @ TID_URO
280         mrc     p15, 0, r5, c13, c0, 4  @ TID_PRIV
281         mrc     p15, 0, r6, c5, c0, 0   @ DFSR
282         mrc     p15, 0, r7, c5, c0, 1   @ IFSR
283         mrc     p15, 0, r8, c5, c1, 0   @ ADFSR
284         mrc     p15, 0, r9, c5, c1, 1   @ AIFSR
285         mrc     p15, 0, r10, c6, c0, 0  @ DFAR
286         mrc     p15, 0, r11, c6, c0, 2  @ IFAR
287         mrc     p15, 0, r12, c12, c0, 0 @ VBAR
289         .if \store_to_vcpu == 0
290         push    {r2-r12}                @ Push CP15 registers
291         .else
292         str     r2, [vcpu, #CP15_OFFSET(c13_CID)]
293         str     r3, [vcpu, #CP15_OFFSET(c13_TID_URW)]
294         str     r4, [vcpu, #CP15_OFFSET(c13_TID_URO)]
295         str     r5, [vcpu, #CP15_OFFSET(c13_TID_PRIV)]
296         str     r6, [vcpu, #CP15_OFFSET(c5_DFSR)]
297         str     r7, [vcpu, #CP15_OFFSET(c5_IFSR)]
298         str     r8, [vcpu, #CP15_OFFSET(c5_ADFSR)]
299         str     r9, [vcpu, #CP15_OFFSET(c5_AIFSR)]
300         str     r10, [vcpu, #CP15_OFFSET(c6_DFAR)]
301         str     r11, [vcpu, #CP15_OFFSET(c6_IFAR)]
302         str     r12, [vcpu, #CP15_OFFSET(c12_VBAR)]
303         .endif
305         mrc     p15, 0, r2, c14, c1, 0  @ CNTKCTL
306         mrrc    p15, 0, r4, r5, c7      @ PAR
307         mrc     p15, 0, r6, c10, c3, 0  @ AMAIR0
308         mrc     p15, 0, r7, c10, c3, 1  @ AMAIR1
310         .if \store_to_vcpu == 0
311         push    {r2,r4-r7}
312         .else
313         str     r2, [vcpu, #CP15_OFFSET(c14_CNTKCTL)]
314         add     r12, vcpu, #CP15_OFFSET(c7_PAR)
315         strd    r4, r5, [r12]
316         str     r6, [vcpu, #CP15_OFFSET(c10_AMAIR0)]
317         str     r7, [vcpu, #CP15_OFFSET(c10_AMAIR1)]
318         .endif
319 .endm
322  * Reads cp15 registers from memory and writes them to hardware
323  * @read_from_vcpu: If 0, registers are read in-order from the stack,
324  *                  otherwise from the VCPU struct pointed to by vcpup
326  * Assumes vcpu pointer in vcpu reg
327  */
328 .macro write_cp15_state read_from_vcpu
329         .if \read_from_vcpu == 0
330         pop     {r2,r4-r7}
331         .else
332         ldr     r2, [vcpu, #CP15_OFFSET(c14_CNTKCTL)]
333         add     r12, vcpu, #CP15_OFFSET(c7_PAR)
334         ldrd    r4, r5, [r12]
335         ldr     r6, [vcpu, #CP15_OFFSET(c10_AMAIR0)]
336         ldr     r7, [vcpu, #CP15_OFFSET(c10_AMAIR1)]
337         .endif
339         mcr     p15, 0, r2, c14, c1, 0  @ CNTKCTL
340         mcrr    p15, 0, r4, r5, c7      @ PAR
341         mcr     p15, 0, r6, c10, c3, 0  @ AMAIR0
342         mcr     p15, 0, r7, c10, c3, 1  @ AMAIR1
344         .if \read_from_vcpu == 0
345         pop     {r2-r12}
346         .else
347         ldr     r2, [vcpu, #CP15_OFFSET(c13_CID)]
348         ldr     r3, [vcpu, #CP15_OFFSET(c13_TID_URW)]
349         ldr     r4, [vcpu, #CP15_OFFSET(c13_TID_URO)]
350         ldr     r5, [vcpu, #CP15_OFFSET(c13_TID_PRIV)]
351         ldr     r6, [vcpu, #CP15_OFFSET(c5_DFSR)]
352         ldr     r7, [vcpu, #CP15_OFFSET(c5_IFSR)]
353         ldr     r8, [vcpu, #CP15_OFFSET(c5_ADFSR)]
354         ldr     r9, [vcpu, #CP15_OFFSET(c5_AIFSR)]
355         ldr     r10, [vcpu, #CP15_OFFSET(c6_DFAR)]
356         ldr     r11, [vcpu, #CP15_OFFSET(c6_IFAR)]
357         ldr     r12, [vcpu, #CP15_OFFSET(c12_VBAR)]
358         .endif
360         mcr     p15, 0, r2, c13, c0, 1  @ CID
361         mcr     p15, 0, r3, c13, c0, 2  @ TID_URW
362         mcr     p15, 0, r4, c13, c0, 3  @ TID_URO
363         mcr     p15, 0, r5, c13, c0, 4  @ TID_PRIV
364         mcr     p15, 0, r6, c5, c0, 0   @ DFSR
365         mcr     p15, 0, r7, c5, c0, 1   @ IFSR
366         mcr     p15, 0, r8, c5, c1, 0   @ ADFSR
367         mcr     p15, 0, r9, c5, c1, 1   @ AIFSR
368         mcr     p15, 0, r10, c6, c0, 0  @ DFAR
369         mcr     p15, 0, r11, c6, c0, 2  @ IFAR
370         mcr     p15, 0, r12, c12, c0, 0 @ VBAR
372         .if \read_from_vcpu == 0
373         pop     {r2-r12}
374         .else
375         ldr     r2, [vcpu, #CP15_OFFSET(c1_SCTLR)]
376         ldr     r3, [vcpu, #CP15_OFFSET(c1_CPACR)]
377         ldr     r4, [vcpu, #CP15_OFFSET(c2_TTBCR)]
378         ldr     r5, [vcpu, #CP15_OFFSET(c3_DACR)]
379         add     r12, vcpu, #CP15_OFFSET(c2_TTBR0)
380         ldrd    r6, r7, [r12]
381         add     r12, vcpu, #CP15_OFFSET(c2_TTBR1)
382         ldrd    r8, r9, [r12]
383         ldr     r10, [vcpu, #CP15_OFFSET(c10_PRRR)]
384         ldr     r11, [vcpu, #CP15_OFFSET(c10_NMRR)]
385         ldr     r12, [vcpu, #CP15_OFFSET(c0_CSSELR)]
386         .endif
388         mcr     p15, 0, r2, c1, c0, 0   @ SCTLR
389         mcr     p15, 0, r3, c1, c0, 2   @ CPACR
390         mcr     p15, 0, r4, c2, c0, 2   @ TTBCR
391         mcr     p15, 0, r5, c3, c0, 0   @ DACR
392         mcrr    p15, 0, r6, r7, c2      @ TTBR 0
393         mcrr    p15, 1, r8, r9, c2      @ TTBR 1
394         mcr     p15, 0, r10, c10, c2, 0 @ PRRR
395         mcr     p15, 0, r11, c10, c2, 1 @ NMRR
396         mcr     p15, 2, r12, c0, c0, 0  @ CSSELR
397 .endm
400  * Save the VGIC CPU state into memory
402  * Assumes vcpu pointer in vcpu reg
403  */
404 .macro save_vgic_state
405 #ifdef CONFIG_KVM_ARM_VGIC
406         /* Get VGIC VCTRL base into r2 */
407         ldr     r2, [vcpu, #VCPU_KVM]
408         ldr     r2, [r2, #KVM_VGIC_VCTRL]
409         cmp     r2, #0
410         beq     2f
412         /* Compute the address of struct vgic_cpu */
413         add     r11, vcpu, #VCPU_VGIC_CPU
415         /* Save all interesting registers */
416         ldr     r3, [r2, #GICH_HCR]
417         ldr     r4, [r2, #GICH_VMCR]
418         ldr     r5, [r2, #GICH_MISR]
419         ldr     r6, [r2, #GICH_EISR0]
420         ldr     r7, [r2, #GICH_EISR1]
421         ldr     r8, [r2, #GICH_ELRSR0]
422         ldr     r9, [r2, #GICH_ELRSR1]
423         ldr     r10, [r2, #GICH_APR]
424 ARM_BE8(rev     r3, r3  )
425 ARM_BE8(rev     r4, r4  )
426 ARM_BE8(rev     r5, r5  )
427 ARM_BE8(rev     r6, r6  )
428 ARM_BE8(rev     r7, r7  )
429 ARM_BE8(rev     r8, r8  )
430 ARM_BE8(rev     r9, r9  )
431 ARM_BE8(rev     r10, r10        )
433         str     r3, [r11, #VGIC_V2_CPU_HCR]
434         str     r4, [r11, #VGIC_V2_CPU_VMCR]
435         str     r5, [r11, #VGIC_V2_CPU_MISR]
436 #ifdef CONFIG_CPU_ENDIAN_BE8
437         str     r6, [r11, #(VGIC_V2_CPU_EISR + 4)]
438         str     r7, [r11, #VGIC_V2_CPU_EISR]
439         str     r8, [r11, #(VGIC_V2_CPU_ELRSR + 4)]
440         str     r9, [r11, #VGIC_V2_CPU_ELRSR]
441 #else
442         str     r6, [r11, #VGIC_V2_CPU_EISR]
443         str     r7, [r11, #(VGIC_V2_CPU_EISR + 4)]
444         str     r8, [r11, #VGIC_V2_CPU_ELRSR]
445         str     r9, [r11, #(VGIC_V2_CPU_ELRSR + 4)]
446 #endif
447         str     r10, [r11, #VGIC_V2_CPU_APR]
449         /* Clear GICH_HCR */
450         mov     r5, #0
451         str     r5, [r2, #GICH_HCR]
453         /* Save list registers */
454         add     r2, r2, #GICH_LR0
455         add     r3, r11, #VGIC_V2_CPU_LR
456         ldr     r4, [r11, #VGIC_CPU_NR_LR]
457 1:      ldr     r6, [r2], #4
458 ARM_BE8(rev     r6, r6  )
459         str     r6, [r3], #4
460         subs    r4, r4, #1
461         bne     1b
463 #endif
464 .endm
467  * Restore the VGIC CPU state from memory
469  * Assumes vcpu pointer in vcpu reg
470  */
471 .macro restore_vgic_state
472 #ifdef CONFIG_KVM_ARM_VGIC
473         /* Get VGIC VCTRL base into r2 */
474         ldr     r2, [vcpu, #VCPU_KVM]
475         ldr     r2, [r2, #KVM_VGIC_VCTRL]
476         cmp     r2, #0
477         beq     2f
479         /* Compute the address of struct vgic_cpu */
480         add     r11, vcpu, #VCPU_VGIC_CPU
482         /* We only restore a minimal set of registers */
483         ldr     r3, [r11, #VGIC_V2_CPU_HCR]
484         ldr     r4, [r11, #VGIC_V2_CPU_VMCR]
485         ldr     r8, [r11, #VGIC_V2_CPU_APR]
486 ARM_BE8(rev     r3, r3  )
487 ARM_BE8(rev     r4, r4  )
488 ARM_BE8(rev     r8, r8  )
490         str     r3, [r2, #GICH_HCR]
491         str     r4, [r2, #GICH_VMCR]
492         str     r8, [r2, #GICH_APR]
494         /* Restore list registers */
495         add     r2, r2, #GICH_LR0
496         add     r3, r11, #VGIC_V2_CPU_LR
497         ldr     r4, [r11, #VGIC_CPU_NR_LR]
498 1:      ldr     r6, [r3], #4
499 ARM_BE8(rev     r6, r6  )
500         str     r6, [r2], #4
501         subs    r4, r4, #1
502         bne     1b
504 #endif
505 .endm
507 #define CNTHCTL_PL1PCTEN        (1 << 0)
508 #define CNTHCTL_PL1PCEN         (1 << 1)
511  * Save the timer state onto the VCPU and allow physical timer/counter access
512  * for the host.
514  * Assumes vcpu pointer in vcpu reg
515  * Clobbers r2-r5
516  */
517 .macro save_timer_state
518 #ifdef CONFIG_KVM_ARM_TIMER
519         ldr     r4, [vcpu, #VCPU_KVM]
520         ldr     r2, [r4, #KVM_TIMER_ENABLED]
521         cmp     r2, #0
522         beq     1f
524         mrc     p15, 0, r2, c14, c3, 1  @ CNTV_CTL
525         str     r2, [vcpu, #VCPU_TIMER_CNTV_CTL]
526         bic     r2, #1                  @ Clear ENABLE
527         mcr     p15, 0, r2, c14, c3, 1  @ CNTV_CTL
528         isb
530         mrrc    p15, 3, rr_lo_hi(r2, r3), c14   @ CNTV_CVAL
531         ldr     r4, =VCPU_TIMER_CNTV_CVAL
532         add     r5, vcpu, r4
533         strd    r2, r3, [r5]
535         @ Ensure host CNTVCT == CNTPCT
536         mov     r2, #0
537         mcrr    p15, 4, r2, r2, c14     @ CNTVOFF
540 #endif
541         @ Allow physical timer/counter access for the host
542         mrc     p15, 4, r2, c14, c1, 0  @ CNTHCTL
543         orr     r2, r2, #(CNTHCTL_PL1PCEN | CNTHCTL_PL1PCTEN)
544         mcr     p15, 4, r2, c14, c1, 0  @ CNTHCTL
545 .endm
548  * Load the timer state from the VCPU and deny physical timer/counter access
549  * for the host.
551  * Assumes vcpu pointer in vcpu reg
552  * Clobbers r2-r5
553  */
554 .macro restore_timer_state
555         @ Disallow physical timer access for the guest
556         @ Physical counter access is allowed
557         mrc     p15, 4, r2, c14, c1, 0  @ CNTHCTL
558         orr     r2, r2, #CNTHCTL_PL1PCTEN
559         bic     r2, r2, #CNTHCTL_PL1PCEN
560         mcr     p15, 4, r2, c14, c1, 0  @ CNTHCTL
562 #ifdef CONFIG_KVM_ARM_TIMER
563         ldr     r4, [vcpu, #VCPU_KVM]
564         ldr     r2, [r4, #KVM_TIMER_ENABLED]
565         cmp     r2, #0
566         beq     1f
568         ldr     r2, [r4, #KVM_TIMER_CNTVOFF]
569         ldr     r3, [r4, #(KVM_TIMER_CNTVOFF + 4)]
570         mcrr    p15, 4, rr_lo_hi(r2, r3), c14   @ CNTVOFF
572         ldr     r4, =VCPU_TIMER_CNTV_CVAL
573         add     r5, vcpu, r4
574         ldrd    r2, r3, [r5]
575         mcrr    p15, 3, rr_lo_hi(r2, r3), c14   @ CNTV_CVAL
576         isb
578         ldr     r2, [vcpu, #VCPU_TIMER_CNTV_CTL]
579         and     r2, r2, #3
580         mcr     p15, 0, r2, c14, c3, 1  @ CNTV_CTL
582 #endif
583 .endm
585 .equ vmentry,   0
586 .equ vmexit,    1
588 /* Configures the HSTR (Hyp System Trap Register) on entry/return
589  * (hardware reset value is 0) */
590 .macro set_hstr operation
591         mrc     p15, 4, r2, c1, c1, 3
592         ldr     r3, =HSTR_T(15)
593         .if \operation == vmentry
594         orr     r2, r2, r3              @ Trap CR{15}
595         .else
596         bic     r2, r2, r3              @ Don't trap any CRx accesses
597         .endif
598         mcr     p15, 4, r2, c1, c1, 3
599 .endm
601 /* Configures the HCPTR (Hyp Coprocessor Trap Register) on entry/return
602  * (hardware reset value is 0). Keep previous value in r2. */
603 .macro set_hcptr operation, mask
604         mrc     p15, 4, r2, c1, c1, 2
605         ldr     r3, =\mask
606         .if \operation == vmentry
607         orr     r3, r2, r3              @ Trap coproc-accesses defined in mask
608         .else
609         bic     r3, r2, r3              @ Don't trap defined coproc-accesses
610         .endif
611         mcr     p15, 4, r3, c1, c1, 2
612 .endm
614 /* Configures the HDCR (Hyp Debug Configuration Register) on entry/return
615  * (hardware reset value is 0) */
616 .macro set_hdcr operation
617         mrc     p15, 4, r2, c1, c1, 1
618         ldr     r3, =(HDCR_TPM|HDCR_TPMCR)
619         .if \operation == vmentry
620         orr     r2, r2, r3              @ Trap some perfmon accesses
621         .else
622         bic     r2, r2, r3              @ Don't trap any perfmon accesses
623         .endif
624         mcr     p15, 4, r2, c1, c1, 1
625 .endm
627 /* Enable/Disable: stage-2 trans., trap interrupts, trap wfi, trap smc */
628 .macro configure_hyp_role operation
629         .if \operation == vmentry
630         ldr     r2, [vcpu, #VCPU_HCR]
631         ldr     r3, [vcpu, #VCPU_IRQ_LINES]
632         orr     r2, r2, r3
633         .else
634         mov     r2, #0
635         .endif
636         mcr     p15, 4, r2, c1, c1, 0   @ HCR
637 .endm
639 .macro load_vcpu
640         mrc     p15, 4, vcpu, c13, c0, 2        @ HTPIDR
641 .endm