mirror of
https://github.com/libretro/mupen64plus-libretro-nx.git
synced 2024-11-23 08:10:05 +00:00
496 lines
14 KiB
C
496 lines
14 KiB
C
/******************************************************************************\
|
|
* Project: Module Subsystem Interface to SP Interpreter Core *
|
|
* Authors: Iconoclast *
|
|
* Release: 2016.03.23 *
|
|
* License: CC0 Public Domain Dedication *
|
|
* *
|
|
* To the extent possible under law, the author(s) have dedicated all copyright *
|
|
* and related and neighboring rights to this software to the public domain *
|
|
* worldwide. This software is distributed without any warranty. *
|
|
* *
|
|
* You should have received a copy of the CC0 Public Domain Dedication along *
|
|
* with this software. *
|
|
* If not, see <http://creativecommons.org/publicdomain/zero/1.0/>. *
|
|
\******************************************************************************/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#ifdef WIN32
|
|
#include <windows.h>
|
|
#endif
|
|
|
|
#include "module.h"
|
|
#include "main/rom.h"
|
|
#include "su.h"
|
|
|
|
RSP_INFO RSP_INFO_NAME;
|
|
|
|
#define RSP_CXD4_VERSION 0x0101
|
|
|
|
#include <stdarg.h>
|
|
|
|
#define RSP_PLUGIN_API_VERSION 0x020000
|
|
#define CONFIG_API_VERSION 0x020100
|
|
#define CONFIG_PARAM_VERSION 1.00
|
|
|
|
static void (*l_DebugCallback)(void *, int, const char *) = NULL;
|
|
static void *l_DebugCallContext = NULL;
|
|
static int l_PluginInit = 0;
|
|
static m64p_handle l_ConfigRsp;
|
|
|
|
#define VERSION_PRINTF_SPLIT(x) (((x) >> 16) & 0xffff), (((x) >> 8) & 0xff), ((x) & 0xff)
|
|
|
|
#define API_PREFIX(x) cxd4##x
|
|
|
|
NOINLINE void update_conf(const char* source)
|
|
{
|
|
memset(conf, 0, sizeof(conf));
|
|
#if 0
|
|
#ifndef __LIBRETRO__
|
|
CFG_HLE_GFX = ConfigGetParamBool(l_ConfigRsp, "DisplayListToGraphicsPlugin");
|
|
CFG_HLE_AUD = ConfigGetParamBool(l_ConfigRsp, "AudioListToAudioPlugin");
|
|
CFG_WAIT_FOR_CPU_HOST = ConfigGetParamBool(l_ConfigRsp, "WaitForCPUHost");
|
|
CFG_MEND_SEMAPHORE_LOCK = ConfigGetParamBool(l_ConfigRsp, "SupportCPUSemaphoreLock");
|
|
#endif
|
|
#endif
|
|
|
|
#if 1
|
|
if (strstr((char*)ROM_HEADER.Name, (const char*)"WORLD DRIVER CHAMP") != NULL)
|
|
CFG_HLE_GFX = 0;
|
|
else if (strstr((char*)ROM_HEADER.Name, (const char*)"Indiana Jones") != NULL)
|
|
CFG_HLE_GFX = 0;
|
|
else if (strstr((char*)ROM_HEADER.Name, (const char*)"Rogue Squadron") != NULL)
|
|
CFG_HLE_GFX = 0;
|
|
else if (strstr((char*)ROM_HEADER.Name, (const char*)"rogue squadron") != NULL)
|
|
CFG_HLE_GFX = 0;
|
|
else if (strstr((char*)ROM_HEADER.Name, (const char*)"Battle for Naboo") != NULL)
|
|
CFG_HLE_GFX = 0;
|
|
else if (strstr((char*)ROM_HEADER.Name, (const char*)"Stunt Racer 64") != NULL)
|
|
CFG_HLE_GFX = 0;
|
|
else if (strstr((char*)ROM_HEADER.Name, (const char*)"GAUNTLET LEGENDS") != NULL)
|
|
CFG_HLE_GFX = 0;
|
|
#endif
|
|
}
|
|
|
|
extern void DebugMessage(int level, const char *message, ...);
|
|
|
|
EXPORT u32 CALL API_PREFIX(DoRspCycles)(u32 cycles)
|
|
{
|
|
OSTask_type task_type;
|
|
register unsigned int i;
|
|
|
|
if (GET_RCP_REG(SP_STATUS_REG) & 0x00000003)
|
|
{
|
|
message("SP_STATUS_HALT");
|
|
return 0x00000000;
|
|
}
|
|
|
|
task_type = 0x00000000
|
|
#ifdef USE_CLIENT_ENDIAN
|
|
| *((pi32)(DMEM + 0x000FC0U))
|
|
#else
|
|
| (u32)DMEM[0xFC0] << 24
|
|
| (u32)DMEM[0xFC1] << 16
|
|
| (u32)DMEM[0xFC2] << 8
|
|
| (u32)DMEM[0xFC3] << 0
|
|
#endif
|
|
;
|
|
switch (task_type) {
|
|
#ifdef EXTERN_COMMAND_LIST_GBI
|
|
case M_GFXTASK:
|
|
if (CFG_HLE_GFX == 0)
|
|
break;
|
|
|
|
if (*(pi32)(DMEM + 0xFF0) == 0x00000000)
|
|
break; /* Resident Evil 2, null task pointers */
|
|
if (GET_RSP_INFO(ProcessDlistList) == NULL)
|
|
{ /* branch */ }
|
|
else
|
|
GET_RSP_INFO(ProcessDlistList)();
|
|
|
|
GET_RCP_REG(SP_STATUS_REG) |=
|
|
SP_STATUS_SIG2 | SP_STATUS_BROKE | SP_STATUS_HALT
|
|
;
|
|
if (GET_RCP_REG(SP_STATUS_REG) & SP_STATUS_INTR_BREAK) {
|
|
GET_RCP_REG(MI_INTR_REG) |= 0x00000001;
|
|
GET_RSP_INFO(CheckInterrupts)();
|
|
}
|
|
GET_RCP_REG(DPC_STATUS_REG) &= ~0x00000002ul; /* DPC_STATUS_FREEZE */
|
|
return 0;
|
|
#endif
|
|
#ifdef EXTERN_COMMAND_LIST_ABI
|
|
case M_AUDTASK:
|
|
if (CFG_HLE_AUD == 0)
|
|
break;
|
|
|
|
if (GET_RSP_INFO(ProcessAlistList) == NULL)
|
|
{ /* branch */ }
|
|
else
|
|
GET_RSP_INFO(ProcessAlistList)();
|
|
|
|
GET_RCP_REG(SP_STATUS_REG) |=
|
|
SP_STATUS_SIG2 | SP_STATUS_BROKE | SP_STATUS_HALT
|
|
;
|
|
if (GET_RCP_REG(SP_STATUS_REG) & SP_STATUS_INTR_BREAK) {
|
|
GET_RCP_REG(MI_INTR_REG) |= 0x00000001;
|
|
GET_RSP_INFO(CheckInterrupts)();
|
|
}
|
|
return 0;
|
|
#endif
|
|
case M_VIDTASK:
|
|
message("M_VIDTASK");
|
|
break;
|
|
case M_NJPEGTASK:
|
|
break; /* Zelda, Pokemon, others */
|
|
case M_NULTASK:
|
|
message("M_NULTASK");
|
|
break;
|
|
case M_HVQTASK:
|
|
message("M_HVQTASK");
|
|
break;
|
|
case M_HVQMTASK:
|
|
if (GET_RSP_INFO(ShowCFB) == NULL) /* Gfx #1.2 or older specs */
|
|
break;
|
|
GET_RSP_INFO(ShowCFB)(); /* forced FB refresh in case gfx plugin skip */
|
|
break;
|
|
}
|
|
|
|
#ifdef WAIT_FOR_CPU_HOST
|
|
for (i = 0; i < 32; i++)
|
|
MFC0_count[i] = 0;
|
|
#endif
|
|
run_task();
|
|
|
|
#if 0
|
|
/*
|
|
* An optional EMMS when compiling with Intel SIMD or MMX support.
|
|
*
|
|
* Whether or not MMX has been executed in this emulator, here is a good time
|
|
* to finally empty the MM state, at the end of a long interpreter loop.
|
|
*/
|
|
#ifdef ARCH_MIN_SSE2
|
|
_mm_empty();
|
|
#endif
|
|
#endif
|
|
|
|
if (*CR[0x4] & SP_STATUS_BROKE) /* normal exit, from executing BREAK */
|
|
return (cycles);
|
|
else if (GET_RCP_REG(MI_INTR_REG) & 1) /* interrupt set by MTC0 to break */
|
|
GET_RSP_INFO(CheckInterrupts)();
|
|
else if (*CR[0x7] != 0x00000000) /* semaphore lock fixes */
|
|
{}
|
|
#ifdef WAIT_FOR_CPU_HOST
|
|
else
|
|
MF_SP_STATUS_TIMEOUT = 16; /* From now on, wait 16 times, not 32767. */
|
|
#else
|
|
else { /* ??? unknown, possibly external intervention from CPU memory map */
|
|
message("SP_SET_HALT");
|
|
return (cycles);
|
|
}
|
|
#endif
|
|
*CR[0x4] &= ~SP_STATUS_HALT; /* CPU restarts with the correct SIGs. */
|
|
return (cycles);
|
|
}
|
|
|
|
EXPORT void CALL API_PREFIX(GetDllInfo)(PLUGIN_INFO *PluginInfo)
|
|
{
|
|
PluginInfo -> Version = PLUGIN_API_VERSION;
|
|
PluginInfo -> Type = PLUGIN_TYPE_RSP;
|
|
my_strcpy(PluginInfo -> Name, "Static Interpreter");
|
|
PluginInfo -> NormalMemory = 0;
|
|
PluginInfo -> MemoryBswaped = USE_CLIENT_ENDIAN;
|
|
return;
|
|
}
|
|
|
|
EXPORT m64p_error CALL API_PREFIX(PluginGetVersion)(m64p_plugin_type *PluginType, int *PluginVersion, int *APIVersion, const char **PluginNamePtr, int *Capabilities)
|
|
{
|
|
/* set version info */
|
|
if (PluginType != NULL)
|
|
*PluginType = M64PLUGIN_RSP;
|
|
|
|
if (PluginVersion != NULL)
|
|
*PluginVersion = RSP_CXD4_VERSION;
|
|
|
|
if (APIVersion != NULL)
|
|
*APIVersion = RSP_PLUGIN_API_VERSION;
|
|
|
|
if (PluginNamePtr != NULL)
|
|
*PluginNamePtr = "Static Interpreter";
|
|
|
|
if (Capabilities != NULL)
|
|
{
|
|
*Capabilities = 0;
|
|
}
|
|
|
|
return M64ERR_SUCCESS;
|
|
}
|
|
|
|
|
|
EXPORT void CALL API_PREFIX(InitiateRSP)(RSP_INFO Rsp_Info, pu32 CycleCount)
|
|
{
|
|
if (CycleCount != NULL) /* cycle-accuracy not doable with today's hosts */
|
|
*CycleCount = 0;
|
|
update_conf(CFG_FILE);
|
|
|
|
RSP_INFO_NAME = Rsp_Info;
|
|
DRAM = GET_RSP_INFO(RDRAM);
|
|
if (Rsp_Info.DMEM == Rsp_Info.IMEM) /* usually dummy RSP data for testing */
|
|
return; /* DMA is not executed just because plugin initiates. */
|
|
DMEM = GET_RSP_INFO(DMEM);
|
|
IMEM = GET_RSP_INFO(IMEM);
|
|
|
|
CR[0x0] = &GET_RCP_REG(SP_MEM_ADDR_REG);
|
|
CR[0x1] = &GET_RCP_REG(SP_DRAM_ADDR_REG);
|
|
CR[0x2] = &GET_RCP_REG(SP_RD_LEN_REG);
|
|
CR[0x3] = &GET_RCP_REG(SP_WR_LEN_REG);
|
|
CR[0x4] = &GET_RCP_REG(SP_STATUS_REG);
|
|
CR[0x5] = &GET_RCP_REG(SP_DMA_FULL_REG);
|
|
CR[0x6] = &GET_RCP_REG(SP_DMA_BUSY_REG);
|
|
CR[0x7] = &GET_RCP_REG(SP_SEMAPHORE_REG);
|
|
GET_RCP_REG(SP_PC_REG) = 0x04001000;
|
|
CR[0x8] = &GET_RCP_REG(DPC_START_REG);
|
|
CR[0x9] = &GET_RCP_REG(DPC_END_REG);
|
|
CR[0xA] = &GET_RCP_REG(DPC_CURRENT_REG);
|
|
CR[0xB] = &GET_RCP_REG(DPC_STATUS_REG);
|
|
CR[0xC] = &GET_RCP_REG(DPC_CLOCK_REG);
|
|
CR[0xD] = &GET_RCP_REG(DPC_BUFBUSY_REG);
|
|
CR[0xE] = &GET_RCP_REG(DPC_PIPEBUSY_REG);
|
|
CR[0xF] = &GET_RCP_REG(DPC_TMEM_REG);
|
|
|
|
MF_SP_STATUS_TIMEOUT = 32767;
|
|
#if 1
|
|
GET_RCP_REG(SP_PC_REG) &= 0x00000FFFu; /* hack to fix Mupen64 */
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
EXPORT void CALL API_PREFIX(RomClosed)(void)
|
|
{
|
|
GET_RCP_REG(SP_PC_REG) = 0x04001000;
|
|
}
|
|
|
|
NOINLINE void message(const char* body)
|
|
{
|
|
#if defined(M64P_PLUGIN_API)
|
|
DebugMessage(M64MSG_ERROR, body);
|
|
#else
|
|
printf("%s\n", body);
|
|
#endif
|
|
|
|
}
|
|
|
|
#ifdef SP_EXECUTE_LOG
|
|
void step_SP_commands(uint32_t inst)
|
|
{
|
|
unsigned char endian_swap[4];
|
|
char text[256];
|
|
char offset[4] = "";
|
|
char code[9] = "";
|
|
|
|
if (output_log == NULL)
|
|
return;
|
|
|
|
endian_swap[00] = (u8)((inst >> 24) & 0xFF);
|
|
endian_swap[01] = (u8)((inst >> 16) & 0xFF);
|
|
endian_swap[02] = (u8)((inst >> 8) & 0xFF);
|
|
endian_swap[03] = (u8)((inst >> 0) & 0xFF);
|
|
sprintf(&offset[0], "%03X", GET_RCP_REG(SP_PC_REG) & 0xFFF);
|
|
sprintf(&code[0], "%08X", inst);
|
|
strcpy(text, offset);
|
|
my_strcat(text, "\n");
|
|
my_strcat(text, code);
|
|
message(text); /* PC offset, MIPS hex. */
|
|
if (output_log != NULL)
|
|
my_fwrite(endian_swap, 4, 1, output_log);
|
|
}
|
|
#endif
|
|
|
|
NOINLINE void export_data_cache(void)
|
|
{
|
|
pu8 DMEM_swapped;
|
|
FILE * out;
|
|
register int i;
|
|
/* const int little_endian = GET_RSP_INFO(MemoryBswaped); */
|
|
|
|
DMEM_swapped = my_calloc(4096, 1);
|
|
for (i = 0; i < 4096; i++)
|
|
DMEM_swapped[i] = DMEM[BES(i)];
|
|
out = my_fopen("rcpcache.dhex", "wb");
|
|
my_fwrite(DMEM_swapped, 16, 4096 / 16, out);
|
|
my_fclose(out);
|
|
my_free(DMEM_swapped);
|
|
return;
|
|
}
|
|
NOINLINE void export_instruction_cache(void)
|
|
{
|
|
pu8 IMEM_swapped;
|
|
FILE * out;
|
|
register int i;
|
|
/* const int little_endian = GET_RSP_INFO(MemoryBswaped); */
|
|
|
|
IMEM_swapped = my_calloc(4096, 1);
|
|
for (i = 0; i < 4096; i++)
|
|
IMEM_swapped[i] = IMEM[BES(i)];
|
|
out = my_fopen("rcpcache.ihex", "wb");
|
|
my_fwrite(IMEM_swapped, 16, 4096 / 16, out);
|
|
my_fclose(out);
|
|
my_free(IMEM_swapped);
|
|
return;
|
|
}
|
|
void export_SP_memory(void)
|
|
{
|
|
export_data_cache();
|
|
export_instruction_cache();
|
|
}
|
|
|
|
/*
|
|
* Microsoft linker defaults to an entry point of `_DllMainCRTStartup',
|
|
* which attaches several CRT dependencies. To eliminate CRT dependencies,
|
|
* we direct the linker to cursor the entry point to the lower-level
|
|
* `DllMain' symbol or, alternatively, link with /NOENTRY for no entry point.
|
|
*/
|
|
#ifdef WIN32
|
|
BOOL WINAPI DllMain(
|
|
HINSTANCE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)
|
|
{
|
|
hModule = lpReserved = NULL; /* unused */
|
|
switch (ul_reason_for_call)
|
|
{
|
|
case 1: /* DLL_PROCESS_ATTACH */
|
|
break;
|
|
case 2: /* DLL_THREAD_ATTACH */
|
|
break;
|
|
case 3: /* DLL_THREAD_DETACH */
|
|
break;
|
|
case 0: /* DLL_PROCESS_DETACH */
|
|
break;
|
|
}
|
|
return 1; /* TRUE */
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* low-level recreations of the C standard library functions for operating
|
|
* systems that define a C run-time or dependency on top of fixed OS calls
|
|
*
|
|
* Currently, this only addresses Microsoft Windows.
|
|
*
|
|
* None of these are meant to out-perform the original functions, by the way
|
|
* (especially with better intrinsic compiler support for stuff like memcpy),
|
|
* just to cut down on I-cache use for performance-irrelevant code sections
|
|
* and to avoid std. lib run-time dependencies on certain operating systems.
|
|
*/
|
|
|
|
NOINLINE p_void my_calloc(size_t count, size_t size)
|
|
{
|
|
#ifdef WIN32
|
|
return GlobalAlloc(GPTR, size * count);
|
|
#else
|
|
return calloc(count, size);
|
|
#endif
|
|
}
|
|
|
|
NOINLINE void my_free(p_void ptr)
|
|
{
|
|
#ifdef WIN32
|
|
while (GlobalFree(ptr) != NULL)
|
|
message("GlobalFree() failure");
|
|
#else
|
|
free(ptr);
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
NOINLINE size_t my_strlen(const char* str)
|
|
{
|
|
size_t ret_slot;
|
|
|
|
for (ret_slot = 0; *str != '\0'; ret_slot++, str++)
|
|
;
|
|
return (ret_slot);
|
|
}
|
|
|
|
NOINLINE char* my_strcpy(char* destination, const char* source)
|
|
{
|
|
register size_t i;
|
|
const size_t length = my_strlen(source) + 1; /* including null terminator */
|
|
|
|
for (i = 0; i < length; i++)
|
|
destination[i] = source[i];
|
|
return (destination);
|
|
}
|
|
|
|
NOINLINE char* my_strcat(char* destination, const char* source)
|
|
{
|
|
const size_t length = my_strlen(destination);
|
|
|
|
my_strcpy(destination + length, source);
|
|
return (destination);
|
|
}
|
|
|
|
NOINLINE FILE* my_fopen(const char * filename, const char* mode)
|
|
{
|
|
#ifdef WIN32
|
|
#if _MSC_VER >= 1400 && WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_APP)
|
|
return NULL;
|
|
#else
|
|
return (FILE *)(HANDLE)CreateFileA(
|
|
filename,
|
|
(mode[0] == 'r') ? GENERIC_READ : GENERIC_WRITE,
|
|
(mode[0] == 'r') ? FILE_SHARE_READ : FILE_SHARE_WRITE,
|
|
NULL,
|
|
(mode[0] == 'r') ? OPEN_EXISTING : CREATE_ALWAYS,
|
|
#if 0
|
|
FILE_FLAG_WRITE_THROUGH | FILE_FLAG_OVERLAPPED | FILE_FLAG_NO_BUFFERING,
|
|
#else
|
|
(mode[0] == 'r') ? FILE_ATTRIBUTE_NORMAL : FILE_FLAG_WRITE_THROUGH,
|
|
#endif
|
|
NULL
|
|
);
|
|
#endif
|
|
#else
|
|
return fopen(filename, mode);
|
|
#endif
|
|
}
|
|
|
|
NOINLINE int my_fclose(FILE* stream)
|
|
{
|
|
int ret_slot;
|
|
#ifdef WIN32
|
|
ret_slot = !CloseHandle((HANDLE)stream);
|
|
#else
|
|
ret_slot = fclose(stream);
|
|
#endif
|
|
return (ret_slot);
|
|
}
|
|
|
|
NOINLINE size_t my_fread(p_void ptr, size_t size, size_t count, FILE* stream)
|
|
{
|
|
#ifdef WIN32
|
|
DWORD ret_slot;
|
|
|
|
ReadFile((HANDLE)stream, ptr, size * count, &ret_slot, NULL);
|
|
#else
|
|
size_t ret_slot;
|
|
|
|
ret_slot = fread(ptr, size, count, stream);
|
|
#endif
|
|
return (size_t)(ret_slot);
|
|
}
|
|
|
|
NOINLINE size_t my_fwrite(p_void ptr, size_t size, size_t count, FILE* stream)
|
|
{
|
|
#ifdef WIN32
|
|
DWORD ret_slot;
|
|
|
|
WriteFile((HANDLE)stream, ptr, size * count, &ret_slot, NULL);
|
|
#else
|
|
size_t ret_slot;
|
|
|
|
ret_slot = fwrite(ptr, size, count, stream);
|
|
#endif
|
|
return (size_t)(ret_slot);
|
|
}
|