target/riscv: store RISCVCPUDef struct directly in the class

Prepare for adding more fields to RISCVCPUDef and reading them in
riscv_cpu_init: instead of storing the misa_mxl_max field in
RISCVCPUClass, ensure that there's always a valid RISCVCPUDef struct
and go through it.

Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
This commit is contained in:
Paolo Bonzini 2025-02-06 13:13:23 +01:00
parent 71fb3aa5eb
commit 5fd23f20e1
8 changed files with 39 additions and 29 deletions

View file

@ -37,7 +37,7 @@
bool riscv_is_32bit(RISCVHartArrayState *harts) bool riscv_is_32bit(RISCVHartArrayState *harts)
{ {
RISCVCPUClass *mcc = RISCV_CPU_GET_CLASS(&harts->harts[0]); RISCVCPUClass *mcc = RISCV_CPU_GET_CLASS(&harts->harts[0]);
return mcc->misa_mxl_max == MXL_RV32; return mcc->def->misa_mxl_max == MXL_RV32;
} }
/* /*

View file

@ -356,7 +356,7 @@ void riscv_cpu_set_misa_ext(CPURISCVState *env, uint32_t ext)
int riscv_cpu_max_xlen(RISCVCPUClass *mcc) int riscv_cpu_max_xlen(RISCVCPUClass *mcc)
{ {
return 16 << mcc->misa_mxl_max; return 16 << mcc->def->misa_mxl_max;
} }
#ifndef CONFIG_USER_ONLY #ifndef CONFIG_USER_ONLY
@ -1047,7 +1047,7 @@ static void riscv_cpu_reset_hold(Object *obj, ResetType type)
mcc->parent_phases.hold(obj, type); mcc->parent_phases.hold(obj, type);
} }
#ifndef CONFIG_USER_ONLY #ifndef CONFIG_USER_ONLY
env->misa_mxl = mcc->misa_mxl_max; env->misa_mxl = mcc->def->misa_mxl_max;
env->priv = PRV_M; env->priv = PRV_M;
env->mstatus &= ~(MSTATUS_MIE | MSTATUS_MPRV); env->mstatus &= ~(MSTATUS_MIE | MSTATUS_MPRV);
if (env->misa_mxl > MXL_RV32) { if (env->misa_mxl > MXL_RV32) {
@ -1449,7 +1449,7 @@ static void riscv_cpu_init(Object *obj)
RISCVCPU *cpu = RISCV_CPU(obj); RISCVCPU *cpu = RISCV_CPU(obj);
CPURISCVState *env = &cpu->env; CPURISCVState *env = &cpu->env;
env->misa_mxl = mcc->misa_mxl_max; env->misa_mxl = mcc->def->misa_mxl_max;
#ifndef CONFIG_USER_ONLY #ifndef CONFIG_USER_ONLY
qdev_init_gpio_in(DEVICE(obj), riscv_cpu_set_irq, qdev_init_gpio_in(DEVICE(obj), riscv_cpu_set_irq,
@ -1543,7 +1543,7 @@ static void riscv_cpu_validate_misa_mxl(RISCVCPUClass *mcc)
CPUClass *cc = CPU_CLASS(mcc); CPUClass *cc = CPU_CLASS(mcc);
/* Validate that MISA_MXL is set properly. */ /* Validate that MISA_MXL is set properly. */
switch (mcc->misa_mxl_max) { switch (mcc->def->misa_mxl_max) {
#ifdef TARGET_RISCV64 #ifdef TARGET_RISCV64
case MXL_RV64: case MXL_RV64:
case MXL_RV128: case MXL_RV128:
@ -3070,12 +3070,24 @@ static void riscv_cpu_common_class_init(ObjectClass *c, const void *data)
device_class_set_props(dc, riscv_cpu_properties); device_class_set_props(dc, riscv_cpu_properties);
} }
static void riscv_cpu_class_base_init(ObjectClass *c, const void *data)
{
RISCVCPUClass *mcc = RISCV_CPU_CLASS(c);
RISCVCPUClass *pcc = RISCV_CPU_CLASS(object_class_get_parent(c));
if (pcc->def) {
mcc->def = g_memdup2(pcc->def, sizeof(*pcc->def));
} else {
mcc->def = g_new0(RISCVCPUDef, 1);
}
}
static void riscv_cpu_class_init(ObjectClass *c, const void *data) static void riscv_cpu_class_init(ObjectClass *c, const void *data)
{ {
RISCVCPUClass *mcc = RISCV_CPU_CLASS(c); RISCVCPUClass *mcc = RISCV_CPU_CLASS(c);
const RISCVCPUDef *def = data; const RISCVCPUDef *def = data;
mcc->misa_mxl_max = def->misa_mxl_max; mcc->def->misa_mxl_max = def->misa_mxl_max;
riscv_cpu_validate_misa_mxl(mcc); riscv_cpu_validate_misa_mxl(mcc);
} }
@ -3226,6 +3238,7 @@ static const TypeInfo riscv_cpu_type_infos[] = {
.abstract = true, .abstract = true,
.class_size = sizeof(RISCVCPUClass), .class_size = sizeof(RISCVCPUClass),
.class_init = riscv_cpu_common_class_init, .class_init = riscv_cpu_common_class_init,
.class_base_init = riscv_cpu_class_base_init,
}, },
{ {
.name = TYPE_RISCV_DYNAMIC_CPU, .name = TYPE_RISCV_DYNAMIC_CPU,

View file

@ -554,7 +554,7 @@ struct RISCVCPUClass {
DeviceRealize parent_realize; DeviceRealize parent_realize;
ResettablePhases parent_phases; ResettablePhases parent_phases;
RISCVMXL misa_mxl_max; /* max mxl for this cpu */ RISCVCPUDef *def;
}; };
static inline int riscv_has_ext(CPURISCVState *env, target_ulong ext) static inline int riscv_has_ext(CPURISCVState *env, target_ulong ext)

View file

@ -62,7 +62,7 @@ int riscv_cpu_gdb_read_register(CPUState *cs, GByteArray *mem_buf, int n)
return 0; return 0;
} }
switch (mcc->misa_mxl_max) { switch (mcc->def->misa_mxl_max) {
case MXL_RV32: case MXL_RV32:
return gdb_get_reg32(mem_buf, tmp); return gdb_get_reg32(mem_buf, tmp);
case MXL_RV64: case MXL_RV64:
@ -82,7 +82,7 @@ int riscv_cpu_gdb_write_register(CPUState *cs, uint8_t *mem_buf, int n)
int length = 0; int length = 0;
target_ulong tmp; target_ulong tmp;
switch (mcc->misa_mxl_max) { switch (mcc->def->misa_mxl_max) {
case MXL_RV32: case MXL_RV32:
tmp = (int32_t)ldl_p(mem_buf); tmp = (int32_t)ldl_p(mem_buf);
length = 4; length = 4;
@ -359,7 +359,7 @@ void riscv_cpu_register_gdb_regs_for_features(CPUState *cs)
ricsv_gen_dynamic_vector_feature(cs, cs->gdb_num_regs), ricsv_gen_dynamic_vector_feature(cs, cs->gdb_num_regs),
0); 0);
} }
switch (mcc->misa_mxl_max) { switch (mcc->def->misa_mxl_max) {
case MXL_RV32: case MXL_RV32:
gdb_register_coprocessor(cs, riscv_gdb_get_virtual, gdb_register_coprocessor(cs, riscv_gdb_get_virtual,
riscv_gdb_set_virtual, riscv_gdb_set_virtual,

View file

@ -2086,22 +2086,19 @@ static void kvm_cpu_accel_register_types(void)
} }
type_init(kvm_cpu_accel_register_types); type_init(kvm_cpu_accel_register_types);
static void riscv_host_cpu_class_init(ObjectClass *c, const void *data)
{
RISCVCPUClass *mcc = RISCV_CPU_CLASS(c);
#if defined(TARGET_RISCV32)
mcc->misa_mxl_max = MXL_RV32;
#elif defined(TARGET_RISCV64)
mcc->misa_mxl_max = MXL_RV64;
#endif
}
static const TypeInfo riscv_kvm_cpu_type_infos[] = { static const TypeInfo riscv_kvm_cpu_type_infos[] = {
{ {
.name = TYPE_RISCV_CPU_HOST, .name = TYPE_RISCV_CPU_HOST,
.parent = TYPE_RISCV_CPU, .parent = TYPE_RISCV_CPU,
.class_init = riscv_host_cpu_class_init, #if defined(TARGET_RISCV32)
.class_data = &(const RISCVCPUDef) {
.misa_mxl_max = MXL_RV32,
},
#elif defined(TARGET_RISCV64)
.class_data = &(const RISCVCPUDef) {
.misa_mxl_max = MXL_RV64,
},
#endif
} }
}; };

View file

@ -170,7 +170,7 @@ static bool rv128_needed(void *opaque)
{ {
RISCVCPUClass *mcc = RISCV_CPU_GET_CLASS(opaque); RISCVCPUClass *mcc = RISCV_CPU_GET_CLASS(opaque);
return mcc->misa_mxl_max == MXL_RV128; return mcc->def->misa_mxl_max == MXL_RV128;
} }
static const VMStateDescription vmstate_rv128 = { static const VMStateDescription vmstate_rv128 = {

View file

@ -691,7 +691,7 @@ void riscv_cpu_validate_set_extensions(RISCVCPU *cpu, Error **errp)
return; return;
} }
if (mcc->misa_mxl_max != MXL_RV32 && cpu->cfg.ext_zcf) { if (mcc->def->misa_mxl_max != MXL_RV32 && cpu->cfg.ext_zcf) {
error_setg(errp, "Zcf extension is only relevant to RV32"); error_setg(errp, "Zcf extension is only relevant to RV32");
return; return;
} }
@ -788,7 +788,7 @@ void riscv_cpu_validate_set_extensions(RISCVCPU *cpu, Error **errp)
return; return;
} }
if (mcc->misa_mxl_max == MXL_RV32 && cpu->cfg.ext_svukte) { if (mcc->def->misa_mxl_max == MXL_RV32 && cpu->cfg.ext_svukte) {
error_setg(errp, "svukte is not supported for RV32"); error_setg(errp, "svukte is not supported for RV32");
return; return;
} }
@ -1026,7 +1026,7 @@ static void cpu_enable_zc_implied_rules(RISCVCPU *cpu)
cpu_cfg_ext_auto_update(cpu, CPU_CFG_OFFSET(ext_zcmp), true); cpu_cfg_ext_auto_update(cpu, CPU_CFG_OFFSET(ext_zcmp), true);
cpu_cfg_ext_auto_update(cpu, CPU_CFG_OFFSET(ext_zcmt), true); cpu_cfg_ext_auto_update(cpu, CPU_CFG_OFFSET(ext_zcmt), true);
if (riscv_has_ext(env, RVF) && mcc->misa_mxl_max == MXL_RV32) { if (riscv_has_ext(env, RVF) && mcc->def->misa_mxl_max == MXL_RV32) {
cpu_cfg_ext_auto_update(cpu, CPU_CFG_OFFSET(ext_zcf), true); cpu_cfg_ext_auto_update(cpu, CPU_CFG_OFFSET(ext_zcf), true);
} }
} }
@ -1035,7 +1035,7 @@ static void cpu_enable_zc_implied_rules(RISCVCPU *cpu)
if (riscv_has_ext(env, RVC) && env->priv_ver >= PRIV_VERSION_1_12_0) { if (riscv_has_ext(env, RVC) && env->priv_ver >= PRIV_VERSION_1_12_0) {
cpu_cfg_ext_auto_update(cpu, CPU_CFG_OFFSET(ext_zca), true); cpu_cfg_ext_auto_update(cpu, CPU_CFG_OFFSET(ext_zca), true);
if (riscv_has_ext(env, RVF) && mcc->misa_mxl_max == MXL_RV32) { if (riscv_has_ext(env, RVF) && mcc->def->misa_mxl_max == MXL_RV32) {
cpu_cfg_ext_auto_update(cpu, CPU_CFG_OFFSET(ext_zcf), true); cpu_cfg_ext_auto_update(cpu, CPU_CFG_OFFSET(ext_zcf), true);
} }
@ -1161,7 +1161,7 @@ static bool riscv_tcg_cpu_realize(CPUState *cs, Error **errp)
#ifndef CONFIG_USER_ONLY #ifndef CONFIG_USER_ONLY
RISCVCPUClass *mcc = RISCV_CPU_GET_CLASS(cpu); RISCVCPUClass *mcc = RISCV_CPU_GET_CLASS(cpu);
if (mcc->misa_mxl_max >= MXL_RV128 && qemu_tcg_mttcg_enabled()) { if (mcc->def->misa_mxl_max >= MXL_RV128 && qemu_tcg_mttcg_enabled()) {
/* Missing 128-bit aligned atomics */ /* Missing 128-bit aligned atomics */
error_setg(errp, error_setg(errp,
"128-bit RISC-V currently does not work with Multi " "128-bit RISC-V currently does not work with Multi "

View file

@ -1276,7 +1276,7 @@ static void riscv_tr_init_disas_context(DisasContextBase *dcbase, CPUState *cs)
ctx->cfg_vta_all_1s = cpu->cfg.rvv_ta_all_1s; ctx->cfg_vta_all_1s = cpu->cfg.rvv_ta_all_1s;
ctx->vstart_eq_zero = FIELD_EX32(tb_flags, TB_FLAGS, VSTART_EQ_ZERO); ctx->vstart_eq_zero = FIELD_EX32(tb_flags, TB_FLAGS, VSTART_EQ_ZERO);
ctx->vl_eq_vlmax = FIELD_EX32(tb_flags, TB_FLAGS, VL_EQ_VLMAX); ctx->vl_eq_vlmax = FIELD_EX32(tb_flags, TB_FLAGS, VL_EQ_VLMAX);
ctx->misa_mxl_max = mcc->misa_mxl_max; ctx->misa_mxl_max = mcc->def->misa_mxl_max;
ctx->xl = FIELD_EX32(tb_flags, TB_FLAGS, XL); ctx->xl = FIELD_EX32(tb_flags, TB_FLAGS, XL);
ctx->address_xl = FIELD_EX32(tb_flags, TB_FLAGS, AXL); ctx->address_xl = FIELD_EX32(tb_flags, TB_FLAGS, AXL);
ctx->cs = cs; ctx->cs = cs;