unsigned int sprites_status;
+int sprites_zoom; // latched sprite zoom flag
+int sprites_sat, sprites_base; // latched sprite table data
+int xscroll; // horizontal scroll
+
/* sprite collision detection */
static int CollisionDetect(u8 *mb, u16 sx, unsigned int pack, int zoomed)
{
u8 *sat;
int xoff = line_offset;
int sprite_base, addr_mask;
- int zoomed = pv->reg[1] & 0x1; // zoomed sprites, e.g. Earthworm Jim
+ int zoomed = sprites_zoom & 0x1; // zoomed sprites, e.g. Earthworm Jim
unsigned int pack;
int i, s, h, m;
if (Pico.m.hardware & PMS_HW_LCD)
xoff -= 48; // GG LCD, adjust to center 160 px
- sat = (u8 *)PicoMem.vram + ((pv->reg[5] & 0x7e) << 7);
- if (pv->reg[1] & 2) {
+ sat = (u8 *)PicoMem.vram + ((sprites_sat & 0x7e) << 7);
+ if (sprites_zoom & 2) {
addr_mask = 0xfe; h = 16;
} else {
addr_mask = 0xff; h = 8;
}
if (zoomed) h *= 2;
- sprite_base = (pv->reg[6] & 4) << (13-2-1);
+ sprite_base = (sprites_base & 4) << (13-2-1);
m = pv->status & SR_C;
memset(sprites_map, 0, sizeof(sprites_map));
sprites_x[s] = xoff + sat[MEM_LE2(0x80 + i*2)];
sprites_addr[s] = sprite_base + ((sat[MEM_LE2(0x80 + i*2 + 1)] & addr_mask) << (5-1)) +
((scanline - y) >> zoomed << (2-1));
- if (Pico.video.reg[1] & 0x40) {
+ if (pv->reg[1] & 0x40) {
// collision detection. Do it here since off-screen lines aren't drawn
pack = CPU_LE2(*(u32 *)(PicoMem.vram + sprites_addr[s]));
// make sprite pixel map by merging the 4 bitplanes
{
struct PicoVideo *pv = &Pico.video;
unsigned int pack;
- int zoomed = pv->reg[1] & 0x1; // zoomed sprites, e.g. Earthworm Jim
+ int zoomed = sprites_zoom & 0x1; // zoomed sprites, e.g. Earthworm Jim
int s = sprites;
// now draw all sprites backwards
nametab2 = nametab + ((scanline>>3) << (6-1));
nametab = nametab + ((line>>3) << (6-1));
- dx = pv->reg[8]; // hscroll
+ dx = xscroll; // hscroll
if (scanline < 16 && (pv->reg[0] & 0x40))
dx = 0; // hscroll disabled for top 2 rows (e.g. Fantasy Zone II)
u8 *sat;
int xoff;
int sprite_base, addr_mask;
- int zoomed = pv->reg[1] & 0x1; // zoomed sprites
+ int zoomed = sprites_zoom & 0x1; // zoomed sprites
int i, s, h, m;
xoff = line_offset;
- sat = (u8 *)PicoMem.vramb + ((pv->reg[5] & 0x7f) << 7);
- if (pv->reg[1] & 2) {
+ sat = (u8 *)PicoMem.vramb + ((sprites_sat & 0x7f) << 7);
+ if (sprites_zoom & 2) {
addr_mask = 0xfc; h = 16;
} else {
addr_mask = 0xff; h = 8;
}
if (zoomed) h *= 2;
- sprite_base = (pv->reg[6] & 0x7) << 11;
+ sprite_base = (sprites_base & 0x7) << 11;
m = pv->status & SR_C;
memset(sprites_map, 0, sizeof(sprites_map));
sprites_x[s] = x;
sprites_addr[s] = sprite_base + ((sat[MEM_LE2(4*i + 2)] & addr_mask) << 3) +
((scanline - y) >> zoomed);
- if (Pico.video.reg[1] & 0x40) {
+ if (pv->reg[1] & 0x40) {
// collision detection. Do it here since off-screen lines aren't drawn
if (sprites_c[s] && x > 0) {
pack = PicoMem.vramb[MEM_LE2(sprites_addr[s])];
if (!m) m = CollisionDetect(sprites_map, x, pack, zoomed);
}
x += (zoomed ? 16:8);
- if (sprites_c[s] && (pv->reg[1] & 0x2) && x > 0 && x < 8+256) {
+ if (sprites_c[s] && (sprites_zoom & 0x2) && x > 0 && x < 8+256) {
pack = PicoMem.vramb[MEM_LE2(sprites_addr[s]+0x10)];
if (!m) m = CollisionDetect(sprites_map, x, pack, zoomed);
}
{
struct PicoVideo *pv = &Pico.video;
unsigned int pack;
- int zoomed = pv->reg[1] & 0x1; // zoomed sprites
+ int zoomed = sprites_zoom & 0x1; // zoomed sprites
int s = sprites;
// now draw all sprites backwards
if (zoomed) TileDoubleSprTMS(x, pack, c);
else TileNormSprTMS(x, pack, c);
}
- if (c && (pv->reg[1] & 0x2) && (x+=w) > 0 && x < 8+256) {
+ if (c && (sprites_zoom & 0x2) && (x+=w) > 0 && x < 8+256) {
pack = PicoMem.vramb[MEM_LE2(sprites_addr[s]+0x10)];
if (zoomed) TileDoubleSprTMS(x, pack, c);
else TileNormSprTMS(x, pack, c);
else DrawDisplayM0(line);
}
+ // latch current register values (may be overwritten by VDP reg writes later)
+ sprites_zoom = (Pico.video.reg[1] & 0x3) | (Pico.video.reg[0] & 0x8);
+ sprites_sat = Pico.video.reg[5];
+ sprites_base = Pico.video.reg[6];
+ xscroll = Pico.video.reg[8];
+
if (FinalizeLineSMS != NULL)
FinalizeLineSMS(line);
extern void YM2413_dataWrite(unsigned data);
extern unsigned sprites_status; // TODO put in some hdr file!
+extern int sprites_zoom, sprites_sat, sprites_base, xscroll;
static unsigned char vdp_data_read(void)
{
pv->pending = 0;
}
+// VDP horizontal timing, total 342 px:
+// 256 px active display,
+// 23 px right border+blanking,
+// 26 px hsync,
+// 37 px left blanking+border
+// VINT is at the beginning of hsync, and HINRT is one px later. Relative TO V/HINT:
+// -10 px sprite mode latching (r1)
+// -8 px sprite mode latching (r0)
+// -6 px sprite attribute table latching (r5)
+// -4 px sprite pattern table latching (r6)
+// -2 px xscroll latching (r8)
+// TODO: off by 2 CPU cycles according to VDPTEST?
+
static NOINLINE void vdp_reg_write(struct PicoVideo *pv, u8 a, u8 d)
{
int l;
pv->reg[a] = d;
switch (a) {
- case 0:
+ case 0: // mode control 1
l = pv->pending_ints & (d >> 3) & 2;
elprintf(EL_INTS, "hint %d", l);
z80_int_assert(l);
+ if (z80_cyclesDone() - Pico.t.z80c_line_start < 228 - (int)(8*1.5)+2)
+ sprites_zoom = (pv->reg[1] & 0x3) | (pv->reg[0] & 0x8);
break;
- case 1:
+ case 1: // mode control 2
l = pv->pending_ints & (d >> 5) & 1;
elprintf(EL_INTS, "vint %d", l);
z80_int_assert(l);
+ if (z80_cyclesDone() - Pico.t.z80c_line_start < 228 - (int)(10*1.5)+2)
+ sprites_zoom = (pv->reg[1] & 0x3) | (pv->reg[0] & 0x8);
+ break;
+ case 5: // sprite attribute table
+ if (z80_cyclesDone() - Pico.t.z80c_line_start < 228 - (int)(6*1.5)+2)
+ sprites_sat = d;
+ break;
+ case 6: // sprite pattern table
+ if (z80_cyclesDone() - Pico.t.z80c_line_start < 228 - (int)(4*1.5)+2)
+ sprites_base = d;
+ break;
+ case 8: // horizontal scroll
+ if (z80_cyclesDone() - Pico.t.z80c_line_start < 228 - (int)(2*1.5)+2)
+ xscroll = d;
break;
}
}
case 0x01:
if ((PicoIn.AHW & PAHW_GG) && a < 0x8) { // GG I/O area
switch (a) {
- case 0: d = (~PicoIn.pad[0] & 0x80) | (!(Pico.m.hardware & PMS_HW_JAP) << 6); break;
+ case 0: d = (~PicoIn.pad[0] & 0x80) |
+ (!(Pico.m.hardware & PMS_HW_JAP) << 6); 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;
else if (y > lines-32)
PicoParseSATSMS(y-1-lines);
- // render next line
- if (y < lines_vis && !skip)
- PicoLineSMS(y);
-
// take over status bits from previously rendered line TODO: cycle exact?
pv->status |= sprites_status;
sprites_status = 0;
z80_int_assert(1);
}
}
+ z80_exec(Pico.t.z80c_line_start + 12); // GG Madou 1, display off after line start
+
+ // render next line
+ if (y < lines_vis && !skip)
+ PicoLineSMS(y);
z80_exec(Pico.t.z80c_line_start + cycles_line);
}