2016-01-27 09:11:59 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* *
|
|
|
|
* Project64 - A Nintendo 64 emulator. *
|
|
|
|
* http://www.pj64-emu.com/ *
|
|
|
|
* Copyright (C) 2012 Project64. All rights reserved. *
|
|
|
|
* *
|
|
|
|
* License: *
|
|
|
|
* GNU/GPLv2 http://www.gnu.org/licenses/gpl-2.0.html *
|
|
|
|
* *
|
|
|
|
****************************************************************************/
|
|
|
|
#include "stdafx.h"
|
|
|
|
#include <Project64-core/N64System/SystemGlobals.h>
|
|
|
|
#include <Project64-core/N64System/Mips/MemoryVirtualMem.h>
|
|
|
|
#include <Project64-core/N64System/Mips/RegisterClass.h>
|
2016-08-07 08:27:14 +00:00
|
|
|
#include <Project64-core/N64System/N64Class.h>
|
2019-01-26 13:31:24 +00:00
|
|
|
#include <Project64-core/N64System/N64DiskClass.h>
|
2016-01-27 09:11:59 +00:00
|
|
|
#include "RSPPlugin.h"
|
|
|
|
#include "GFXPlugin.h"
|
|
|
|
#include <Project64-core/Plugins/AudioPlugin.h>
|
|
|
|
|
|
|
|
void DummyFunc1(int a) { a += 1; }
|
|
|
|
|
|
|
|
CRSP_Plugin::CRSP_Plugin(void) :
|
|
|
|
DoRspCycles(NULL),
|
|
|
|
EnableDebugging(NULL),
|
|
|
|
m_CycleCount(0),
|
|
|
|
GetDebugInfo(NULL),
|
|
|
|
InitiateDebugger(NULL)
|
|
|
|
{
|
|
|
|
memset(&m_RSPDebug, 0, sizeof(m_RSPDebug));
|
|
|
|
}
|
|
|
|
|
|
|
|
CRSP_Plugin::~CRSP_Plugin()
|
|
|
|
{
|
2016-04-18 09:38:20 +00:00
|
|
|
Close(NULL);
|
2016-01-27 09:11:59 +00:00
|
|
|
UnloadPlugin();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CRSP_Plugin::LoadFunctions(void)
|
|
|
|
{
|
|
|
|
// Find entries for functions in DLL
|
|
|
|
void(CALL *InitiateRSP)(void);
|
|
|
|
LoadFunction(InitiateRSP);
|
|
|
|
LoadFunction(DoRspCycles);
|
|
|
|
_LoadFunction("GetRspDebugInfo", GetDebugInfo);
|
|
|
|
_LoadFunction("InitiateRSPDebugger", InitiateDebugger);
|
|
|
|
LoadFunction(EnableDebugging);
|
|
|
|
if (EnableDebugging == NULL) { EnableDebugging = DummyFunc1; }
|
|
|
|
|
|
|
|
//Make sure dll had all needed functions
|
|
|
|
if (DoRspCycles == NULL) { UnloadPlugin(); return false; }
|
|
|
|
if (InitiateRSP == NULL) { UnloadPlugin(); return false; }
|
|
|
|
if (RomClosed == NULL) { UnloadPlugin(); return false; }
|
|
|
|
if (CloseDLL == NULL) { UnloadPlugin(); return false; }
|
|
|
|
|
|
|
|
if (m_PluginInfo.Version >= 0x0102)
|
|
|
|
{
|
|
|
|
if (PluginOpened == NULL) { UnloadPlugin(); return false; }
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get debug info if able
|
|
|
|
if (GetDebugInfo != NULL)
|
|
|
|
{
|
|
|
|
GetDebugInfo(&m_RSPDebug);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CRSP_Plugin::Initiate(CPlugins * Plugins, CN64System * System)
|
|
|
|
{
|
2016-02-01 06:33:19 +00:00
|
|
|
WriteTrace(TraceRSPPlugin, TraceDebug, "Starting");
|
2016-01-27 09:11:59 +00:00
|
|
|
if (m_PluginInfo.Version == 1 || m_PluginInfo.Version == 0x100)
|
|
|
|
{
|
2016-08-07 08:27:14 +00:00
|
|
|
WriteTrace(TraceRSPPlugin, TraceDebug, "Invalid Version: %X", m_PluginInfo.Version);
|
|
|
|
WriteTrace(TraceRSPPlugin, TraceDebug, "Done (res: false)");
|
2016-01-27 09:11:59 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-10-02 23:57:07 +00:00
|
|
|
if (m_PluginInfo.Version >= 0x103)
|
2016-01-27 09:11:59 +00:00
|
|
|
{
|
2016-10-02 23:57:07 +00:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
void * hInst;
|
|
|
|
int MemoryBswaped; /* If this is set to TRUE, then the memory has been pre
|
|
|
|
bswap on a dword (32 bits) boundry */
|
|
|
|
uint8_t * HEADER; // This is the rom header (first 40h bytes of the rom
|
|
|
|
// This will be in the same memory format as the rest of the memory.
|
|
|
|
uint8_t * RDRAM;
|
|
|
|
uint8_t * DMEM;
|
|
|
|
uint8_t * IMEM;
|
|
|
|
|
|
|
|
uint32_t * MI__INTR_REG;
|
|
|
|
|
|
|
|
uint32_t * SP__MEM_ADDR_REG;
|
|
|
|
uint32_t * SP__DRAM_ADDR_REG;
|
|
|
|
uint32_t * SP__RD_LEN_REG;
|
|
|
|
uint32_t * SP__WR_LEN_REG;
|
|
|
|
uint32_t * SP__STATUS_REG;
|
|
|
|
uint32_t * SP__DMA_FULL_REG;
|
|
|
|
uint32_t * SP__DMA_BUSY_REG;
|
|
|
|
uint32_t * SP__PC_REG;
|
|
|
|
uint32_t * SP__SEMAPHORE_REG;
|
|
|
|
|
|
|
|
uint32_t * DPC__START_REG;
|
|
|
|
uint32_t * DPC__END_REG;
|
|
|
|
uint32_t * DPC__CURRENT_REG;
|
|
|
|
uint32_t * DPC__STATUS_REG;
|
|
|
|
uint32_t * DPC__CLOCK_REG;
|
|
|
|
uint32_t * DPC__BUFBUSY_REG;
|
|
|
|
uint32_t * DPC__PIPEBUSY_REG;
|
|
|
|
uint32_t * DPC__TMEM_REG;
|
|
|
|
|
|
|
|
void(CALL *CheckInterrupts)(void);
|
|
|
|
void(CALL *ProcessDlist)(void);
|
|
|
|
void(CALL *ProcessAlist)(void);
|
|
|
|
void(CALL *ProcessRdpList)(void);
|
|
|
|
void(CALL *ShowCFB)(void);
|
|
|
|
} RSP_INFO_1_3;
|
|
|
|
|
|
|
|
RSP_INFO_1_3 Info = { 0 };
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2016-04-18 09:38:20 +00:00
|
|
|
#ifdef _WIN32
|
2016-10-02 23:57:07 +00:00
|
|
|
Info.hInst = (Plugins != NULL && Plugins->MainWindow() != NULL) ? Plugins->MainWindow()->GetModuleInstance() : NULL;
|
2016-04-18 09:38:20 +00:00
|
|
|
#else
|
2016-10-02 23:57:07 +00:00
|
|
|
Info.hInst = NULL;
|
2016-04-18 09:38:20 +00:00
|
|
|
#endif
|
2016-10-02 23:57:07 +00:00
|
|
|
Info.CheckInterrupts = DummyCheckInterrupts;
|
|
|
|
Info.MemoryBswaped = (System == NULL); // only true when the system's not yet loaded
|
2016-08-07 08:27:14 +00:00
|
|
|
|
2016-10-02 23:57:07 +00:00
|
|
|
//Get Function from DLL
|
|
|
|
void(CALL *InitiateRSP) (RSP_INFO_1_3 RSP_Info, uint32_t * Cycles);
|
|
|
|
LoadFunction(InitiateRSP);
|
|
|
|
if (InitiateRSP == NULL)
|
|
|
|
{
|
|
|
|
WriteTrace(TraceRSPPlugin, TraceDebug, "Failed to find InitiateRSP");
|
|
|
|
WriteTrace(TraceRSPPlugin, TraceDebug, "Done (res: false)");
|
|
|
|
return false;
|
|
|
|
}
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2016-10-02 23:57:07 +00:00
|
|
|
// We are initializing the plugin before any rom is loaded so we do not have any correct
|
|
|
|
// parameters here.. just needed to we can config the DLL.
|
|
|
|
if (System == NULL)
|
|
|
|
{
|
|
|
|
static uint8_t Buffer[100];
|
|
|
|
static uint32_t Value = 0;
|
|
|
|
|
|
|
|
Info.ProcessDlist = DummyCheckInterrupts;
|
|
|
|
Info.ProcessRdpList = DummyCheckInterrupts;
|
|
|
|
Info.ShowCFB = DummyCheckInterrupts;
|
|
|
|
Info.ProcessAlist = DummyCheckInterrupts;
|
|
|
|
|
|
|
|
Info.HEADER = Buffer;
|
|
|
|
Info.RDRAM = Buffer;
|
|
|
|
Info.DMEM = Buffer;
|
|
|
|
Info.IMEM = Buffer;
|
|
|
|
|
|
|
|
Info.MI__INTR_REG = &Value;
|
|
|
|
|
|
|
|
Info.SP__MEM_ADDR_REG = &Value;
|
|
|
|
Info.SP__DRAM_ADDR_REG = &Value;
|
|
|
|
Info.SP__RD_LEN_REG = &Value;
|
|
|
|
Info.SP__WR_LEN_REG = &Value;
|
|
|
|
Info.SP__STATUS_REG = &Value;
|
|
|
|
Info.SP__DMA_FULL_REG = &Value;
|
|
|
|
Info.SP__DMA_BUSY_REG = &Value;
|
|
|
|
Info.SP__PC_REG = &Value;
|
|
|
|
Info.SP__SEMAPHORE_REG = &Value;
|
|
|
|
|
|
|
|
Info.DPC__START_REG = &Value;
|
|
|
|
Info.DPC__END_REG = &Value;
|
|
|
|
Info.DPC__CURRENT_REG = &Value;
|
|
|
|
Info.DPC__STATUS_REG = &Value;
|
|
|
|
Info.DPC__CLOCK_REG = &Value;
|
|
|
|
Info.DPC__BUFBUSY_REG = &Value;
|
|
|
|
Info.DPC__PIPEBUSY_REG = &Value;
|
|
|
|
Info.DPC__TMEM_REG = &Value;
|
|
|
|
}
|
|
|
|
// Send initialization information to the DLL
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Info.ProcessDlist = Plugins->Gfx()->ProcessDList;
|
|
|
|
Info.ProcessRdpList = Plugins->Gfx()->ProcessRDPList;
|
|
|
|
Info.ShowCFB = Plugins->Gfx()->ShowCFB;
|
|
|
|
Info.ProcessAlist = Plugins->Audio()->ProcessAList;
|
|
|
|
|
|
|
|
CMipsMemoryVM & MMU = System->m_MMU_VM;
|
|
|
|
CRegisters & Reg = System->m_Reg;
|
|
|
|
|
2019-01-27 12:06:36 +00:00
|
|
|
if (g_Rom->IsLoadedRomDDIPL() && g_Disk != NULL)
|
2019-01-26 13:31:24 +00:00
|
|
|
Info.HEADER = g_Disk->GetDiskHeader();
|
|
|
|
else
|
|
|
|
Info.HEADER = g_Rom->GetRomAddress();
|
2016-10-02 23:57:07 +00:00
|
|
|
Info.RDRAM = MMU.Rdram();
|
|
|
|
Info.DMEM = MMU.Dmem();
|
|
|
|
Info.IMEM = MMU.Imem();
|
|
|
|
|
|
|
|
Info.MI__INTR_REG = &Reg.m_RspIntrReg;
|
|
|
|
|
|
|
|
Info.SP__MEM_ADDR_REG = &Reg.SP_MEM_ADDR_REG;
|
|
|
|
Info.SP__DRAM_ADDR_REG = &Reg.SP_DRAM_ADDR_REG;
|
|
|
|
Info.SP__RD_LEN_REG = &Reg.SP_RD_LEN_REG;
|
|
|
|
Info.SP__WR_LEN_REG = &Reg.SP_WR_LEN_REG;
|
|
|
|
Info.SP__STATUS_REG = &Reg.SP_STATUS_REG;
|
|
|
|
Info.SP__DMA_FULL_REG = &Reg.SP_DMA_FULL_REG;
|
|
|
|
Info.SP__DMA_BUSY_REG = &Reg.SP_DMA_BUSY_REG;
|
|
|
|
Info.SP__PC_REG = &Reg.SP_PC_REG;
|
|
|
|
Info.SP__SEMAPHORE_REG = &Reg.SP_SEMAPHORE_REG;
|
|
|
|
|
|
|
|
Info.DPC__START_REG = &Reg.DPC_START_REG;
|
|
|
|
Info.DPC__END_REG = &Reg.DPC_END_REG;
|
|
|
|
Info.DPC__CURRENT_REG = &Reg.DPC_CURRENT_REG;
|
|
|
|
Info.DPC__STATUS_REG = &Reg.DPC_STATUS_REG;
|
|
|
|
Info.DPC__CLOCK_REG = &Reg.DPC_CLOCK_REG;
|
|
|
|
Info.DPC__BUFBUSY_REG = &Reg.DPC_BUFBUSY_REG;
|
|
|
|
Info.DPC__PIPEBUSY_REG = &Reg.DPC_PIPEBUSY_REG;
|
|
|
|
Info.DPC__TMEM_REG = &Reg.DPC_TMEM_REG;
|
|
|
|
}
|
|
|
|
|
|
|
|
InitiateRSP(Info, &m_CycleCount);
|
2016-01-27 09:11:59 +00:00
|
|
|
}
|
2016-10-02 23:57:07 +00:00
|
|
|
else
|
2016-01-27 09:11:59 +00:00
|
|
|
{
|
2016-10-02 23:57:07 +00:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
void * hInst;
|
|
|
|
int MemoryBswaped; /* If this is set to TRUE, then the memory has been pre
|
|
|
|
bswap on a dword (32 bits) boundry */
|
|
|
|
uint8_t * RDRAM;
|
|
|
|
uint8_t * DMEM;
|
|
|
|
uint8_t * IMEM;
|
|
|
|
|
|
|
|
uint32_t * MI__INTR_REG;
|
|
|
|
|
|
|
|
uint32_t * SP__MEM_ADDR_REG;
|
|
|
|
uint32_t * SP__DRAM_ADDR_REG;
|
|
|
|
uint32_t * SP__RD_LEN_REG;
|
|
|
|
uint32_t * SP__WR_LEN_REG;
|
|
|
|
uint32_t * SP__STATUS_REG;
|
|
|
|
uint32_t * SP__DMA_FULL_REG;
|
|
|
|
uint32_t * SP__DMA_BUSY_REG;
|
|
|
|
uint32_t * SP__PC_REG;
|
|
|
|
uint32_t * SP__SEMAPHORE_REG;
|
|
|
|
|
|
|
|
uint32_t * DPC__START_REG;
|
|
|
|
uint32_t * DPC__END_REG;
|
|
|
|
uint32_t * DPC__CURRENT_REG;
|
|
|
|
uint32_t * DPC__STATUS_REG;
|
|
|
|
uint32_t * DPC__CLOCK_REG;
|
|
|
|
uint32_t * DPC__BUFBUSY_REG;
|
|
|
|
uint32_t * DPC__PIPEBUSY_REG;
|
|
|
|
uint32_t * DPC__TMEM_REG;
|
|
|
|
|
|
|
|
void(CALL *CheckInterrupts)(void);
|
|
|
|
void(CALL *ProcessDlist)(void);
|
|
|
|
void(CALL *ProcessAlist)(void);
|
|
|
|
void(CALL *ProcessRdpList)(void);
|
|
|
|
void(CALL *ShowCFB)(void);
|
|
|
|
} RSP_INFO_1_1;
|
|
|
|
|
|
|
|
RSP_INFO_1_1 Info = { 0 };
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
Info.hInst = (Plugins != NULL && Plugins->MainWindow() != NULL) ? Plugins->MainWindow()->GetModuleInstance() : NULL;
|
|
|
|
#else
|
|
|
|
Info.hInst = NULL;
|
|
|
|
#endif
|
|
|
|
Info.CheckInterrupts = DummyCheckInterrupts;
|
|
|
|
Info.MemoryBswaped = (System == NULL); // only true when the system's not yet loaded
|
|
|
|
|
|
|
|
//Get Function from DLL
|
|
|
|
void(CALL *InitiateRSP) (RSP_INFO_1_1 RSP_Info, uint32_t * Cycles);
|
|
|
|
LoadFunction(InitiateRSP);
|
|
|
|
if (InitiateRSP == NULL)
|
|
|
|
{
|
|
|
|
WriteTrace(TraceRSPPlugin, TraceDebug, "Failed to find InitiateRSP");
|
|
|
|
WriteTrace(TraceRSPPlugin, TraceDebug, "Done (res: false)");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We are initializing the plugin before any rom is loaded so we do not have any correct
|
|
|
|
// parameters here.. just needed to we can config the DLL.
|
|
|
|
if (System == NULL)
|
|
|
|
{
|
|
|
|
static uint8_t Buffer[100];
|
|
|
|
static uint32_t Value = 0;
|
|
|
|
|
|
|
|
Info.ProcessDlist = DummyCheckInterrupts;
|
|
|
|
Info.ProcessRdpList = DummyCheckInterrupts;
|
|
|
|
Info.ShowCFB = DummyCheckInterrupts;
|
|
|
|
Info.ProcessAlist = DummyCheckInterrupts;
|
|
|
|
|
|
|
|
Info.RDRAM = Buffer;
|
|
|
|
Info.DMEM = Buffer;
|
|
|
|
Info.IMEM = Buffer;
|
|
|
|
|
|
|
|
Info.MI__INTR_REG = &Value;
|
|
|
|
|
|
|
|
Info.SP__MEM_ADDR_REG = &Value;
|
|
|
|
Info.SP__DRAM_ADDR_REG = &Value;
|
|
|
|
Info.SP__RD_LEN_REG = &Value;
|
|
|
|
Info.SP__WR_LEN_REG = &Value;
|
|
|
|
Info.SP__STATUS_REG = &Value;
|
|
|
|
Info.SP__DMA_FULL_REG = &Value;
|
|
|
|
Info.SP__DMA_BUSY_REG = &Value;
|
|
|
|
Info.SP__PC_REG = &Value;
|
|
|
|
Info.SP__SEMAPHORE_REG = &Value;
|
|
|
|
|
|
|
|
Info.DPC__START_REG = &Value;
|
|
|
|
Info.DPC__END_REG = &Value;
|
|
|
|
Info.DPC__CURRENT_REG = &Value;
|
|
|
|
Info.DPC__STATUS_REG = &Value;
|
|
|
|
Info.DPC__CLOCK_REG = &Value;
|
|
|
|
Info.DPC__BUFBUSY_REG = &Value;
|
|
|
|
Info.DPC__PIPEBUSY_REG = &Value;
|
|
|
|
Info.DPC__TMEM_REG = &Value;
|
|
|
|
}
|
|
|
|
// Send initialization information to the DLL
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Info.ProcessDlist = Plugins->Gfx()->ProcessDList;
|
|
|
|
Info.ProcessRdpList = Plugins->Gfx()->ProcessRDPList;
|
|
|
|
Info.ShowCFB = Plugins->Gfx()->ShowCFB;
|
|
|
|
Info.ProcessAlist = Plugins->Audio()->ProcessAList;
|
|
|
|
|
|
|
|
CMipsMemoryVM & MMU = System->m_MMU_VM;
|
|
|
|
CRegisters & Reg = System->m_Reg;
|
|
|
|
|
|
|
|
Info.RDRAM = MMU.Rdram();
|
|
|
|
Info.DMEM = MMU.Dmem();
|
|
|
|
Info.IMEM = MMU.Imem();
|
|
|
|
|
|
|
|
Info.MI__INTR_REG = &Reg.m_RspIntrReg;
|
|
|
|
|
|
|
|
Info.SP__MEM_ADDR_REG = &Reg.SP_MEM_ADDR_REG;
|
|
|
|
Info.SP__DRAM_ADDR_REG = &Reg.SP_DRAM_ADDR_REG;
|
|
|
|
Info.SP__RD_LEN_REG = &Reg.SP_RD_LEN_REG;
|
|
|
|
Info.SP__WR_LEN_REG = &Reg.SP_WR_LEN_REG;
|
|
|
|
Info.SP__STATUS_REG = &Reg.SP_STATUS_REG;
|
|
|
|
Info.SP__DMA_FULL_REG = &Reg.SP_DMA_FULL_REG;
|
|
|
|
Info.SP__DMA_BUSY_REG = &Reg.SP_DMA_BUSY_REG;
|
|
|
|
Info.SP__PC_REG = &Reg.SP_PC_REG;
|
|
|
|
Info.SP__SEMAPHORE_REG = &Reg.SP_SEMAPHORE_REG;
|
|
|
|
|
|
|
|
Info.DPC__START_REG = &Reg.DPC_START_REG;
|
|
|
|
Info.DPC__END_REG = &Reg.DPC_END_REG;
|
|
|
|
Info.DPC__CURRENT_REG = &Reg.DPC_CURRENT_REG;
|
|
|
|
Info.DPC__STATUS_REG = &Reg.DPC_STATUS_REG;
|
|
|
|
Info.DPC__CLOCK_REG = &Reg.DPC_CLOCK_REG;
|
|
|
|
Info.DPC__BUFBUSY_REG = &Reg.DPC_BUFBUSY_REG;
|
|
|
|
Info.DPC__PIPEBUSY_REG = &Reg.DPC_PIPEBUSY_REG;
|
|
|
|
Info.DPC__TMEM_REG = &Reg.DPC_TMEM_REG;
|
|
|
|
}
|
|
|
|
|
|
|
|
InitiateRSP(Info, &m_CycleCount);
|
2016-01-27 09:11:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
m_Initialized = true;
|
|
|
|
|
2016-08-07 08:27:14 +00:00
|
|
|
WriteTrace(TraceRSPPlugin, TraceDebug, "Done (res: %s)", m_Initialized ? "true" : "false");
|
2016-01-27 09:11:59 +00:00
|
|
|
return m_Initialized;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CRSP_Plugin::UnloadPluginDetails(void)
|
|
|
|
{
|
|
|
|
memset(&m_RSPDebug, 0, sizeof(m_RSPDebug));
|
|
|
|
DoRspCycles = NULL;
|
|
|
|
EnableDebugging = NULL;
|
|
|
|
GetDebugInfo = NULL;
|
|
|
|
InitiateDebugger = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CRSP_Plugin::ProcessMenuItem(int id)
|
|
|
|
{
|
|
|
|
if (m_RSPDebug.ProcessMenuItem)
|
|
|
|
{
|
|
|
|
m_RSPDebug.ProcessMenuItem(id);
|
|
|
|
}
|
2017-02-05 01:10:01 +00:00
|
|
|
}
|