diff options
author | Christoffer Dall <christoffer.dall@linaro.org> | 2014-12-12 15:19:23 -0500 |
---|---|---|
committer | Christoffer Dall <christoffer.dall@linaro.org> | 2014-12-15 05:50:42 -0500 |
commit | 05971120fca43e0357789a14b3386bb56eef2201 (patch) | |
tree | e839f7efda57e1d53fa69c90315df4d0775716bf /arch/arm/kvm | |
parent | ca7d9c829d419c06e450afa5f785d58198c37caa (diff) |
arm/arm64: KVM: Require in-kernel vgic for the arch timers
It is curently possible to run a VM with architected timers support
without creating an in-kernel VGIC, which will result in interrupts from
the virtual timer going nowhere.
To address this issue, move the architected timers initialization to the
time when we run a VCPU for the first time, and then only initialize
(and enable) the architected timers if we have a properly created and
initialized in-kernel VGIC.
When injecting interrupts from the virtual timer to the vgic, the
current setup should ensure that this never calls an on-demand init of
the VGIC, which is the only call path that could return an error from
kvm_vgic_inject_irq(), so capture the return value and raise a warning
if there's an error there.
We also change the kvm_timer_init() function from returning an int to be
a void function, since the function always succeeds.
Reviewed-by: Marc Zyngier <marc.zyngier@arm.com>
Signed-off-by: Christoffer Dall <christoffer.dall@linaro.org>
Diffstat (limited to 'arch/arm/kvm')
-rw-r--r-- | arch/arm/kvm/arm.c | 13 |
1 files changed, 11 insertions, 2 deletions
diff --git a/arch/arm/kvm/arm.c b/arch/arm/kvm/arm.c index 66f37c4cdf13..2d6d91001062 100644 --- a/arch/arm/kvm/arm.c +++ b/arch/arm/kvm/arm.c | |||
@@ -425,6 +425,7 @@ static void update_vttbr(struct kvm *kvm) | |||
425 | 425 | ||
426 | static int kvm_vcpu_first_run_init(struct kvm_vcpu *vcpu) | 426 | static int kvm_vcpu_first_run_init(struct kvm_vcpu *vcpu) |
427 | { | 427 | { |
428 | struct kvm *kvm = vcpu->kvm; | ||
428 | int ret; | 429 | int ret; |
429 | 430 | ||
430 | if (likely(vcpu->arch.has_run_once)) | 431 | if (likely(vcpu->arch.has_run_once)) |
@@ -436,12 +437,20 @@ static int kvm_vcpu_first_run_init(struct kvm_vcpu *vcpu) | |||
436 | * Map the VGIC hardware resources before running a vcpu the first | 437 | * Map the VGIC hardware resources before running a vcpu the first |
437 | * time on this VM. | 438 | * time on this VM. |
438 | */ | 439 | */ |
439 | if (unlikely(!vgic_ready(vcpu->kvm))) { | 440 | if (unlikely(!vgic_ready(kvm))) { |
440 | ret = kvm_vgic_map_resources(vcpu->kvm); | 441 | ret = kvm_vgic_map_resources(kvm); |
441 | if (ret) | 442 | if (ret) |
442 | return ret; | 443 | return ret; |
443 | } | 444 | } |
444 | 445 | ||
446 | /* | ||
447 | * Enable the arch timers only if we have an in-kernel VGIC | ||
448 | * and it has been properly initialized, since we cannot handle | ||
449 | * interrupts from the virtual timer with a userspace gic. | ||
450 | */ | ||
451 | if (irqchip_in_kernel(kvm) && vgic_initialized(kvm)) | ||
452 | kvm_timer_enable(kvm); | ||
453 | |||
445 | return 0; | 454 | return 0; |
446 | } | 455 | } |
447 | 456 | ||