2016-06-26 00:34:39 +02:00
|
|
|
static int opMOV_w_seg_a16(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
fetch_ea_16(fetchdat);
|
|
|
|
|
|
|
|
|
|
switch (rmdat & 0x38)
|
|
|
|
|
{
|
|
|
|
|
case 0x00: /*ES*/
|
|
|
|
|
seteaw(ES);
|
|
|
|
|
break;
|
|
|
|
|
case 0x08: /*CS*/
|
|
|
|
|
seteaw(CS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x18: /*DS*/
|
|
|
|
|
seteaw(DS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x10: /*SS*/
|
|
|
|
|
seteaw(SS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x20: /*FS*/
|
|
|
|
|
seteaw(FS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x28: /*GS*/
|
|
|
|
|
seteaw(GS);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-15 01:34:46 +02:00
|
|
|
CLOCK_CYCLES((cpu_mod == 3) ? 2 : 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
return abrt;
|
|
|
|
|
}
|
|
|
|
|
static int opMOV_w_seg_a32(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
fetch_ea_32(fetchdat);
|
|
|
|
|
|
|
|
|
|
switch (rmdat & 0x38)
|
|
|
|
|
{
|
|
|
|
|
case 0x00: /*ES*/
|
|
|
|
|
seteaw(ES);
|
|
|
|
|
break;
|
|
|
|
|
case 0x08: /*CS*/
|
|
|
|
|
seteaw(CS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x18: /*DS*/
|
|
|
|
|
seteaw(DS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x10: /*SS*/
|
|
|
|
|
seteaw(SS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x20: /*FS*/
|
|
|
|
|
seteaw(FS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x28: /*GS*/
|
|
|
|
|
seteaw(GS);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-15 01:34:46 +02:00
|
|
|
CLOCK_CYCLES((cpu_mod == 3) ? 2 : 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
return abrt;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int opMOV_l_seg_a16(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
fetch_ea_16(fetchdat);
|
|
|
|
|
|
|
|
|
|
switch (rmdat & 0x38)
|
|
|
|
|
{
|
|
|
|
|
case 0x00: /*ES*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = ES;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(ES);
|
|
|
|
|
break;
|
|
|
|
|
case 0x08: /*CS*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = CS;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(CS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x18: /*DS*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = DS;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(DS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x10: /*SS*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = SS;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(SS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x20: /*FS*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = FS;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(FS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x28: /*GS*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = GS;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(GS);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-15 01:34:46 +02:00
|
|
|
CLOCK_CYCLES((cpu_mod == 3) ? 2 : 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
return abrt;
|
|
|
|
|
}
|
|
|
|
|
static int opMOV_l_seg_a32(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
fetch_ea_32(fetchdat);
|
|
|
|
|
|
|
|
|
|
switch (rmdat & 0x38)
|
|
|
|
|
{
|
|
|
|
|
case 0x00: /*ES*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = ES;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(ES);
|
|
|
|
|
break;
|
|
|
|
|
case 0x08: /*CS*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = CS;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(CS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x18: /*DS*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = DS;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(DS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x10: /*SS*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = SS;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(SS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x20: /*FS*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = FS;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(FS);
|
|
|
|
|
break;
|
|
|
|
|
case 0x28: /*GS*/
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3) cpu_state.regs[cpu_rm].l = GS;
|
2016-06-26 00:34:39 +02:00
|
|
|
else seteaw(GS);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-15 01:34:46 +02:00
|
|
|
CLOCK_CYCLES((cpu_mod == 3) ? 2 : 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
return abrt;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int opMOV_seg_w_a16(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
uint16_t new_seg;
|
|
|
|
|
|
|
|
|
|
fetch_ea_16(fetchdat);
|
|
|
|
|
|
|
|
|
|
new_seg=geteaw(); if (abrt) return 1;
|
|
|
|
|
|
|
|
|
|
switch (rmdat & 0x38)
|
|
|
|
|
{
|
|
|
|
|
case 0x00: /*ES*/
|
|
|
|
|
loadseg(new_seg, &_es);
|
|
|
|
|
break;
|
|
|
|
|
case 0x18: /*DS*/
|
|
|
|
|
loadseg(new_seg, &_ds);
|
|
|
|
|
break;
|
|
|
|
|
case 0x10: /*SS*/
|
|
|
|
|
loadseg(new_seg, &_ss);
|
|
|
|
|
if (abrt) return 1;
|
|
|
|
|
oldpc = cpu_state.pc;
|
|
|
|
|
op32 = use32;
|
|
|
|
|
ssegs = 0;
|
|
|
|
|
ea_seg = &_ds;
|
|
|
|
|
fetchdat = fastreadl(cs + cpu_state.pc);
|
|
|
|
|
cpu_state.pc++;
|
|
|
|
|
if (abrt) return 1;
|
|
|
|
|
x86_opcodes[(fetchdat & 0xff) | op32](fetchdat >> 8);
|
|
|
|
|
return 1;
|
|
|
|
|
case 0x20: /*FS*/
|
|
|
|
|
loadseg(new_seg, &_fs);
|
|
|
|
|
break;
|
|
|
|
|
case 0x28: /*GS*/
|
|
|
|
|
loadseg(new_seg, &_gs);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-15 01:34:46 +02:00
|
|
|
CLOCK_CYCLES((cpu_mod == 3) ? 2 : 5);
|
2016-06-26 00:34:39 +02:00
|
|
|
return abrt;
|
|
|
|
|
}
|
|
|
|
|
static int opMOV_seg_w_a32(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
uint16_t new_seg;
|
|
|
|
|
|
|
|
|
|
fetch_ea_32(fetchdat);
|
|
|
|
|
|
|
|
|
|
new_seg=geteaw(); if (abrt) return 1;
|
|
|
|
|
|
|
|
|
|
switch (rmdat & 0x38)
|
|
|
|
|
{
|
|
|
|
|
case 0x00: /*ES*/
|
|
|
|
|
loadseg(new_seg, &_es);
|
|
|
|
|
break;
|
|
|
|
|
case 0x18: /*DS*/
|
|
|
|
|
loadseg(new_seg, &_ds);
|
|
|
|
|
break;
|
|
|
|
|
case 0x10: /*SS*/
|
|
|
|
|
loadseg(new_seg, &_ss);
|
|
|
|
|
if (abrt) return 1;
|
|
|
|
|
oldpc = cpu_state.pc;
|
|
|
|
|
op32 = use32;
|
|
|
|
|
ssegs = 0;
|
|
|
|
|
ea_seg = &_ds;
|
|
|
|
|
fetchdat = fastreadl(cs + cpu_state.pc);
|
|
|
|
|
cpu_state.pc++;
|
|
|
|
|
if (abrt) return 1;
|
|
|
|
|
x86_opcodes[(fetchdat & 0xff) | op32](fetchdat >> 8);
|
|
|
|
|
return 1;
|
|
|
|
|
case 0x20: /*FS*/
|
|
|
|
|
loadseg(new_seg, &_fs);
|
|
|
|
|
break;
|
|
|
|
|
case 0x28: /*GS*/
|
|
|
|
|
loadseg(new_seg, &_gs);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-15 01:34:46 +02:00
|
|
|
CLOCK_CYCLES((cpu_mod == 3) ? 2 : 5);
|
2016-06-26 00:34:39 +02:00
|
|
|
return abrt;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int opLDS_w_a16(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
uint16_t addr, seg;
|
|
|
|
|
|
|
|
|
|
fetch_ea_16(fetchdat);
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmemw(easeg, eaaddr);
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 2); if (abrt) return 1;
|
|
|
|
|
loadseg(seg, &_ds); if (abrt) return 1;
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].w = addr;
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
CLOCK_CYCLES(7);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
static int opLDS_w_a32(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
uint16_t addr, seg;
|
|
|
|
|
|
|
|
|
|
fetch_ea_32(fetchdat);
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmemw(easeg, eaaddr);
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 2); if (abrt) return 1;
|
|
|
|
|
loadseg(seg, &_ds); if (abrt) return 1;
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].w = addr;
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
CLOCK_CYCLES(7);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
static int opLDS_l_a16(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
uint32_t addr;
|
|
|
|
|
uint16_t seg;
|
|
|
|
|
|
|
|
|
|
fetch_ea_16(fetchdat);
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmeml(easeg, eaaddr);
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 4); if (abrt) return 1;
|
|
|
|
|
loadseg(seg, &_ds); if (abrt) return 1;
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].l = addr;
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
CLOCK_CYCLES(7);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
static int opLDS_l_a32(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
uint32_t addr;
|
|
|
|
|
uint16_t seg;
|
|
|
|
|
|
|
|
|
|
fetch_ea_32(fetchdat);
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmeml(easeg, eaaddr);
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 4); if (abrt) return 1;
|
|
|
|
|
loadseg(seg, &_ds); if (abrt) return 1;
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].l = addr;
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
CLOCK_CYCLES(7);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int opLSS_w_a16(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
uint16_t addr, seg;
|
|
|
|
|
|
|
|
|
|
fetch_ea_16(fetchdat);
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmemw(easeg, eaaddr);
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 2); if (abrt) return 1;
|
|
|
|
|
loadseg(seg, &_ss); if (abrt) return 1;
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].w = addr;
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
CLOCK_CYCLES(7);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
static int opLSS_w_a32(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
uint16_t addr, seg;
|
|
|
|
|
|
|
|
|
|
fetch_ea_32(fetchdat);
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmemw(easeg, eaaddr);
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 2); if (abrt) return 1;
|
|
|
|
|
loadseg(seg, &_ss); if (abrt) return 1;
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].w = addr;
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
CLOCK_CYCLES(7);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
static int opLSS_l_a16(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
uint32_t addr;
|
|
|
|
|
uint16_t seg;
|
|
|
|
|
|
|
|
|
|
fetch_ea_16(fetchdat);
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmeml(easeg, eaaddr);
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 4); if (abrt) return 1;
|
|
|
|
|
loadseg(seg, &_ss); if (abrt) return 1;
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].l = addr;
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
CLOCK_CYCLES(7);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
static int opLSS_l_a32(uint32_t fetchdat)
|
|
|
|
|
{
|
|
|
|
|
uint32_t addr;
|
|
|
|
|
uint16_t seg;
|
|
|
|
|
|
|
|
|
|
fetch_ea_32(fetchdat);
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3);
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmeml(easeg, eaaddr);
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 4); if (abrt) return 1;
|
|
|
|
|
loadseg(seg, &_ss); if (abrt) return 1;
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].l = addr;
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
CLOCK_CYCLES(7);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define opLsel(name, sel) \
|
|
|
|
|
static int opL ## name ## _w_a16(uint32_t fetchdat) \
|
|
|
|
|
{ \
|
|
|
|
|
uint16_t addr, seg; \
|
|
|
|
|
\
|
|
|
|
|
fetch_ea_16(fetchdat); \
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3); \
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmemw(easeg, eaaddr); \
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 2); if (abrt) return 1; \
|
|
|
|
|
loadseg(seg, &sel); if (abrt) return 1; \
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].w = addr; \
|
2016-06-26 00:34:39 +02:00
|
|
|
\
|
|
|
|
|
CLOCK_CYCLES(7); \
|
|
|
|
|
return 0; \
|
|
|
|
|
} \
|
|
|
|
|
\
|
|
|
|
|
static int opL ## name ## _w_a32(uint32_t fetchdat) \
|
|
|
|
|
{ \
|
|
|
|
|
uint16_t addr, seg; \
|
|
|
|
|
\
|
|
|
|
|
fetch_ea_32(fetchdat); \
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3); \
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmemw(easeg, eaaddr); \
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 2); if (abrt) return 1; \
|
|
|
|
|
loadseg(seg, &sel); if (abrt) return 1; \
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].w = addr; \
|
2016-06-26 00:34:39 +02:00
|
|
|
\
|
|
|
|
|
CLOCK_CYCLES(7); \
|
|
|
|
|
return 0; \
|
|
|
|
|
} \
|
|
|
|
|
\
|
|
|
|
|
static int opL ## name ## _l_a16(uint32_t fetchdat) \
|
|
|
|
|
{ \
|
|
|
|
|
uint32_t addr; \
|
|
|
|
|
uint16_t seg; \
|
|
|
|
|
\
|
|
|
|
|
fetch_ea_16(fetchdat); \
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3); \
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmeml(easeg, eaaddr); \
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 4); if (abrt) return 1; \
|
|
|
|
|
loadseg(seg, &sel); if (abrt) return 1; \
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].l = addr; \
|
2016-06-26 00:34:39 +02:00
|
|
|
\
|
|
|
|
|
CLOCK_CYCLES(7); \
|
|
|
|
|
return 0; \
|
|
|
|
|
} \
|
|
|
|
|
\
|
|
|
|
|
static int opL ## name ## _l_a32(uint32_t fetchdat) \
|
|
|
|
|
{ \
|
|
|
|
|
uint32_t addr; \
|
|
|
|
|
uint16_t seg; \
|
|
|
|
|
\
|
|
|
|
|
fetch_ea_32(fetchdat); \
|
2016-08-15 01:34:46 +02:00
|
|
|
ILLEGAL_ON(cpu_mod == 3); \
|
2016-06-26 00:34:39 +02:00
|
|
|
addr = readmeml(easeg, eaaddr); \
|
|
|
|
|
seg = readmemw(easeg, eaaddr + 4); if (abrt) return 1; \
|
|
|
|
|
loadseg(seg, &sel); if (abrt) return 1; \
|
2016-08-15 01:34:46 +02:00
|
|
|
cpu_state.regs[cpu_reg].l = addr; \
|
2016-06-26 00:34:39 +02:00
|
|
|
\
|
|
|
|
|
CLOCK_CYCLES(7); \
|
|
|
|
|
return 0; \
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
opLsel(ES, _es)
|
|
|
|
|
opLsel(FS, _fs)
|
|
|
|
|
opLsel(GS, _gs)
|