2017-06-03 00:45:12 +02:00
|
|
|
/*
|
|
|
|
|
* 86Box A hypervisor and IBM PC system emulator that specializes in
|
|
|
|
|
* running old operating systems and software designed for IBM
|
|
|
|
|
* PC systems and compatibles from 1981 through fairly recent
|
|
|
|
|
* system designs based on the PCI bus.
|
|
|
|
|
*
|
|
|
|
|
* This file is part of the 86Box distribution.
|
|
|
|
|
*
|
|
|
|
|
* Common 386 CPU code.
|
|
|
|
|
*
|
|
|
|
|
* Version: @(#)386_common.h 1.0.0 2017/05/30
|
|
|
|
|
*
|
|
|
|
|
* Author: Sarah Walker, <http://pcem-emulator.co.uk/>
|
|
|
|
|
* Miran Grca, <mgrca8@gmail.com>
|
|
|
|
|
* Copyright 2008-2017 Sarah Walker.
|
|
|
|
|
* Copyright 2016-2017 Miran Grca.
|
|
|
|
|
*/
|
|
|
|
|
|
2016-06-26 00:34:39 +02:00
|
|
|
extern uint16_t ea_rseg;
|
|
|
|
|
|
|
|
|
|
#undef readmemb
|
|
|
|
|
#undef writememb
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#define readmemb(s,a) ((readlookup2[(uint32_t)((s)+(a))>>12]==-1 || (s)==0xFFFFFFFF)?readmemb386l(s,a): *(uint8_t *)(readlookup2[(uint32_t)((s)+(a))>>12] + (uint32_t)((s) + (a))) )
|
|
|
|
|
#define readmemq(s,a) ((readlookup2[(uint32_t)((s)+(a))>>12]==-1 || (s)==0xFFFFFFFF || (((s)+(a))&0xFFF)>0xFF8)?readmemql(s,a):*(uint64_t *)(readlookup2[(uint32_t)((s)+(a))>>12]+(uint32_t)((s)+(a))))
|
|
|
|
|
|
|
|
|
|
#define writememb(s,a,v) if (writelookup2[(uint32_t)((s)+(a))>>12]==-1 || (s)==0xFFFFFFFF) writememb386l(s,a,v); else *(uint8_t *)(writelookup2[(uint32_t)((s) + (a)) >> 12] + (uint32_t)((s) + (a))) = v
|
|
|
|
|
|
|
|
|
|
#define writememw(s,a,v) if (writelookup2[(uint32_t)((s)+(a))>>12]==-1 || (s)==0xFFFFFFFF || (((s)+(a))&0xFFF)>0xFFE) writememwl(s,a,v); else *(uint16_t *)(writelookup2[(uint32_t)((s) + (a)) >> 12] + (uint32_t)((s) + (a))) = v
|
|
|
|
|
#define writememl(s,a,v) if (writelookup2[(uint32_t)((s)+(a))>>12]==-1 || (s)==0xFFFFFFFF || (((s)+(a))&0xFFF)>0xFFC) writememll(s,a,v); else *(uint32_t *)(writelookup2[(uint32_t)((s) + (a)) >> 12] + (uint32_t)((s) + (a))) = v
|
|
|
|
|
#define writememq(s,a,v) if (writelookup2[(uint32_t)((s)+(a))>>12]==-1 || (s)==0xFFFFFFFF || (((s)+(a))&0xFFF)>0xFF8) writememql(s,a,v); else *(uint64_t *)(writelookup2[(uint32_t)((s) + (a)) >> 12] + (uint32_t)((s) + (a))) = v
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
#define check_io_perm(port) if (!IOPLp || (eflags&VM_FLAG)) \
|
|
|
|
|
{ \
|
|
|
|
|
int tempi = checkio(port); \
|
2016-08-31 22:49:56 +02:00
|
|
|
if (cpu_state.abrt) return 1; \
|
2016-06-26 00:34:39 +02:00
|
|
|
if (tempi) \
|
|
|
|
|
{ \
|
|
|
|
|
x86gpf("check_io_perm(): no permission",0); \
|
|
|
|
|
return 1; \
|
|
|
|
|
} \
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define checkio_perm(port) if (!IOPLp || (eflags&VM_FLAG)) \
|
|
|
|
|
{ \
|
|
|
|
|
tempi = checkio(port); \
|
2016-08-31 22:49:56 +02:00
|
|
|
if (cpu_state.abrt) break; \
|
2016-06-26 00:34:39 +02:00
|
|
|
if (tempi) \
|
|
|
|
|
{ \
|
|
|
|
|
x86gpf("checkio_perm(): no permission",0); \
|
|
|
|
|
break; \
|
|
|
|
|
} \
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#define check_io_perm(port) if (msw&1 && ((CPL > IOPL) || (eflags&VM_FLAG))) \
|
|
|
|
|
{ \
|
|
|
|
|
int tempi = checkio(port); \
|
2016-08-31 22:49:56 +02:00
|
|
|
if (cpu_state.abrt) return 1; \
|
2016-06-26 00:34:39 +02:00
|
|
|
if (tempi) \
|
|
|
|
|
{ \
|
|
|
|
|
x86gpf("check_io_perm(): no permission",0); \
|
|
|
|
|
return 1; \
|
|
|
|
|
} \
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define checkio_perm(port) if (msw&1 && ((CPL > IOPL) || (eflags&VM_FLAG))) \
|
|
|
|
|
{ \
|
|
|
|
|
tempi = checkio(port); \
|
2016-08-31 22:49:56 +02:00
|
|
|
if (cpu_state.abrt) break; \
|
2016-06-26 00:34:39 +02:00
|
|
|
if (tempi) \
|
|
|
|
|
{ \
|
|
|
|
|
x86gpf("checkio_perm(): no permission",0); \
|
|
|
|
|
break; \
|
|
|
|
|
} \
|
|
|
|
|
}
|
2016-07-19 02:44:32 +02:00
|
|
|
|
|
|
|
|
#define CHECK_READ(chseg, low, high) \
|
2017-05-05 01:49:42 +02:00
|
|
|
if ((low < (chseg)->limit_low) || (high > (chseg)->limit_high) || ((msw & 1) && !(eflags & VM_FLAG) && (((chseg)->access & 10) == 8))) \
|
2016-07-19 02:44:32 +02:00
|
|
|
{ \
|
|
|
|
|
x86gpf("Limit check", 0); \
|
|
|
|
|
return 1; \
|
|
|
|
|
} \
|
|
|
|
|
if (msw&1 && !(eflags&VM_FLAG) && !((chseg)->access & 0x80)) \
|
|
|
|
|
{ \
|
|
|
|
|
if ((chseg) == &_ss) \
|
|
|
|
|
x86ss(NULL,(chseg)->seg & 0xfffc); \
|
|
|
|
|
else \
|
|
|
|
|
x86np("Read from seg not present", (chseg)->seg & 0xfffc); \
|
|
|
|
|
return 1; \
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define CHECK_WRITE(chseg, low, high) \
|
2017-05-05 01:49:42 +02:00
|
|
|
if ((low < (chseg)->limit_low) || (high > (chseg)->limit_high) || !((chseg)->access & 2) || ((msw & 1) && !(eflags & VM_FLAG) && ((chseg)->access & 8))) \
|
2016-07-19 02:44:32 +02:00
|
|
|
{ \
|
|
|
|
|
x86gpf("Limit check", 0); \
|
|
|
|
|
return 1; \
|
|
|
|
|
} \
|
|
|
|
|
if (msw&1 && !(eflags&VM_FLAG) && !((chseg)->access & 0x80)) \
|
|
|
|
|
{ \
|
|
|
|
|
if ((chseg) == &_ss) \
|
|
|
|
|
x86ss(NULL,(chseg)->seg & 0xfffc); \
|
|
|
|
|
else \
|
|
|
|
|
x86np("Write to seg not present", (chseg)->seg & 0xfffc); \
|
|
|
|
|
return 1; \
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define CHECK_WRITE_REP(chseg, low, high) \
|
|
|
|
|
if ((low < (chseg)->limit_low) || (high > (chseg)->limit_high)) \
|
|
|
|
|
{ \
|
|
|
|
|
x86gpf("Limit check", 0); \
|
|
|
|
|
break; \
|
|
|
|
|
} \
|
|
|
|
|
if (msw&1 && !(eflags&VM_FLAG) && !((chseg)->access & 0x80)) \
|
|
|
|
|
{ \
|
|
|
|
|
if ((chseg) == &_ss) \
|
|
|
|
|
x86ss(NULL,(chseg)->seg & 0xfffc); \
|
|
|
|
|
else \
|
|
|
|
|
x86np("Write (REP) to seg not present", (chseg)->seg & 0xfffc); \
|
|
|
|
|
break; \
|
|
|
|
|
}
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
#define NOTRM if (!(msw & 1) || (eflags & VM_FLAG))\
|
|
|
|
|
{ \
|
|
|
|
|
x86_int(6); \
|
|
|
|
|
return 1; \
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint8_t fastreadb(uint32_t a)
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
uint8_t *t;
|
|
|
|
|
|
|
|
|
|
if ((a >> 12) == pccache)
|
|
|
|
|
return *((uint8_t *)&pccache2[a]);
|
|
|
|
|
t = getpccache(a);
|
2016-08-31 22:49:56 +02:00
|
|
|
if (cpu_state.abrt)
|
2017-05-05 01:49:42 +02:00
|
|
|
return 0xFF;
|
2016-06-26 00:34:39 +02:00
|
|
|
pccache = a >> 12;
|
|
|
|
|
pccache2 = t;
|
|
|
|
|
return *((uint8_t *)&pccache2[a]);
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint16_t fastreadw(uint32_t a)
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
uint8_t *t;
|
|
|
|
|
uint16_t val;
|
|
|
|
|
if ((a&0xFFF)>0xFFE)
|
|
|
|
|
{
|
|
|
|
|
val = readmemb(0, a);
|
|
|
|
|
val |= (readmemb(0, a + 1) << 8);
|
|
|
|
|
return val;
|
|
|
|
|
}
|
|
|
|
|
if ((a>>12)==pccache) return *((uint16_t *)&pccache2[a]);
|
|
|
|
|
t = getpccache(a);
|
2016-08-31 22:49:56 +02:00
|
|
|
if (cpu_state.abrt)
|
2017-05-05 01:49:42 +02:00
|
|
|
return 0xff;
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
pccache = a >> 12;
|
|
|
|
|
pccache2 = t;
|
|
|
|
|
return *((uint16_t *)&pccache2[a]);
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint32_t fastreadl(uint32_t a)
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
uint8_t *t;
|
|
|
|
|
uint32_t val;
|
|
|
|
|
if ((a&0xFFF)<0xFFD)
|
|
|
|
|
{
|
|
|
|
|
if ((a>>12)!=pccache)
|
|
|
|
|
{
|
|
|
|
|
t = getpccache(a);
|
2016-08-31 22:49:56 +02:00
|
|
|
if (cpu_state.abrt)
|
2016-06-26 00:34:39 +02:00
|
|
|
return 0;
|
|
|
|
|
pccache2 = t;
|
|
|
|
|
pccache=a>>12;
|
2017-05-05 01:49:42 +02:00
|
|
|
/* return *((uint32_t *)&pccache2[a]); */
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
|
|
|
|
return *((uint32_t *)&pccache2[a]);
|
|
|
|
|
}
|
|
|
|
|
val =readmemb(0,a);
|
|
|
|
|
val |=(readmemb(0,a+1)<<8);
|
|
|
|
|
val |=(readmemb(0,a+2)<<16);
|
|
|
|
|
val |=(readmemb(0,a+3)<<24);
|
|
|
|
|
return val;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint8_t getbyte()
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
cpu_state.pc++;
|
|
|
|
|
return fastreadb(cs + (cpu_state.pc - 1));
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint16_t getword()
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
cpu_state.pc+=2;
|
|
|
|
|
return fastreadw(cs+(cpu_state.pc-2));
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint32_t getlong()
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
cpu_state.pc+=4;
|
|
|
|
|
return fastreadl(cs+(cpu_state.pc-4));
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint64_t getquad()
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
cpu_state.pc+=8;
|
|
|
|
|
return fastreadl(cs+(cpu_state.pc-8)) | ((uint64_t)fastreadl(cs+(cpu_state.pc-4)) << 32);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint8_t geteab()
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3)
|
|
|
|
|
return (cpu_rm & 4) ? cpu_state.regs[cpu_rm & 3].b.h : cpu_state.regs[cpu_rm&3].b.l;
|
2016-06-26 00:34:39 +02:00
|
|
|
if (eal_r)
|
|
|
|
|
return *(uint8_t *)eal_r;
|
2016-08-20 03:40:12 +02:00
|
|
|
return readmemb(easeg, cpu_state.eaaddr);
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint16_t geteaw()
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3)
|
|
|
|
|
return cpu_state.regs[cpu_rm].w;
|
2017-05-05 01:49:42 +02:00
|
|
|
/* cycles-=3; */
|
2016-06-26 00:34:39 +02:00
|
|
|
if (eal_r)
|
|
|
|
|
return *(uint16_t *)eal_r;
|
2016-08-20 03:40:12 +02:00
|
|
|
return readmemw(easeg, cpu_state.eaaddr);
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint32_t geteal()
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
2016-08-15 01:34:46 +02:00
|
|
|
if (cpu_mod == 3)
|
|
|
|
|
return cpu_state.regs[cpu_rm].l;
|
2017-05-05 01:49:42 +02:00
|
|
|
/* cycles-=3; */
|
2016-06-26 00:34:39 +02:00
|
|
|
if (eal_r)
|
|
|
|
|
return *eal_r;
|
2016-08-20 03:40:12 +02:00
|
|
|
return readmeml(easeg, cpu_state.eaaddr);
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint64_t geteaq()
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
2016-08-20 03:40:12 +02:00
|
|
|
return readmemq(easeg, cpu_state.eaaddr);
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint8_t geteab_mem()
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
if (eal_r) return *(uint8_t *)eal_r;
|
2016-08-20 03:40:12 +02:00
|
|
|
return readmemb(easeg,cpu_state.eaaddr);
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint16_t geteaw_mem()
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
if (eal_r) return *(uint16_t *)eal_r;
|
2016-08-20 03:40:12 +02:00
|
|
|
return readmemw(easeg,cpu_state.eaaddr);
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline uint32_t geteal_mem()
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
|
|
|
|
if (eal_r) return *eal_r;
|
2016-08-20 03:40:12 +02:00
|
|
|
return readmeml(easeg,cpu_state.eaaddr);
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
|
|
|
|
|
2017-05-05 01:49:42 +02:00
|
|
|
static __inline void seteaq(uint64_t v)
|
2016-06-26 00:34:39 +02:00
|
|
|
{
|
2016-08-20 03:40:12 +02:00
|
|
|
writememql(easeg, cpu_state.eaaddr, v);
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
|
|
|
|
|
2016-08-20 03:40:12 +02:00
|
|
|
#define seteab(v) if (cpu_mod!=3) { if (eal_w) *(uint8_t *)eal_w=v; else writememb386l(easeg,cpu_state.eaaddr,v); } else if (cpu_rm&4) cpu_state.regs[cpu_rm&3].b.h=v; else cpu_state.regs[cpu_rm].b.l=v
|
|
|
|
|
#define seteaw(v) if (cpu_mod!=3) { if (eal_w) *(uint16_t *)eal_w=v; else writememwl(easeg,cpu_state.eaaddr,v); } else cpu_state.regs[cpu_rm].w=v
|
|
|
|
|
#define seteal(v) if (cpu_mod!=3) { if (eal_w) *eal_w=v; else writememll(easeg,cpu_state.eaaddr,v); } else cpu_state.regs[cpu_rm].l=v
|
|
|
|
|
|
|
|
|
|
#define seteab_mem(v) if (eal_w) *(uint8_t *)eal_w=v; else writememb386l(easeg,cpu_state.eaaddr,v);
|
|
|
|
|
#define seteaw_mem(v) if (eal_w) *(uint16_t *)eal_w=v; else writememwl(easeg,cpu_state.eaaddr,v);
|
|
|
|
|
#define seteal_mem(v) if (eal_w) *eal_w=v; else writememll(easeg,cpu_state.eaaddr,v);
|
|
|
|
|
|
2016-06-26 00:34:39 +02:00
|
|
|
#define getbytef() ((uint8_t)(fetchdat)); cpu_state.pc++
|
|
|
|
|
#define getwordf() ((uint16_t)(fetchdat)); cpu_state.pc+=2
|
|
|
|
|
#define getbyte2f() ((uint8_t)(fetchdat>>8)); cpu_state.pc++
|
|
|
|
|
#define getword2f() ((uint16_t)(fetchdat>>8)); cpu_state.pc+=2
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#define rmdat rmdat32
|
|
|
|
|
#define fetchdat rmdat32
|
|
|
|
|
|
|
|
|
|
void x86_int(int num);
|