added CPL/IOPL support - fixed subtle inc/dec flag optimisation bug - added HLT instruction
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@134 c046a42c-6fe2-441c-8c8c-71466251a162
This commit is contained in:
parent
bf7c65bdf4
commit
982b431579
@ -130,6 +130,8 @@ typedef struct DisasContext {
|
|||||||
int addseg; /* non zero if either DS/ES/SS have a non zero base */
|
int addseg; /* non zero if either DS/ES/SS have a non zero base */
|
||||||
int f_st; /* currently unused */
|
int f_st; /* currently unused */
|
||||||
int vm86; /* vm86 mode */
|
int vm86; /* vm86 mode */
|
||||||
|
int cpl;
|
||||||
|
int iopl;
|
||||||
} DisasContext;
|
} DisasContext;
|
||||||
|
|
||||||
/* i386 arith/logic operations */
|
/* i386 arith/logic operations */
|
||||||
@ -2766,6 +2768,10 @@ long disas_insn(DisasContext *s, uint8_t *pc_start)
|
|||||||
break;
|
break;
|
||||||
case 0x6c: /* insS */
|
case 0x6c: /* insS */
|
||||||
case 0x6d:
|
case 0x6d:
|
||||||
|
if (s->cpl > s->iopl || s->vm86) {
|
||||||
|
/* NOTE: even for (E)CX = 0 the exception is raised */
|
||||||
|
gen_op_gpf(pc_start - s->cs_base);
|
||||||
|
} else {
|
||||||
if ((b & 1) == 0)
|
if ((b & 1) == 0)
|
||||||
ot = OT_BYTE;
|
ot = OT_BYTE;
|
||||||
else
|
else
|
||||||
@ -2775,9 +2781,14 @@ long disas_insn(DisasContext *s, uint8_t *pc_start)
|
|||||||
} else {
|
} else {
|
||||||
gen_string_es(s, ot, gen_op_ins);
|
gen_string_es(s, ot, gen_op_ins);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case 0x6e: /* outsS */
|
case 0x6e: /* outsS */
|
||||||
case 0x6f:
|
case 0x6f:
|
||||||
|
if (s->cpl > s->iopl || s->vm86) {
|
||||||
|
/* NOTE: even for (E)CX = 0 the exception is raised */
|
||||||
|
gen_op_gpf(pc_start - s->cs_base);
|
||||||
|
} else {
|
||||||
if ((b & 1) == 0)
|
if ((b & 1) == 0)
|
||||||
ot = OT_BYTE;
|
ot = OT_BYTE;
|
||||||
else
|
else
|
||||||
@ -2787,12 +2798,16 @@ long disas_insn(DisasContext *s, uint8_t *pc_start)
|
|||||||
} else {
|
} else {
|
||||||
gen_string_ds(s, ot, gen_op_outs);
|
gen_string_ds(s, ot, gen_op_outs);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/************************/
|
/************************/
|
||||||
/* port I/O */
|
/* port I/O */
|
||||||
case 0xe4:
|
case 0xe4:
|
||||||
case 0xe5:
|
case 0xe5:
|
||||||
|
if (s->cpl > s->iopl || s->vm86) {
|
||||||
|
gen_op_gpf(pc_start - s->cs_base);
|
||||||
|
} else {
|
||||||
if ((b & 1) == 0)
|
if ((b & 1) == 0)
|
||||||
ot = OT_BYTE;
|
ot = OT_BYTE;
|
||||||
else
|
else
|
||||||
@ -2801,9 +2816,13 @@ long disas_insn(DisasContext *s, uint8_t *pc_start)
|
|||||||
gen_op_movl_T0_im(val);
|
gen_op_movl_T0_im(val);
|
||||||
gen_op_in[ot]();
|
gen_op_in[ot]();
|
||||||
gen_op_mov_reg_T1[ot][R_EAX]();
|
gen_op_mov_reg_T1[ot][R_EAX]();
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case 0xe6:
|
case 0xe6:
|
||||||
case 0xe7:
|
case 0xe7:
|
||||||
|
if (s->cpl > s->iopl || s->vm86) {
|
||||||
|
gen_op_gpf(pc_start - s->cs_base);
|
||||||
|
} else {
|
||||||
if ((b & 1) == 0)
|
if ((b & 1) == 0)
|
||||||
ot = OT_BYTE;
|
ot = OT_BYTE;
|
||||||
else
|
else
|
||||||
@ -2812,9 +2831,13 @@ long disas_insn(DisasContext *s, uint8_t *pc_start)
|
|||||||
gen_op_movl_T0_im(val);
|
gen_op_movl_T0_im(val);
|
||||||
gen_op_mov_TN_reg[ot][1][R_EAX]();
|
gen_op_mov_TN_reg[ot][1][R_EAX]();
|
||||||
gen_op_out[ot]();
|
gen_op_out[ot]();
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case 0xec:
|
case 0xec:
|
||||||
case 0xed:
|
case 0xed:
|
||||||
|
if (s->cpl > s->iopl || s->vm86) {
|
||||||
|
gen_op_gpf(pc_start - s->cs_base);
|
||||||
|
} else {
|
||||||
if ((b & 1) == 0)
|
if ((b & 1) == 0)
|
||||||
ot = OT_BYTE;
|
ot = OT_BYTE;
|
||||||
else
|
else
|
||||||
@ -2822,9 +2845,13 @@ long disas_insn(DisasContext *s, uint8_t *pc_start)
|
|||||||
gen_op_mov_TN_reg[OT_WORD][0][R_EDX]();
|
gen_op_mov_TN_reg[OT_WORD][0][R_EDX]();
|
||||||
gen_op_in[ot]();
|
gen_op_in[ot]();
|
||||||
gen_op_mov_reg_T1[ot][R_EAX]();
|
gen_op_mov_reg_T1[ot][R_EAX]();
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case 0xee:
|
case 0xee:
|
||||||
case 0xef:
|
case 0xef:
|
||||||
|
if (s->cpl > s->iopl || s->vm86) {
|
||||||
|
gen_op_gpf(pc_start - s->cs_base);
|
||||||
|
} else {
|
||||||
if ((b & 1) == 0)
|
if ((b & 1) == 0)
|
||||||
ot = OT_BYTE;
|
ot = OT_BYTE;
|
||||||
else
|
else
|
||||||
@ -2832,6 +2859,7 @@ long disas_insn(DisasContext *s, uint8_t *pc_start)
|
|||||||
gen_op_mov_TN_reg[OT_WORD][0][R_EDX]();
|
gen_op_mov_TN_reg[OT_WORD][0][R_EDX]();
|
||||||
gen_op_mov_TN_reg[ot][1][R_EAX]();
|
gen_op_mov_TN_reg[ot][1][R_EAX]();
|
||||||
gen_op_out[ot]();
|
gen_op_out[ot]();
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/************************/
|
/************************/
|
||||||
@ -3219,8 +3247,7 @@ long disas_insn(DisasContext *s, uint8_t *pc_start)
|
|||||||
break;
|
break;
|
||||||
case 0xcd: /* int N */
|
case 0xcd: /* int N */
|
||||||
val = ldub(s->pc++);
|
val = ldub(s->pc++);
|
||||||
/* XXX: currently we ignore the interrupt number */
|
gen_op_int_im(val, pc_start - s->cs_base);
|
||||||
gen_op_int_im(pc_start - s->cs_base);
|
|
||||||
s->is_jmp = 1;
|
s->is_jmp = 1;
|
||||||
break;
|
break;
|
||||||
case 0xce: /* into */
|
case 0xce: /* into */
|
||||||
@ -3229,16 +3256,30 @@ long disas_insn(DisasContext *s, uint8_t *pc_start)
|
|||||||
gen_op_into();
|
gen_op_into();
|
||||||
break;
|
break;
|
||||||
case 0xfa: /* cli */
|
case 0xfa: /* cli */
|
||||||
if (s->vm86)
|
if (!s->vm86) {
|
||||||
gen_op_cli_vm();
|
if (s->cpl <= s->iopl)
|
||||||
else
|
|
||||||
gen_op_cli();
|
gen_op_cli();
|
||||||
|
else
|
||||||
|
gen_op_gpf(pc_start - s->cs_base);
|
||||||
|
} else {
|
||||||
|
if (s->iopl == 3)
|
||||||
|
gen_op_cli();
|
||||||
|
else
|
||||||
|
gen_op_cli_vm();
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case 0xfb: /* sti */
|
case 0xfb: /* sti */
|
||||||
if (s->vm86)
|
if (!s->vm86) {
|
||||||
gen_op_sti_vm(pc_start - s->cs_base);
|
if (s->cpl <= s->iopl)
|
||||||
else
|
|
||||||
gen_op_sti();
|
gen_op_sti();
|
||||||
|
else
|
||||||
|
gen_op_gpf(pc_start - s->cs_base);
|
||||||
|
} else {
|
||||||
|
if (s->iopl == 3)
|
||||||
|
gen_op_sti();
|
||||||
|
else
|
||||||
|
gen_op_sti_vm(pc_start - s->cs_base);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case 0x62: /* bound */
|
case 0x62: /* bound */
|
||||||
ot = dflag ? OT_LONG : OT_WORD;
|
ot = dflag ? OT_LONG : OT_WORD;
|
||||||
@ -3286,6 +3327,13 @@ long disas_insn(DisasContext *s, uint8_t *pc_start)
|
|||||||
case 0x1a2: /* cpuid */
|
case 0x1a2: /* cpuid */
|
||||||
gen_op_cpuid();
|
gen_op_cpuid();
|
||||||
break;
|
break;
|
||||||
|
case 0xf4: /* hlt */
|
||||||
|
if (s->cpl == 0) {
|
||||||
|
/* ignored */
|
||||||
|
} else {
|
||||||
|
gen_op_gpf(pc_start - s->cs_base);
|
||||||
|
}
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
goto illegal_op;
|
goto illegal_op;
|
||||||
}
|
}
|
||||||
@ -3315,6 +3363,10 @@ static uint16_t opc_read_flags[NB_OPS] = {
|
|||||||
[INDEX_op_sbbw_T0_T1_cc] = CC_C,
|
[INDEX_op_sbbw_T0_T1_cc] = CC_C,
|
||||||
[INDEX_op_sbbl_T0_T1_cc] = CC_C,
|
[INDEX_op_sbbl_T0_T1_cc] = CC_C,
|
||||||
|
|
||||||
|
/* subtle: due to the incl/decl implementation, C is used */
|
||||||
|
[INDEX_op_incl_T0_cc] = CC_C,
|
||||||
|
[INDEX_op_decl_T0_cc] = CC_C,
|
||||||
|
|
||||||
[INDEX_op_into] = CC_O,
|
[INDEX_op_into] = CC_O,
|
||||||
|
|
||||||
[INDEX_op_jo_cc] = CC_O,
|
[INDEX_op_jo_cc] = CC_O,
|
||||||
@ -3416,8 +3468,9 @@ static uint16_t opc_write_flags[NB_OPS] = {
|
|||||||
[INDEX_op_xorl_T0_T1_cc] = CC_OSZAPC,
|
[INDEX_op_xorl_T0_T1_cc] = CC_OSZAPC,
|
||||||
[INDEX_op_cmpl_T0_T1_cc] = CC_OSZAPC,
|
[INDEX_op_cmpl_T0_T1_cc] = CC_OSZAPC,
|
||||||
[INDEX_op_negl_T0_cc] = CC_OSZAPC,
|
[INDEX_op_negl_T0_cc] = CC_OSZAPC,
|
||||||
[INDEX_op_incl_T0_cc] = CC_OSZAP,
|
/* subtle: due to the incl/decl implementation, C is used */
|
||||||
[INDEX_op_decl_T0_cc] = CC_OSZAP,
|
[INDEX_op_incl_T0_cc] = CC_OSZAPC,
|
||||||
|
[INDEX_op_decl_T0_cc] = CC_OSZAPC,
|
||||||
[INDEX_op_testl_T0_T1_cc] = CC_OSZAPC,
|
[INDEX_op_testl_T0_T1_cc] = CC_OSZAPC,
|
||||||
|
|
||||||
[INDEX_op_mulb_AL_T0] = CC_OSZAPC,
|
[INDEX_op_mulb_AL_T0] = CC_OSZAPC,
|
||||||
@ -3659,6 +3712,8 @@ int cpu_x86_gen_code(uint8_t *gen_code_buf, int max_code_size,
|
|||||||
dc->addseg = (flags >> GEN_FLAG_ADDSEG_SHIFT) & 1;
|
dc->addseg = (flags >> GEN_FLAG_ADDSEG_SHIFT) & 1;
|
||||||
dc->f_st = (flags >> GEN_FLAG_ST_SHIFT) & 7;
|
dc->f_st = (flags >> GEN_FLAG_ST_SHIFT) & 7;
|
||||||
dc->vm86 = (flags >> GEN_FLAG_VM_SHIFT) & 1;
|
dc->vm86 = (flags >> GEN_FLAG_VM_SHIFT) & 1;
|
||||||
|
dc->cpl = (flags >> GEN_FLAG_CPL_SHIFT) & 3;
|
||||||
|
dc->iopl = (flags >> GEN_FLAG_IOPL_SHIFT) & 3;
|
||||||
dc->cc_op = CC_OP_DYNAMIC;
|
dc->cc_op = CC_OP_DYNAMIC;
|
||||||
dc->cs_base = cs_base;
|
dc->cs_base = cs_base;
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user