dosbox-x/src/cpu/core_simple.cpp
Jonathan Campbell 16929fa921 cleanup
2025-01-20 02:38:48 -08:00

238 lines
6.4 KiB
C++

/*
* Copyright (C) 2002-2021 The DOSBox Team
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include <stdio.h>
#include <stdlib.h>
#include "cpu.h"
#include "lazyflags.h"
#include "callback.h"
#include "pic.h"
#include "fpu.h"
#define PRE_EXCEPTION { }
#define CPU_CORE CPU_ARCHTYPE_386
extern bool ignore_opcode_63;
#if C_DEBUG
#include "debug.h"
#endif
#include "paging.h"
#define SegBase(c) SegPhys(c)
#define LoadMb(off) mem_readb(off)
#define LoadMw(off) mem_readw(off)
#define LoadMd(off) mem_readd(off)
#define LoadMq(off) ((uint64_t)((uint64_t)mem_readd(off+4)<<32 | (uint64_t)mem_readd(off)))
#define SaveMb(off,val) mem_writeb(off,val)
#define SaveMw(off,val) mem_writew(off,val)
#define SaveMd(off,val) mem_writed(off,val)
#define SaveMq(off,val) {mem_writed(off,val&0xffffffff);mem_writed(off+4,(val>>32)&0xffffffff);}
extern Bitu cycle_count;
#if C_FPU
#define CPU_FPU 1u //Enable FPU escape instructions
#endif
#define CPU_PIC_CHECK 1u
#define CPU_TRAP_CHECK 1u
#define CPU_TRAP_DECODER CPU_Core_Simple_Trap_Run
#define OPCODE_NONE 0x000u
#define OPCODE_0F 0x100u
#define OPCODE_SIZE 0x200u
#define PREFIX_ADDR 0x1u
#define PREFIX_REP 0x2u
#define TEST_PREFIX_ADDR (core.prefixes & PREFIX_ADDR)
#define TEST_PREFIX_REP (core.prefixes & PREFIX_REP)
#define DO_PREFIX_SEG(_SEG) \
BaseDS=SegBase(_SEG); \
BaseSS=SegBase(_SEG); \
core.base_val_ds=_SEG; \
goto restart_opcode;
#define DO_PREFIX_ADDR() \
core.prefixes=(core.prefixes & ~PREFIX_ADDR) | \
(cpu.code.big ^ PREFIX_ADDR); \
core.ea_table=&EATable[(core.prefixes&1u) * 256u]; \
goto restart_opcode;
#define DO_PREFIX_REP(_ZERO) \
core.prefixes|=PREFIX_REP; \
core.rep_zero=_ZERO; \
goto restart_opcode;
#define REMEMBER_PREFIX(_x) last_prefix = (_x)
static uint8_t last_prefix;
typedef PhysPt (*GetEAHandler)(void);
static const uint32_t AddrMaskTable[2]={0x0000ffffu,0xffffffffu};
static struct {
Bitu opcode_index;
#if defined (_MSC_VER)
volatile HostPt cseip;
#else
HostPt cseip;
#endif
PhysPt base_ds,base_ss;
SegNames base_val_ds;
bool rep_zero;
Bitu prefixes;
GetEAHandler* ea_table;
} core;
#define GETIP ((uint32_t) ((uintptr_t)core.cseip - (uintptr_t)SegBase(cs) - (uintptr_t)MemBase))
#define SAVEIP reg_eip=GETIP;
#define LOADIP core.cseip=((HostPt) ((uintptr_t)MemBase + (uintptr_t)SegBase(cs) + (uintptr_t)reg_eip));
#define SegBase(c) SegPhys(c)
#define BaseDS core.base_ds
#define BaseSS core.base_ss
static INLINE void FetchDiscardb() {
core.cseip+=1;
}
static INLINE uint8_t FetchPeekb() {
uint8_t temp=host_readb(core.cseip);
return temp;
}
static INLINE uint8_t Fetchb() {
uint8_t temp=host_readb(core.cseip);
core.cseip+=1;
return temp;
}
static INLINE uint16_t Fetchw() {
uint16_t temp=host_readw(core.cseip);
core.cseip+=2;
return temp;
}
static INLINE uint32_t Fetchd() {
uint32_t temp=host_readd(core.cseip);
core.cseip+=4;
return temp;
}
#define Push_16 CPU_Push16
#define Push_32 CPU_Push32
#define Pop_16 CPU_Pop16
#define Pop_32 CPU_Pop32
bool CPU_RDMSR();
bool CPU_WRMSR();
bool CPU_SYSENTER();
bool CPU_SYSEXIT();
#include "instructions.h"
#include "core_normal/support.h"
#include "core_normal/string.h"
#define EALookupTable (core.ea_table)
/* NTS: This code fetches opcodes directly from MemBase plus an offset.
* Because of this, the simple core cannot be used to execute directly
* from ROM provided by an adapter since that requires memory I/O callbacks. */
Bits CPU_Core_Simple_Run(void) {
HostPt safety_limit;
/* simple core is incompatible with paging */
if (paging.enabled)
return CPU_Core_Normal_Run();
safety_limit = (HostPt)((size_t)MemBase + ((size_t)MEM_TotalPages() * (size_t)4096) - (size_t)16384); /* safety margin */
LOADIP;
if (core.cseip >= safety_limit) /* beyond the safety limit, use the normal core */
return CPU_Core_Normal_Run();
while (CPU_Cycles-->0) {
LOADIP;
/* Simple core optimizes for non-paged linear memory access and can break (segfault) if beyond end of memory */
if (core.cseip >= safety_limit) break;
last_prefix=MP_NONE;
core.opcode_index=cpu.code.big*0x200u;
core.prefixes=cpu.code.big;
core.ea_table=&EATable[cpu.code.big*256u];
BaseDS=SegBase(ds);
BaseSS=SegBase(ss);
core.base_val_ds=ds;
#if C_DEBUG
#if C_HEAVY_DEBUG
if (DEBUG_HeavyIsBreakpoint()) {
FillFlags();
return (Bits)debugCallback;
}
#endif
#endif
cycle_count++;
restart_opcode:
switch (core.opcode_index+Fetchb()) {
#include "core_normal/prefix_none.h"
#include "core_normal/prefix_0f.h"
#include "core_normal/prefix_66.h"
#include "core_normal/prefix_66_0f.h"
default:
illegal_opcode:
CPU_Exception(6,0);
continue;
gp_fault:
CPU_Exception(EXCEPTION_GP,0);
continue;
}
SAVEIP;
}
decode_end:
SAVEIP;
FillFlags();
return CBRET_NONE;
}
Bits CPU_Core_Simple_Trap_Run(void) {
Bits oldCycles = CPU_Cycles;
CPU_Cycles = 1;
cpu.trap_skip = false;
Bits ret=CPU_Core_Simple_Run();
if (!cpu.trap_skip) CPU_DebugException(DBINT_STEP,reg_eip);
CPU_Cycles = oldCycles-1;
cpudecoder = &CPU_Core_Simple_Run;
return ret;
}
void CPU_Core_Simple_Init(void) {
}