diff options
author | Anup Patel <anup.patel@linaro.org> | 2014-04-29 01:54:24 -0400 |
---|---|---|
committer | Christoffer Dall <christoffer.dall@linaro.org> | 2014-04-30 07:18:58 -0400 |
commit | b376d02b53b87f8684f91f13ba4ee43331850fcd (patch) | |
tree | 720c11fca0b9ad59dca59f026e8e631e50a0f5b0 /arch/arm/kvm | |
parent | aa8aeefe5e567637bbec7d7a3031cc057e3af303 (diff) |
ARM/ARM64: KVM: Emulate PSCI v0.2 CPU_SUSPEND
This patch adds emulation of PSCI v0.2 CPU_SUSPEND function call for
KVM ARM/ARM64. This is a CPU-level function call which can suspend
current CPU or current CPU cluster. We don't have VCPU clusters in
KVM so we only suspend the current VCPU.
The CPU_SUSPEND emulation is not tested much because currently there
is no CPUIDLE driver in Linux kernel that uses PSCI CPU_SUSPEND. The
PSCI CPU_SUSPEND implementation in ARM64 kernel was tested using a
Simple CPUIDLE driver which is not published due to unstable DT-bindings
for PSCI.
(For more info, http://lwn.net/Articles/574950/)
For simplicity, we implement CPU_SUSPEND emulation similar to WFI
(Wait-for-interrupt) emulation and we also treat power-down request
to be same as stand-by request. This is consistent with section
5.4.1 and section 5.4.2 of PSCI v0.2 specification.
Signed-off-by: Anup Patel <anup.patel@linaro.org>
Signed-off-by: Pranavkumar Sawargaonkar <pranavkumar@linaro.org>
Acked-by: Christoffer Dall <christoffer.dall@linaro.org>
Acked-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/psci.c | 28 |
1 files changed, 24 insertions, 4 deletions
diff --git a/arch/arm/kvm/psci.c b/arch/arm/kvm/psci.c index 1067579c7336..09cf37737ee2 100644 --- a/arch/arm/kvm/psci.c +++ b/arch/arm/kvm/psci.c | |||
@@ -37,6 +37,26 @@ static unsigned long psci_affinity_mask(unsigned long affinity_level) | |||
37 | return 0; | 37 | return 0; |
38 | } | 38 | } |
39 | 39 | ||
40 | static unsigned long kvm_psci_vcpu_suspend(struct kvm_vcpu *vcpu) | ||
41 | { | ||
42 | /* | ||
43 | * NOTE: For simplicity, we make VCPU suspend emulation to be | ||
44 | * same-as WFI (Wait-for-interrupt) emulation. | ||
45 | * | ||
46 | * This means for KVM the wakeup events are interrupts and | ||
47 | * this is consistent with intended use of StateID as described | ||
48 | * in section 5.4.1 of PSCI v0.2 specification (ARM DEN 0022A). | ||
49 | * | ||
50 | * Further, we also treat power-down request to be same as | ||
51 | * stand-by request as-per section 5.4.2 clause 3 of PSCI v0.2 | ||
52 | * specification (ARM DEN 0022A). This means all suspend states | ||
53 | * for KVM will preserve the register state. | ||
54 | */ | ||
55 | kvm_vcpu_block(vcpu); | ||
56 | |||
57 | return PSCI_RET_SUCCESS; | ||
58 | } | ||
59 | |||
40 | static void kvm_psci_vcpu_off(struct kvm_vcpu *vcpu) | 60 | static void kvm_psci_vcpu_off(struct kvm_vcpu *vcpu) |
41 | { | 61 | { |
42 | vcpu->arch.pause = true; | 62 | vcpu->arch.pause = true; |
@@ -183,6 +203,10 @@ static int kvm_psci_0_2_call(struct kvm_vcpu *vcpu) | |||
183 | */ | 203 | */ |
184 | val = 2; | 204 | val = 2; |
185 | break; | 205 | break; |
206 | case PSCI_0_2_FN_CPU_SUSPEND: | ||
207 | case PSCI_0_2_FN64_CPU_SUSPEND: | ||
208 | val = kvm_psci_vcpu_suspend(vcpu); | ||
209 | break; | ||
186 | case PSCI_0_2_FN_CPU_OFF: | 210 | case PSCI_0_2_FN_CPU_OFF: |
187 | kvm_psci_vcpu_off(vcpu); | 211 | kvm_psci_vcpu_off(vcpu); |
188 | val = PSCI_RET_SUCCESS; | 212 | val = PSCI_RET_SUCCESS; |
@@ -235,10 +259,6 @@ static int kvm_psci_0_2_call(struct kvm_vcpu *vcpu) | |||
235 | val = PSCI_RET_INTERNAL_FAILURE; | 259 | val = PSCI_RET_INTERNAL_FAILURE; |
236 | ret = 0; | 260 | ret = 0; |
237 | break; | 261 | break; |
238 | case PSCI_0_2_FN_CPU_SUSPEND: | ||
239 | case PSCI_0_2_FN64_CPU_SUSPEND: | ||
240 | val = PSCI_RET_NOT_SUPPORTED; | ||
241 | break; | ||
242 | default: | 262 | default: |
243 | return -EINVAL; | 263 | return -EINVAL; |
244 | } | 264 | } |