forked from libretro/parallel-rsp
-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathparallel.cpp
158 lines (131 loc) · 3.97 KB
/
parallel.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
#ifdef DEBUG_JIT
#include "debug_rsp.hpp"
#else
#include "rsp_jit.hpp"
#endif
#include <stdint.h>
#include "Zilmar_Rsp.h"
#define RSP_PARALLEL_VERSION 0x0101
#define RSP_PLUGIN_API_VERSION 0x020000
namespace RSP
{
RSP_INFO rsp;
#ifdef DEBUG_JIT
RSP::CPU cpu;
#else
RSP::JIT::CPU cpu;
#endif
short MFC0_count[32];
int SP_STATUS_TIMEOUT;
} // namespace RSP
extern "C"
{
// Hack entry point to use when loading savestates when we're tracing.
void rsp_clear_registers()
{
memset(RSP::cpu.get_state().sr, 0, sizeof(uint32_t) * 32);
memset(&RSP::cpu.get_state().cp2, 0, sizeof(RSP::cpu.get_state().cp2));
}
#ifdef INTENSE_DEBUG
// Need super-fast hash here.
static uint64_t hash_imem(const uint8_t *data, size_t size)
{
uint64_t h = 0xcbf29ce484222325ull;
size_t i;
for (i = 0; i < size; i++)
h = (h * 0x100000001b3ull) ^ data[i];
return h;
}
void log_rsp_mem_parallel(void)
{
fprintf(stderr, "IMEM HASH: 0x%016llx\n", hash_imem(RSP::rsp.IMEM, 0x1000));
fprintf(stderr, "DMEM HASH: 0x%016llx\n", hash_imem(RSP::rsp.DMEM, 0x1000));
}
#endif
EXPORT unsigned int CALL DoRspCycles(unsigned int cycles)
{
if (*RSP::rsp.SP_STATUS_REG & (SP_STATUS_HALT | SP_STATUS_BROKE))
return 0;
// We don't know if Mupen from the outside invalidated our IMEM.
RSP::cpu.invalidate_imem();
// Run CPU until we either break or we need to fire an IRQ.
RSP::cpu.get_state().pc = *RSP::rsp.SP_PC_REG & 0xfff;
#ifdef INTENSE_DEBUG
fprintf(stderr, "RUN TASK: %u\n", RSP::cpu.get_state().pc);
log_rsp_mem_parallel();
#endif
for (auto &count : RSP::MFC0_count)
count = 0;
while (!(*RSP::rsp.SP_STATUS_REG & SP_STATUS_HALT))
{
auto mode = RSP::cpu.run();
if (mode == RSP::MODE_CHECK_FLAGS && (*RSP::cpu.get_state().cp0.irq & 1))
break;
}
*RSP::rsp.SP_PC_REG = 0x04001000 | (RSP::cpu.get_state().pc & 0xffc);
// From CXD4.
if (*RSP::rsp.SP_STATUS_REG & SP_STATUS_BROKE)
return cycles;
else if (*RSP::cpu.get_state().cp0.irq & 1)
RSP::rsp.CheckInterrupts();
else if (*RSP::rsp.SP_SEMAPHORE_REG != 0) // Semaphore lock fixes.
{
}
else
RSP::SP_STATUS_TIMEOUT = 16; // From now on, wait 16 times, not 0x7fff
// CPU restarts with the correct SIGs.
*RSP::rsp.SP_STATUS_REG &= ~SP_STATUS_HALT;
return cycles;
}
EXPORT void CALL GetDllInfo(PLUGIN_INFO * PluginInfo)
{
PluginInfo->Version = 0x0101;
PluginInfo->Type = PLUGIN_TYPE_RSP;
strcpy(PluginInfo->Name, "ParaLLel RSP Plugin");
PluginInfo->NormalMemory = 1;
PluginInfo->MemoryBswaped = 1;
}
EXPORT void CALL RomClosed(void)
{
*RSP::rsp.SP_PC_REG = 0x00000000;
}
EXPORT void CALL InitiateRSP(RSP_INFO Rsp_Info, unsigned int *CycleCount)
{
if (CycleCount)
*CycleCount = 0;
if (Rsp_Info.DMEM == Rsp_Info.IMEM) /* usually dummy RSP data for testing */
return; /* DMA is not executed just because plugin initiates. */
RSP::rsp = Rsp_Info;
*RSP::rsp.SP_PC_REG = 0x04001000 & 0x00000FFF; /* task init bug on Mupen64 */
auto **cr = RSP::cpu.get_state().cp0.cr;
cr[0x0] = RSP::rsp.SP_MEM_ADDR_REG;
cr[0x1] = RSP::rsp.SP_DRAM_ADDR_REG;
cr[0x2] = RSP::rsp.SP_RD_LEN_REG;
cr[0x3] = RSP::rsp.SP_WR_LEN_REG;
cr[0x4] = RSP::rsp.SP_STATUS_REG;
cr[0x5] = RSP::rsp.SP_DMA_FULL_REG;
cr[0x6] = RSP::rsp.SP_DMA_BUSY_REG;
cr[0x7] = RSP::rsp.SP_SEMAPHORE_REG;
cr[0x8] = RSP::rsp.DPC_START_REG;
cr[0x9] = RSP::rsp.DPC_END_REG;
cr[0xA] = RSP::rsp.DPC_CURRENT_REG;
cr[0xB] = RSP::rsp.DPC_STATUS_REG;
cr[0xC] = RSP::rsp.DPC_CLOCK_REG;
cr[0xD] = RSP::rsp.DPC_BUFBUSY_REG;
cr[0xE] = RSP::rsp.DPC_PIPEBUSY_REG;
cr[0xF] = RSP::rsp.DPC_TMEM_REG;
*cr[RSP::CP0_REGISTER_SP_STATUS] = SP_STATUS_HALT;
RSP::cpu.get_state().cp0.irq = RSP::rsp.MI_INTR_REG;
// From CXD4.
RSP::SP_STATUS_TIMEOUT = 0x7fff;
RSP::cpu.set_dmem(reinterpret_cast<uint32_t *>(Rsp_Info.DMEM));
RSP::cpu.set_imem(reinterpret_cast<uint32_t *>(Rsp_Info.IMEM));
RSP::cpu.set_rdram(reinterpret_cast<uint32_t *>(Rsp_Info.RDRAM));
}
EXPORT void CALL CloseDLL(void)
{
}
EXPORT void CALL DllConfig(int hWnd)
{
}
}