diff options
-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 | } |