xref: /openbmc/qemu/target/mips/system/cp0.c (revision d9a4282c4b690e45d25c2b933f318bb41eeb271d)
132cad1ffSPhilippe Mathieu-Daudé /*
232cad1ffSPhilippe Mathieu-Daudé  * QEMU MIPS CPU
332cad1ffSPhilippe Mathieu-Daudé  *
432cad1ffSPhilippe Mathieu-Daudé  * Copyright (c) 2012 SUSE LINUX Products GmbH
532cad1ffSPhilippe Mathieu-Daudé  *
632cad1ffSPhilippe Mathieu-Daudé  * This library is free software; you can redistribute it and/or
732cad1ffSPhilippe Mathieu-Daudé  * modify it under the terms of the GNU Lesser General Public
832cad1ffSPhilippe Mathieu-Daudé  * License as published by the Free Software Foundation; either
932cad1ffSPhilippe Mathieu-Daudé  * version 2.1 of the License, or (at your option) any later version.
1032cad1ffSPhilippe Mathieu-Daudé  *
1132cad1ffSPhilippe Mathieu-Daudé  * This library is distributed in the hope that it will be useful,
1232cad1ffSPhilippe Mathieu-Daudé  * but WITHOUT ANY WARRANTY; without even the implied warranty of
1332cad1ffSPhilippe Mathieu-Daudé  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
1432cad1ffSPhilippe Mathieu-Daudé  * Lesser General Public License for more details.
1532cad1ffSPhilippe Mathieu-Daudé  *
1632cad1ffSPhilippe Mathieu-Daudé  * You should have received a copy of the GNU Lesser General Public
1732cad1ffSPhilippe Mathieu-Daudé  * License along with this library; if not, see
1832cad1ffSPhilippe Mathieu-Daudé  * <http://www.gnu.org/licenses/lgpl-2.1.html>
1932cad1ffSPhilippe Mathieu-Daudé  */
2032cad1ffSPhilippe Mathieu-Daudé 
2132cad1ffSPhilippe Mathieu-Daudé #include "qemu/osdep.h"
2232cad1ffSPhilippe Mathieu-Daudé #include "cpu.h"
2332cad1ffSPhilippe Mathieu-Daudé #include "internal.h"
24*6ff5da16SPhilippe Mathieu-Daudé #include "exec/cputlb.h"
2532cad1ffSPhilippe Mathieu-Daudé 
2632cad1ffSPhilippe Mathieu-Daudé /* Called for updates to CP0_Status.  */
sync_c0_status(CPUMIPSState * env,CPUMIPSState * cpu,int tc)2732cad1ffSPhilippe Mathieu-Daudé void sync_c0_status(CPUMIPSState *env, CPUMIPSState *cpu, int tc)
2832cad1ffSPhilippe Mathieu-Daudé {
2932cad1ffSPhilippe Mathieu-Daudé     int32_t tcstatus, *tcst;
3032cad1ffSPhilippe Mathieu-Daudé     uint32_t v = cpu->CP0_Status;
3132cad1ffSPhilippe Mathieu-Daudé     uint32_t cu, mx, asid, ksu;
3232cad1ffSPhilippe Mathieu-Daudé     uint32_t mask = ((1 << CP0TCSt_TCU3)
3332cad1ffSPhilippe Mathieu-Daudé                        | (1 << CP0TCSt_TCU2)
3432cad1ffSPhilippe Mathieu-Daudé                        | (1 << CP0TCSt_TCU1)
3532cad1ffSPhilippe Mathieu-Daudé                        | (1 << CP0TCSt_TCU0)
3632cad1ffSPhilippe Mathieu-Daudé                        | (1 << CP0TCSt_TMX)
3732cad1ffSPhilippe Mathieu-Daudé                        | (3 << CP0TCSt_TKSU)
3832cad1ffSPhilippe Mathieu-Daudé                        | (0xff << CP0TCSt_TASID));
3932cad1ffSPhilippe Mathieu-Daudé 
4032cad1ffSPhilippe Mathieu-Daudé     cu = (v >> CP0St_CU0) & 0xf;
4132cad1ffSPhilippe Mathieu-Daudé     mx = (v >> CP0St_MX) & 0x1;
4232cad1ffSPhilippe Mathieu-Daudé     ksu = (v >> CP0St_KSU) & 0x3;
4332cad1ffSPhilippe Mathieu-Daudé     asid = env->CP0_EntryHi & env->CP0_EntryHi_ASID_mask;
4432cad1ffSPhilippe Mathieu-Daudé 
4532cad1ffSPhilippe Mathieu-Daudé     tcstatus = cu << CP0TCSt_TCU0;
4632cad1ffSPhilippe Mathieu-Daudé     tcstatus |= mx << CP0TCSt_TMX;
4732cad1ffSPhilippe Mathieu-Daudé     tcstatus |= ksu << CP0TCSt_TKSU;
4832cad1ffSPhilippe Mathieu-Daudé     tcstatus |= asid;
4932cad1ffSPhilippe Mathieu-Daudé 
5032cad1ffSPhilippe Mathieu-Daudé     if (tc == cpu->current_tc) {
5132cad1ffSPhilippe Mathieu-Daudé         tcst = &cpu->active_tc.CP0_TCStatus;
5232cad1ffSPhilippe Mathieu-Daudé     } else {
5332cad1ffSPhilippe Mathieu-Daudé         tcst = &cpu->tcs[tc].CP0_TCStatus;
5432cad1ffSPhilippe Mathieu-Daudé     }
5532cad1ffSPhilippe Mathieu-Daudé 
5632cad1ffSPhilippe Mathieu-Daudé     *tcst &= ~mask;
5732cad1ffSPhilippe Mathieu-Daudé     *tcst |= tcstatus;
5832cad1ffSPhilippe Mathieu-Daudé     compute_hflags(cpu);
5932cad1ffSPhilippe Mathieu-Daudé }
6032cad1ffSPhilippe Mathieu-Daudé 
cpu_mips_store_status(CPUMIPSState * env,target_ulong val)6132cad1ffSPhilippe Mathieu-Daudé void cpu_mips_store_status(CPUMIPSState *env, target_ulong val)
6232cad1ffSPhilippe Mathieu-Daudé {
6332cad1ffSPhilippe Mathieu-Daudé     uint32_t mask = env->CP0_Status_rw_bitmask;
6432cad1ffSPhilippe Mathieu-Daudé     target_ulong old = env->CP0_Status;
6532cad1ffSPhilippe Mathieu-Daudé 
6632cad1ffSPhilippe Mathieu-Daudé     if (env->insn_flags & ISA_MIPS_R6) {
6732cad1ffSPhilippe Mathieu-Daudé         bool has_supervisor = extract32(mask, CP0St_KSU, 2) == 0x3;
6832cad1ffSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
6932cad1ffSPhilippe Mathieu-Daudé         uint32_t ksux = (1 << CP0St_KX) & val;
7032cad1ffSPhilippe Mathieu-Daudé         ksux |= (ksux >> 1) & val; /* KX = 0 forces SX to be 0 */
7132cad1ffSPhilippe Mathieu-Daudé         ksux |= (ksux >> 1) & val; /* SX = 0 forces UX to be 0 */
7232cad1ffSPhilippe Mathieu-Daudé         val = (val & ~(7 << CP0St_UX)) | ksux;
7332cad1ffSPhilippe Mathieu-Daudé #endif
7432cad1ffSPhilippe Mathieu-Daudé         if (has_supervisor && extract32(val, CP0St_KSU, 2) == 0x3) {
7532cad1ffSPhilippe Mathieu-Daudé             mask &= ~(3 << CP0St_KSU);
7632cad1ffSPhilippe Mathieu-Daudé         }
7732cad1ffSPhilippe Mathieu-Daudé         mask &= ~(((1 << CP0St_SR) | (1 << CP0St_NMI)) & val);
7832cad1ffSPhilippe Mathieu-Daudé     }
7932cad1ffSPhilippe Mathieu-Daudé 
8032cad1ffSPhilippe Mathieu-Daudé     env->CP0_Status = (old & ~mask) | (val & mask);
8132cad1ffSPhilippe Mathieu-Daudé #if defined(TARGET_MIPS64)
8232cad1ffSPhilippe Mathieu-Daudé     if ((env->CP0_Status ^ old) & (old & (7 << CP0St_UX))) {
8332cad1ffSPhilippe Mathieu-Daudé         /* Access to at least one of the 64-bit segments has been disabled */
8432cad1ffSPhilippe Mathieu-Daudé         tlb_flush(env_cpu(env));
8532cad1ffSPhilippe Mathieu-Daudé     }
8632cad1ffSPhilippe Mathieu-Daudé #endif
8732cad1ffSPhilippe Mathieu-Daudé     if (ase_mt_available(env)) {
8832cad1ffSPhilippe Mathieu-Daudé         sync_c0_status(env, env, env->current_tc);
8932cad1ffSPhilippe Mathieu-Daudé     } else {
9032cad1ffSPhilippe Mathieu-Daudé         compute_hflags(env);
9132cad1ffSPhilippe Mathieu-Daudé     }
9232cad1ffSPhilippe Mathieu-Daudé }
9332cad1ffSPhilippe Mathieu-Daudé 
cpu_mips_store_cause(CPUMIPSState * env,target_ulong val)9432cad1ffSPhilippe Mathieu-Daudé void cpu_mips_store_cause(CPUMIPSState *env, target_ulong val)
9532cad1ffSPhilippe Mathieu-Daudé {
9632cad1ffSPhilippe Mathieu-Daudé     uint32_t mask = 0x00C00300;
9732cad1ffSPhilippe Mathieu-Daudé     uint32_t old = env->CP0_Cause;
9832cad1ffSPhilippe Mathieu-Daudé     int i;
9932cad1ffSPhilippe Mathieu-Daudé 
10032cad1ffSPhilippe Mathieu-Daudé     if (env->insn_flags & ISA_MIPS_R2) {
10132cad1ffSPhilippe Mathieu-Daudé         mask |= 1 << CP0Ca_DC;
10232cad1ffSPhilippe Mathieu-Daudé     }
10332cad1ffSPhilippe Mathieu-Daudé     if (env->insn_flags & ISA_MIPS_R6) {
10432cad1ffSPhilippe Mathieu-Daudé         mask &= ~((1 << CP0Ca_WP) & val);
10532cad1ffSPhilippe Mathieu-Daudé     }
10632cad1ffSPhilippe Mathieu-Daudé 
10732cad1ffSPhilippe Mathieu-Daudé     env->CP0_Cause = (env->CP0_Cause & ~mask) | (val & mask);
10832cad1ffSPhilippe Mathieu-Daudé 
10932cad1ffSPhilippe Mathieu-Daudé     if ((old ^ env->CP0_Cause) & (1 << CP0Ca_DC)) {
11032cad1ffSPhilippe Mathieu-Daudé         if (env->CP0_Cause & (1 << CP0Ca_DC)) {
11132cad1ffSPhilippe Mathieu-Daudé             cpu_mips_stop_count(env);
11232cad1ffSPhilippe Mathieu-Daudé         } else {
11332cad1ffSPhilippe Mathieu-Daudé             cpu_mips_start_count(env);
11432cad1ffSPhilippe Mathieu-Daudé         }
11532cad1ffSPhilippe Mathieu-Daudé     }
11632cad1ffSPhilippe Mathieu-Daudé 
11732cad1ffSPhilippe Mathieu-Daudé     /* Set/reset software interrupts */
11832cad1ffSPhilippe Mathieu-Daudé     for (i = 0 ; i < 2 ; i++) {
11932cad1ffSPhilippe Mathieu-Daudé         if ((old ^ env->CP0_Cause) & (1 << (CP0Ca_IP + i))) {
12032cad1ffSPhilippe Mathieu-Daudé             cpu_mips_soft_irq(env, i, env->CP0_Cause & (1 << (CP0Ca_IP + i)));
12132cad1ffSPhilippe Mathieu-Daudé         }
12232cad1ffSPhilippe Mathieu-Daudé     }
12332cad1ffSPhilippe Mathieu-Daudé }
124