Update lightrec 20220911 (#687)
[pcsx_rearmed.git] / libpcsxcore / lightrec / plugin.c
CommitLineData
6f1edc3c
PC
1#include <lightrec.h>
2#include <stdbool.h>
3#include <stdio.h>
4#include <unistd.h>
5#include <signal.h>
6
7#include "../cdrom.h"
8#include "../gpu.h"
9#include "../gte.h"
10#include "../mdec.h"
11#include "../psxdma.h"
12#include "../psxhw.h"
13#include "../psxmem.h"
14#include "../r3000a.h"
15
16#include "../frontend/main.h"
17
f8548105 18#include "mem.h"
70939d49 19#include "plugin.h"
f8548105 20
46a38bda 21#if (defined(__arm__) || defined(__aarch64__)) && !defined(ALLOW_LIGHTREC_ON_ARM)
22#error "Lightrec should not be used on ARM (please specify DYNAREC=ari64 to make)"
23#endif
24
6f1edc3c
PC
25#define ARRAY_SIZE(x) (sizeof(x) ? sizeof(x) / sizeof((x)[0]) : 0)
26
27#if __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__
28# define LE32TOH(x) __builtin_bswap32(x)
29# define HTOLE32(x) __builtin_bswap32(x)
30# define LE16TOH(x) __builtin_bswap16(x)
31# define HTOLE16(x) __builtin_bswap16(x)
32#else
33# define LE32TOH(x) (x)
34# define HTOLE32(x) (x)
35# define LE16TOH(x) (x)
36# define HTOLE16(x) (x)
37#endif
38
39#ifdef __GNUC__
40# define likely(x) __builtin_expect(!!(x),1)
41# define unlikely(x) __builtin_expect(!!(x),0)
42#else
43# define likely(x) (x)
44# define unlikely(x) (x)
45#endif
46
630b122b 47psxRegisters psxRegs;
48Rcnt rcnts[4];
49
6f1edc3c
PC
50static struct lightrec_state *lightrec_state;
51
52static char *name = "retroarch.exe";
53
54static bool use_lightrec_interpreter;
6b02f240 55static bool use_pcsx_interpreter;
2bf88032 56static bool booting;
6f1edc3c
PC
57static u32 lightrec_begin_cycles;
58
6f1edc3c
PC
59enum my_cp2_opcodes {
60 OP_CP2_RTPS = 0x01,
61 OP_CP2_NCLIP = 0x06,
62 OP_CP2_OP = 0x0c,
63 OP_CP2_DPCS = 0x10,
64 OP_CP2_INTPL = 0x11,
65 OP_CP2_MVMVA = 0x12,
66 OP_CP2_NCDS = 0x13,
67 OP_CP2_CDP = 0x14,
68 OP_CP2_NCDT = 0x16,
69 OP_CP2_NCCS = 0x1b,
70 OP_CP2_CC = 0x1c,
71 OP_CP2_NCS = 0x1e,
72 OP_CP2_NCT = 0x20,
73 OP_CP2_SQR = 0x28,
74 OP_CP2_DCPL = 0x29,
75 OP_CP2_DPCT = 0x2a,
76 OP_CP2_AVSZ3 = 0x2d,
77 OP_CP2_AVSZ4 = 0x2e,
78 OP_CP2_RTPT = 0x30,
79 OP_CP2_GPF = 0x3d,
80 OP_CP2_GPL = 0x3e,
81 OP_CP2_NCCT = 0x3f,
82};
83
84static void (*cp2_ops[])(struct psxCP2Regs *) = {
85 [OP_CP2_RTPS] = gteRTPS,
86 [OP_CP2_RTPS] = gteRTPS,
87 [OP_CP2_NCLIP] = gteNCLIP,
88 [OP_CP2_OP] = gteOP,
89 [OP_CP2_DPCS] = gteDPCS,
90 [OP_CP2_INTPL] = gteINTPL,
91 [OP_CP2_MVMVA] = gteMVMVA,
92 [OP_CP2_NCDS] = gteNCDS,
93 [OP_CP2_CDP] = gteCDP,
94 [OP_CP2_NCDT] = gteNCDT,
95 [OP_CP2_NCCS] = gteNCCS,
96 [OP_CP2_CC] = gteCC,
97 [OP_CP2_NCS] = gteNCS,
98 [OP_CP2_NCT] = gteNCT,
99 [OP_CP2_SQR] = gteSQR,
100 [OP_CP2_DCPL] = gteDCPL,
101 [OP_CP2_DPCT] = gteDPCT,
102 [OP_CP2_AVSZ3] = gteAVSZ3,
103 [OP_CP2_AVSZ4] = gteAVSZ4,
104 [OP_CP2_RTPT] = gteRTPT,
105 [OP_CP2_GPF] = gteGPF,
106 [OP_CP2_GPL] = gteGPL,
107 [OP_CP2_NCCT] = gteNCCT,
108};
109
110static char cache_buf[64 * 1024];
111
0733c3ab 112static void cop2_op(struct lightrec_state *state, u32 func)
6f1edc3c 113{
0733c3ab 114 struct lightrec_registers *regs = lightrec_get_registers(state);
6f1edc3c 115
0733c3ab 116 psxRegs.code = func;
6f1edc3c 117
0733c3ab
PC
118 if (unlikely(!cp2_ops[func & 0x3f])) {
119 fprintf(stderr, "Invalid CP2 function %u\n", func);
120 } else {
121 /* This works because regs->cp2c comes right after regs->cp2d,
122 * so it can be cast to a pcsxCP2Regs pointer. */
123 cp2_ops[func & 0x3f]((psxCP2Regs *) regs->cp2d);
124 }
6f1edc3c
PC
125}
126
2bf88032
PC
127static bool has_interrupt(void)
128{
6ce0b00a
PC
129 struct lightrec_registers *regs = lightrec_get_registers(lightrec_state);
130
2bf88032 131 return ((psxHu32(0x1070) & psxHu32(0x1074)) &&
6ce0b00a
PC
132 (regs->cp0[12] & 0x401) == 0x401) ||
133 (regs->cp0[12] & regs->cp0[13] & 0x0300);
2bf88032
PC
134}
135
f4f9f2a4
PC
136static void lightrec_restore_state(struct lightrec_state *state)
137{
138 lightrec_reset_cycle_count(state, psxRegs.cycle);
2bf88032
PC
139
140 if (booting || has_interrupt())
141 lightrec_set_exit_flags(state, LIGHTREC_EXIT_CHECK_INTERRUPT);
142 else
143 lightrec_set_target_cycle_count(state, next_interupt);
f4f9f2a4
PC
144}
145
6b02f240 146static void hw_write_byte(struct lightrec_state *state,
147 u32 op, void *host, u32 mem, u8 val)
6f1edc3c
PC
148{
149 psxRegs.cycle = lightrec_current_cycle_count(state);
150
151 psxHwWrite8(mem, val);
6f1edc3c 152
f4f9f2a4 153 lightrec_restore_state(state);
6f1edc3c
PC
154}
155
6b02f240 156static void hw_write_half(struct lightrec_state *state,
157 u32 op, void *host, u32 mem, u16 val)
6f1edc3c
PC
158{
159 psxRegs.cycle = lightrec_current_cycle_count(state);
160
161 psxHwWrite16(mem, val);
6f1edc3c 162
f4f9f2a4 163 lightrec_restore_state(state);
6f1edc3c
PC
164}
165
6b02f240 166static void hw_write_word(struct lightrec_state *state,
167 u32 op, void *host, u32 mem, u32 val)
6f1edc3c
PC
168{
169 psxRegs.cycle = lightrec_current_cycle_count(state);
170
171 psxHwWrite32(mem, val);
6f1edc3c 172
f4f9f2a4 173 lightrec_restore_state(state);
6f1edc3c
PC
174}
175
6b02f240 176static u8 hw_read_byte(struct lightrec_state *state, u32 op, void *host, u32 mem)
6f1edc3c
PC
177{
178 u8 val;
179
180 psxRegs.cycle = lightrec_current_cycle_count(state);
181
6f1edc3c 182 val = psxHwRead8(mem);
f4f9f2a4
PC
183
184 lightrec_restore_state(state);
6f1edc3c
PC
185
186 return val;
187}
188
6b02f240 189static u16 hw_read_half(struct lightrec_state *state,
190 u32 op, void *host, u32 mem)
6f1edc3c
PC
191{
192 u16 val;
193
194 psxRegs.cycle = lightrec_current_cycle_count(state);
195
6f1edc3c 196 val = psxHwRead16(mem);
f4f9f2a4
PC
197
198 lightrec_restore_state(state);
6f1edc3c
PC
199
200 return val;
201}
202
6b02f240 203static u32 hw_read_word(struct lightrec_state *state,
204 u32 op, void *host, u32 mem)
6f1edc3c
PC
205{
206 u32 val;
207
208 psxRegs.cycle = lightrec_current_cycle_count(state);
209
6f1edc3c 210 val = psxHwRead32(mem);
f4f9f2a4
PC
211
212 lightrec_restore_state(state);
6f1edc3c
PC
213
214 return val;
215}
216
217static struct lightrec_mem_map_ops hw_regs_ops = {
218 .sb = hw_write_byte,
219 .sh = hw_write_half,
220 .sw = hw_write_word,
221 .lb = hw_read_byte,
222 .lh = hw_read_half,
223 .lw = hw_read_word,
224};
225
226static u32 cache_ctrl;
227
6b02f240 228static void cache_ctrl_write_word(struct lightrec_state *state,
229 u32 op, void *host, u32 mem, u32 val)
6f1edc3c
PC
230{
231 cache_ctrl = val;
232}
233
6b02f240 234static u32 cache_ctrl_read_word(struct lightrec_state *state,
235 u32 op, void *host, u32 mem)
6f1edc3c
PC
236{
237 return cache_ctrl;
238}
239
240static struct lightrec_mem_map_ops cache_ctrl_ops = {
241 .sw = cache_ctrl_write_word,
242 .lw = cache_ctrl_read_word,
243};
244
245static struct lightrec_mem_map lightrec_map[] = {
246 [PSX_MAP_KERNEL_USER_RAM] = {
247 /* Kernel and user memory */
248 .pc = 0x00000000,
249 .length = 0x200000,
250 },
251 [PSX_MAP_BIOS] = {
252 /* BIOS */
253 .pc = 0x1fc00000,
254 .length = 0x80000,
255 },
256 [PSX_MAP_SCRATCH_PAD] = {
257 /* Scratch pad */
258 .pc = 0x1f800000,
259 .length = 0x400,
260 },
261 [PSX_MAP_PARALLEL_PORT] = {
262 /* Parallel port */
263 .pc = 0x1f000000,
264 .length = 0x10000,
265 },
266 [PSX_MAP_HW_REGISTERS] = {
267 /* Hardware registers */
268 .pc = 0x1f801000,
269 .length = 0x2000,
270 .ops = &hw_regs_ops,
271 },
272 [PSX_MAP_CACHE_CONTROL] = {
273 /* Cache control */
274 .pc = 0x5ffe0130,
275 .length = 4,
276 .ops = &cache_ctrl_ops,
277 },
278
279 /* Mirrors of the kernel/user memory */
280 [PSX_MAP_MIRROR1] = {
281 .pc = 0x00200000,
282 .length = 0x200000,
283 .mirror_of = &lightrec_map[PSX_MAP_KERNEL_USER_RAM],
284 },
285 [PSX_MAP_MIRROR2] = {
286 .pc = 0x00400000,
287 .length = 0x200000,
288 .mirror_of = &lightrec_map[PSX_MAP_KERNEL_USER_RAM],
289 },
290 [PSX_MAP_MIRROR3] = {
291 .pc = 0x00600000,
292 .length = 0x200000,
293 .mirror_of = &lightrec_map[PSX_MAP_KERNEL_USER_RAM],
294 },
f8548105
PC
295 [PSX_MAP_CODE_BUFFER] = {
296 .length = CODE_BUFFER_SIZE,
297 },
6f1edc3c
PC
298};
299
0733c3ab
PC
300static void lightrec_enable_ram(struct lightrec_state *state, bool enable)
301{
302 if (enable)
303 memcpy(psxM, cache_buf, sizeof(cache_buf));
304 else
305 memcpy(cache_buf, psxM, sizeof(cache_buf));
306}
307
ba3814c1
PC
308static bool lightrec_can_hw_direct(u32 kaddr, bool is_write, u8 size)
309{
310 switch (size) {
311 case 8:
312 switch (kaddr) {
313 case 0x1f801040:
314 case 0x1f801050:
315 case 0x1f801800:
316 case 0x1f801801:
317 case 0x1f801802:
318 case 0x1f801803:
319 return false;
320 default:
321 return true;
322 }
323 case 16:
324 switch (kaddr) {
325 case 0x1f801040:
326 case 0x1f801044:
327 case 0x1f801048:
328 case 0x1f80104a:
329 case 0x1f80104e:
330 case 0x1f801050:
331 case 0x1f801054:
332 case 0x1f80105a:
333 case 0x1f80105e:
334 case 0x1f801100:
335 case 0x1f801104:
336 case 0x1f801108:
337 case 0x1f801110:
338 case 0x1f801114:
339 case 0x1f801118:
340 case 0x1f801120:
341 case 0x1f801124:
342 case 0x1f801128:
343 return false;
344 case 0x1f801070:
345 case 0x1f801074:
346 return !is_write;
347 default:
348 return is_write || kaddr < 0x1f801c00 || kaddr >= 0x1f801e00;
349 }
350 default:
351 switch (kaddr) {
352 case 0x1f801040:
353 case 0x1f801050:
354 case 0x1f801100:
355 case 0x1f801104:
356 case 0x1f801108:
357 case 0x1f801110:
358 case 0x1f801114:
359 case 0x1f801118:
360 case 0x1f801120:
361 case 0x1f801124:
362 case 0x1f801128:
363 case 0x1f801810:
364 case 0x1f801814:
365 case 0x1f801820:
366 case 0x1f801824:
367 return false;
368 case 0x1f801070:
369 case 0x1f801074:
370 case 0x1f801088:
371 case 0x1f801098:
372 case 0x1f8010a8:
373 case 0x1f8010b8:
374 case 0x1f8010c8:
375 case 0x1f8010e8:
376 case 0x1f8010f4:
377 return !is_write;
378 default:
379 return !is_write || kaddr < 0x1f801c00 || kaddr >= 0x1f801e00;
380 }
381 }
382}
383
6f1edc3c 384static const struct lightrec_ops lightrec_ops = {
0733c3ab
PC
385 .cop2_op = cop2_op,
386 .enable_ram = lightrec_enable_ram,
ba3814c1 387 .hw_direct = lightrec_can_hw_direct,
6f1edc3c
PC
388};
389
390static int lightrec_plugin_init(void)
391{
392 lightrec_map[PSX_MAP_KERNEL_USER_RAM].address = psxM;
393 lightrec_map[PSX_MAP_BIOS].address = psxR;
394 lightrec_map[PSX_MAP_SCRATCH_PAD].address = psxH;
6ce0b00a 395 lightrec_map[PSX_MAP_HW_REGISTERS].address = psxH + 0x1000;
6f1edc3c
PC
396 lightrec_map[PSX_MAP_PARALLEL_PORT].address = psxP;
397
479d58cf
PC
398 if (LIGHTREC_CUSTOM_MAP) {
399 lightrec_map[PSX_MAP_MIRROR1].address = psxM + 0x200000;
400 lightrec_map[PSX_MAP_MIRROR2].address = psxM + 0x400000;
401 lightrec_map[PSX_MAP_MIRROR3].address = psxM + 0x600000;
f8548105 402 lightrec_map[PSX_MAP_CODE_BUFFER].address = code_buffer;
479d58cf
PC
403 }
404
6f1edc3c
PC
405 use_lightrec_interpreter = !!getenv("LIGHTREC_INTERPRETER");
406 if (getenv("LIGHTREC_BEGIN_CYCLES"))
407 lightrec_begin_cycles = (unsigned int) strtol(
408 getenv("LIGHTREC_BEGIN_CYCLES"), NULL, 0);
409
410 lightrec_state = lightrec_init(name,
411 lightrec_map, ARRAY_SIZE(lightrec_map),
412 &lightrec_ops);
413
9361a5aa
EC
414 // fprintf(stderr, "M=0x%lx, P=0x%lx, R=0x%lx, H=0x%lx\n",
415 // (uintptr_t) psxM,
416 // (uintptr_t) psxP,
417 // (uintptr_t) psxR,
418 // (uintptr_t) psxH);
6f1edc3c
PC
419
420#ifndef _WIN32
421 signal(SIGPIPE, exit);
422#endif
423 return 0;
424}
425
0733c3ab
PC
426static void lightrec_dump_regs(struct lightrec_state *state)
427{
428 struct lightrec_registers *regs = lightrec_get_registers(state);
429
430 if (unlikely(booting))
431 memcpy(&psxRegs.GPR, regs->gpr, sizeof(regs->gpr));
432 psxRegs.CP0.n.Status = regs->cp0[12];
433 psxRegs.CP0.n.Cause = regs->cp0[13];
434}
435
436static void lightrec_restore_regs(struct lightrec_state *state)
437{
438 struct lightrec_registers *regs = lightrec_get_registers(state);
439
440 if (unlikely(booting))
441 memcpy(regs->gpr, &psxRegs.GPR, sizeof(regs->gpr));
442 regs->cp0[12] = psxRegs.CP0.n.Status;
443 regs->cp0[13] = psxRegs.CP0.n.Cause;
444 regs->cp0[14] = psxRegs.CP0.n.EPC;
445}
6b02f240 446
447extern void intExecuteBlock();
2bf88032 448extern void gen_interupt();
6b02f240 449
6f1edc3c
PC
450static void lightrec_plugin_execute_block(void)
451{
452 u32 old_pc = psxRegs.pc;
453 u32 flags;
454
2bf88032
PC
455 gen_interupt();
456
ba3814c1
PC
457 // step during early boot so that 0x80030000 fastboot hack works
458 if (booting)
459 next_interupt = psxRegs.cycle;
460
6b02f240 461 if (use_pcsx_interpreter) {
462 intExecuteBlock();
463 } else {
464 lightrec_reset_cycle_count(lightrec_state, psxRegs.cycle);
0733c3ab 465 lightrec_restore_regs(lightrec_state);
6f1edc3c 466
ba3814c1 467 if (unlikely(use_lightrec_interpreter)) {
6b02f240 468 psxRegs.pc = lightrec_run_interpreter(lightrec_state,
ba3814c1
PC
469 psxRegs.pc,
470 next_interupt);
471 } else {
2bf88032
PC
472 psxRegs.pc = lightrec_execute(lightrec_state,
473 psxRegs.pc, next_interupt);
ba3814c1 474 }
6f1edc3c 475
6b02f240 476 psxRegs.cycle = lightrec_current_cycle_count(lightrec_state);
6f1edc3c 477
0733c3ab 478 lightrec_dump_regs(lightrec_state);
6b02f240 479 flags = lightrec_exit_flags(lightrec_state);
6f1edc3c 480
6b02f240 481 if (flags & LIGHTREC_EXIT_SEGFAULT) {
482 fprintf(stderr, "Exiting at cycle 0x%08x\n",
483 psxRegs.cycle);
484 exit(1);
485 }
6f1edc3c 486
6b02f240 487 if (flags & LIGHTREC_EXIT_SYSCALL)
488 psxException(0x20, 0);
6f1edc3c 489
2bf88032
PC
490 if (booting && (psxRegs.pc & 0xff800000) == 0x80000000)
491 booting = false;
492 }
6f1edc3c 493
6f1edc3c
PC
494 if ((psxRegs.CP0.n.Cause & psxRegs.CP0.n.Status & 0x300) &&
495 (psxRegs.CP0.n.Status & 0x1)) {
496 /* Handle software interrupts */
497 psxRegs.CP0.n.Cause &= ~0x7c;
498 psxException(psxRegs.CP0.n.Cause, 0);
499 }
6f1edc3c
PC
500}
501
502static void lightrec_plugin_execute(void)
503{
504 extern int stop;
505
506 while (!stop)
507 lightrec_plugin_execute_block();
508}
509
510static void lightrec_plugin_clear(u32 addr, u32 size)
511{
8c9468f1
ZC
512 if (addr == 0 && size == UINT32_MAX)
513 lightrec_invalidate_all(lightrec_state);
514 else
515 /* size * 4: PCSX uses DMA units */
516 lightrec_invalidate(lightrec_state, addr, size * 4);
6f1edc3c
PC
517}
518
7a811716 519static void lightrec_plugin_notify(int note, void *data)
520{
521 /*
522 To change once proper icache emulation is emulated
523 switch (note)
524 {
525 case R3000ACPU_NOTIFY_CACHE_UNISOLATED:
526 lightrec_plugin_clear(0, 0x200000/4);
527 break;
528 case R3000ACPU_NOTIFY_CACHE_ISOLATED:
529 // Sent from psxDma3().
530 case R3000ACPU_NOTIFY_DMA3_EXE_LOAD:
531 default:
532 break;
533 }*/
534}
630b122b 535
536static void lightrec_plugin_apply_config()
537{
538}
7a811716 539
6f1edc3c
PC
540static void lightrec_plugin_shutdown(void)
541{
542 lightrec_destroy(lightrec_state);
543}
544
545static void lightrec_plugin_reset(void)
546{
0733c3ab
PC
547 struct lightrec_registers *regs;
548
0733c3ab
PC
549 regs = lightrec_get_registers(lightrec_state);
550
03535202
PC
551 /* Invalidate all blocks */
552 lightrec_invalidate_all(lightrec_state);
553
554 /* Reset registers */
555 memset(regs, 0, sizeof(*regs));
556
0733c3ab
PC
557 regs->cp0[12] = 0x10900000; // COP0 enabled | BEV = 1 | TS = 1
558 regs->cp0[15] = 0x00000002; // PRevID = Revision ID, same as R3000A
559
2bf88032 560 booting = true;
6f1edc3c
PC
561}
562
6962f770
PC
563void lightrec_plugin_prepare_load_state(void)
564{
565 struct lightrec_registers *regs;
566
567 regs = lightrec_get_registers(lightrec_state);
568 memcpy(regs->cp2d, &psxRegs.CP2, sizeof(regs->cp2d) + sizeof(regs->cp2c));
569 memcpy(regs->cp0, &psxRegs.CP0, sizeof(regs->cp0));
570 memcpy(regs->gpr, &psxRegs.GPR, sizeof(regs->gpr));
571
572 lightrec_invalidate_all(lightrec_state);
573}
574
575void lightrec_plugin_prepare_save_state(void)
576{
577 struct lightrec_registers *regs;
578
579 regs = lightrec_get_registers(lightrec_state);
580 memcpy(&psxRegs.CP2, regs->cp2d, sizeof(regs->cp2d) + sizeof(regs->cp2c));
581 memcpy(&psxRegs.CP0, regs->cp0, sizeof(regs->cp0));
582 memcpy(&psxRegs.GPR, regs->gpr, sizeof(regs->gpr));
583}
584
6f1edc3c
PC
585R3000Acpu psxRec =
586{
587 lightrec_plugin_init,
588 lightrec_plugin_reset,
589 lightrec_plugin_execute,
590 lightrec_plugin_execute_block,
591 lightrec_plugin_clear,
7a811716 592 lightrec_plugin_notify,
630b122b 593 lightrec_plugin_apply_config,
6f1edc3c
PC
594 lightrec_plugin_shutdown,
595};