mirror of
https://github.com/RetroDECK/Supermodel.git
synced 2024-11-23 14:15:40 +00:00
341 lines
10 KiB
C++
341 lines
10 KiB
C++
/**
|
|
** Supermodel
|
|
** A Sega Model 3 Arcade Emulator.
|
|
** Copyright 2011 Bart Trzynadlowski, Nik Henson
|
|
**
|
|
** This file is part of Supermodel.
|
|
**
|
|
** Supermodel 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 3 of the License, or (at your option)
|
|
** any later version.
|
|
**
|
|
** Supermodel 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 Supermodel. If not, see <http://www.gnu.org/licenses/>.
|
|
**/
|
|
|
|
/*
|
|
* Turbo68KDebug.h
|
|
*/
|
|
|
|
#ifdef SUPERMODEL_DEBUGGER
|
|
#ifndef INCLUDED_TURBO68KDEBUG_H
|
|
#define INCLUDED_TURBO68KDEBUG_H
|
|
|
|
#include "Debugger/CPU/68KDebug.h"
|
|
#include "Types.h"
|
|
|
|
#include "CPU/68K/Turbo68K/Turbo68K.h"
|
|
|
|
#define TBO68K_REG_SP 0
|
|
#define TBO68K_REG_SR 1
|
|
|
|
#define USE_NATIVE_READ 0
|
|
#define USE_NATIVE_WRITE 0
|
|
|
|
namespace Debugger
|
|
{
|
|
class CTurbo68KDebug;
|
|
|
|
static CTurbo68KDebug *debug = NULL;
|
|
|
|
typedef bool (*DebugPtr)(TURBO68K_INT32 pc, TURBO68K_INT32 opcode);
|
|
typedef void (*IntAckPtr)(TURBO68K_UINT32 intVec);
|
|
|
|
static DebugPtr origDebugPtr = NULL;
|
|
static IntAckPtr origIntAckPtr = NULL;
|
|
|
|
static TURBO68K_DATAREGION *origRead8Regions;
|
|
static TURBO68K_DATAREGION *origRead16Regions;
|
|
static TURBO68K_DATAREGION *origRead32Regions;
|
|
static TURBO68K_DATAREGION *origWrite8Regions;
|
|
static TURBO68K_DATAREGION *origWrite16Regions;
|
|
static TURBO68K_DATAREGION *origWrite32Regions;
|
|
|
|
static TURBO68K_DATAREGION debugRead8Regions[] =
|
|
{
|
|
{ 0x000000, 0xFFFFFF, TURBO68K_NULL, TURBO68K_NULL },
|
|
{ -1, -1, TURBO68K_NULL, TURBO68K_NULL }
|
|
};
|
|
static TURBO68K_DATAREGION debugRead16Regions[] =
|
|
{
|
|
{ 0x000000, 0xFFFFFF, TURBO68K_NULL, TURBO68K_NULL },
|
|
{ -1, -1, TURBO68K_NULL, TURBO68K_NULL }
|
|
};
|
|
static TURBO68K_DATAREGION debugRead32Regions[] =
|
|
{
|
|
{ 0x000000, 0xFFFFFF, TURBO68K_NULL, TURBO68K_NULL },
|
|
{ -1, -1, TURBO68K_NULL, TURBO68K_NULL }
|
|
};
|
|
static TURBO68K_DATAREGION debugWrite8Regions[] =
|
|
{
|
|
{ 0x000000, 0xFFFFFF, TURBO68K_NULL, TURBO68K_NULL },
|
|
{ -1, -1, TURBO68K_NULL, TURBO68K_NULL }
|
|
};
|
|
static TURBO68K_DATAREGION debugWrite16Regions[] =
|
|
{
|
|
{ 0x000000, 0xFFFFFF, TURBO68K_NULL, TURBO68K_NULL },
|
|
{ -1, -1, TURBO68K_NULL, TURBO68K_NULL }
|
|
};
|
|
static TURBO68K_DATAREGION debugWrite32Regions[] =
|
|
{
|
|
{ 0x000000, 0xFFFFFF, TURBO68K_NULL, TURBO68K_NULL },
|
|
{ -1, -1, TURBO68K_NULL, TURBO68K_NULL }
|
|
};
|
|
|
|
static UINT32 GetSpecialReg(CCPUDebug *cpu, unsigned id);
|
|
static bool SetSpecialReg(CCPUDebug *cpu, unsigned id, UINT32 data);
|
|
static UINT32 GetDataReg(CCPUDebug *cpu, unsigned id);
|
|
static bool SetDataReg(CCPUDebug *cpu, unsigned id, UINT32 data) ;
|
|
static UINT32 GetAddressReg(CCPUDebug *cpu, unsigned id);
|
|
static bool SetAddressReg(CCPUDebug *cpu, unsigned id, UINT32 data);
|
|
|
|
static TURBO68K_UINT8 ReadByteDebug(TURBO68K_UINT32 addr);
|
|
static TURBO68K_UINT16 ReadWordDebug(TURBO68K_UINT32 addr);
|
|
static TURBO68K_UINT32 ReadLongDebug(TURBO68K_UINT32 addr);
|
|
static void WriteByteDebug(TURBO68K_UINT32 addr, TURBO68K_UINT8 data);
|
|
static void WriteWordDebug(TURBO68K_UINT32 addr, TURBO68K_UINT16 data);
|
|
static void WriteLongDebug(TURBO68K_UINT32 addr, TURBO68K_UINT32 data);
|
|
|
|
static TURBO68K_UINT8 ReadByteDirect(TURBO68K_UINT32 addr);
|
|
static TURBO68K_UINT16 ReadWordDirect(TURBO68K_UINT32 addr);
|
|
static TURBO68K_UINT32 ReadLongDirect(TURBO68K_UINT32 addr);
|
|
static void WriteByteDirect(TURBO68K_UINT32 addr, TURBO68K_UINT8 data);
|
|
static void WriteWordDirect(TURBO68K_UINT32 addr, TURBO68K_UINT16 data);
|
|
static void WriteLongDirect(TURBO68K_UINT32 addr, TURBO68K_UINT32 data);
|
|
|
|
/*
|
|
* CCPUDebug implementation for the Turbo68K Motorola 68000 emulator.
|
|
*/
|
|
class CTurbo68KDebug : public C68KDebug
|
|
{
|
|
private:
|
|
char m_drNames[8][3];
|
|
char m_arNames[8][3];
|
|
|
|
UINT32 m_resetAddr;
|
|
|
|
protected:
|
|
UINT32 GetSP();
|
|
|
|
public:
|
|
CTurbo68KDebug(const char *name);
|
|
|
|
virtual ~CTurbo68KDebug();
|
|
|
|
// CCPUDebug methods
|
|
|
|
void AttachToCPU();
|
|
|
|
void DetachFromCPU();
|
|
|
|
UINT32 GetResetAddr();
|
|
|
|
bool UpdatePC(UINT32 pc);
|
|
|
|
bool ForceException(CException *ex);
|
|
|
|
bool ForceInterrupt(CInterrupt *in);
|
|
|
|
UINT64 ReadMem(UINT32 addr, unsigned dataSize);
|
|
|
|
bool WriteMem(UINT32 addr, unsigned dataSize, UINT64 data);
|
|
};
|
|
|
|
//
|
|
// Inlined functions
|
|
//
|
|
|
|
typedef TURBO68K_UINT8 (*ReadByteFPtr)(TURBO68K_UINT32);
|
|
typedef TURBO68K_UINT16 (*ReadWordFPtr)(TURBO68K_UINT32);
|
|
typedef TURBO68K_UINT32 (*ReadLongFPtr)(TURBO68K_UINT32);
|
|
typedef void (*WriteByteFPtr)(TURBO68K_UINT32, TURBO68K_UINT8);
|
|
typedef void (*WriteWordFPtr)(TURBO68K_UINT32, TURBO68K_UINT16);
|
|
typedef void (*WriteLongFPtr)(TURBO68K_UINT32, TURBO68K_UINT32);
|
|
|
|
inline TURBO68K_UINT8 ReadByteDirect(TURBO68K_UINT32 addr)
|
|
{
|
|
#if USE_NATIVE_READ
|
|
Turbo68KSetReadByte(origRead8Regions, TURBO68K_NULL);
|
|
TURBO68K_UINT8 data = Turbo68KReadByte(addr);
|
|
Turbo68KSetReadByte(debugRead8Regions, TURBO68K_NULL);
|
|
return data;
|
|
#else
|
|
for (TURBO68K_DATAREGION *region = origRead8Regions; region->ptr != TURBO68K_NULL || region->handler != TURBO68K_NULL; region++)
|
|
{
|
|
if (region->base <= addr && addr <= region->limit)
|
|
{
|
|
if (region->ptr != TURBO68K_NULL)
|
|
{
|
|
// Turbo68K requires native memory to be byte swapped, so must reverse this when reading bytes
|
|
TURBO68K_UINT8 *dataP = (TURBO68K_UINT8*)(region->ptr + (addr^1));
|
|
return *dataP;
|
|
}
|
|
else
|
|
{
|
|
ReadByteFPtr fPtr = (ReadByteFPtr)region->handler;
|
|
return fPtr(addr);
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
#endif
|
|
}
|
|
|
|
inline TURBO68K_UINT16 ReadWordDirect(TURBO68K_UINT32 addr)
|
|
{
|
|
#if USE_NATIVE_READ
|
|
Turbo68KSetReadWord(origRead16Regions, TURBO68K_NULL);
|
|
TURBO68K_UINT16 data = Turbo68KReadWord(addr);
|
|
Turbo68KSetReadWord(debugRead16Regions, TURBO68K_NULL);
|
|
return data;
|
|
#else
|
|
for (TURBO68K_DATAREGION *region = origRead16Regions; region->ptr != TURBO68K_NULL || region->handler != TURBO68K_NULL; region++)
|
|
{
|
|
if (region->base <= addr && addr <= region->limit)
|
|
{
|
|
if (region->ptr != TURBO68K_NULL)
|
|
{
|
|
// Turbo68K requires native memory to be byte swapped, so must reverse this when reading bytes
|
|
TURBO68K_UINT16 *dataP = (TURBO68K_UINT16*)(region->ptr + addr);
|
|
return *dataP;
|
|
}
|
|
else
|
|
{
|
|
ReadWordFPtr fPtr = (ReadWordFPtr)region->handler;
|
|
return fPtr(addr);
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
#endif
|
|
}
|
|
|
|
inline TURBO68K_UINT32 ReadLongDirect(TURBO68K_UINT32 addr)
|
|
{
|
|
#if USE_NATIVE_READ
|
|
Turbo68KSetReadLong(origRead32Regions, TURBO68K_NULL);
|
|
TURBO68K_UINT32 data = Turbo68KReadLong(addr);
|
|
Turbo68KSetReadLong(debugRead32Regions, TURBO68K_NULL);
|
|
return data;
|
|
#else
|
|
for (TURBO68K_DATAREGION *region = origRead32Regions; region->ptr != TURBO68K_NULL || region->handler != TURBO68K_NULL; region++)
|
|
{
|
|
if (region->base <= addr && addr <= region->limit)
|
|
{
|
|
if (region->ptr != TURBO68K_NULL)
|
|
{
|
|
// Turbo68K requires native memory to be byte swapped, so must reverse this when reading bytes
|
|
TURBO68K_UINT16 *dataP = (TURBO68K_UINT16*)(region->ptr + addr);
|
|
return (TURBO68K_UINT32)dataP[1] | ((TURBO68K_UINT32)dataP[0])<<16;
|
|
}
|
|
else
|
|
{
|
|
ReadLongFPtr fPtr = (ReadLongFPtr)region->handler;
|
|
return fPtr(addr);
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
#endif
|
|
}
|
|
|
|
inline void WriteByteDirect(TURBO68K_UINT32 addr, TURBO68K_UINT8 data)
|
|
{
|
|
#if USE_NATIVE_WRITE
|
|
Turbo68KSetWriteByte(origWrite8Regions, TURBO68K_NULL);
|
|
Turbo68KWriteByte(addr, data);
|
|
Turbo68KSetWriteByte(debugWrite8Regions, TURBO68K_NULL);
|
|
#else
|
|
for (TURBO68K_DATAREGION *region = origWrite8Regions; region->ptr != TURBO68K_NULL || region->handler != TURBO68K_NULL; region++)
|
|
{
|
|
if (region->base <= addr && addr <= region->limit)
|
|
{
|
|
if (region->ptr != TURBO68K_NULL)
|
|
{
|
|
// Turbo68K requires native memory to be byte swapped, so must reverse this when writing bytes
|
|
TURBO68K_UINT8 *dataP = (TURBO68K_UINT8*)(region->ptr + (addr^1));
|
|
*dataP = data;
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
WriteByteFPtr fPtr = (WriteByteFPtr)region->handler;
|
|
fPtr(addr, data);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
return;
|
|
#endif
|
|
}
|
|
|
|
inline void WriteWordDirect(TURBO68K_UINT32 addr, TURBO68K_UINT16 data)
|
|
{
|
|
#if USE_NATIVE_WRITE
|
|
Turbo68KSetWriteWord(origWrite16Regions, TURBO68K_NULL);
|
|
Turbo68KWriteWord(addr, data);
|
|
Turbo68KSetWriteWord(debugWrite16Regions, TURBO68K_NULL);
|
|
#else
|
|
for (TURBO68K_DATAREGION *region = origWrite16Regions; region->ptr != TURBO68K_NULL || region->handler != TURBO68K_NULL; region++)
|
|
{
|
|
if (region->base <= addr && addr <= region->limit)
|
|
{
|
|
if (region->ptr != TURBO68K_NULL)
|
|
{
|
|
// Turbo68K requires native memory to be byte swapped, so must reverse this when writing bytes
|
|
TURBO68K_UINT16 *dataP = (TURBO68K_UINT16*)(region->ptr + addr);
|
|
*dataP = data;
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
WriteWordFPtr fPtr = (WriteWordFPtr)region->handler;
|
|
fPtr(addr, data);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
return;
|
|
#endif
|
|
}
|
|
|
|
inline void WriteLongDirect(TURBO68K_UINT32 addr, TURBO68K_UINT32 data)
|
|
{
|
|
#if USE_NATIVE_WRITE
|
|
Turbo68KSetWriteLong(origWrite32Regions, TURBO68K_NULL);
|
|
Turbo68KWriteLong(addr, data);
|
|
Turbo68KSetWriteLong(debugWrite32Regions, TURBO68K_NULL);
|
|
#else
|
|
for (TURBO68K_DATAREGION *region = origWrite32Regions; region->ptr != TURBO68K_NULL || region->handler != TURBO68K_NULL; region++)
|
|
{
|
|
if (region->base <= addr && addr <= region->limit)
|
|
{
|
|
if (region->ptr != TURBO68K_NULL)
|
|
{
|
|
// Turbo68K requires native memory to be byte swapped, so must reverse this when writing bytes
|
|
TURBO68K_UINT16 *dataP = (TURBO68K_UINT16*)(region->ptr + addr);
|
|
dataP[0] = data>>16;
|
|
dataP[1] = data&0xFFFF;
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
WriteLongFPtr fPtr = (WriteLongFPtr)region->handler;
|
|
fPtr(addr, data);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
return;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
#endif // INCLUDED_TURBO68KDEBUG_H
|
|
#endif // SUPERMODEL_DEBUGGER
|