Update lightrec 20220911 (#687)
[pcsx_rearmed.git] / libpcsxcore / lightrec / plugin.c
... / ...
CommitLineData
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
18#include "mem.h"
19#include "plugin.h"
20
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
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
47psxRegisters psxRegs;
48Rcnt rcnts[4];
49
50static struct lightrec_state *lightrec_state;
51
52static char *name = "retroarch.exe";
53
54static bool use_lightrec_interpreter;
55static bool use_pcsx_interpreter;
56static bool booting;
57static u32 lightrec_begin_cycles;
58
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
112static void cop2_op(struct lightrec_state *state, u32 func)
113{
114 struct lightrec_registers *regs = lightrec_get_registers(state);
115
116 psxRegs.code = func;
117
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 }
125}
126
127static bool has_interrupt(void)
128{
129 struct lightrec_registers *regs = lightrec_get_registers(lightrec_state);
130
131 return ((psxHu32(0x1070) & psxHu32(0x1074)) &&
132 (regs->cp0[12] & 0x401) == 0x401) ||
133 (regs->cp0[12] & regs->cp0[13] & 0x0300);
134}
135
136static void lightrec_restore_state(struct lightrec_state *state)
137{
138 lightrec_reset_cycle_count(state, psxRegs.cycle);
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);
144}
145
146static void hw_write_byte(struct lightrec_state *state,
147 u32 op, void *host, u32 mem, u8 val)
148{
149 psxRegs.cycle = lightrec_current_cycle_count(state);
150
151 psxHwWrite8(mem, val);
152
153 lightrec_restore_state(state);
154}
155
156static void hw_write_half(struct lightrec_state *state,
157 u32 op, void *host, u32 mem, u16 val)
158{
159 psxRegs.cycle = lightrec_current_cycle_count(state);
160
161 psxHwWrite16(mem, val);
162
163 lightrec_restore_state(state);
164}
165
166static void hw_write_word(struct lightrec_state *state,
167 u32 op, void *host, u32 mem, u32 val)
168{
169 psxRegs.cycle = lightrec_current_cycle_count(state);
170
171 psxHwWrite32(mem, val);
172
173 lightrec_restore_state(state);
174}
175
176static u8 hw_read_byte(struct lightrec_state *state, u32 op, void *host, u32 mem)
177{
178 u8 val;
179
180 psxRegs.cycle = lightrec_current_cycle_count(state);
181
182 val = psxHwRead8(mem);
183
184 lightrec_restore_state(state);
185
186 return val;
187}
188
189static u16 hw_read_half(struct lightrec_state *state,
190 u32 op, void *host, u32 mem)
191{
192 u16 val;
193
194 psxRegs.cycle = lightrec_current_cycle_count(state);
195
196 val = psxHwRead16(mem);
197
198 lightrec_restore_state(state);
199
200 return val;
201}
202
203static u32 hw_read_word(struct lightrec_state *state,
204 u32 op, void *host, u32 mem)
205{
206 u32 val;
207
208 psxRegs.cycle = lightrec_current_cycle_count(state);
209
210 val = psxHwRead32(mem);
211
212 lightrec_restore_state(state);
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
228static void cache_ctrl_write_word(struct lightrec_state *state,
229 u32 op, void *host, u32 mem, u32 val)
230{
231 cache_ctrl = val;
232}
233
234static u32 cache_ctrl_read_word(struct lightrec_state *state,
235 u32 op, void *host, u32 mem)
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 },
295 [PSX_MAP_CODE_BUFFER] = {
296 .length = CODE_BUFFER_SIZE,
297 },
298};
299
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
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
384static const struct lightrec_ops lightrec_ops = {
385 .cop2_op = cop2_op,
386 .enable_ram = lightrec_enable_ram,
387 .hw_direct = lightrec_can_hw_direct,
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;
395 lightrec_map[PSX_MAP_HW_REGISTERS].address = psxH + 0x1000;
396 lightrec_map[PSX_MAP_PARALLEL_PORT].address = psxP;
397
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;
402 lightrec_map[PSX_MAP_CODE_BUFFER].address = code_buffer;
403 }
404
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
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);
419
420#ifndef _WIN32
421 signal(SIGPIPE, exit);
422#endif
423 return 0;
424}
425
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}
446
447extern void intExecuteBlock();
448extern void gen_interupt();
449
450static void lightrec_plugin_execute_block(void)
451{
452 u32 old_pc = psxRegs.pc;
453 u32 flags;
454
455 gen_interupt();
456
457 // step during early boot so that 0x80030000 fastboot hack works
458 if (booting)
459 next_interupt = psxRegs.cycle;
460
461 if (use_pcsx_interpreter) {
462 intExecuteBlock();
463 } else {
464 lightrec_reset_cycle_count(lightrec_state, psxRegs.cycle);
465 lightrec_restore_regs(lightrec_state);
466
467 if (unlikely(use_lightrec_interpreter)) {
468 psxRegs.pc = lightrec_run_interpreter(lightrec_state,
469 psxRegs.pc,
470 next_interupt);
471 } else {
472 psxRegs.pc = lightrec_execute(lightrec_state,
473 psxRegs.pc, next_interupt);
474 }
475
476 psxRegs.cycle = lightrec_current_cycle_count(lightrec_state);
477
478 lightrec_dump_regs(lightrec_state);
479 flags = lightrec_exit_flags(lightrec_state);
480
481 if (flags & LIGHTREC_EXIT_SEGFAULT) {
482 fprintf(stderr, "Exiting at cycle 0x%08x\n",
483 psxRegs.cycle);
484 exit(1);
485 }
486
487 if (flags & LIGHTREC_EXIT_SYSCALL)
488 psxException(0x20, 0);
489
490 if (booting && (psxRegs.pc & 0xff800000) == 0x80000000)
491 booting = false;
492 }
493
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 }
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{
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);
517}
518
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}
535
536static void lightrec_plugin_apply_config()
537{
538}
539
540static void lightrec_plugin_shutdown(void)
541{
542 lightrec_destroy(lightrec_state);
543}
544
545static void lightrec_plugin_reset(void)
546{
547 struct lightrec_registers *regs;
548
549 regs = lightrec_get_registers(lightrec_state);
550
551 /* Invalidate all blocks */
552 lightrec_invalidate_all(lightrec_state);
553
554 /* Reset registers */
555 memset(regs, 0, sizeof(*regs));
556
557 regs->cp0[12] = 0x10900000; // COP0 enabled | BEV = 1 | TS = 1
558 regs->cp0[15] = 0x00000002; // PRevID = Revision ID, same as R3000A
559
560 booting = true;
561}
562
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
585R3000Acpu psxRec =
586{
587 lightrec_plugin_init,
588 lightrec_plugin_reset,
589 lightrec_plugin_execute,
590 lightrec_plugin_execute_block,
591 lightrec_plugin_clear,
592 lightrec_plugin_notify,
593 lightrec_plugin_apply_config,
594 lightrec_plugin_shutdown,
595};