target-mips: Fix CP0.Config3.ISAOnExc write accesses

Fix CP0.Config3.ISAOnExc write accesses on microMIPS processors. This
bit is mandatory for any processor that implements the microMIPS
instruction set. This bit is r/w for processors that implement both the
standard MIPS and the microMIPS instruction set. This bit is r/o and
hardwired to 1 if only the microMIPS instruction set is implemented.

There is no other bit ever writable in CP0.Config3 so defining a
corresponding `CP0_Config3_rw_bitmask' member in `CPUMIPSState' is I
think an overkill. Therefore make the ability to write the bit rely on
the presence of ASE_MICROMIPS set in the instruction flags.

The read-only case of the microMIPS instruction set being implemented
only can be added when we add support for such a configuration. We do
not currently have such support, we have no instruction flag that would
control the presence of the standard MIPS instruction set nor any
associated code in instruction decoding.

This change is needed to boot a microMIPS Linux kernel successfully,
otherwise it hangs early on as interrupts are enabled and then the
exception handler invoked loops as its first instruction is interpreted
in the wrong execution mode and triggers another exception right away.
And then over and over again.

We already check the current setting of the CP0.Config3.ISAOnExc in
`set_hflags_for_handler' to set the ISA bit correctly on the exception
handler entry so it is the ability to set it that is missing only.

Backports commit 90f12d735d66ac1196d9a2bced039a432eefc03d from qemu
This commit is contained in:
Maciej W. Rozycki 2018-02-11 16:20:45 -05:00 committed by Lioncash
parent 611933d18d
commit a4c4834545
No known key found for this signature in database
GPG key ID: 4E3C3CC1031BA9C7
8 changed files with 20 additions and 2 deletions

View file

@ -3593,6 +3593,7 @@ mips_symbols = (
'helper_mftc0_configx', 'helper_mftc0_configx',
'helper_mtc0_config0', 'helper_mtc0_config0',
'helper_mtc0_config2', 'helper_mtc0_config2',
'helper_mtc0_config3',
'helper_mtc0_config4', 'helper_mtc0_config4',
'helper_mtc0_config5', 'helper_mtc0_config5',
'helper_mtc0_lladdr', 'helper_mtc0_lladdr',

View file

@ -3540,6 +3540,7 @@
#define helper_mftc0_configx helper_mftc0_configx_mips #define helper_mftc0_configx helper_mftc0_configx_mips
#define helper_mtc0_config0 helper_mtc0_config0_mips #define helper_mtc0_config0 helper_mtc0_config0_mips
#define helper_mtc0_config2 helper_mtc0_config2_mips #define helper_mtc0_config2 helper_mtc0_config2_mips
#define helper_mtc0_config3 helper_mtc0_config3_mips
#define helper_mtc0_config4 helper_mtc0_config4_mips #define helper_mtc0_config4 helper_mtc0_config4_mips
#define helper_mtc0_config5 helper_mtc0_config5_mips #define helper_mtc0_config5 helper_mtc0_config5_mips
#define helper_mtc0_lladdr helper_mtc0_lladdr_mips #define helper_mtc0_lladdr helper_mtc0_lladdr_mips

View file

@ -3540,6 +3540,7 @@
#define helper_mftc0_configx helper_mftc0_configx_mips64 #define helper_mftc0_configx helper_mftc0_configx_mips64
#define helper_mtc0_config0 helper_mtc0_config0_mips64 #define helper_mtc0_config0 helper_mtc0_config0_mips64
#define helper_mtc0_config2 helper_mtc0_config2_mips64 #define helper_mtc0_config2 helper_mtc0_config2_mips64
#define helper_mtc0_config3 helper_mtc0_config3_mips64
#define helper_mtc0_config4 helper_mtc0_config4_mips64 #define helper_mtc0_config4 helper_mtc0_config4_mips64
#define helper_mtc0_config5 helper_mtc0_config5_mips64 #define helper_mtc0_config5 helper_mtc0_config5_mips64
#define helper_mtc0_lladdr helper_mtc0_lladdr_mips64 #define helper_mtc0_lladdr helper_mtc0_lladdr_mips64

View file

@ -3540,6 +3540,7 @@
#define helper_mftc0_configx helper_mftc0_configx_mips64el #define helper_mftc0_configx helper_mftc0_configx_mips64el
#define helper_mtc0_config0 helper_mtc0_config0_mips64el #define helper_mtc0_config0 helper_mtc0_config0_mips64el
#define helper_mtc0_config2 helper_mtc0_config2_mips64el #define helper_mtc0_config2 helper_mtc0_config2_mips64el
#define helper_mtc0_config3 helper_mtc0_config3_mips64el
#define helper_mtc0_config4 helper_mtc0_config4_mips64el #define helper_mtc0_config4 helper_mtc0_config4_mips64el
#define helper_mtc0_config5 helper_mtc0_config5_mips64el #define helper_mtc0_config5 helper_mtc0_config5_mips64el
#define helper_mtc0_lladdr helper_mtc0_lladdr_mips64el #define helper_mtc0_lladdr helper_mtc0_lladdr_mips64el

View file

@ -3540,6 +3540,7 @@
#define helper_mftc0_configx helper_mftc0_configx_mipsel #define helper_mftc0_configx helper_mftc0_configx_mipsel
#define helper_mtc0_config0 helper_mtc0_config0_mipsel #define helper_mtc0_config0 helper_mtc0_config0_mipsel
#define helper_mtc0_config2 helper_mtc0_config2_mipsel #define helper_mtc0_config2 helper_mtc0_config2_mipsel
#define helper_mtc0_config3 helper_mtc0_config3_mipsel
#define helper_mtc0_config4 helper_mtc0_config4_mipsel #define helper_mtc0_config4 helper_mtc0_config4_mipsel
#define helper_mtc0_config5 helper_mtc0_config5_mipsel #define helper_mtc0_config5 helper_mtc0_config5_mipsel
#define helper_mtc0_lladdr helper_mtc0_lladdr_mipsel #define helper_mtc0_lladdr helper_mtc0_lladdr_mipsel

View file

@ -139,6 +139,7 @@ DEF_HELPER_2(mtc0_ebase, void, env, tl)
DEF_HELPER_2(mttc0_ebase, void, env, tl) DEF_HELPER_2(mttc0_ebase, void, env, tl)
DEF_HELPER_2(mtc0_config0, void, env, tl) DEF_HELPER_2(mtc0_config0, void, env, tl)
DEF_HELPER_2(mtc0_config2, void, env, tl) DEF_HELPER_2(mtc0_config2, void, env, tl)
DEF_HELPER_2(mtc0_config3, void, env, tl)
DEF_HELPER_2(mtc0_config4, void, env, tl) DEF_HELPER_2(mtc0_config4, void, env, tl)
DEF_HELPER_2(mtc0_config5, void, env, tl) DEF_HELPER_2(mtc0_config5, void, env, tl)
DEF_HELPER_2(mtc0_lladdr, void, env, tl) DEF_HELPER_2(mtc0_lladdr, void, env, tl)

View file

@ -1502,6 +1502,14 @@ void helper_mtc0_config2(CPUMIPSState *env, target_ulong arg1)
env->CP0_Config2 = (env->CP0_Config2 & 0x8FFF0FFF); env->CP0_Config2 = (env->CP0_Config2 & 0x8FFF0FFF);
} }
void helper_mtc0_config3(CPUMIPSState *env, target_ulong arg1)
{
if (env->insn_flags & ASE_MICROMIPS) {
env->CP0_Config3 = (env->CP0_Config3 & ~(1 << CP0C3_ISA_ON_EXC)) |
(arg1 & (1 << CP0C3_ISA_ON_EXC));
}
}
void helper_mtc0_config4(CPUMIPSState *env, target_ulong arg1) void helper_mtc0_config4(CPUMIPSState *env, target_ulong arg1)
{ {
env->CP0_Config4 = (env->CP0_Config4 & (~env->CP0_Config4_rw_bitmask)) | env->CP0_Config4 = (env->CP0_Config4 & (~env->CP0_Config4_rw_bitmask)) |

View file

@ -5912,8 +5912,10 @@ static void gen_mtc0(DisasContext *ctx, TCGv arg, int reg, int sel)
ctx->bstate = BS_STOP; ctx->bstate = BS_STOP;
break; break;
case 3: case 3:
/* ignored, read only */ gen_helper_mtc0_config3(tcg_ctx, tcg_ctx->cpu_env, arg);
rn = "Config3"; rn = "Config3";
/* Stop translation as we may have switched the execution mode */
ctx->bstate = BS_STOP;
break; break;
case 4: case 4:
gen_helper_mtc0_config4(tcg_ctx, tcg_ctx->cpu_env, arg); gen_helper_mtc0_config4(tcg_ctx, tcg_ctx->cpu_env, arg);
@ -7165,8 +7167,10 @@ static void gen_dmtc0(DisasContext *ctx, TCGv arg, int reg, int sel)
ctx->bstate = BS_STOP; ctx->bstate = BS_STOP;
break; break;
case 3: case 3:
/* ignored */ gen_helper_mtc0_config3(tcg_ctx, tcg_ctx->cpu_env, arg);
rn = "Config3"; rn = "Config3";
/* Stop translation as we may have switched the execution mode */
ctx->bstate = BS_STOP;
break; break;
case 4: case 4:
/* currently ignored */ /* currently ignored */