diff options
author | Marc Zyngier <marc.zyngier@arm.com> | 2013-01-21 19:36:15 -0500 |
---|---|---|
committer | Marc Zyngier <marc.zyngier@arm.com> | 2013-02-11 14:00:03 -0500 |
commit | 348b2b0708f6cdd3d0db95f8d02aa4ad2b3e2fa9 (patch) | |
tree | 2ecf6a0ef7ed9359545f426f78277b4fb1a5fde9 | |
parent | 5863c2ce7269a7b24d60006430aa79a750b226ec (diff) |
ARM: KVM: VGIC control interface world switch
Enable the VGIC control interface to be save-restored on world switch.
Reviewed-by: Will Deacon <will.deacon@arm.com>
Signed-off-by: Christoffer Dall <c.dall@virtualopensystems.com>
Signed-off-by: Marc Zyngier <marc.zyngier@arm.com>
-rw-r--r-- | arch/arm/kernel/asm-offsets.c | 12 | ||||
-rw-r--r-- | arch/arm/kvm/interrupts_head.S | 74 |
2 files changed, 86 insertions, 0 deletions
diff --git a/arch/arm/kernel/asm-offsets.c b/arch/arm/kernel/asm-offsets.c index c8b3272dfed1..17cea2e78d88 100644 --- a/arch/arm/kernel/asm-offsets.c +++ b/arch/arm/kernel/asm-offsets.c | |||
@@ -169,6 +169,18 @@ int main(void) | |||
169 | DEFINE(VCPU_HxFAR, offsetof(struct kvm_vcpu, arch.hxfar)); | 169 | DEFINE(VCPU_HxFAR, offsetof(struct kvm_vcpu, arch.hxfar)); |
170 | DEFINE(VCPU_HPFAR, offsetof(struct kvm_vcpu, arch.hpfar)); | 170 | DEFINE(VCPU_HPFAR, offsetof(struct kvm_vcpu, arch.hpfar)); |
171 | DEFINE(VCPU_HYP_PC, offsetof(struct kvm_vcpu, arch.hyp_pc)); | 171 | DEFINE(VCPU_HYP_PC, offsetof(struct kvm_vcpu, arch.hyp_pc)); |
172 | #ifdef CONFIG_KVM_ARM_VGIC | ||
173 | DEFINE(VCPU_VGIC_CPU, offsetof(struct kvm_vcpu, arch.vgic_cpu)); | ||
174 | DEFINE(VGIC_CPU_HCR, offsetof(struct vgic_cpu, vgic_hcr)); | ||
175 | DEFINE(VGIC_CPU_VMCR, offsetof(struct vgic_cpu, vgic_vmcr)); | ||
176 | DEFINE(VGIC_CPU_MISR, offsetof(struct vgic_cpu, vgic_misr)); | ||
177 | DEFINE(VGIC_CPU_EISR, offsetof(struct vgic_cpu, vgic_eisr)); | ||
178 | DEFINE(VGIC_CPU_ELRSR, offsetof(struct vgic_cpu, vgic_elrsr)); | ||
179 | DEFINE(VGIC_CPU_APR, offsetof(struct vgic_cpu, vgic_apr)); | ||
180 | DEFINE(VGIC_CPU_LR, offsetof(struct vgic_cpu, vgic_lr)); | ||
181 | DEFINE(VGIC_CPU_NR_LR, offsetof(struct vgic_cpu, nr_lr)); | ||
182 | DEFINE(KVM_VGIC_VCTRL, offsetof(struct kvm, arch.vgic.vctrl_base)); | ||
183 | #endif | ||
172 | DEFINE(KVM_VTTBR, offsetof(struct kvm, arch.vttbr)); | 184 | DEFINE(KVM_VTTBR, offsetof(struct kvm, arch.vttbr)); |
173 | #endif | 185 | #endif |
174 | return 0; | 186 | return 0; |
diff --git a/arch/arm/kvm/interrupts_head.S b/arch/arm/kvm/interrupts_head.S index 6a95d341e9c5..8c875d54a089 100644 --- a/arch/arm/kvm/interrupts_head.S +++ b/arch/arm/kvm/interrupts_head.S | |||
@@ -1,3 +1,5 @@ | |||
1 | #include <linux/irqchip/arm-gic.h> | ||
2 | |||
1 | #define VCPU_USR_REG(_reg_nr) (VCPU_USR_REGS + (_reg_nr * 4)) | 3 | #define VCPU_USR_REG(_reg_nr) (VCPU_USR_REGS + (_reg_nr * 4)) |
2 | #define VCPU_USR_SP (VCPU_USR_REG(13)) | 4 | #define VCPU_USR_SP (VCPU_USR_REG(13)) |
3 | #define VCPU_USR_LR (VCPU_USR_REG(14)) | 5 | #define VCPU_USR_LR (VCPU_USR_REG(14)) |
@@ -369,6 +371,49 @@ vcpu .req r0 @ vcpu pointer always in r0 | |||
369 | * Assumes vcpu pointer in vcpu reg | 371 | * Assumes vcpu pointer in vcpu reg |
370 | */ | 372 | */ |
371 | .macro save_vgic_state | 373 | .macro save_vgic_state |
374 | #ifdef CONFIG_KVM_ARM_VGIC | ||
375 | /* Get VGIC VCTRL base into r2 */ | ||
376 | ldr r2, [vcpu, #VCPU_KVM] | ||
377 | ldr r2, [r2, #KVM_VGIC_VCTRL] | ||
378 | cmp r2, #0 | ||
379 | beq 2f | ||
380 | |||
381 | /* Compute the address of struct vgic_cpu */ | ||
382 | add r11, vcpu, #VCPU_VGIC_CPU | ||
383 | |||
384 | /* Save all interesting registers */ | ||
385 | ldr r3, [r2, #GICH_HCR] | ||
386 | ldr r4, [r2, #GICH_VMCR] | ||
387 | ldr r5, [r2, #GICH_MISR] | ||
388 | ldr r6, [r2, #GICH_EISR0] | ||
389 | ldr r7, [r2, #GICH_EISR1] | ||
390 | ldr r8, [r2, #GICH_ELRSR0] | ||
391 | ldr r9, [r2, #GICH_ELRSR1] | ||
392 | ldr r10, [r2, #GICH_APR] | ||
393 | |||
394 | str r3, [r11, #VGIC_CPU_HCR] | ||
395 | str r4, [r11, #VGIC_CPU_VMCR] | ||
396 | str r5, [r11, #VGIC_CPU_MISR] | ||
397 | str r6, [r11, #VGIC_CPU_EISR] | ||
398 | str r7, [r11, #(VGIC_CPU_EISR + 4)] | ||
399 | str r8, [r11, #VGIC_CPU_ELRSR] | ||
400 | str r9, [r11, #(VGIC_CPU_ELRSR + 4)] | ||
401 | str r10, [r11, #VGIC_CPU_APR] | ||
402 | |||
403 | /* Clear GICH_HCR */ | ||
404 | mov r5, #0 | ||
405 | str r5, [r2, #GICH_HCR] | ||
406 | |||
407 | /* Save list registers */ | ||
408 | add r2, r2, #GICH_LR0 | ||
409 | add r3, r11, #VGIC_CPU_LR | ||
410 | ldr r4, [r11, #VGIC_CPU_NR_LR] | ||
411 | 1: ldr r6, [r2], #4 | ||
412 | str r6, [r3], #4 | ||
413 | subs r4, r4, #1 | ||
414 | bne 1b | ||
415 | 2: | ||
416 | #endif | ||
372 | .endm | 417 | .endm |
373 | 418 | ||
374 | /* | 419 | /* |
@@ -377,6 +422,35 @@ vcpu .req r0 @ vcpu pointer always in r0 | |||
377 | * Assumes vcpu pointer in vcpu reg | 422 | * Assumes vcpu pointer in vcpu reg |
378 | */ | 423 | */ |
379 | .macro restore_vgic_state | 424 | .macro restore_vgic_state |
425 | #ifdef CONFIG_KVM_ARM_VGIC | ||
426 | /* Get VGIC VCTRL base into r2 */ | ||
427 | ldr r2, [vcpu, #VCPU_KVM] | ||
428 | ldr r2, [r2, #KVM_VGIC_VCTRL] | ||
429 | cmp r2, #0 | ||
430 | beq 2f | ||
431 | |||
432 | /* Compute the address of struct vgic_cpu */ | ||
433 | add r11, vcpu, #VCPU_VGIC_CPU | ||
434 | |||
435 | /* We only restore a minimal set of registers */ | ||
436 | ldr r3, [r11, #VGIC_CPU_HCR] | ||
437 | ldr r4, [r11, #VGIC_CPU_VMCR] | ||
438 | ldr r8, [r11, #VGIC_CPU_APR] | ||
439 | |||
440 | str r3, [r2, #GICH_HCR] | ||
441 | str r4, [r2, #GICH_VMCR] | ||
442 | str r8, [r2, #GICH_APR] | ||
443 | |||
444 | /* Restore list registers */ | ||
445 | add r2, r2, #GICH_LR0 | ||
446 | add r3, r11, #VGIC_CPU_LR | ||
447 | ldr r4, [r11, #VGIC_CPU_NR_LR] | ||
448 | 1: ldr r6, [r3], #4 | ||
449 | str r6, [r2], #4 | ||
450 | subs r4, r4, #1 | ||
451 | bne 1b | ||
452 | 2: | ||
453 | #endif | ||
380 | .endm | 454 | .endm |
381 | 455 | ||
382 | .equ vmentry, 0 | 456 | .equ vmentry, 0 |