X-Git-Url: https://notaz.gp2x.de/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=pico%2F32x%2F32x.c;h=5581b0940bf33bf70d152627310add6998cd14f6;hb=ed4402a7dfd12dbbf34c547b438a671ae8114197;hp=9a981f6b35c3ee94e708a3f12204dd44d2a91796;hpb=c987bb5c36d92c1441e4dcfd9bded3e5dc1f1ff6;p=picodrive.git diff --git a/pico/32x/32x.c b/pico/32x/32x.c index 9a981f6..5581b09 100644 --- a/pico/32x/32x.c +++ b/pico/32x/32x.c @@ -1,15 +1,32 @@ +/* + * PicoDrive + * (C) notaz, 2009,2010 + * + * This work is licensed under the terms of MAME license. + * See COPYING file in the top-level directory. + */ #include "../pico_int.h" #include "../sound/ym2612.h" struct Pico32x Pico32x; +SH2 sh2s[2]; -static void sh2_irq_cb(int id, int level) +static int REGPARM(2) sh2_irq_cb(SH2 *sh2, int level) { - // diagnostic for now - elprintf(EL_32X, "%csh2 ack %d @ %08x", id ? 's' : 'm', level, sh2_pc(id)); + if (sh2->pending_irl > sh2->pending_int_irq) { + elprintf(EL_32X, "%csh2 ack/irl %d @ %08x", + sh2->is_slave ? 's' : 'm', level, sh2->pc); + return 64 + sh2->pending_irl / 2; + } else { + elprintf(EL_32X, "%csh2 ack/int %d/%d @ %08x", + sh2->is_slave ? 's' : 'm', level, sh2->pending_int_vector, sh2->pc); + sh2->pending_int_irq = 0; // auto-clear + sh2->pending_level = sh2->pending_irl; + return sh2->pending_int_vector; + } } -void p32x_update_irls(void) +void p32x_update_irls(int nested_call) { int irqs, mlvl = 0, slvl = 0; @@ -26,8 +43,8 @@ void p32x_update_irls(void) slvl *= 2; elprintf(EL_32X, "update_irls: m %d, s %d", mlvl, slvl); - sh2_irl_irq(&msh2, mlvl); - sh2_irl_irq(&ssh2, slvl); + sh2_irl_irq(&msh2, mlvl, nested_call); + sh2_irl_irq(&ssh2, slvl, nested_call); mlvl = mlvl ? 1 : 0; slvl = slvl ? 1 : 0; p32x_poll_event(mlvl | (slvl << 1), 0); @@ -37,32 +54,89 @@ void Pico32xStartup(void) { elprintf(EL_STATUS|EL_32X, "32X startup"); + // TODO: OOM handling PicoAHW |= PAHW_32X; - PicoMemSetup32x(); - sh2_init(&msh2, 0); msh2.irq_callback = sh2_irq_cb; - sh2_reset(&msh2); - sh2_init(&ssh2, 1); ssh2.irq_callback = sh2_irq_cb; - sh2_reset(&ssh2); + + PicoMemSetup32x(); if (!Pico.m.pal) Pico32x.vdp_regs[0] |= P32XV_nPAL; + PREG8(Pico32xMem->sh2_peri_regs[0], 4) = + PREG8(Pico32xMem->sh2_peri_regs[1], 4) = 0x84; // SCI SSR + + rendstatus_old = -1; + emu_32x_startup(); } +#define HWSWAP(x) (((x) << 16) | ((x) >> 16)) +void p32x_reset_sh2s(void) +{ + elprintf(EL_32X, "sh2 reset"); + + sh2_reset(&msh2); + sh2_reset(&ssh2); + + // if we don't have BIOS set, perform it's work here. + // MSH2 + if (p32x_bios_m == NULL) { + unsigned int idl_src, idl_dst, idl_size; // initial data load + unsigned int vbr; + + // initial data + idl_src = HWSWAP(*(unsigned int *)(Pico.rom + 0x3d4)) & ~0xf0000000; + idl_dst = HWSWAP(*(unsigned int *)(Pico.rom + 0x3d8)) & ~0xf0000000; + idl_size= HWSWAP(*(unsigned int *)(Pico.rom + 0x3dc)); + if (idl_size > Pico.romsize || idl_src + idl_size > Pico.romsize || + idl_size > 0x40000 || idl_dst + idl_size > 0x40000 || (idl_src & 3) || (idl_dst & 3)) { + elprintf(EL_STATUS|EL_ANOMALY, "32x: invalid initial data ptrs: %06x -> %06x, %06x", + idl_src, idl_dst, idl_size); + } + else + memcpy(Pico32xMem->sdram + idl_dst, Pico.rom + idl_src, idl_size); + + // GBR/VBR + vbr = HWSWAP(*(unsigned int *)(Pico.rom + 0x3e8)); + sh2_set_gbr(0, 0x20004000); + sh2_set_vbr(0, vbr); + + // checksum and M_OK + Pico32x.regs[0x28 / 2] = *(unsigned short *)(Pico.rom + 0x18e); + // program will set M_OK + } + + // SSH2 + if (p32x_bios_s == NULL) { + unsigned int vbr; + + // GBR/VBR + vbr = HWSWAP(*(unsigned int *)(Pico.rom + 0x3ec)); + sh2_set_gbr(1, 0x20004000); + sh2_set_vbr(1, vbr); + // program will set S_OK + } + + msh2.m68krcycles_done = ssh2.m68krcycles_done = SekCyclesDoneT(); +} + void Pico32xInit(void) { + if (msh2.mult_m68k_to_sh2 == 0 || msh2.mult_sh2_to_m68k == 0) + Pico32xSetClocks(PICO_MSH2_HZ, 0); + if (ssh2.mult_m68k_to_sh2 == 0 || ssh2.mult_sh2_to_m68k == 0) + Pico32xSetClocks(0, PICO_MSH2_HZ); } void PicoPower32x(void) { memset(&Pico32x, 0, sizeof(Pico32x)); - Pico32x.regs[0] = 0x0082; // SH2 reset? + Pico32x.regs[0] = P32XS_REN|P32XS_nRES; // verified Pico32x.vdp_regs[0x0a/2] = P32XV_VBLK|P32XV_HBLK|P32XV_PEN; Pico32x.sh2_regs[0] = P32XS2_ADEN; } @@ -70,20 +144,52 @@ void PicoPower32x(void) void PicoUnload32x(void) { if (Pico32xMem != NULL) - free(Pico32xMem); + plat_munmap(Pico32xMem, sizeof(*Pico32xMem)); Pico32xMem = NULL; + sh2_finish(&msh2); + sh2_finish(&ssh2); PicoAHW &= ~PAHW_32X; } void PicoReset32x(void) { - extern int p32x_csum_faked; - p32x_csum_faked = 0; // tmp + if (PicoAHW & PAHW_32X) { + Pico32x.sh2irqs |= P32XI_VRES; + p32x_update_irls(0); + p32x_poll_event(3, 0); + } } static void p32x_start_blank(void) { + if (Pico32xDrawMode != PDM32X_OFF && !PicoSkipFrame) { + int offs, lines; + + pprof_start(draw); + + offs = 8; lines = 224; + if ((Pico.video.reg[1] & 8) && !(PicoOpt & POPT_ALT_RENDERER)) { + offs = 0; + lines = 240; + } + + // XXX: no proper handling of 32col mode.. + if ((Pico32x.vdp_regs[0] & P32XV_Mx) != 0 && // 32x not blanking + (Pico.video.reg[12] & 1) && // 40col mode + (PicoDrawMask & PDRAW_32X_ON)) + { + int md_bg = Pico.video.reg[7] & 0x3f; + + // we draw full layer (not line-by-line) + PicoDraw32xLayer(offs, lines, md_bg); + } + else if (Pico32xDrawMode != PDM32X_32X_ONLY) + PicoDraw32xLayerMdOnly(offs, lines); + + pprof_end(draw); + } + // enter vblank Pico32x.vdp_regs[0x0a/2] |= P32XV_VBLK|P32XV_PEN; @@ -95,71 +201,80 @@ static void p32x_start_blank(void) } Pico32x.sh2irqs |= P32XI_VINT; - p32x_update_irls(); + p32x_update_irls(0); p32x_poll_event(3, 1); } -// FIXME.. -static __inline void SekRunM68k(int cyc) +#define sync_sh2s_normal p32x_sync_sh2s +//#define sync_sh2s_lockstep p32x_sync_sh2s + +void sync_sh2s_normal(unsigned int m68k_target) { - int cyc_do; - SekCycleAim += cyc; - if (Pico32x.emu_flags & P32XF_68KPOLL) { - SekCycleCnt = SekCycleAim; - return; + unsigned int target = m68k_target; + int msh2_cycles, ssh2_cycles; + int done; + + elprintf(EL_32X, "sh2 sync to %u (%u)", m68k_target, SekCycleCnt); + + if (!(Pico32x.regs[0] & P32XS_nRES)) + return; // rare + + { + msh2_cycles = C_M68K_TO_SH2(msh2, target - msh2.m68krcycles_done); + ssh2_cycles = C_M68K_TO_SH2(ssh2, target - ssh2.m68krcycles_done); + + while (msh2_cycles > 0 || ssh2_cycles > 0) { + elprintf(EL_32X, "sh2 exec %u,%u->%u", + msh2.m68krcycles_done, ssh2.m68krcycles_done, target); + + if (Pico32x.emu_flags & (P32XF_SSH2POLL|P32XF_SSH2VPOLL)) { + ssh2.m68krcycles_done = target; + ssh2_cycles = 0; + } + else if (ssh2_cycles > 0) { + done = sh2_execute(&ssh2, ssh2_cycles); + ssh2.m68krcycles_done += C_SH2_TO_M68K(ssh2, done); + + ssh2_cycles = C_M68K_TO_SH2(ssh2, target - ssh2.m68krcycles_done); + } + + if (Pico32x.emu_flags & (P32XF_MSH2POLL|P32XF_MSH2VPOLL)) { + msh2.m68krcycles_done = target; + msh2_cycles = 0; + } + else if (msh2_cycles > 0) { + done = sh2_execute(&msh2, msh2_cycles); + msh2.m68krcycles_done += C_SH2_TO_M68K(msh2, done); + + msh2_cycles = C_M68K_TO_SH2(msh2, target - msh2.m68krcycles_done); + } + } } - if ((cyc_do = SekCycleAim - SekCycleCnt) <= 0) - return; -#if defined(EMU_CORE_DEBUG) - // this means we do run-compare - SekCycleCnt+=CM_compareRun(cyc_do, 0); -#elif defined(EMU_C68K) - PicoCpuCM68k.cycles=cyc_do; - CycloneRun(&PicoCpuCM68k); - SekCycleCnt+=cyc_do-PicoCpuCM68k.cycles; -#elif defined(EMU_M68K) - SekCycleCnt+=m68k_execute(cyc_do); -#elif defined(EMU_F68K) - SekCycleCnt+=fm68k_emulate(cyc_do+1, 0, 0); -#endif } -// ~1463.8, but due to cache misses and slow mem -// it's much lower than that -//#define SH2_LINE_CYCLES 735 -#define CYCLES_M68K2SH2(x) ((x) * 6 / 4) - -#define PICO_32X -#define CPUS_RUN_SIMPLE(m68k_cycles,s68k_cycles) \ - SekRunM68k(m68k_cycles); \ - if (!(Pico32x.emu_flags & (P32XF_MSH2POLL|P32XF_MSH2VPOLL))) \ - sh2_execute(&msh2, CYCLES_M68K2SH2(m68k_cycles)); \ - if (!(Pico32x.emu_flags & (P32XF_SSH2POLL|P32XF_SSH2VPOLL))) \ - sh2_execute(&ssh2, CYCLES_M68K2SH2(m68k_cycles)) - #define STEP_68K 24 -#define CPUS_RUN_LOCKSTEP(m68k_cycles,s68k_cycles) \ -{ \ - int i; \ - for (i = 0; i <= (m68k_cycles) - STEP_68K; i += STEP_68K) { \ - SekRunM68k(STEP_68K); \ - if (!(Pico32x.emu_flags & (P32XF_MSH2POLL|P32XF_MSH2VPOLL))) \ - sh2_execute(&msh2, CYCLES_M68K2SH2(STEP_68K)); \ - if (!(Pico32x.emu_flags & (P32XF_SSH2POLL|P32XF_SSH2VPOLL))) \ - sh2_execute(&ssh2, CYCLES_M68K2SH2(STEP_68K)); \ - } \ - /* last step */ \ - i = (m68k_cycles) - i; \ - SekRunM68k(i); \ - if (!(Pico32x.emu_flags & (P32XF_MSH2POLL|P32XF_MSH2VPOLL))) \ - sh2_execute(&msh2, CYCLES_M68K2SH2(i)); \ - if (!(Pico32x.emu_flags & (P32XF_SSH2POLL|P32XF_SSH2VPOLL))) \ - sh2_execute(&ssh2, CYCLES_M68K2SH2(i)); \ + +void sync_sh2s_lockstep(unsigned int m68k_target) +{ + unsigned int mcycles; + + mcycles = msh2.m68krcycles_done; + if (ssh2.m68krcycles_done < mcycles) + mcycles = ssh2.m68krcycles_done; + + while (mcycles < m68k_target) { + mcycles += STEP_68K; + sync_sh2s_normal(mcycles); + } } -//#define CPUS_RUN CPUS_RUN_SIMPLE -#define CPUS_RUN CPUS_RUN_LOCKSTEP +#define CPUS_RUN(m68k_cycles,s68k_cycles) do { \ + SekRunM68k(m68k_cycles); \ + if (SekIsStoppedM68k()) \ + p32x_sync_sh2s(SekCycleCntT + SekCycleCnt); \ +} while (0) +#define PICO_32X #include "../pico_cmn.c" void PicoFrame32x(void) @@ -177,3 +292,20 @@ void PicoFrame32x(void) elprintf(EL_32X, "poll: %02x", Pico32x.emu_flags); } +// calculate multipliers against 68k clock (7670442) +// normally * 3, but effectively slower due to high latencies everywhere +// however using something lower breaks MK2 animations +void Pico32xSetClocks(int msh2_hz, int ssh2_hz) +{ + float m68k_clk = (float)(OSC_NTSC / 7); + if (msh2_hz > 0) { + msh2.mult_m68k_to_sh2 = (int)((float)msh2_hz * (1 << CYCLE_MULT_SHIFT) / m68k_clk); + msh2.mult_sh2_to_m68k = (int)(m68k_clk * (1 << CYCLE_MULT_SHIFT) / (float)msh2_hz); + } + if (ssh2_hz > 0) { + ssh2.mult_m68k_to_sh2 = (int)((float)ssh2_hz * (1 << CYCLE_MULT_SHIFT) / m68k_clk); + ssh2.mult_sh2_to_m68k = (int)(m68k_clk * (1 << CYCLE_MULT_SHIFT) / (float)ssh2_hz); + } +} + +// vim:shiftwidth=2:ts=2:expandtab