diff options
Diffstat (limited to 'arch/arm/kvm/interrupts_head.S')
-rw-r--r-- | arch/arm/kvm/interrupts_head.S | 164 |
1 files changed, 164 insertions, 0 deletions
diff --git a/arch/arm/kvm/interrupts_head.S b/arch/arm/kvm/interrupts_head.S index 6a95d341e9c5..3c8f2f0b4c5e 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)) |
@@ -298,6 +300,14 @@ vcpu .req r0 @ vcpu pointer always in r0 | |||
298 | str r11, [vcpu, #CP15_OFFSET(c6_IFAR)] | 300 | str r11, [vcpu, #CP15_OFFSET(c6_IFAR)] |
299 | str r12, [vcpu, #CP15_OFFSET(c12_VBAR)] | 301 | str r12, [vcpu, #CP15_OFFSET(c12_VBAR)] |
300 | .endif | 302 | .endif |
303 | |||
304 | mrc p15, 0, r2, c14, c1, 0 @ CNTKCTL | ||
305 | |||
306 | .if \store_to_vcpu == 0 | ||
307 | push {r2} | ||
308 | .else | ||
309 | str r2, [vcpu, #CP15_OFFSET(c14_CNTKCTL)] | ||
310 | .endif | ||
301 | .endm | 311 | .endm |
302 | 312 | ||
303 | /* | 313 | /* |
@@ -309,6 +319,14 @@ vcpu .req r0 @ vcpu pointer always in r0 | |||
309 | */ | 319 | */ |
310 | .macro write_cp15_state read_from_vcpu | 320 | .macro write_cp15_state read_from_vcpu |
311 | .if \read_from_vcpu == 0 | 321 | .if \read_from_vcpu == 0 |
322 | pop {r2} | ||
323 | .else | ||
324 | ldr r2, [vcpu, #CP15_OFFSET(c14_CNTKCTL)] | ||
325 | .endif | ||
326 | |||
327 | mcr p15, 0, r2, c14, c1, 0 @ CNTKCTL | ||
328 | |||
329 | .if \read_from_vcpu == 0 | ||
312 | pop {r2-r12} | 330 | pop {r2-r12} |
313 | .else | 331 | .else |
314 | ldr r2, [vcpu, #CP15_OFFSET(c13_CID)] | 332 | ldr r2, [vcpu, #CP15_OFFSET(c13_CID)] |
@@ -369,6 +387,49 @@ vcpu .req r0 @ vcpu pointer always in r0 | |||
369 | * Assumes vcpu pointer in vcpu reg | 387 | * Assumes vcpu pointer in vcpu reg |
370 | */ | 388 | */ |
371 | .macro save_vgic_state | 389 | .macro save_vgic_state |
390 | #ifdef CONFIG_KVM_ARM_VGIC | ||
391 | /* Get VGIC VCTRL base into r2 */ | ||
392 | ldr r2, [vcpu, #VCPU_KVM] | ||
393 | ldr r2, [r2, #KVM_VGIC_VCTRL] | ||
394 | cmp r2, #0 | ||
395 | beq 2f | ||
396 | |||
397 | /* Compute the address of struct vgic_cpu */ | ||
398 | add r11, vcpu, #VCPU_VGIC_CPU | ||
399 | |||
400 | /* Save all interesting registers */ | ||
401 | ldr r3, [r2, #GICH_HCR] | ||
402 | ldr r4, [r2, #GICH_VMCR] | ||
403 | ldr r5, [r2, #GICH_MISR] | ||
404 | ldr r6, [r2, #GICH_EISR0] | ||
405 | ldr r7, [r2, #GICH_EISR1] | ||
406 | ldr r8, [r2, #GICH_ELRSR0] | ||
407 | ldr r9, [r2, #GICH_ELRSR1] | ||
408 | ldr r10, [r2, #GICH_APR] | ||
409 | |||
410 | str r3, [r11, #VGIC_CPU_HCR] | ||
411 | str r4, [r11, #VGIC_CPU_VMCR] | ||
412 | str r5, [r11, #VGIC_CPU_MISR] | ||
413 | str r6, [r11, #VGIC_CPU_EISR] | ||
414 | str r7, [r11, #(VGIC_CPU_EISR + 4)] | ||
415 | str r8, [r11, #VGIC_CPU_ELRSR] | ||
416 | str r9, [r11, #(VGIC_CPU_ELRSR + 4)] | ||
417 | str r10, [r11, #VGIC_CPU_APR] | ||
418 | |||
419 | /* Clear GICH_HCR */ | ||
420 | mov r5, #0 | ||
421 | str r5, [r2, #GICH_HCR] | ||
422 | |||
423 | /* Save list registers */ | ||
424 | add r2, r2, #GICH_LR0 | ||
425 | add r3, r11, #VGIC_CPU_LR | ||
426 | ldr r4, [r11, #VGIC_CPU_NR_LR] | ||
427 | 1: ldr r6, [r2], #4 | ||
428 | str r6, [r3], #4 | ||
429 | subs r4, r4, #1 | ||
430 | bne 1b | ||
431 | 2: | ||
432 | #endif | ||
372 | .endm | 433 | .endm |
373 | 434 | ||
374 | /* | 435 | /* |
@@ -377,6 +438,109 @@ vcpu .req r0 @ vcpu pointer always in r0 | |||
377 | * Assumes vcpu pointer in vcpu reg | 438 | * Assumes vcpu pointer in vcpu reg |
378 | */ | 439 | */ |
379 | .macro restore_vgic_state | 440 | .macro restore_vgic_state |
441 | #ifdef CONFIG_KVM_ARM_VGIC | ||
442 | /* Get VGIC VCTRL base into r2 */ | ||
443 | ldr r2, [vcpu, #VCPU_KVM] | ||
444 | ldr r2, [r2, #KVM_VGIC_VCTRL] | ||
445 | cmp r2, #0 | ||
446 | beq 2f | ||
447 | |||
448 | /* Compute the address of struct vgic_cpu */ | ||
449 | add r11, vcpu, #VCPU_VGIC_CPU | ||
450 | |||
451 | /* We only restore a minimal set of registers */ | ||
452 | ldr r3, [r11, #VGIC_CPU_HCR] | ||
453 | ldr r4, [r11, #VGIC_CPU_VMCR] | ||
454 | ldr r8, [r11, #VGIC_CPU_APR] | ||
455 | |||
456 | str r3, [r2, #GICH_HCR] | ||
457 | str r4, [r2, #GICH_VMCR] | ||
458 | str r8, [r2, #GICH_APR] | ||
459 | |||
460 | /* Restore list registers */ | ||
461 | add r2, r2, #GICH_LR0 | ||
462 | add r3, r11, #VGIC_CPU_LR | ||
463 | ldr r4, [r11, #VGIC_CPU_NR_LR] | ||
464 | 1: ldr r6, [r3], #4 | ||
465 | str r6, [r2], #4 | ||
466 | subs r4, r4, #1 | ||
467 | bne 1b | ||
468 | 2: | ||
469 | #endif | ||
470 | .endm | ||
471 | |||
472 | #define CNTHCTL_PL1PCTEN (1 << 0) | ||
473 | #define CNTHCTL_PL1PCEN (1 << 1) | ||
474 | |||
475 | /* | ||
476 | * Save the timer state onto the VCPU and allow physical timer/counter access | ||
477 | * for the host. | ||
478 | * | ||
479 | * Assumes vcpu pointer in vcpu reg | ||
480 | * Clobbers r2-r5 | ||
481 | */ | ||
482 | .macro save_timer_state | ||
483 | #ifdef CONFIG_KVM_ARM_TIMER | ||
484 | ldr r4, [vcpu, #VCPU_KVM] | ||
485 | ldr r2, [r4, #KVM_TIMER_ENABLED] | ||
486 | cmp r2, #0 | ||
487 | beq 1f | ||
488 | |||
489 | mrc p15, 0, r2, c14, c3, 1 @ CNTV_CTL | ||
490 | str r2, [vcpu, #VCPU_TIMER_CNTV_CTL] | ||
491 | bic r2, #1 @ Clear ENABLE | ||
492 | mcr p15, 0, r2, c14, c3, 1 @ CNTV_CTL | ||
493 | isb | ||
494 | |||
495 | mrrc p15, 3, r2, r3, c14 @ CNTV_CVAL | ||
496 | ldr r4, =VCPU_TIMER_CNTV_CVAL | ||
497 | add r5, vcpu, r4 | ||
498 | strd r2, r3, [r5] | ||
499 | |||
500 | 1: | ||
501 | #endif | ||
502 | @ Allow physical timer/counter access for the host | ||
503 | mrc p15, 4, r2, c14, c1, 0 @ CNTHCTL | ||
504 | orr r2, r2, #(CNTHCTL_PL1PCEN | CNTHCTL_PL1PCTEN) | ||
505 | mcr p15, 4, r2, c14, c1, 0 @ CNTHCTL | ||
506 | .endm | ||
507 | |||
508 | /* | ||
509 | * Load the timer state from the VCPU and deny physical timer/counter access | ||
510 | * for the host. | ||
511 | * | ||
512 | * Assumes vcpu pointer in vcpu reg | ||
513 | * Clobbers r2-r5 | ||
514 | */ | ||
515 | .macro restore_timer_state | ||
516 | @ Disallow physical timer access for the guest | ||
517 | @ Physical counter access is allowed | ||
518 | mrc p15, 4, r2, c14, c1, 0 @ CNTHCTL | ||
519 | orr r2, r2, #CNTHCTL_PL1PCTEN | ||
520 | bic r2, r2, #CNTHCTL_PL1PCEN | ||
521 | mcr p15, 4, r2, c14, c1, 0 @ CNTHCTL | ||
522 | |||
523 | #ifdef CONFIG_KVM_ARM_TIMER | ||
524 | ldr r4, [vcpu, #VCPU_KVM] | ||
525 | ldr r2, [r4, #KVM_TIMER_ENABLED] | ||
526 | cmp r2, #0 | ||
527 | beq 1f | ||
528 | |||
529 | ldr r2, [r4, #KVM_TIMER_CNTVOFF] | ||
530 | ldr r3, [r4, #(KVM_TIMER_CNTVOFF + 4)] | ||
531 | mcrr p15, 4, r2, r3, c14 @ CNTVOFF | ||
532 | |||
533 | ldr r4, =VCPU_TIMER_CNTV_CVAL | ||
534 | add r5, vcpu, r4 | ||
535 | ldrd r2, r3, [r5] | ||
536 | mcrr p15, 3, r2, r3, c14 @ CNTV_CVAL | ||
537 | isb | ||
538 | |||
539 | ldr r2, [vcpu, #VCPU_TIMER_CNTV_CTL] | ||
540 | and r2, r2, #3 | ||
541 | mcr p15, 0, r2, c14, c3, 1 @ CNTV_CTL | ||
542 | 1: | ||
543 | #endif | ||
380 | .endm | 544 | .endm |
381 | 545 | ||
382 | .equ vmentry, 0 | 546 | .equ vmentry, 0 |