mirror of
https://github.com/yuzu-emu/unicorn.git
synced 2025-01-22 09:51:05 +00:00
target/arm: Add support for DC CVAP & DC CVADP ins
ARMv8.2 introduced support for Data Cache Clean instructions to PoP (point-of-persistence) - DC CVAP and PoDP (point-of-deep-persistence) - DV CVADP. Both specify conceptual points in a memory system where all writes that are to reach them are considered persistent. The support provided considers both to be actually the same so there is no distinction between the two. If none is available (there is no backing store for given memory) both will result in Data Cache Clean up to the point of coherency. Otherwise sync for the specified range shall be performed. Backports commit 0d57b49992200a926c4436eead97ecfc8cc710be from qemu
This commit is contained in:
parent
0716794d86
commit
81c14bb595
|
@ -3529,6 +3529,16 @@ static inline bool isar_feature_aa64_frint(const ARMISARegisters *id)
|
||||||
return FIELD_EX64(id->id_aa64isar1, ID_AA64ISAR1, FRINTTS) != 0;
|
return FIELD_EX64(id->id_aa64isar1, ID_AA64ISAR1, FRINTTS) != 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline bool isar_feature_aa64_dcpop(const ARMISARegisters *id)
|
||||||
|
{
|
||||||
|
return FIELD_EX64(id->id_aa64isar1, ID_AA64ISAR1, DPB) != 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline bool isar_feature_aa64_dcpodp(const ARMISARegisters *id)
|
||||||
|
{
|
||||||
|
return FIELD_EX64(id->id_aa64isar1, ID_AA64ISAR1, DPB) >= 2;
|
||||||
|
}
|
||||||
|
|
||||||
static inline bool isar_feature_aa64_fp16(const ARMISARegisters *id)
|
static inline bool isar_feature_aa64_fp16(const ARMISARegisters *id)
|
||||||
{
|
{
|
||||||
/* We always set the AdvSIMD and FP fields identically wrt FP16. */
|
/* We always set the AdvSIMD and FP fields identically wrt FP16. */
|
||||||
|
|
|
@ -291,6 +291,7 @@ static void aarch64_max_initfn(struct uc_struct *uc, Object *obj, void *opaque)
|
||||||
cpu->isar.id_aa64isar0 = t;
|
cpu->isar.id_aa64isar0 = t;
|
||||||
|
|
||||||
t = cpu->isar.id_aa64isar1;
|
t = cpu->isar.id_aa64isar1;
|
||||||
|
t = FIELD_DP64(t, ID_AA64ISAR1, DPB, 2);
|
||||||
t = FIELD_DP64(t, ID_AA64ISAR1, JSCVT, 1);
|
t = FIELD_DP64(t, ID_AA64ISAR1, JSCVT, 1);
|
||||||
t = FIELD_DP64(t, ID_AA64ISAR1, FCMA, 1);
|
t = FIELD_DP64(t, ID_AA64ISAR1, FCMA, 1);
|
||||||
t = FIELD_DP64(t, ID_AA64ISAR1, APA, 1); /* PAuth, architected only */
|
t = FIELD_DP64(t, ID_AA64ISAR1, APA, 1); /* PAuth, architected only */
|
||||||
|
|
|
@ -5766,6 +5766,52 @@ static const ARMCPRegInfo rndr_reginfo[] = {
|
||||||
.access = PL0_R, .readfn = rndr_readfn },
|
.access = PL0_R, .readfn = rndr_readfn },
|
||||||
REGINFO_SENTINEL
|
REGINFO_SENTINEL
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#ifndef CONFIG_USER_ONLY
|
||||||
|
static void dccvap_writefn(CPUARMState *env, const ARMCPRegInfo *opaque,
|
||||||
|
uint64_t value)
|
||||||
|
{
|
||||||
|
ARMCPU *cpu = env_archcpu(env);
|
||||||
|
/* CTR_EL0 System register -> DminLine, bits [19:16] */
|
||||||
|
uint64_t dline_size = 4 << ((cpu->ctr >> 16) & 0xF);
|
||||||
|
uint64_t vaddr_in = (uint64_t) value;
|
||||||
|
uint64_t vaddr = vaddr_in & ~(dline_size - 1);
|
||||||
|
void *haddr;
|
||||||
|
int mem_idx = cpu_mmu_index(env, false);
|
||||||
|
|
||||||
|
/* This won't be crossing page boundaries */
|
||||||
|
haddr = probe_read(env, vaddr, dline_size, mem_idx, GETPC());
|
||||||
|
if (haddr) {
|
||||||
|
|
||||||
|
ram_addr_t offset;
|
||||||
|
MemoryRegion *mr;
|
||||||
|
|
||||||
|
/* RCU lock is already being held */
|
||||||
|
mr = memory_region_from_host(env->uc, haddr, &offset);
|
||||||
|
|
||||||
|
if (mr) {
|
||||||
|
memory_region_do_writeback(mr, offset, dline_size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static const ARMCPRegInfo dcpop_reg[] = {
|
||||||
|
{ .name = "DC_CVAP", .state = ARM_CP_STATE_AA64,
|
||||||
|
.opc0 = 1, .opc1 = 3, .crn = 7, .crm = 12, .opc2 = 1,
|
||||||
|
.access = PL0_W, .type = ARM_CP_NO_RAW | ARM_CP_SUPPRESS_TB_END,
|
||||||
|
.accessfn = aa64_cacheop_access, .writefn = dccvap_writefn },
|
||||||
|
REGINFO_SENTINEL
|
||||||
|
};
|
||||||
|
|
||||||
|
static const ARMCPRegInfo dcpodp_reg[] = {
|
||||||
|
{ .name = "DC_CVADP", .state = ARM_CP_STATE_AA64,
|
||||||
|
.opc0 = 1, .opc1 = 3, .crn = 7, .crm = 13, .opc2 = 1,
|
||||||
|
.access = PL0_W, .type = ARM_CP_NO_RAW | ARM_CP_SUPPRESS_TB_END,
|
||||||
|
.accessfn = aa64_cacheop_access, .writefn = dccvap_writefn },
|
||||||
|
REGINFO_SENTINEL
|
||||||
|
};
|
||||||
|
#endif /*CONFIG_USER_ONLY*/
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static CPAccessResult access_predinv(CPUARMState *env, const ARMCPRegInfo *ri,
|
static CPAccessResult access_predinv(CPUARMState *env, const ARMCPRegInfo *ri,
|
||||||
|
@ -6824,6 +6870,16 @@ void register_cp_regs_for_features(ARMCPU *cpu)
|
||||||
if (cpu_isar_feature(aa64_rndr, cpu)) {
|
if (cpu_isar_feature(aa64_rndr, cpu)) {
|
||||||
define_arm_cp_regs(cpu, rndr_reginfo);
|
define_arm_cp_regs(cpu, rndr_reginfo);
|
||||||
}
|
}
|
||||||
|
#ifndef CONFIG_USER_ONLY
|
||||||
|
/* Data Cache clean instructions up to PoP */
|
||||||
|
if (cpu_isar_feature(aa64_dcpop, cpu)) {
|
||||||
|
define_one_arm_cp_reg(cpu, dcpop_reg);
|
||||||
|
|
||||||
|
if (cpu_isar_feature(aa64_dcpodp, cpu)) {
|
||||||
|
define_one_arm_cp_reg(cpu, dcpodp_reg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif /*CONFIG_USER_ONLY*/
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
Loading…
Reference in a new issue