2019-06-03 13:44:50 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2013-01-21 07:28:13 +08:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2012 - ARM Ltd
|
|
|
|
* Author: Marc Zyngier <marc.zyngier@arm.com>
|
|
|
|
*/
|
|
|
|
|
2018-02-07 01:56:12 +08:00
|
|
|
#include <linux/arm-smccc.h>
|
2014-10-16 23:00:18 +08:00
|
|
|
#include <linux/preempt.h>
|
2013-01-21 07:28:13 +08:00
|
|
|
#include <linux/kvm_host.h>
|
2018-01-22 00:42:56 +08:00
|
|
|
#include <linux/uaccess.h>
|
2013-01-21 07:28:13 +08:00
|
|
|
#include <linux/wait.h>
|
|
|
|
|
2013-10-19 01:19:03 +08:00
|
|
|
#include <asm/cputype.h>
|
2013-01-21 07:28:13 +08:00
|
|
|
#include <asm/kvm_emulate.h>
|
|
|
|
|
2018-02-07 01:56:08 +08:00
|
|
|
#include <kvm/arm_psci.h>
|
2019-10-21 23:28:15 +08:00
|
|
|
#include <kvm/arm_hypercalls.h>
|
2018-02-07 01:56:08 +08:00
|
|
|
|
2013-01-21 07:28:13 +08:00
|
|
|
/*
|
|
|
|
* This is an implementation of the Power State Coordination Interface
|
|
|
|
* as described in ARM document number ARM DEN 0022A.
|
|
|
|
*/
|
|
|
|
|
2014-04-29 13:54:21 +08:00
|
|
|
#define AFFINITY_MASK(level) ~((0x1UL << ((level) * MPIDR_LEVEL_BITS)) - 1)
|
|
|
|
|
|
|
|
static unsigned long psci_affinity_mask(unsigned long affinity_level)
|
|
|
|
{
|
|
|
|
if (affinity_level <= 3)
|
|
|
|
return MPIDR_HWID_BITMASK & AFFINITY_MASK(affinity_level);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-04-29 13:54:24 +08:00
|
|
|
static unsigned long kvm_psci_vcpu_suspend(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* NOTE: For simplicity, we make VCPU suspend emulation to be
|
|
|
|
* same-as WFI (Wait-for-interrupt) emulation.
|
|
|
|
*
|
|
|
|
* This means for KVM the wakeup events are interrupts and
|
|
|
|
* this is consistent with intended use of StateID as described
|
|
|
|
* in section 5.4.1 of PSCI v0.2 specification (ARM DEN 0022A).
|
|
|
|
*
|
|
|
|
* Further, we also treat power-down request to be same as
|
|
|
|
* stand-by request as-per section 5.4.2 clause 3 of PSCI v0.2
|
|
|
|
* specification (ARM DEN 0022A). This means all suspend states
|
|
|
|
* for KVM will preserve the register state.
|
|
|
|
*/
|
2022-02-17 18:12:42 +08:00
|
|
|
kvm_vcpu_wfi(vcpu);
|
2014-04-29 13:54:24 +08:00
|
|
|
|
|
|
|
return PSCI_RET_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2021-08-19 04:21:32 +08:00
|
|
|
static inline bool kvm_psci_valid_affinity(struct kvm_vcpu *vcpu,
|
|
|
|
unsigned long affinity)
|
|
|
|
{
|
|
|
|
return !(affinity & ~MPIDR_HWID_BITMASK);
|
|
|
|
}
|
|
|
|
|
2013-01-21 07:28:13 +08:00
|
|
|
static unsigned long kvm_psci_vcpu_on(struct kvm_vcpu *source_vcpu)
|
|
|
|
{
|
2018-12-20 19:36:07 +08:00
|
|
|
struct vcpu_reset_state *reset_state;
|
2013-01-21 07:28:13 +08:00
|
|
|
struct kvm *kvm = source_vcpu->kvm;
|
2014-06-02 21:37:13 +08:00
|
|
|
struct kvm_vcpu *vcpu = NULL;
|
2013-01-21 07:28:13 +08:00
|
|
|
unsigned long cpu_id;
|
|
|
|
|
2021-08-19 04:21:32 +08:00
|
|
|
cpu_id = smccc_get_arg1(source_vcpu);
|
|
|
|
if (!kvm_psci_valid_affinity(source_vcpu, cpu_id))
|
|
|
|
return PSCI_RET_INVALID_PARAMS;
|
2013-01-21 07:28:13 +08:00
|
|
|
|
2014-06-02 21:37:13 +08:00
|
|
|
vcpu = kvm_mpidr_to_vcpu(kvm, cpu_id);
|
2013-10-19 01:19:03 +08:00
|
|
|
|
2013-11-20 09:43:19 +08:00
|
|
|
/*
|
|
|
|
* Make sure the caller requested a valid CPU and that the CPU is
|
|
|
|
* turned off.
|
|
|
|
*/
|
2014-04-29 13:54:23 +08:00
|
|
|
if (!vcpu)
|
2014-04-29 13:54:16 +08:00
|
|
|
return PSCI_RET_INVALID_PARAMS;
|
2022-05-04 11:24:37 +08:00
|
|
|
if (!kvm_arm_vcpu_stopped(vcpu)) {
|
2022-02-08 09:27:05 +08:00
|
|
|
if (kvm_psci_version(source_vcpu) != KVM_ARM_PSCI_0_1)
|
2014-04-29 13:54:23 +08:00
|
|
|
return PSCI_RET_ALREADY_ON;
|
|
|
|
else
|
|
|
|
return PSCI_RET_INVALID_PARAMS;
|
|
|
|
}
|
2013-01-21 07:28:13 +08:00
|
|
|
|
2018-12-20 19:36:07 +08:00
|
|
|
reset_state = &vcpu->arch.reset_state;
|
2013-01-21 07:28:13 +08:00
|
|
|
|
2018-12-20 19:36:07 +08:00
|
|
|
reset_state->pc = smccc_get_arg2(source_vcpu);
|
2013-01-21 07:28:13 +08:00
|
|
|
|
2013-11-05 22:12:15 +08:00
|
|
|
/* Propagate caller endianness */
|
2018-12-20 19:36:07 +08:00
|
|
|
reset_state->be = kvm_vcpu_is_be(source_vcpu);
|
2013-11-05 22:12:15 +08:00
|
|
|
|
2014-04-29 13:54:23 +08:00
|
|
|
/*
|
|
|
|
* NOTE: We always update r0 (or x0) because for PSCI v0.1
|
2020-04-01 22:03:10 +08:00
|
|
|
* the general purpose registers are undefined upon CPU_ON.
|
2014-04-29 13:54:23 +08:00
|
|
|
*/
|
2018-12-20 19:36:07 +08:00
|
|
|
reset_state->r0 = smccc_get_arg3(source_vcpu);
|
|
|
|
|
|
|
|
WRITE_ONCE(reset_state->reset, true);
|
|
|
|
kvm_make_request(KVM_REQ_VCPU_RESET, vcpu);
|
2013-01-21 07:28:13 +08:00
|
|
|
|
2018-12-20 19:36:07 +08:00
|
|
|
/*
|
2022-05-04 11:24:37 +08:00
|
|
|
* Make sure the reset request is observed if the RUNNABLE mp_state is
|
|
|
|
* observed.
|
2018-12-20 19:36:07 +08:00
|
|
|
*/
|
|
|
|
smp_wmb();
|
|
|
|
|
2022-05-04 11:24:37 +08:00
|
|
|
vcpu->arch.mp_state.mp_state = KVM_MP_STATE_RUNNABLE;
|
2018-12-20 19:36:07 +08:00
|
|
|
kvm_vcpu_wake_up(vcpu);
|
2013-01-21 07:28:13 +08:00
|
|
|
|
2014-04-29 13:54:16 +08:00
|
|
|
return PSCI_RET_SUCCESS;
|
2013-01-21 07:28:13 +08:00
|
|
|
}
|
|
|
|
|
2014-04-29 13:54:21 +08:00
|
|
|
static unsigned long kvm_psci_vcpu_affinity_info(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
2021-11-17 00:04:02 +08:00
|
|
|
int matching_cpus = 0;
|
|
|
|
unsigned long i, mpidr;
|
2014-04-29 13:54:21 +08:00
|
|
|
unsigned long target_affinity;
|
|
|
|
unsigned long target_affinity_mask;
|
|
|
|
unsigned long lowest_affinity_level;
|
|
|
|
struct kvm *kvm = vcpu->kvm;
|
|
|
|
struct kvm_vcpu *tmp;
|
|
|
|
|
2018-02-07 01:56:10 +08:00
|
|
|
target_affinity = smccc_get_arg1(vcpu);
|
|
|
|
lowest_affinity_level = smccc_get_arg2(vcpu);
|
2014-04-29 13:54:21 +08:00
|
|
|
|
2021-08-19 04:21:32 +08:00
|
|
|
if (!kvm_psci_valid_affinity(vcpu, target_affinity))
|
|
|
|
return PSCI_RET_INVALID_PARAMS;
|
|
|
|
|
2014-04-29 13:54:21 +08:00
|
|
|
/* Determine target affinity mask */
|
|
|
|
target_affinity_mask = psci_affinity_mask(lowest_affinity_level);
|
|
|
|
if (!target_affinity_mask)
|
|
|
|
return PSCI_RET_INVALID_PARAMS;
|
|
|
|
|
|
|
|
/* Ignore other bits of target affinity */
|
|
|
|
target_affinity &= target_affinity_mask;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If one or more VCPU matching target affinity are running
|
|
|
|
* then ON else OFF
|
|
|
|
*/
|
|
|
|
kvm_for_each_vcpu(i, tmp, kvm) {
|
2014-06-02 21:37:13 +08:00
|
|
|
mpidr = kvm_vcpu_get_mpidr_aff(tmp);
|
2015-09-04 23:06:24 +08:00
|
|
|
if ((mpidr & target_affinity_mask) == target_affinity) {
|
|
|
|
matching_cpus++;
|
2022-05-04 11:24:37 +08:00
|
|
|
if (!kvm_arm_vcpu_stopped(tmp))
|
2015-09-04 23:06:24 +08:00
|
|
|
return PSCI_0_2_AFFINITY_LEVEL_ON;
|
2014-04-29 13:54:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-04 23:06:24 +08:00
|
|
|
if (!matching_cpus)
|
|
|
|
return PSCI_RET_INVALID_PARAMS;
|
|
|
|
|
2014-04-29 13:54:21 +08:00
|
|
|
return PSCI_0_2_AFFINITY_LEVEL_OFF;
|
|
|
|
}
|
|
|
|
|
2022-02-21 23:35:24 +08:00
|
|
|
static void kvm_prepare_system_event(struct kvm_vcpu *vcpu, u32 type, u64 flags)
|
2014-04-29 13:54:20 +08:00
|
|
|
{
|
2021-11-17 00:04:02 +08:00
|
|
|
unsigned long i;
|
2014-10-16 23:00:18 +08:00
|
|
|
struct kvm_vcpu *tmp;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The KVM ABI specifies that a system event exit may call KVM_RUN
|
|
|
|
* again and may perform shutdown/reboot at a later time that when the
|
|
|
|
* actual request is made. Since we are implementing PSCI and a
|
|
|
|
* caller of PSCI reboot and shutdown expects that the system shuts
|
|
|
|
* down or reboots immediately, let's make sure that VCPUs are not run
|
|
|
|
* after this call is handled and before the VCPUs have been
|
|
|
|
* re-initialized.
|
|
|
|
*/
|
2017-06-04 20:43:56 +08:00
|
|
|
kvm_for_each_vcpu(i, tmp, vcpu->kvm)
|
2022-05-04 11:24:37 +08:00
|
|
|
tmp->arch.mp_state.mp_state = KVM_MP_STATE_STOPPED;
|
2017-06-04 20:43:58 +08:00
|
|
|
kvm_make_all_cpus_request(vcpu->kvm, KVM_REQ_SLEEP);
|
2014-10-16 23:00:18 +08:00
|
|
|
|
2014-04-29 13:54:20 +08:00
|
|
|
memset(&vcpu->run->system_event, 0, sizeof(vcpu->run->system_event));
|
|
|
|
vcpu->run->system_event.type = type;
|
KVM: fix bad user ABI for KVM_EXIT_SYSTEM_EVENT
When KVM_EXIT_SYSTEM_EVENT was introduced, it included a flags
member that at the time was unused. Unfortunately this extensibility
mechanism has several issues:
- x86 is not writing the member, so it would not be possible to use it
on x86 except for new events
- the member is not aligned to 64 bits, so the definition of the
uAPI struct is incorrect for 32- on 64-bit userspace. This is a
problem for RISC-V, which supports CONFIG_KVM_COMPAT, but fortunately
usage of flags was only introduced in 5.18.
Since padding has to be introduced, place a new field in there
that tells if the flags field is valid. To allow further extensibility,
in fact, change flags to an array of 16 values, and store how many
of the values are valid. The availability of the new ndata field
is tied to a system capability; all architectures are changed to
fill in the field.
To avoid breaking compilation of userspace that was using the flags
field, provide a userspace-only union to overlap flags with data[0].
The new field is placed at the same offset for both 32- and 64-bit
userspace.
Cc: Will Deacon <will@kernel.org>
Cc: Marc Zyngier <maz@kernel.org>
Cc: Peter Gonda <pgonda@google.com>
Cc: Sean Christopherson <seanjc@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Reported-by: kernel test robot <lkp@intel.com>
Message-Id: <20220422103013.34832-1-pbonzini@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2022-04-22 18:30:13 +08:00
|
|
|
vcpu->run->system_event.ndata = 1;
|
|
|
|
vcpu->run->system_event.data[0] = flags;
|
2014-04-29 13:54:20 +08:00
|
|
|
vcpu->run->exit_reason = KVM_EXIT_SYSTEM_EVENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void kvm_psci_system_off(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
2022-02-21 23:35:24 +08:00
|
|
|
kvm_prepare_system_event(vcpu, KVM_SYSTEM_EVENT_SHUTDOWN, 0);
|
2014-04-29 13:54:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void kvm_psci_system_reset(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
2022-02-21 23:35:24 +08:00
|
|
|
kvm_prepare_system_event(vcpu, KVM_SYSTEM_EVENT_RESET, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void kvm_psci_system_reset2(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
kvm_prepare_system_event(vcpu, KVM_SYSTEM_EVENT_RESET,
|
|
|
|
KVM_SYSTEM_EVENT_RESET_FLAG_PSCI_RESET2);
|
2014-04-29 13:54:20 +08:00
|
|
|
}
|
|
|
|
|
2022-05-04 11:24:41 +08:00
|
|
|
static void kvm_psci_system_suspend(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct kvm_run *run = vcpu->run;
|
|
|
|
|
|
|
|
memset(&run->system_event, 0, sizeof(vcpu->run->system_event));
|
|
|
|
run->system_event.type = KVM_SYSTEM_EVENT_SUSPEND;
|
|
|
|
run->exit_reason = KVM_EXIT_SYSTEM_EVENT;
|
|
|
|
}
|
|
|
|
|
2020-04-01 19:25:05 +08:00
|
|
|
static void kvm_psci_narrow_to_32bit(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Zero the input registers' upper 32 bits. They will be fully
|
|
|
|
* zeroed on exit, so we're fine changing them in place.
|
|
|
|
*/
|
|
|
|
for (i = 1; i < 4; i++)
|
|
|
|
vcpu_set_reg(vcpu, i, lower_32_bits(vcpu_get_reg(vcpu, i)));
|
|
|
|
}
|
|
|
|
|
2020-04-01 19:38:49 +08:00
|
|
|
static unsigned long kvm_psci_check_allowed_function(struct kvm_vcpu *vcpu, u32 fn)
|
|
|
|
{
|
2022-03-23 02:35:36 +08:00
|
|
|
/*
|
|
|
|
* Prevent 32 bit guests from calling 64 bit PSCI functions.
|
|
|
|
*/
|
|
|
|
if ((fn & PSCI_0_2_64BIT) && vcpu_mode_is_32bit(vcpu))
|
|
|
|
return PSCI_RET_NOT_SUPPORTED;
|
2020-04-01 19:38:49 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-04-29 13:54:18 +08:00
|
|
|
static int kvm_psci_0_2_call(struct kvm_vcpu *vcpu)
|
2014-04-29 13:54:16 +08:00
|
|
|
{
|
2017-04-18 23:59:58 +08:00
|
|
|
struct kvm *kvm = vcpu->kvm;
|
2018-02-07 01:56:10 +08:00
|
|
|
u32 psci_fn = smccc_get_function(vcpu);
|
2014-04-29 13:54:16 +08:00
|
|
|
unsigned long val;
|
2017-04-18 23:59:58 +08:00
|
|
|
int ret = 1;
|
2014-04-29 13:54:16 +08:00
|
|
|
|
|
|
|
switch (psci_fn) {
|
|
|
|
case PSCI_0_2_FN_PSCI_VERSION:
|
|
|
|
/*
|
|
|
|
* Bits[31:16] = Major Version = 0
|
|
|
|
* Bits[15:0] = Minor Version = 2
|
|
|
|
*/
|
2018-02-07 01:56:09 +08:00
|
|
|
val = KVM_ARM_PSCI_0_2;
|
2014-04-29 13:54:16 +08:00
|
|
|
break;
|
2014-04-29 13:54:24 +08:00
|
|
|
case PSCI_0_2_FN_CPU_SUSPEND:
|
|
|
|
case PSCI_0_2_FN64_CPU_SUSPEND:
|
|
|
|
val = kvm_psci_vcpu_suspend(vcpu);
|
|
|
|
break;
|
2014-04-29 13:54:16 +08:00
|
|
|
case PSCI_0_2_FN_CPU_OFF:
|
2022-05-04 11:24:36 +08:00
|
|
|
kvm_arm_vcpu_power_off(vcpu);
|
2014-04-29 13:54:16 +08:00
|
|
|
val = PSCI_RET_SUCCESS;
|
|
|
|
break;
|
|
|
|
case PSCI_0_2_FN_CPU_ON:
|
2020-04-01 19:25:05 +08:00
|
|
|
kvm_psci_narrow_to_32bit(vcpu);
|
|
|
|
fallthrough;
|
2014-04-29 13:54:16 +08:00
|
|
|
case PSCI_0_2_FN64_CPU_ON:
|
2017-04-18 23:59:58 +08:00
|
|
|
mutex_lock(&kvm->lock);
|
2014-04-29 13:54:16 +08:00
|
|
|
val = kvm_psci_vcpu_on(vcpu);
|
2017-04-18 23:59:58 +08:00
|
|
|
mutex_unlock(&kvm->lock);
|
2014-04-29 13:54:16 +08:00
|
|
|
break;
|
2014-04-29 13:54:21 +08:00
|
|
|
case PSCI_0_2_FN_AFFINITY_INFO:
|
2020-04-01 19:25:05 +08:00
|
|
|
kvm_psci_narrow_to_32bit(vcpu);
|
|
|
|
fallthrough;
|
2014-04-29 13:54:21 +08:00
|
|
|
case PSCI_0_2_FN64_AFFINITY_INFO:
|
|
|
|
val = kvm_psci_vcpu_affinity_info(vcpu);
|
|
|
|
break;
|
2014-04-29 13:54:22 +08:00
|
|
|
case PSCI_0_2_FN_MIGRATE_INFO_TYPE:
|
|
|
|
/*
|
|
|
|
* Trusted OS is MP hence does not require migration
|
|
|
|
* or
|
|
|
|
* Trusted OS is not present
|
|
|
|
*/
|
|
|
|
val = PSCI_0_2_TOS_MP;
|
|
|
|
break;
|
2014-04-29 13:54:20 +08:00
|
|
|
case PSCI_0_2_FN_SYSTEM_OFF:
|
|
|
|
kvm_psci_system_off(vcpu);
|
|
|
|
/*
|
2020-04-01 22:03:10 +08:00
|
|
|
* We shouldn't be going back to guest VCPU after
|
2014-04-29 13:54:20 +08:00
|
|
|
* receiving SYSTEM_OFF request.
|
|
|
|
*
|
2020-04-01 22:03:10 +08:00
|
|
|
* If user space accidentally/deliberately resumes
|
2014-04-29 13:54:20 +08:00
|
|
|
* guest VCPU after SYSTEM_OFF request then guest
|
|
|
|
* VCPU should see internal failure from PSCI return
|
|
|
|
* value. To achieve this, we preload r0 (or x0) with
|
|
|
|
* PSCI return value INTERNAL_FAILURE.
|
|
|
|
*/
|
|
|
|
val = PSCI_RET_INTERNAL_FAILURE;
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
case PSCI_0_2_FN_SYSTEM_RESET:
|
|
|
|
kvm_psci_system_reset(vcpu);
|
|
|
|
/*
|
|
|
|
* Same reason as SYSTEM_OFF for preloading r0 (or x0)
|
|
|
|
* with PSCI return value INTERNAL_FAILURE.
|
|
|
|
*/
|
|
|
|
val = PSCI_RET_INTERNAL_FAILURE;
|
|
|
|
ret = 0;
|
|
|
|
break;
|
2014-04-29 13:54:16 +08:00
|
|
|
default:
|
2015-06-10 22:19:24 +08:00
|
|
|
val = PSCI_RET_NOT_SUPPORTED;
|
|
|
|
break;
|
2014-04-29 13:54:16 +08:00
|
|
|
}
|
|
|
|
|
2018-02-07 01:56:10 +08:00
|
|
|
smccc_set_retval(vcpu, val, 0, 0, 0);
|
2014-04-29 13:54:20 +08:00
|
|
|
return ret;
|
2014-04-29 13:54:16 +08:00
|
|
|
}
|
|
|
|
|
2022-02-21 23:35:22 +08:00
|
|
|
static int kvm_psci_1_x_call(struct kvm_vcpu *vcpu, u32 minor)
|
2018-02-07 01:56:11 +08:00
|
|
|
{
|
2022-05-04 11:24:35 +08:00
|
|
|
unsigned long val = PSCI_RET_NOT_SUPPORTED;
|
2018-02-07 01:56:11 +08:00
|
|
|
u32 psci_fn = smccc_get_function(vcpu);
|
2022-05-04 11:24:41 +08:00
|
|
|
struct kvm *kvm = vcpu->kvm;
|
2022-02-21 23:35:23 +08:00
|
|
|
u32 arg;
|
2018-02-07 01:56:11 +08:00
|
|
|
int ret = 1;
|
|
|
|
|
|
|
|
switch(psci_fn) {
|
|
|
|
case PSCI_0_2_FN_PSCI_VERSION:
|
2022-02-21 23:35:22 +08:00
|
|
|
val = minor == 0 ? KVM_ARM_PSCI_1_0 : KVM_ARM_PSCI_1_1;
|
2018-02-07 01:56:11 +08:00
|
|
|
break;
|
|
|
|
case PSCI_1_0_FN_PSCI_FEATURES:
|
2022-02-21 23:35:23 +08:00
|
|
|
arg = smccc_get_arg1(vcpu);
|
|
|
|
val = kvm_psci_check_allowed_function(vcpu, arg);
|
2020-04-01 19:38:49 +08:00
|
|
|
if (val)
|
|
|
|
break;
|
|
|
|
|
2022-05-04 11:24:35 +08:00
|
|
|
val = PSCI_RET_NOT_SUPPORTED;
|
|
|
|
|
2022-02-21 23:35:23 +08:00
|
|
|
switch(arg) {
|
2018-02-07 01:56:11 +08:00
|
|
|
case PSCI_0_2_FN_PSCI_VERSION:
|
|
|
|
case PSCI_0_2_FN_CPU_SUSPEND:
|
|
|
|
case PSCI_0_2_FN64_CPU_SUSPEND:
|
|
|
|
case PSCI_0_2_FN_CPU_OFF:
|
|
|
|
case PSCI_0_2_FN_CPU_ON:
|
|
|
|
case PSCI_0_2_FN64_CPU_ON:
|
|
|
|
case PSCI_0_2_FN_AFFINITY_INFO:
|
|
|
|
case PSCI_0_2_FN64_AFFINITY_INFO:
|
|
|
|
case PSCI_0_2_FN_MIGRATE_INFO_TYPE:
|
|
|
|
case PSCI_0_2_FN_SYSTEM_OFF:
|
|
|
|
case PSCI_0_2_FN_SYSTEM_RESET:
|
|
|
|
case PSCI_1_0_FN_PSCI_FEATURES:
|
2018-02-07 01:56:12 +08:00
|
|
|
case ARM_SMCCC_VERSION_FUNC_ID:
|
2018-02-07 01:56:11 +08:00
|
|
|
val = 0;
|
|
|
|
break;
|
2022-05-04 11:24:41 +08:00
|
|
|
case PSCI_1_0_FN_SYSTEM_SUSPEND:
|
|
|
|
case PSCI_1_0_FN64_SYSTEM_SUSPEND:
|
|
|
|
if (test_bit(KVM_ARCH_FLAG_SYSTEM_SUSPEND_ENABLED, &kvm->arch.flags))
|
|
|
|
val = 0;
|
|
|
|
break;
|
2022-02-21 23:35:23 +08:00
|
|
|
case PSCI_1_1_FN_SYSTEM_RESET2:
|
|
|
|
case PSCI_1_1_FN64_SYSTEM_RESET2:
|
2022-05-04 11:24:35 +08:00
|
|
|
if (minor >= 1)
|
2022-02-21 23:35:23 +08:00
|
|
|
val = 0;
|
2018-02-07 01:56:11 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2022-05-04 11:24:41 +08:00
|
|
|
case PSCI_1_0_FN_SYSTEM_SUSPEND:
|
|
|
|
kvm_psci_narrow_to_32bit(vcpu);
|
|
|
|
fallthrough;
|
|
|
|
case PSCI_1_0_FN64_SYSTEM_SUSPEND:
|
|
|
|
/*
|
|
|
|
* Return directly to userspace without changing the vCPU's
|
|
|
|
* registers. Userspace depends on reading the SMCCC parameters
|
|
|
|
* to implement SYSTEM_SUSPEND.
|
|
|
|
*/
|
|
|
|
if (test_bit(KVM_ARCH_FLAG_SYSTEM_SUSPEND_ENABLED, &kvm->arch.flags)) {
|
|
|
|
kvm_psci_system_suspend(vcpu);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
break;
|
2022-02-21 23:35:23 +08:00
|
|
|
case PSCI_1_1_FN_SYSTEM_RESET2:
|
|
|
|
kvm_psci_narrow_to_32bit(vcpu);
|
|
|
|
fallthrough;
|
|
|
|
case PSCI_1_1_FN64_SYSTEM_RESET2:
|
|
|
|
if (minor >= 1) {
|
|
|
|
arg = smccc_get_arg1(vcpu);
|
|
|
|
|
2022-03-10 02:13:08 +08:00
|
|
|
if (arg <= PSCI_1_1_RESET_TYPE_SYSTEM_WARM_RESET ||
|
|
|
|
arg >= PSCI_1_1_RESET_TYPE_VENDOR_START) {
|
2022-02-21 23:35:24 +08:00
|
|
|
kvm_psci_system_reset2(vcpu);
|
2022-03-10 02:13:08 +08:00
|
|
|
vcpu_set_reg(vcpu, 0, PSCI_RET_INTERNAL_FAILURE);
|
|
|
|
return 0;
|
2022-02-21 23:35:23 +08:00
|
|
|
}
|
2022-03-10 02:13:08 +08:00
|
|
|
|
|
|
|
val = PSCI_RET_INVALID_PARAMS;
|
2022-02-21 23:35:23 +08:00
|
|
|
break;
|
2022-02-23 17:27:50 +08:00
|
|
|
}
|
2022-05-04 11:24:35 +08:00
|
|
|
break;
|
2018-02-07 01:56:11 +08:00
|
|
|
default:
|
|
|
|
return kvm_psci_0_2_call(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
smccc_set_retval(vcpu, val, 0, 0, 0);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-04-29 13:54:18 +08:00
|
|
|
static int kvm_psci_0_1_call(struct kvm_vcpu *vcpu)
|
2013-01-21 07:28:13 +08:00
|
|
|
{
|
2017-04-18 23:59:58 +08:00
|
|
|
struct kvm *kvm = vcpu->kvm;
|
2018-02-07 01:56:10 +08:00
|
|
|
u32 psci_fn = smccc_get_function(vcpu);
|
2013-01-21 07:28:13 +08:00
|
|
|
unsigned long val;
|
|
|
|
|
|
|
|
switch (psci_fn) {
|
|
|
|
case KVM_PSCI_FN_CPU_OFF:
|
2022-05-04 11:24:36 +08:00
|
|
|
kvm_arm_vcpu_power_off(vcpu);
|
2014-04-29 13:54:16 +08:00
|
|
|
val = PSCI_RET_SUCCESS;
|
2013-01-21 07:28:13 +08:00
|
|
|
break;
|
|
|
|
case KVM_PSCI_FN_CPU_ON:
|
2017-04-18 23:59:58 +08:00
|
|
|
mutex_lock(&kvm->lock);
|
2013-01-21 07:28:13 +08:00
|
|
|
val = kvm_psci_vcpu_on(vcpu);
|
2017-04-18 23:59:58 +08:00
|
|
|
mutex_unlock(&kvm->lock);
|
2013-01-21 07:28:13 +08:00
|
|
|
break;
|
2015-06-10 22:19:24 +08:00
|
|
|
default:
|
2014-04-29 13:54:16 +08:00
|
|
|
val = PSCI_RET_NOT_SUPPORTED;
|
2013-01-21 07:28:13 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-02-07 01:56:10 +08:00
|
|
|
smccc_set_retval(vcpu, val, 0, 0, 0);
|
2014-04-29 13:54:18 +08:00
|
|
|
return 1;
|
2013-01-21 07:28:13 +08:00
|
|
|
}
|
2014-04-29 13:54:16 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* kvm_psci_call - handle PSCI call if r0 value is in range
|
|
|
|
* @vcpu: Pointer to the VCPU struct
|
|
|
|
*
|
|
|
|
* Handle PSCI calls from guests through traps from HVC instructions.
|
2014-04-29 13:54:18 +08:00
|
|
|
* The calling convention is similar to SMC calls to the secure world
|
|
|
|
* where the function number is placed in r0.
|
|
|
|
*
|
|
|
|
* This function returns: > 0 (success), 0 (success but exit to user
|
|
|
|
* space), and < 0 (errors)
|
|
|
|
*
|
|
|
|
* Errors:
|
|
|
|
* -EINVAL: Unrecognized PSCI function
|
2014-04-29 13:54:16 +08:00
|
|
|
*/
|
2019-10-21 23:28:15 +08:00
|
|
|
int kvm_psci_call(struct kvm_vcpu *vcpu)
|
2014-04-29 13:54:16 +08:00
|
|
|
{
|
2022-03-23 02:35:37 +08:00
|
|
|
u32 psci_fn = smccc_get_function(vcpu);
|
|
|
|
unsigned long val;
|
|
|
|
|
|
|
|
val = kvm_psci_check_allowed_function(vcpu, psci_fn);
|
|
|
|
if (val) {
|
|
|
|
smccc_set_retval(vcpu, val, 0, 0, 0);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2022-02-08 09:27:05 +08:00
|
|
|
switch (kvm_psci_version(vcpu)) {
|
2022-02-21 23:35:22 +08:00
|
|
|
case KVM_ARM_PSCI_1_1:
|
|
|
|
return kvm_psci_1_x_call(vcpu, 1);
|
2018-02-07 01:56:11 +08:00
|
|
|
case KVM_ARM_PSCI_1_0:
|
2022-02-21 23:35:22 +08:00
|
|
|
return kvm_psci_1_x_call(vcpu, 0);
|
2014-04-29 13:54:16 +08:00
|
|
|
case KVM_ARM_PSCI_0_2:
|
|
|
|
return kvm_psci_0_2_call(vcpu);
|
|
|
|
case KVM_ARM_PSCI_0_1:
|
|
|
|
return kvm_psci_0_1_call(vcpu);
|
|
|
|
default:
|
2014-04-29 13:54:18 +08:00
|
|
|
return -EINVAL;
|
2022-02-23 17:27:50 +08:00
|
|
|
}
|
2014-04-29 13:54:16 +08:00
|
|
|
}
|