|
@@ -4879,6 +4879,53 @@ DISAS_INSN(trap)
|
|
gen_exception(s, s->pc, EXCP_TRAP0 + (insn & 0xf));
|
|
gen_exception(s, s->pc, EXCP_TRAP0 + (insn & 0xf));
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static void do_trapcc(DisasContext *s, DisasCompare *c)
|
|
|
|
+{
|
|
|
|
+ if (c->tcond != TCG_COND_NEVER) {
|
|
|
|
+ TCGLabel *over = NULL;
|
|
|
|
+
|
|
|
|
+ update_cc_op(s);
|
|
|
|
+
|
|
|
|
+ if (c->tcond != TCG_COND_ALWAYS) {
|
|
|
|
+ /* Jump over if !c. */
|
|
|
|
+ over = gen_new_label();
|
|
|
|
+ tcg_gen_brcond_i32(tcg_invert_cond(c->tcond), c->v1, c->v2, over);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ tcg_gen_movi_i32(QREG_PC, s->pc);
|
|
|
|
+ gen_raise_exception_format2(s, EXCP_TRAPCC, s->base.pc_next);
|
|
|
|
+
|
|
|
|
+ if (over != NULL) {
|
|
|
|
+ gen_set_label(over);
|
|
|
|
+ s->base.is_jmp = DISAS_NEXT;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ free_cond(c);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+DISAS_INSN(trapcc)
|
|
|
|
+{
|
|
|
|
+ DisasCompare c;
|
|
|
|
+
|
|
|
|
+ /* Consume and discard the immediate operand. */
|
|
|
|
+ switch (extract32(insn, 0, 3)) {
|
|
|
|
+ case 2: /* trapcc.w */
|
|
|
|
+ (void)read_im16(env, s);
|
|
|
|
+ break;
|
|
|
|
+ case 3: /* trapcc.l */
|
|
|
|
+ (void)read_im32(env, s);
|
|
|
|
+ break;
|
|
|
|
+ case 4: /* trapcc (no operand) */
|
|
|
|
+ break;
|
|
|
|
+ default:
|
|
|
|
+ /* trapcc registered with only valid opmodes */
|
|
|
|
+ g_assert_not_reached();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ gen_cc_cond(&c, s, extract32(insn, 8, 4));
|
|
|
|
+ do_trapcc(s, &c);
|
|
|
|
+}
|
|
|
|
+
|
|
static void gen_load_fcr(DisasContext *s, TCGv res, int reg)
|
|
static void gen_load_fcr(DisasContext *s, TCGv res, int reg)
|
|
{
|
|
{
|
|
switch (reg) {
|
|
switch (reg) {
|
|
@@ -6051,6 +6098,8 @@ void register_m68k_insns (CPUM68KState *env)
|
|
INSN(scc, 50c0, f0f8, CF_ISA_A); /* Scc.B Dx */
|
|
INSN(scc, 50c0, f0f8, CF_ISA_A); /* Scc.B Dx */
|
|
INSN(scc, 50c0, f0c0, M68000); /* Scc.B <EA> */
|
|
INSN(scc, 50c0, f0c0, M68000); /* Scc.B <EA> */
|
|
INSN(dbcc, 50c8, f0f8, M68000);
|
|
INSN(dbcc, 50c8, f0f8, M68000);
|
|
|
|
+ INSN(trapcc, 50fa, f0fe, TRAPCC); /* opmode 010, 011 */
|
|
|
|
+ INSN(trapcc, 50fc, f0ff, TRAPCC); /* opmode 100 */
|
|
INSN(tpf, 51f8, fff8, CF_ISA_A);
|
|
INSN(tpf, 51f8, fff8, CF_ISA_A);
|
|
|
|
|
|
/* Branch instructions. */
|
|
/* Branch instructions. */
|