/* * SMS emulation * (C) notaz, 2009-2010 * (C) kub, 2021 * * This work is licensed under the terms of MAME license. * See COPYING file in the top-level directory. */ /* * TODO: * - start in a state as if BIOS ran (partly done for VDP registers, RAM) * - region support (currently only very limited PAL and Mark-III support) * - mappers: EEPROM, some obscure Korean (Nemesis, Janggun) */ #include "pico_int.h" #include "memory.h" #include "sound/sn76496.h" #include "sound/emu2413/emu2413.h" extern void YM2413_regWrite(unsigned reg); extern void YM2413_dataWrite(unsigned data); static unsigned char vdp_data_read(void) { struct PicoVideo *pv = &Pico.video; unsigned char d; d = Pico.ms.vdp_buffer; Pico.ms.vdp_buffer = PicoMem.vramb[MEM_LE2(pv->addr)]; pv->addr = (pv->addr + 1) & 0x3fff; pv->pending = 0; return d; } static unsigned char vdp_ctl_read(void) { struct PicoVideo *pv = &Pico.video; unsigned char d; z80_int_assert(0); d = pv->status | (pv->pending_ints << 7); pv->pending = pv->pending_ints = 0; pv->status = 0; if (pv->reg[0] & 0x04) d |= 0x1f; // unused bits in mode 4 read as 1 elprintf(EL_SR, "VDP sr: %02x", d); return d; } static void vdp_data_write(unsigned char d) { struct PicoVideo *pv = &Pico.video; if (pv->type == 3) { if (Pico.m.hardware & 0x1) { // GG, same layout as MD unsigned a = pv->addr & 0x3f; if (a & 0x1) { // write complete color on high byte write u16 c = ((d&0x0f) << 8) | Pico.ms.vdp_buffer; if (PicoMem.cram[a >> 1] != c) Pico.m.dirtyPal = 1; PicoMem.cram[a >> 1] = c; } } else { // SMS, convert to MD layout (00BbGgRr to 0000BbBbGgGgRrRr) unsigned a = pv->addr & 0x1f; u16 c = ((d&0x30)<<6) + ((d&0x0c)<<4) + ((d&0x03)<<2); if (PicoMem.cram[a] != (c | (c>>2))) Pico.m.dirtyPal = 1; PicoMem.cram[a] = c | (c>>2); } } else { PicoMem.vramb[MEM_LE2(pv->addr)] = d; } pv->addr = (pv->addr + 1) & 0x3fff; Pico.ms.vdp_buffer = d; pv->pending = 0; } static NOINLINE void vdp_reg_write(struct PicoVideo *pv, u8 a, u8 d) { int l; pv->reg[a] = d; switch (a) { case 0: l = pv->pending_ints & (d >> 3) & 2; elprintf(EL_INTS, "hint %d", l); z80_int_assert(l); break; case 1: l = pv->pending_ints & (d >> 5) & 1; elprintf(EL_INTS, "vint %d", l); z80_int_assert(l); break; } } static void vdp_ctl_write(u8 d) { struct PicoVideo *pv = &Pico.video; if (pv->pending) { pv->type = d >> 6; if (pv->type == 2) { elprintf(EL_IO, " VDP r%02x=%02x", d & 0x0f, pv->addr & 0xff); if (pv->reg[d & 0x0f] != (u8)pv->addr) vdp_reg_write(pv, d & 0x0f, pv->addr); } pv->addr &= 0x00ff; pv->addr |= (d & 0x3f) << 8; if (pv->type == 0) { Pico.ms.vdp_buffer = PicoMem.vramb[MEM_LE2(pv->addr)]; pv->addr = (pv->addr + 1) & 0x3fff; } } else { pv->addr &= 0x3f00; pv->addr |= d; } pv->pending ^= 1; } static u8 vdp_hcounter(int cycles) { // 171 slots per scanline of 228 clocks, counted 0xf4-0x93, 0xe9-0xf3 // this matches h counter tables in SMSVDPTest: // hc = (cycles+2) * 171 /228 -1 + 0xf4; int hc = (((cycles+2) * ((171<<8)/228))>>8)-1 + 0xf4; // Q8 to avoid dividing if (hc > 0x193) hc += 0xe9-0x93-1; return hc; } static unsigned char z80_sms_in(unsigned short a) { unsigned char d = 0xff; a &= 0xff; elprintf(EL_IO, "z80 port %04x read", a); if(a >= 0xf0){ if (PicoIn.opt & POPT_EN_YM2413){ switch(a) { case 0xf0: // FM reg port break; case 0xf1: // FM data port break; case 0xf2: // bit 0 = 1 active FM Pac d = 0xf8 | Pico.ms.fm_ctl; break; } } } else{ switch (a & 0xc1) { case 0x00: case 0x01: if ((Pico.m.hardware & 0x1) && a < 0x8) { // GG I/O area switch (a) { case 0: d = 0xff & ~(PicoIn.pad[0] & 0x80); break; case 1: d = Pico.ms.io_gg[1] | (Pico.ms.io_gg[2] & 0x7f); break; case 5: d = Pico.ms.io_gg[5] & 0xf8; break; default: d = Pico.ms.io_gg[a]; break; } } break; case 0x40: /* V counter */ d = Pico.video.v_counter; elprintf(EL_HVCNT, "V counter read: %02x", d); break; case 0x41: /* H counter */ d = Pico.ms.vdp_hlatch; elprintf(EL_HVCNT, "H counter read: %02x", d); break; case 0x80: d = vdp_data_read(); break; case 0x81: d = vdp_ctl_read(); break; case 0xc0: /* I/O port A and B */ d = ~((PicoIn.pad[0] & 0x3f) | (PicoIn.pad[1] << 6)); break; case 0xc1: /* I/O port B and miscellaneous */ d = (Pico.ms.io_ctl & 0x80) | ((Pico.ms.io_ctl << 1) & 0x40) | 0x30; d |= ~(PicoIn.pad[1] >> 2) & 0x0f; if (Pico.ms.io_ctl & 0x08) d |= 0x80; // TH as input is unconnected if (Pico.ms.io_ctl & 0x02) d |= 0x40; break; } } elprintf(EL_IO, "ret = %02x", d); return d; } static void z80_sms_out(unsigned short a, unsigned char d) { elprintf(EL_IO, "z80 port %04x write %02x", a, d); a &= 0xff; if(a >= 0xf0){ if (PicoIn.opt & POPT_EN_YM2413){ switch(a) { case 0xf0: // FM reg port YM2413_regWrite(d); break; case 0xf1: // FM data port YM2413_dataWrite(d); break; case 0xf2: // bit 0 = 1 active FM Pac Pico.ms.fm_ctl = d & 0x1; break; } } } else{ switch (a & 0xc1) { case 0x00: if ((Pico.m.hardware & 0x1) && a < 0x8) // GG I/O area Pico.ms.io_gg[a] = d; break; case 0x01: if ((Pico.m.hardware & 0x1) && a < 0x8) { // GG I/O area Pico.ms.io_gg[a] = d; } else { // pad. latch hcounter if one of the TH lines is switched to 1 if ((Pico.ms.io_ctl ^ d) & d & 0xa0) Pico.ms.vdp_hlatch = vdp_hcounter(228 - z80_cyclesLeft); Pico.ms.io_ctl = d; } break; case 0x40: case 0x41: PsndDoPSG(z80_cyclesDone()); SN76496Write(d); break; case 0x80: vdp_data_write(d); break; case 0x81: vdp_ctl_write(d); break; } } } static int bank_mask; static void xwrite(unsigned int a, unsigned char d); static void write_sram(unsigned short a, unsigned char d) { // SRAM is mapped in 2 16KB banks, selected by bit 2 in control reg a &= 0x3fff; a += ((Pico.ms.carthw[0x0c] & 0x04) >> 2) * 0x4000; Pico.sv.changed |= (Pico.sv.data[a] != d); Pico.sv.data[a] = d; } // 16KB bank mapping for Sega mapper static void write_bank_sega(unsigned short a, unsigned char d) { // avoid mapper detection for RAM fill with 0 if (Pico.ms.mapper != PMS_MAP_SEGA && (Pico.ms.mapper || d == 0)) return; elprintf(EL_Z80BNK, "bank 16k %04x %02x @ %04x", a, d, z80_pc()); Pico.ms.mapper = PMS_MAP_SEGA; Pico.ms.carthw[a & 0x0f] = d; switch (a & 0x0f) { case 0x0d: d &= bank_mask; z80_map_set(z80_read_map, 0x0400, 0x3fff, Pico.rom+0x400 + (d << 14), 0); break; case 0x0e: d &= bank_mask; z80_map_set(z80_read_map, 0x4000, 0x7fff, Pico.rom + (d << 14), 0); break; case 0x0c: if (d & ~0x8c) elprintf(EL_STATUS|EL_ANOMALY, "%02x written to control reg!", d); /*FALLTHROUGH*/ case 0x0f: if (Pico.ms.carthw[0xc] & 0x08) { d = (Pico.ms.carthw[0xc] & 0x04) >> 2; z80_map_set(z80_read_map, 0x8000, 0xbfff, Pico.sv.data + d*0x4000, 0); z80_map_set(z80_write_map, 0x8000, 0xbfff, write_sram, 1); } else { d = Pico.ms.carthw[0xf] & bank_mask; z80_map_set(z80_read_map, 0x8000, 0xbfff, Pico.rom + (d << 14), 0); z80_map_set(z80_write_map, 0x8000, 0xbfff, xwrite, 1); } break; } } // 16KB bank mapping for Codemasters mapper, TODO: SRAM support static void write_bank_codem(unsigned short a, unsigned char d) { // don't detect @0x0 to avoid confusing with MSX if (Pico.ms.mapper != PMS_MAP_CODEM && (Pico.ms.mapper || (a|d) == 0)) return; elprintf(EL_Z80BNK, "bank 16k %04x %02x @ %04x", a, d, z80_pc()); Pico.ms.mapper = PMS_MAP_CODEM; Pico.ms.carthw[a>>14] = d; d &= bank_mask; z80_map_set(z80_read_map, a, a+0x3fff, Pico.rom + (d << 14), 0); } // 8KB bank mapping for MSX mapper static void write_bank_msx(unsigned short a, unsigned char d) { // don't detect @0x0 to avoid confusing with Codemasters if (Pico.ms.mapper != PMS_MAP_MSX && (Pico.ms.mapper || (a|d) == 0)) return; elprintf(EL_Z80BNK, "bank 8k %04x %02x @ %04x", a, d, z80_pc()); Pico.ms.mapper = PMS_MAP_MSX; Pico.ms.carthw[a] = d; a = (a^2)*0x2000 + 0x4000; d &= 2*bank_mask + 1; z80_map_set(z80_read_map, a, a+0x1fff, Pico.rom + (d << 13), 0); } // 32KB bank mapping for Korean n-in-1 packs static void write_bank_n32k(unsigned short a, unsigned char d) { // code must be in RAM since all visible ROM space is swapped if (Pico.ms.mapper != PMS_MAP_N32K && (Pico.ms.mapper || z80_pc() < 0xc000)) return; elprintf(EL_Z80BNK, "bank 32k %04x %02x @ %04x", a, d, z80_pc()); Pico.ms.mapper = PMS_MAP_N32K; Pico.ms.carthw[0xf] = d; d &= bank_mask >> 1; z80_map_set(z80_read_map, 0, 0x7fff, Pico.rom + (d << 15), 0); } // 16KB bank mapping for Korean n-in-1 packs static void write_bank_n16k(unsigned short a, unsigned char d) { // code must be in RAM since all visible ROM space is swapped if (Pico.ms.mapper != PMS_MAP_N16K && (Pico.ms.mapper || z80_pc() < 0xc000)) return; elprintf(EL_Z80BNK, "bank 16k %04x %02x @ %04x", a, d, z80_pc()); Pico.ms.mapper = PMS_MAP_N16K; Pico.ms.carthw[a>>14] = d; d &= bank_mask; a = a & 0xc000; // the top bank shifts with the bottom bank. if (a == 0x8000) d += Pico.ms.carthw[0] & 0x30; z80_map_set(z80_read_map, a, a+0x3fff, Pico.rom + (d << 14), 0); } // TODO mapping is currently auto-selecting, but that's not totally reliable. // Before adding more mappers this should be revised. static void xwrite(unsigned int a, unsigned char d) { elprintf(EL_IO, "z80 write [%04x] %02x", a, d); if (a >= 0xc000) PicoMem.zram[a & 0x1fff] = d; // NB the sequence of mappers is crucial for the auto detection // Korean n-in-1. 1 selectable 32KB bank at the bottom if (a == 0xffff) write_bank_n32k(a, d); // Sega. Maps 3 banks 16KB each, SRAM if (a >= 0xfff8) write_bank_sega(a, d); // MSX. 4 selectable 8KB banks at the top if (a <= 0x0003) write_bank_msx(a, d); // Codemasters. Similar to Sega, but different addresses if (a < 0xc000 && (a & 0x3fff) == 0) write_bank_codem(a, d); // Korean. 1 selectable 16KB bank at the top if (a == 0xa000) write_bank_codem(0x8000, d); // Korean n-in-1. 2 selectable 16KB banks, top bank has offset from bottom one if (a == 0x3ffe || a == 0x7fff || a == 0xbfff) write_bank_n16k(a, d); } void PicoResetMS(void) { z80_reset(); PsndReset(); // pal must be known here Pico.ms.fm_ctl = 0xff; Pico.m.dirtyPal = 1; if (PicoIn.hwSelect) { switch (PicoIn.hwSelect) { case PHWS_GG: Pico.m.hardware |= 0x1; break; default: Pico.m.hardware &= ~0x1; break; } } else { unsigned tmr; // check if the ROM header contains more system information to detect GG for (tmr = 0x2000; tmr < 0xbfff && tmr <= Pico.romsize; tmr *= 2) { if (!memcmp(Pico.rom + tmr-16, "TMR SEGA", 8)) { if (Pico.rom[tmr-1] >= 0x50 && Pico.rom[tmr-1] < 0x80) Pico.m.hardware |= 0x1; break; } } } // reset memory mapping PicoMemSetupMS(); // BIOS, VDP intialisation Pico.video.reg[0] = 0x36; Pico.video.reg[1] = 0xa0; Pico.video.reg[2] = 0xff; Pico.video.reg[3] = 0xff; Pico.video.reg[4] = 0xff; Pico.video.reg[5] = 0xff; Pico.video.reg[6] = 0xfb; Pico.video.reg[7] = 0x00; Pico.video.reg[8] = 0x00; Pico.video.reg[9] = 0x00; Pico.video.reg[10] = 0xff; // BIOS, clear zram (fake unitialized on japanese NTSC devices) memset(PicoMem.zram, PicoIn.regionOverride&1 ? 0xf0:0, sizeof(PicoMem.zram)); } void PicoPowerMS(void) { int s, tmp; memset(&PicoMem,0,sizeof(PicoMem)); memset(&Pico.video,0,sizeof(Pico.video)); memset(&Pico.m,0,sizeof(Pico.m)); Pico.m.pal = 0; // calculate a mask for bank writes. // ROM loader has aligned the size for us, so this is safe. s = 0; tmp = Pico.romsize; while ((tmp >>= 1) != 0) s++; if (Pico.romsize > (1 << s)) s++; tmp = 1 << s; bank_mask = (tmp - 1) >> 14; PicoMem.ioports[0] = 0xc3; // hack to jump @0 at end of RAM to wrap around Pico.ms.mapper = 0; PicoReset(); } void PicoMemSetupMS(void) { z80_map_set(z80_read_map, 0x0000, 0xbfff, Pico.rom, 0); z80_map_set(z80_read_map, 0xc000, 0xdfff, PicoMem.zram, 0); z80_map_set(z80_read_map, 0xe000, 0xffff, PicoMem.zram, 0); z80_map_set(z80_write_map, 0x0000, 0xbfff, xwrite, 1); z80_map_set(z80_write_map, 0xc000, 0xdfff, PicoMem.zram, 0); z80_map_set(z80_write_map, 0xe000, 0xffff, xwrite, 1); #ifdef _USE_DRZ80 drZ80.z80_in = z80_sms_in; drZ80.z80_out = z80_sms_out; #endif #ifdef _USE_CZ80 Cz80_Set_INPort(&CZ80, z80_sms_in); Cz80_Set_OUTPort(&CZ80, z80_sms_out); #endif // memory mapper setup, linear mapping of 1st 48KB u8 mapper = Pico.ms.mapper; if (Pico.ms.mapper == PMS_MAP_MSX) { xwrite(0x0000, 4); xwrite(0x0001, 5); xwrite(0x0002, 2); xwrite(0x0003, 3); } else if (Pico.ms.mapper == PMS_MAP_N32K) { xwrite(0xffff, 0); } else if (Pico.ms.mapper == PMS_MAP_CODEM) { xwrite(0x0000, 0); xwrite(0x4000, 1); xwrite(0x8000, 2); } else { xwrite(0xfffc, 0); xwrite(0xfffd, 0); xwrite(0xfffe, 1); xwrite(0xffff, 2); } Pico.ms.mapper = mapper; } void PicoStateLoadedMS(void) { u8 mapper = Pico.ms.mapper; if (Pico.ms.mapper == PMS_MAP_MSX) { xwrite(0x0000, Pico.ms.carthw[0]); xwrite(0x0001, Pico.ms.carthw[1]); xwrite(0x0002, Pico.ms.carthw[2]); xwrite(0x0003, Pico.ms.carthw[3]); } else if (Pico.ms.mapper == PMS_MAP_N32K) { xwrite(0xffff, Pico.ms.carthw[0x0f]); } else if (Pico.ms.mapper == PMS_MAP_CODEM) { xwrite(0x0000, Pico.ms.carthw[0]); xwrite(0x4000, Pico.ms.carthw[1]); xwrite(0x8000, Pico.ms.carthw[2]); } else { xwrite(0xfffc, Pico.ms.carthw[0x0c]); xwrite(0xfffd, Pico.ms.carthw[0x0d]); xwrite(0xfffe, Pico.ms.carthw[0x0e]); xwrite(0xffff, Pico.ms.carthw[0x0f]); } Pico.ms.mapper = mapper; } void PicoFrameMS(void) { struct PicoVideo *pv = &Pico.video; int is_pal = Pico.m.pal; int lines = is_pal ? 313 : 262; int cycles_line = 228; int cycles_done = 0, cycles_aim = 0; int skip = PicoIn.skipFrame; int lines_vis = 192; int hint; // Hint counter int nmi; int y; PsndStartFrame(); nmi = (PicoIn.pad[0] >> 7) & 1; if (!Pico.ms.nmi_state && nmi) z80_nmi(); Pico.ms.nmi_state = nmi; if ((pv->reg[0] & 6) == 6 && (pv->reg[1] & 0x18)) lines_vis = (pv->reg[1] & 0x08) ? 240 : 224; PicoFrameStartSMS(); hint = pv->reg[0x0a]; for (y = 0; y < lines; y++) { pv->v_counter = Pico.m.scanline = y; if (y > 218) pv->v_counter = y - 6; if (y < lines_vis && !skip) PicoLineSMS(y); // Interrupt handling. Simulate interrupt flagged and immediately reset in // same insn by flagging the irq, execute for 1 insn, then checking if the // irq is still pending. (GG Chicago, SMS Back to the Future III) if (y <= lines_vis) { if (--hint < 0) { hint = pv->reg[0x0a]; pv->pending_ints |= 2; cycles_done += z80_run(1); if ((pv->reg[0] & 0x10) && (pv->pending_ints & 2)) { elprintf(EL_INTS, "hint"); z80_int_assert(1); } pv->pending_ints &= ~2; } } else if (y == lines_vis + 1) { pv->pending_ints &= ~2; pv->pending_ints |= 1; cycles_done += z80_run(1); if ((pv->reg[1] & 0x20) && (pv->pending_ints & 1)) { elprintf(EL_INTS, "vint"); z80_int_assert(1); } } cycles_aim += cycles_line; Pico.t.z80c_aim = cycles_aim; cycles_done += z80_run(cycles_aim - cycles_done); } if (PicoIn.sndOut) PsndGetSamplesMS(lines); } void PicoFrameDrawOnlyMS(void) { int lines_vis = 192; int y; PicoFrameStartSMS(); for (y = 0; y < lines_vis; y++) PicoLineSMS(y); } // vim:ts=2:sw=2:expandtab