hvf: arm: Implement PSCI handling
We need to handle PSCI calls. Most of the TCG code works for us, but we can simplify it to only handle aa64 mode and we need to handle SUSPEND differently. This patch takes the TCG code as template and duplicates it in HVF. To tell the guest that we support PSCI 0.2 now, update the check in arm_cpu_initfn() as well. Signed-off-by: Alexander Graf <agraf@csgraf.de> Reviewed-by: Sergio Lopez <slp@redhat.com> Reviewed-by: Peter Maydell <peter.maydell@linaro.org> Message-id: 20210916155404.86958-8-agraf@csgraf.de Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
parent
585df85efe
commit
2c9c0bf9d1
@ -1113,8 +1113,8 @@ static void arm_cpu_initfn(Object *obj)
|
|||||||
cpu->psci_version = 1; /* By default assume PSCI v0.1 */
|
cpu->psci_version = 1; /* By default assume PSCI v0.1 */
|
||||||
cpu->kvm_target = QEMU_KVM_ARM_TARGET_NONE;
|
cpu->kvm_target = QEMU_KVM_ARM_TARGET_NONE;
|
||||||
|
|
||||||
if (tcg_enabled()) {
|
if (tcg_enabled() || hvf_enabled()) {
|
||||||
cpu->psci_version = 2; /* TCG implements PSCI 0.2 */
|
cpu->psci_version = 2; /* TCG and HVF implement PSCI 0.2 */
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -25,6 +25,7 @@
|
|||||||
#include "hw/irq.h"
|
#include "hw/irq.h"
|
||||||
#include "qemu/main-loop.h"
|
#include "qemu/main-loop.h"
|
||||||
#include "sysemu/cpus.h"
|
#include "sysemu/cpus.h"
|
||||||
|
#include "arm-powerctl.h"
|
||||||
#include "target/arm/cpu.h"
|
#include "target/arm/cpu.h"
|
||||||
#include "target/arm/internals.h"
|
#include "target/arm/internals.h"
|
||||||
#include "trace/trace-target_arm_hvf.h"
|
#include "trace/trace-target_arm_hvf.h"
|
||||||
@ -48,6 +49,8 @@
|
|||||||
#define TMR_CTL_IMASK (1 << 1)
|
#define TMR_CTL_IMASK (1 << 1)
|
||||||
#define TMR_CTL_ISTATUS (1 << 2)
|
#define TMR_CTL_ISTATUS (1 << 2)
|
||||||
|
|
||||||
|
static void hvf_wfi(CPUState *cpu);
|
||||||
|
|
||||||
typedef struct HVFVTimer {
|
typedef struct HVFVTimer {
|
||||||
/* Vtimer value during migration and paused state */
|
/* Vtimer value during migration and paused state */
|
||||||
uint64_t vtimer_val;
|
uint64_t vtimer_val;
|
||||||
@ -603,6 +606,117 @@ static void hvf_raise_exception(CPUState *cpu, uint32_t excp,
|
|||||||
arm_cpu_do_interrupt(cpu);
|
arm_cpu_do_interrupt(cpu);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void hvf_psci_cpu_off(ARMCPU *arm_cpu)
|
||||||
|
{
|
||||||
|
int32_t ret = arm_set_cpu_off(arm_cpu->mp_affinity);
|
||||||
|
assert(ret == QEMU_ARM_POWERCTL_RET_SUCCESS);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Handle a PSCI call.
|
||||||
|
*
|
||||||
|
* Returns 0 on success
|
||||||
|
* -1 when the PSCI call is unknown,
|
||||||
|
*/
|
||||||
|
static bool hvf_handle_psci_call(CPUState *cpu)
|
||||||
|
{
|
||||||
|
ARMCPU *arm_cpu = ARM_CPU(cpu);
|
||||||
|
CPUARMState *env = &arm_cpu->env;
|
||||||
|
uint64_t param[4] = {
|
||||||
|
env->xregs[0],
|
||||||
|
env->xregs[1],
|
||||||
|
env->xregs[2],
|
||||||
|
env->xregs[3]
|
||||||
|
};
|
||||||
|
uint64_t context_id, mpidr;
|
||||||
|
bool target_aarch64 = true;
|
||||||
|
CPUState *target_cpu_state;
|
||||||
|
ARMCPU *target_cpu;
|
||||||
|
target_ulong entry;
|
||||||
|
int target_el = 1;
|
||||||
|
int32_t ret = 0;
|
||||||
|
|
||||||
|
trace_hvf_psci_call(param[0], param[1], param[2], param[3],
|
||||||
|
arm_cpu->mp_affinity);
|
||||||
|
|
||||||
|
switch (param[0]) {
|
||||||
|
case QEMU_PSCI_0_2_FN_PSCI_VERSION:
|
||||||
|
ret = QEMU_PSCI_0_2_RET_VERSION_0_2;
|
||||||
|
break;
|
||||||
|
case QEMU_PSCI_0_2_FN_MIGRATE_INFO_TYPE:
|
||||||
|
ret = QEMU_PSCI_0_2_RET_TOS_MIGRATION_NOT_REQUIRED; /* No trusted OS */
|
||||||
|
break;
|
||||||
|
case QEMU_PSCI_0_2_FN_AFFINITY_INFO:
|
||||||
|
case QEMU_PSCI_0_2_FN64_AFFINITY_INFO:
|
||||||
|
mpidr = param[1];
|
||||||
|
|
||||||
|
switch (param[2]) {
|
||||||
|
case 0:
|
||||||
|
target_cpu_state = arm_get_cpu_by_id(mpidr);
|
||||||
|
if (!target_cpu_state) {
|
||||||
|
ret = QEMU_PSCI_RET_INVALID_PARAMS;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
target_cpu = ARM_CPU(target_cpu_state);
|
||||||
|
|
||||||
|
ret = target_cpu->power_state;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
/* Everything above affinity level 0 is always on. */
|
||||||
|
ret = 0;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case QEMU_PSCI_0_2_FN_SYSTEM_RESET:
|
||||||
|
qemu_system_reset_request(SHUTDOWN_CAUSE_GUEST_RESET);
|
||||||
|
/*
|
||||||
|
* QEMU reset and shutdown are async requests, but PSCI
|
||||||
|
* mandates that we never return from the reset/shutdown
|
||||||
|
* call, so power the CPU off now so it doesn't execute
|
||||||
|
* anything further.
|
||||||
|
*/
|
||||||
|
hvf_psci_cpu_off(arm_cpu);
|
||||||
|
break;
|
||||||
|
case QEMU_PSCI_0_2_FN_SYSTEM_OFF:
|
||||||
|
qemu_system_shutdown_request(SHUTDOWN_CAUSE_GUEST_SHUTDOWN);
|
||||||
|
hvf_psci_cpu_off(arm_cpu);
|
||||||
|
break;
|
||||||
|
case QEMU_PSCI_0_1_FN_CPU_ON:
|
||||||
|
case QEMU_PSCI_0_2_FN_CPU_ON:
|
||||||
|
case QEMU_PSCI_0_2_FN64_CPU_ON:
|
||||||
|
mpidr = param[1];
|
||||||
|
entry = param[2];
|
||||||
|
context_id = param[3];
|
||||||
|
ret = arm_set_cpu_on(mpidr, entry, context_id,
|
||||||
|
target_el, target_aarch64);
|
||||||
|
break;
|
||||||
|
case QEMU_PSCI_0_1_FN_CPU_OFF:
|
||||||
|
case QEMU_PSCI_0_2_FN_CPU_OFF:
|
||||||
|
hvf_psci_cpu_off(arm_cpu);
|
||||||
|
break;
|
||||||
|
case QEMU_PSCI_0_1_FN_CPU_SUSPEND:
|
||||||
|
case QEMU_PSCI_0_2_FN_CPU_SUSPEND:
|
||||||
|
case QEMU_PSCI_0_2_FN64_CPU_SUSPEND:
|
||||||
|
/* Affinity levels are not supported in QEMU */
|
||||||
|
if (param[1] & 0xfffe0000) {
|
||||||
|
ret = QEMU_PSCI_RET_INVALID_PARAMS;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* Powerdown is not supported, we always go into WFI */
|
||||||
|
env->xregs[0] = 0;
|
||||||
|
hvf_wfi(cpu);
|
||||||
|
break;
|
||||||
|
case QEMU_PSCI_0_1_FN_MIGRATE:
|
||||||
|
case QEMU_PSCI_0_2_FN_MIGRATE:
|
||||||
|
ret = QEMU_PSCI_RET_NOT_SUPPORTED;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
env->xregs[0] = ret;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
static int hvf_sysreg_read(CPUState *cpu, uint32_t reg, uint32_t rt)
|
static int hvf_sysreg_read(CPUState *cpu, uint32_t reg, uint32_t rt)
|
||||||
{
|
{
|
||||||
ARMCPU *arm_cpu = ARM_CPU(cpu);
|
ARMCPU *arm_cpu = ARM_CPU(cpu);
|
||||||
@ -905,14 +1019,31 @@ int hvf_vcpu_exec(CPUState *cpu)
|
|||||||
break;
|
break;
|
||||||
case EC_AA64_HVC:
|
case EC_AA64_HVC:
|
||||||
cpu_synchronize_state(cpu);
|
cpu_synchronize_state(cpu);
|
||||||
trace_hvf_unknown_hvc(env->xregs[0]);
|
if (arm_cpu->psci_conduit == QEMU_PSCI_CONDUIT_HVC) {
|
||||||
/* SMCCC 1.3 section 5.2 says every unknown SMCCC call returns -1 */
|
if (!hvf_handle_psci_call(cpu)) {
|
||||||
env->xregs[0] = -1;
|
trace_hvf_unknown_hvc(env->xregs[0]);
|
||||||
|
/* SMCCC 1.3 section 5.2 says every unknown SMCCC call returns -1 */
|
||||||
|
env->xregs[0] = -1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
trace_hvf_unknown_hvc(env->xregs[0]);
|
||||||
|
hvf_raise_exception(cpu, EXCP_UDEF, syn_uncategorized());
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case EC_AA64_SMC:
|
case EC_AA64_SMC:
|
||||||
cpu_synchronize_state(cpu);
|
cpu_synchronize_state(cpu);
|
||||||
trace_hvf_unknown_smc(env->xregs[0]);
|
if (arm_cpu->psci_conduit == QEMU_PSCI_CONDUIT_SMC) {
|
||||||
hvf_raise_exception(cpu, EXCP_UDEF, syn_uncategorized());
|
advance_pc = true;
|
||||||
|
|
||||||
|
if (!hvf_handle_psci_call(cpu)) {
|
||||||
|
trace_hvf_unknown_smc(env->xregs[0]);
|
||||||
|
/* SMCCC 1.3 section 5.2 says every unknown SMCCC call returns -1 */
|
||||||
|
env->xregs[0] = -1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
trace_hvf_unknown_smc(env->xregs[0]);
|
||||||
|
hvf_raise_exception(cpu, EXCP_UDEF, syn_uncategorized());
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
cpu_synchronize_state(cpu);
|
cpu_synchronize_state(cpu);
|
||||||
|
@ -8,3 +8,4 @@ hvf_sysreg_write(uint32_t reg, uint32_t op0, uint32_t op1, uint32_t crn, uint32_
|
|||||||
hvf_unknown_hvc(uint64_t x0) "unknown HVC! 0x%016"PRIx64
|
hvf_unknown_hvc(uint64_t x0) "unknown HVC! 0x%016"PRIx64
|
||||||
hvf_unknown_smc(uint64_t x0) "unknown SMC! 0x%016"PRIx64
|
hvf_unknown_smc(uint64_t x0) "unknown SMC! 0x%016"PRIx64
|
||||||
hvf_exit(uint64_t syndrome, uint32_t ec, uint64_t pc) "exit: 0x%"PRIx64" [ec=0x%x pc=0x%"PRIx64"]"
|
hvf_exit(uint64_t syndrome, uint32_t ec, uint64_t pc) "exit: 0x%"PRIx64" [ec=0x%x pc=0x%"PRIx64"]"
|
||||||
|
hvf_psci_call(uint64_t x0, uint64_t x1, uint64_t x2, uint64_t x3, uint32_t cpuid) "PSCI Call x0=0x%016"PRIx64" x1=0x%016"PRIx64" x2=0x%016"PRIx64" x3=0x%016"PRIx64" cpu=0x%x"
|
||||||
|
Loading…
Reference in New Issue
Block a user