
In expression 1ULL << tlb_ps, left shifting by more than 63 bits has undefined behavior. The shift amount, tlb_ps, is as much as 64. check "tlb_ps >=64" to fix. Resolves: Coverity CID 1593475 Fixes: d882c284a3 ("target/loongarch: check tlb_ps") Suggested-by: Peter Maydell <peter.maydell@linaro.org> Signed-off-by: Song Gao <gaosong@loongson.cn> Reviewed-by: Bibo Mao <maobibo@loongson.cn> Signed-off-by: Bibo Mao <maobibo@loongson.cn>
139 lines
3.2 KiB
C
139 lines
3.2 KiB
C
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
/*
|
|
* LoongArch emulation helpers for CSRs
|
|
*
|
|
* Copyright (c) 2021 Loongson Technology Corporation Limited
|
|
*/
|
|
|
|
#include "qemu/osdep.h"
|
|
#include "qemu/log.h"
|
|
#include "qemu/main-loop.h"
|
|
#include "cpu.h"
|
|
#include "internals.h"
|
|
#include "qemu/host-utils.h"
|
|
#include "exec/helper-proto.h"
|
|
#include "exec/cputlb.h"
|
|
#include "exec/cpu_ldst.h"
|
|
#include "hw/irq.h"
|
|
#include "cpu-csr.h"
|
|
|
|
target_ulong helper_csrwr_stlbps(CPULoongArchState *env, target_ulong val)
|
|
{
|
|
int64_t old_v = env->CSR_STLBPS;
|
|
|
|
/*
|
|
* The real hardware only supports the min tlb_ps is 12
|
|
* tlb_ps=0 may cause undefined-behavior.
|
|
*/
|
|
uint8_t tlb_ps = FIELD_EX64(env->CSR_STLBPS, CSR_STLBPS, PS);
|
|
if (!check_ps(env, tlb_ps)) {
|
|
qemu_log_mask(LOG_GUEST_ERROR,
|
|
"Attempted set ps %d\n", tlb_ps);
|
|
}
|
|
return old_v;
|
|
}
|
|
|
|
target_ulong helper_csrrd_pgd(CPULoongArchState *env)
|
|
{
|
|
int64_t v;
|
|
|
|
if (env->CSR_TLBRERA & 0x1) {
|
|
v = env->CSR_TLBRBADV;
|
|
} else {
|
|
v = env->CSR_BADV;
|
|
}
|
|
|
|
if ((v >> 63) & 0x1) {
|
|
v = env->CSR_PGDH;
|
|
} else {
|
|
v = env->CSR_PGDL;
|
|
}
|
|
|
|
return v;
|
|
}
|
|
|
|
target_ulong helper_csrrd_cpuid(CPULoongArchState *env)
|
|
{
|
|
LoongArchCPU *lac = env_archcpu(env);
|
|
|
|
env->CSR_CPUID = CPU(lac)->cpu_index;
|
|
|
|
return env->CSR_CPUID;
|
|
}
|
|
|
|
target_ulong helper_csrrd_tval(CPULoongArchState *env)
|
|
{
|
|
LoongArchCPU *cpu = env_archcpu(env);
|
|
|
|
return cpu_loongarch_get_constant_timer_ticks(cpu);
|
|
}
|
|
|
|
target_ulong helper_csrwr_estat(CPULoongArchState *env, target_ulong val)
|
|
{
|
|
int64_t old_v = env->CSR_ESTAT;
|
|
|
|
/* Only IS[1:0] can be written */
|
|
env->CSR_ESTAT = deposit64(env->CSR_ESTAT, 0, 2, val);
|
|
|
|
return old_v;
|
|
}
|
|
|
|
target_ulong helper_csrwr_asid(CPULoongArchState *env, target_ulong val)
|
|
{
|
|
int64_t old_v = env->CSR_ASID;
|
|
|
|
/* Only ASID filed of CSR_ASID can be written */
|
|
env->CSR_ASID = deposit64(env->CSR_ASID, 0, 10, val);
|
|
if (old_v != env->CSR_ASID) {
|
|
tlb_flush(env_cpu(env));
|
|
}
|
|
return old_v;
|
|
}
|
|
|
|
target_ulong helper_csrwr_tcfg(CPULoongArchState *env, target_ulong val)
|
|
{
|
|
LoongArchCPU *cpu = env_archcpu(env);
|
|
int64_t old_v = env->CSR_TCFG;
|
|
|
|
cpu_loongarch_store_constant_timer_config(cpu, val);
|
|
|
|
return old_v;
|
|
}
|
|
|
|
target_ulong helper_csrwr_ticlr(CPULoongArchState *env, target_ulong val)
|
|
{
|
|
LoongArchCPU *cpu = env_archcpu(env);
|
|
int64_t old_v = 0;
|
|
|
|
if (val & 0x1) {
|
|
bql_lock();
|
|
loongarch_cpu_set_irq(cpu, IRQ_TIMER, 0);
|
|
bql_unlock();
|
|
}
|
|
return old_v;
|
|
}
|
|
|
|
target_ulong helper_csrwr_pwcl(CPULoongArchState *env, target_ulong val)
|
|
{
|
|
uint8_t shift, ptbase;
|
|
int64_t old_v = env->CSR_PWCL;
|
|
|
|
/*
|
|
* The real hardware only supports 64bit PTE width now, 128bit or others
|
|
* treated as illegal.
|
|
*/
|
|
shift = FIELD_EX64(val, CSR_PWCL, PTEWIDTH);
|
|
ptbase = FIELD_EX64(val, CSR_PWCL, PTBASE);
|
|
if (shift) {
|
|
qemu_log_mask(LOG_GUEST_ERROR,
|
|
"Attempted set pte width with %d bit\n", 64 << shift);
|
|
val = FIELD_DP64(val, CSR_PWCL, PTEWIDTH, 0);
|
|
}
|
|
if (!check_ps(env, ptbase)) {
|
|
qemu_log_mask(LOG_GUEST_ERROR,
|
|
"Attrmpted set ptbase 2^%d\n", ptbase);
|
|
}
|
|
env->CSR_PWCL =val;
|
|
return old_v;
|
|
}
|