mirror of https://gitlab.com/qemu-project/qemu
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
157 lines
5.9 KiB
C
157 lines
5.9 KiB
C
/*
|
|
* QEMU ARM CP Register GCS regiters and instructions
|
|
* SPDX-License-Identifier: GPL-2.0-or-later
|
|
*/
|
|
|
|
#include "qemu/osdep.h"
|
|
#include "qemu/timer.h"
|
|
#include "exec/icount.h"
|
|
#include "hw/irq.h"
|
|
#include "cpu.h"
|
|
#include "cpu-features.h"
|
|
#include "cpregs.h"
|
|
#include "internals.h"
|
|
|
|
|
|
static CPAccessResult access_gcs(CPUARMState *env, const ARMCPRegInfo *ri,
|
|
bool isread)
|
|
{
|
|
if (arm_current_el(env) < 3
|
|
&& arm_feature(env, ARM_FEATURE_EL3)
|
|
&& !(env->cp15.scr_el3 & SCR_GCSEN)) {
|
|
return CP_ACCESS_TRAP_EL3;
|
|
}
|
|
return CP_ACCESS_OK;
|
|
}
|
|
|
|
static CPAccessResult access_gcs_el0(CPUARMState *env, const ARMCPRegInfo *ri,
|
|
bool isread)
|
|
{
|
|
if (arm_current_el(env) == 0 && !(env->cp15.gcscr_el[0] & GCSCRE0_NTR)) {
|
|
return CP_ACCESS_TRAP_EL1;
|
|
}
|
|
return access_gcs(env, ri, isread);
|
|
}
|
|
|
|
static void gcspr_write(CPUARMState *env, const ARMCPRegInfo *ri,
|
|
uint64_t value)
|
|
{
|
|
/*
|
|
* Bits [2:0] are RES0, so we might as well clear them now,
|
|
* rather than upon each usage a-la GetCurrentGCSPointer.
|
|
*/
|
|
raw_write(env, ri, value & ~7);
|
|
}
|
|
|
|
static CPAccessResult access_gcspushm(CPUARMState *env, const ARMCPRegInfo *ri,
|
|
bool isread)
|
|
{
|
|
int el = arm_current_el(env);
|
|
if (!(env->cp15.gcscr_el[el] & GCSCR_PUSHMEN)) {
|
|
return CP_ACCESS_TRAP_BIT | (el ? el : 1);
|
|
}
|
|
return CP_ACCESS_OK;
|
|
}
|
|
|
|
static CPAccessResult access_gcspushx(CPUARMState *env, const ARMCPRegInfo *ri,
|
|
bool isread)
|
|
{
|
|
/* Trap if lock taken, and enabled. */
|
|
if (!(env->pstate & PSTATE_EXLOCK)) {
|
|
int el = arm_current_el(env);
|
|
if (env->cp15.gcscr_el[el] & GCSCR_EXLOCKEN) {
|
|
return CP_ACCESS_EXLOCK;
|
|
}
|
|
}
|
|
return CP_ACCESS_OK;
|
|
}
|
|
|
|
static CPAccessResult access_gcspopcx(CPUARMState *env, const ARMCPRegInfo *ri,
|
|
bool isread)
|
|
{
|
|
/* Trap if lock not taken, and enabled. */
|
|
if (env->pstate & PSTATE_EXLOCK) {
|
|
int el = arm_current_el(env);
|
|
if (env->cp15.gcscr_el[el] & GCSCR_EXLOCKEN) {
|
|
return CP_ACCESS_EXLOCK;
|
|
}
|
|
}
|
|
return CP_ACCESS_OK;
|
|
}
|
|
|
|
static const ARMCPRegInfo gcs_reginfo[] = {
|
|
{ .name = "GCSCRE0_EL1", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 3, .opc1 = 0, .crn = 2, .crm = 5, .opc2 = 2,
|
|
.access = PL1_RW, .accessfn = access_gcs, .fgt = FGT_NGCS_EL0,
|
|
.fieldoffset = offsetof(CPUARMState, cp15.gcscr_el[0]) },
|
|
{ .name = "GCSCR_EL1", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 3, .opc1 = 0, .crn = 2, .crm = 5, .opc2 = 0,
|
|
.access = PL1_RW, .accessfn = access_gcs, .fgt = FGT_NGCS_EL1,
|
|
.nv2_redirect_offset = 0x8d0 | NV2_REDIR_NV1,
|
|
.vhe_redir_to_el2 = ENCODE_AA64_CP_REG(3, 4, 2, 5, 0),
|
|
.vhe_redir_to_el01 = ENCODE_AA64_CP_REG(3, 5, 2, 5, 0),
|
|
.fieldoffset = offsetof(CPUARMState, cp15.gcscr_el[1]) },
|
|
{ .name = "GCSCR_EL2", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 3, .opc1 = 4, .crn = 2, .crm = 5, .opc2 = 0,
|
|
.access = PL2_RW, .accessfn = access_gcs,
|
|
.fieldoffset = offsetof(CPUARMState, cp15.gcscr_el[2]) },
|
|
{ .name = "GCSCR_EL3", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 3, .opc1 = 6, .crn = 2, .crm = 5, .opc2 = 0,
|
|
.access = PL3_RW,
|
|
.fieldoffset = offsetof(CPUARMState, cp15.gcscr_el[3]) },
|
|
|
|
{ .name = "GCSPR_EL0", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 3, .opc1 = 3, .crn = 2, .crm = 5, .opc2 = 1,
|
|
.access = PL0_R | PL1_W, .accessfn = access_gcs_el0,
|
|
.fgt = FGT_NGCS_EL0, .writefn = gcspr_write,
|
|
.fieldoffset = offsetof(CPUARMState, cp15.gcspr_el[0]) },
|
|
{ .name = "GCSPR_EL1", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 3, .opc1 = 0, .crn = 2, .crm = 5, .opc2 = 1,
|
|
.access = PL1_RW, .accessfn = access_gcs,
|
|
.fgt = FGT_NGCS_EL1, .writefn = gcspr_write,
|
|
.nv2_redirect_offset = 0x8c0 | NV2_REDIR_NV1,
|
|
.vhe_redir_to_el2 = ENCODE_AA64_CP_REG(3, 4, 2, 5, 1),
|
|
.vhe_redir_to_el01 = ENCODE_AA64_CP_REG(3, 5, 2, 5, 1),
|
|
.fieldoffset = offsetof(CPUARMState, cp15.gcspr_el[1]) },
|
|
{ .name = "GCSPR_EL2", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 3, .opc1 = 4, .crn = 2, .crm = 5, .opc2 = 1,
|
|
.access = PL2_RW, .accessfn = access_gcs, .writefn = gcspr_write,
|
|
.fieldoffset = offsetof(CPUARMState, cp15.gcspr_el[2]) },
|
|
{ .name = "GCSPR_EL3", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 3, .opc1 = 6, .crn = 2, .crm = 5, .opc2 = 1,
|
|
.access = PL3_RW, .writefn = gcspr_write,
|
|
.fieldoffset = offsetof(CPUARMState, cp15.gcspr_el[2]) },
|
|
|
|
{ .name = "GCSPUSHM", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 1, .opc1 = 3, .crn = 7, .crm = 7, .opc2 = 0,
|
|
.access = PL0_W, .accessfn = access_gcspushm,
|
|
.fgt = FGT_NGCSPUSHM_EL1, .type = ARM_CP_GCSPUSHM },
|
|
{ .name = "GCSPOPM", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 1, .opc1 = 3, .crn = 7, .crm = 7, .opc2 = 1,
|
|
.access = PL0_R, .type = ARM_CP_GCSPOPM },
|
|
{ .name = "GCSSS1", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 1, .opc1 = 3, .crn = 7, .crm = 7, .opc2 = 2,
|
|
.access = PL0_W, .type = ARM_CP_GCSSS1 },
|
|
{ .name = "GCSSS2", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 1, .opc1 = 3, .crn = 7, .crm = 7, .opc2 = 3,
|
|
.access = PL0_R, .type = ARM_CP_GCSSS2 },
|
|
{ .name = "GCSPUSHX", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 1, .opc1 = 0, .crn = 7, .crm = 7, .opc2 = 4,
|
|
.access = PL1_W, .accessfn = access_gcspushx, .fgt = FGT_NGCSEPP,
|
|
.type = ARM_CP_GCSPUSHX },
|
|
{ .name = "GCSPOPCX", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 1, .opc1 = 0, .crn = 7, .crm = 7, .opc2 = 5,
|
|
.access = PL1_W, .accessfn = access_gcspopcx, .fgt = FGT_NGCSEPP,
|
|
.type = ARM_CP_GCSPOPCX },
|
|
{ .name = "GCSPOPX", .state = ARM_CP_STATE_AA64,
|
|
.opc0 = 1, .opc1 = 0, .crn = 7, .crm = 7, .opc2 = 6,
|
|
.access = PL1_W, .type = ARM_CP_GCSPOPX },
|
|
};
|
|
|
|
void define_gcs_cpregs(ARMCPU *cpu)
|
|
{
|
|
if (cpu_isar_feature(aa64_gcs, cpu)) {
|
|
define_arm_cp_regs(cpu, gcs_reginfo);
|
|
}
|
|
}
|