GBHawk: Pass many more tests

This commit is contained in:
alyosha-tas 2018-11-10 12:47:46 -06:00
parent 1961d8b8d0
commit f0271b1f72
9 changed files with 171 additions and 198 deletions

View File

@ -381,9 +381,9 @@ namespace BizHawk.Emulation.Common.Components.LR35902
else
{
cur_instr = new ushort[]
{IDLE,
HALT_CHK,
IDLE,
{HALT_CHK,
IDLE,
IDLE,
HALT, 0 };
}

View File

@ -10,7 +10,7 @@ namespace BizHawk.Emulation.Common.Components.LR35902
private void NOP_()
{
cur_instr = new ushort[]
{IDLE,
{IDLE,
IDLE,
HALT_CHK,
OP };

View File

@ -130,7 +130,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 0xFF43: // SCX
scroll_x = value;
break;
case 0xFF44: // LY
LY = 0; /*reset*/
@ -418,12 +417,15 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
}
if ((cycle == 4) && (LY == 144)) {
HBL_INT = false;
if ((cycle == 2) && (LY == 144))
{
// there is an edge case where a VBL INT is triggered if STAT bit 5 is set
if (STAT.Bit(5)) { VBL_INT = true; }
}
if ((cycle == 4) && (LY == 144))
{
HBL_INT = false;
// set STAT mode to 1 (VBlank) and interrupt flag if it is enabled
STAT &= 0xFC;
@ -433,12 +435,12 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
Core.REG_FF0F |= 0x01;
}
if ((cycle == 84) && (LY == 144))
if ((cycle == 4) && (LY == 144))
{
if (STAT.Bit(5)) { VBL_INT = false; }
}
if ((LY == 153) && (cycle == 6))
if ((cycle == 6) && (LY == 153))
{
LY = 0;
LY_inc = 0;
@ -528,7 +530,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
else if ((cycle >= 80) && (LY < 144))
{
if (cycle >= 84)
if (cycle >= 83)
{
if (cycle == 84)
{
@ -540,7 +542,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
// render the screen and handle hblank
render(cycle - 84);
render(cycle - 83);
}
else if (cycle == 80)
{
@ -552,13 +554,15 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
}
if ((LY_inc == 0))
if (LY_inc == 0)
{
if (cycle == 12)
if (cycle == 10)
{
LYC_INT = false;
STAT &= 0xFB;
}
else if (cycle == 12)
{
// Special case of LY = LYC
if ((LY == LYC) && !STAT.Bit(2))
{
@ -650,12 +654,11 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
read_case = 0;
internal_cycle = 0;
pre_render = true;
pre_render_2 = true;
tile_inc = 0;
pixel_counter = -8;
sl_use_index = 0;
fetch_sprite = false;
fetch_sprite_01 = false;
fetch_sprite_4 = false;
going_to_fetch = false;
first_fetch = true;
no_sprites = false;
@ -855,7 +858,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
if (!fetch_sprite)
{
if (!pre_render)
if (!pre_render_2)
{
// before we go on to read case 3, we need to know if we stall there or not
// Gekkio's tests show that if sprites are at position 0 or 1 (mod 8)
@ -871,15 +874,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
{
going_to_fetch = true;
fetch_sprite = true;
if ((SL_sprites[i * 4 + 1] % 8) < 2)
{
fetch_sprite_01 = true;
}
if ((SL_sprites[i * 4 + 1] % 8) > 3)
{
fetch_sprite_4 = true;
}
}
}
}
@ -888,20 +882,18 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
switch (read_case)
{
case 0: // read a background tile
if ((internal_cycle % 2) == 0)
if ((internal_cycle % 2) == 1)
{
// calculate the row number of the tiles to be fetched
y_tile = ((int)Math.Floor((float)(scroll_y + LY) / 8)) % 32;
temp_fetch = y_tile * 32 + (x_tile + tile_inc) % 32;
tile_byte = Core.VRAM[0x1800 + (LCDC.Bit(3) ? 1 : 0) * 0x400 + temp_fetch];
tile_byte = Core.VRAM[0x1800 + (LCDC.Bit(3) ? 1 : 0) * 0x400 + temp_fetch];
tile_data[2] = Core.VRAM[0x3800 + (LCDC.Bit(3) ? 1 : 0) * 0x400 + temp_fetch];
VRAM_sel = tile_data[2].Bit(3) ? 1 : 0;
BG_V_flip = tile_data[2].Bit(6) & Core.GBC_compat;
}
else
{
read_case = 1;
if (!pre_render)
{
@ -911,7 +903,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 1: // read from tile graphics (0)
if ((internal_cycle % 2) == 0)
if ((internal_cycle % 2) == 1)
{
y_scroll_offset = (scroll_y + LY) % 8;
@ -933,15 +925,17 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
tile_data[0] = Core.VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2];
}
}
else
{
read_case = 2;
}
break;
case 2: // read from tile graphics (1)
if ((internal_cycle % 2) == 0)
{
pre_render_2 = false;
}
else
{
y_scroll_offset = (scroll_y + LY) % 8;
@ -968,9 +962,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
tile_data[1] = Core.VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
}
}
else
{
if (pre_render)
{
// here we set up rendering
@ -988,11 +980,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 3: // read from sprite data
if ((internal_cycle % 2) == 0)
{
// nothing to do if not fetching
}
else
if ((internal_cycle % 2) == 1)
{
read_case = 0;
latch_new_data = true;
@ -1000,16 +988,14 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 4: // read from window data
if ((window_counter % 2) == 0)
if ((window_counter % 2) == 1)
{
temp_fetch = window_y_tile * 32 + (window_x_tile + window_tile_inc) % 32;
tile_byte = Core.VRAM[0x1800 + (LCDC.Bit(6) ? 1 : 0) * 0x400 + temp_fetch];
tile_data[2] = Core.VRAM[0x3800 + (LCDC.Bit(6) ? 1 : 0) * 0x400 + temp_fetch];
VRAM_sel = tile_data[2].Bit(3) ? 1 : 0;
BG_V_flip = tile_data[2].Bit(6) & Core.GBC_compat;
}
else
{
BG_V_flip = tile_data[2].Bit(6) & Core.GBC_compat;
window_tile_inc++;
read_case = 5;
}
@ -1017,7 +1003,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 5: // read from tile graphics (for the window)
if ((window_counter % 2) == 0)
if ((window_counter % 2) == 1)
{
y_scroll_offset = (window_y_tile_inc) % 8;
@ -1027,8 +1013,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
if (LCDC.Bit(4))
{
tile_data[0] = Core.VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2];
{
tile_data[0] = Core.VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2];
}
else
{
@ -1036,19 +1022,17 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
if (tile_byte.Bit(7))
{
tile_byte -= 256;
}
tile_data[0] = Core.VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2];
}
tile_data[0] = Core.VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2];
}
}
else
{
read_case = 6;
}
window_counter++;
break;
case 6: // read from tile graphics (for the window)
if ((window_counter % 2) == 0)
if ((window_counter % 2) == 1)
{
y_scroll_offset = (window_y_tile_inc) % 8;
@ -1063,8 +1047,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
if (tile_byte < 0)
{
tile_byte += 256;
}
tile_data[1] = Core.VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2 + 1];
}
tile_data[1] = Core.VRAM[(VRAM_sel * 0x2000) + tile_byte * 16 + y_scroll_offset * 2 + 1];
}
else
{
@ -1072,13 +1056,10 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
if (tile_byte.Bit(7) && tile_byte > 0)
{
tile_byte -= 256;
}
tile_data[1] = Core.VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
}
tile_data[1] = Core.VRAM[(VRAM_sel * 0x2000) + 0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
}
}
else
{
if (window_pre_render)
{
// here we set up rendering
@ -1101,11 +1082,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 7: // read from sprite data
if ((window_counter % 2) == 0)
{
// nothing to do if not fetching
}
else
if ((window_counter % 2) == 1)
{
read_case = 4;
latch_new_data = true;
@ -1114,18 +1091,17 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 8: // done reading, we are now in phase 0
pre_render = true;
// the other interrupts appear to be delayed by 1 CPU cycle, so do the same here
if (hbl_countdown > 0)
{
hbl_countdown--;
STAT &= 0xFC;
STAT |= 0x00;
if (hbl_countdown == 0)
{
STAT &= 0xFC;
STAT |= 0x00;
if (STAT.Bit(3)) { HBL_INT = true; }
OAM_access_read = true;
@ -1163,22 +1139,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
if (going_to_fetch)
{
going_to_fetch = false;
sprite_fetch_counter = first_fetch ? 2 : 0;
first_fetch = false;
if (fetch_sprite_01)
{
sprite_fetch_counter += 2;
fetch_sprite_01 = false;
}
if (fetch_sprite_4)
{
sprite_fetch_counter -= 2;
fetch_sprite_4 = false;
}
int last_eval = 0;
last_eval = 0;
// at this time it is unknown what each cycle does, but we only need to accurately keep track of cycles
for (int i = 0; i < SL_sprites_index; i++)
@ -1193,16 +1155,23 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
}
// if we didn't evaluate all the sprites immediately, 2 more cycles are added to restart it
if (evaled_sprites != (Math.Pow(2,SL_sprites_index) - 1))
// there is no penalty if the next sprites to be fetched are within the currentfetch block (8 pixels)
if (first_fetch || (last_eval >= consecutive_sprite))
{
if ((last_eval % 8) == 0) { sprite_fetch_counter += 3; }
else if ((last_eval % 8) == 1) { sprite_fetch_counter += 2; }
if ((last_eval % 8) == 0) { sprite_fetch_counter += 5; }
else if ((last_eval % 8) == 1) { sprite_fetch_counter += 4; }
else if ((last_eval % 8) == 2) { sprite_fetch_counter += 3; }
else if ((last_eval % 8) == 3) { sprite_fetch_counter += 2; }
else if ((last_eval % 8) == 4) { sprite_fetch_counter += 3; }
else { sprite_fetch_counter += 2; }
else if ((last_eval % 8) == 4) { sprite_fetch_counter += 1; }
else if ((last_eval % 8) == 5) { sprite_fetch_counter += 0; }
else if ((last_eval % 8) == 6) { sprite_fetch_counter += 0; }
else if ((last_eval % 8) == 7) { sprite_fetch_counter += 0; }
}
total_counter += sprite_fetch_counter;
consecutive_sprite = last_eval + (8 - (last_eval % 8));
first_fetch = false;
}
else
{

View File

@ -162,9 +162,10 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
//if (ticker > 10000000) { vblank_rise = true; }//throw new Exception("ERROR: Unable to Resolve Frame"); }
in_vblank_old = in_vblank;
REG_FF0F_OLD = REG_FF0F;
}
vblank_rise = false;
vblank_rise = false;
}
// Switch Speed (GBC only)

View File

@ -72,6 +72,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
ser.Sync("REG_FFFF", ref REG_FFFF);
ser.Sync("REG_FF0F", ref REG_FF0F);
ser.Sync("REG_FF0F_OLD", ref REG_FF0F_OLD);
// memory domains
ser.Sync("RAM", ref RAM, false);

View File

@ -28,6 +28,9 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
public byte REG_FFFF;
// The unused bits in this register (interrupt flags) are always set
public byte REG_FF0F = 0xE0;
// Updating reg FF0F seemsto be delayed by one cycle
// tests
public byte REG_FF0F_OLD = 0xE0;
// memory domains
public byte[] RAM = new byte[0x8000]; // only 0x2000 available to GB
@ -109,12 +112,12 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
{
if (game.System == "GB")
{
Bios = comm.CoreFileProvider.GetFirmware("GB", "World", true, "BIOS Not Found, Cannot Load");
Bios = comm.CoreFileProvider.GetFirmware("GBS", "World", true, "BIOS Not Found, Cannot Load");
ppu = new GB_PPU();
}
else
{
Bios = comm.CoreFileProvider.GetFirmware("GBC", "World", true, "BIOS Not Found, Cannot Load");
Bios = comm.CoreFileProvider.GetFirmware("GBCS", "World", true, "BIOS Not Found, Cannot Load");
ppu = new GBC_PPU();
is_GBC = true;
}
@ -122,12 +125,12 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
else if (_syncSettings.ConsoleMode == GBSyncSettings.ConsoleModeType.GB)
{
Bios = comm.CoreFileProvider.GetFirmware("GB", "World", true, "BIOS Not Found, Cannot Load");
Bios = comm.CoreFileProvider.GetFirmware("GBS", "World", true, "BIOS Not Found, Cannot Load");
ppu = new GB_PPU();
}
else
{
Bios = comm.CoreFileProvider.GetFirmware("GBC", "World", true, "BIOS Not Found, Cannot Load");
Bios = comm.CoreFileProvider.GetFirmware("GBCS", "World", true, "BIOS Not Found, Cannot Load");
ppu = new GBC_PPU();
is_GBC = true;
}

View File

@ -53,16 +53,27 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 0xFF41: // STAT
// writing to STAT during mode 0 or 1 causes a STAT IRQ
// this appears to be a glitchy LYC compare
if (LCDC.Bit(7))
{
if (((STAT & 3) == 0) || ((STAT & 3) == 1))
{
LYC_INT = true;
//if (Core.REG_FFFF.Bit(1)) { Core.cpu.FlagI = true; }
//Core.REG_FF0F |= 0x02;
}
else
{
if (value.Bit(6))
{
if (LY == LYC) { LYC_INT = true; }
else { LYC_INT = false; }
}
}
}
STAT = (byte)((value & 0xF8) | (STAT & 7) | 0x80);
if (!STAT.Bit(6)) { LYC_INT = false; }
//if (!STAT.Bit(6)) { LYC_INT = false; }
if (!STAT.Bit(4)) { VBL_INT = false; }
break;
case 0xFF42: // SCY
@ -78,8 +89,17 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
LYC = value;
if (LCDC.Bit(7))
{
if (LY != LYC) { STAT &= 0xFB; }
else { STAT |= 0x4; }
if (LY != LYC) { STAT &= 0xFB; LYC_INT = false; }
else { STAT |= 0x4; LYC_INT = true; }
// special case: the transition from 153 -> 0 acts strange
// the comparison to 153 expects to be true for longer then the value of LY expects to be 153
// this appears to be fixed in CGB
if ((LY_inc == 0) && cycle == 8)
{
if (153 != LYC) { STAT &= 0xFB; LYC_INT = false; }
else { STAT |= 0x4; LYC_INT = true; }
}
}
break;
case 0xFF46: // DMA
@ -191,12 +211,15 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
}
if ((cycle == 4) && (LY == 144)) {
HBL_INT = false;
if ((cycle == 2) && (LY == 144))
{
// there is an edge case where a VBL INT is triggered if STAT bit 5 is set
if (STAT.Bit(5)) { VBL_INT = true; }
}
if ((cycle == 4) && (LY == 144))
{
HBL_INT = false;
// set STAT mode to 1 (VBlank) and interrupt flag if it is enabled
STAT &= 0xFC;
@ -206,12 +229,12 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
Core.REG_FF0F |= 0x01;
}
if ((cycle == 84) && (LY == 144))
if ((cycle == 4) && (LY == 144))
{
if (STAT.Bit(5)) { VBL_INT = false; }
}
if ((LY == 153) && (cycle == 6))
if ((cycle == 6) && (LY == 153))
{
LY = 0;
LY_inc = 0;
@ -301,7 +324,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
else if ((cycle >= 80) && (LY < 144))
{
if (cycle >= 84)
if (cycle >= 83)
{
if (cycle == 84)
{
@ -313,7 +336,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
// render the screen and handle hblank
render(cycle - 84);
render(cycle - 83);
}
else if (cycle == 80)
{
@ -325,13 +348,15 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
}
if ((LY_inc == 0))
if (LY_inc == 0)
{
if (cycle == 12)
if (cycle == 10)
{
LYC_INT = false;
STAT &= 0xFB;
}
else if (cycle == 12)
{
// Special case of LY = LYC
if ((LY == LYC) && !STAT.Bit(2))
{
@ -423,12 +448,11 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
read_case = 0;
internal_cycle = 0;
pre_render = true;
pre_render_2 = true;
tile_inc = 0;
pixel_counter = -8;
sl_use_index = 0;
fetch_sprite = false;
fetch_sprite_01 = false;
fetch_sprite_4 = false;
going_to_fetch = false;
first_fetch = true;
no_sprites = false;
@ -436,6 +460,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
window_pre_render = false;
window_latch = LCDC.Bit(5);
total_counter = 0;
// TODO: If Window is turned on midscanline what happens? When is this check done exactly?
if ((window_started && window_latch) || (window_is_reset && !window_latch && (LY >= window_y)))
{
@ -577,7 +603,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
if (!fetch_sprite)
{
if (!pre_render)
if (!pre_render_2)
{
// before we go on to read case 3, we need to know if we stall there or not
// Gekkio's tests show that if sprites are at position 0 or 1 (mod 8)
@ -593,15 +619,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
{
going_to_fetch = true;
fetch_sprite = true;
if ((SL_sprites[i * 4 + 1] % 8) < 2)
{
fetch_sprite_01 = true;
}
if ((SL_sprites[i * 4 + 1] % 8) > 3)
{
fetch_sprite_4 = true;
}
}
}
}
@ -610,16 +627,14 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
switch (read_case)
{
case 0: // read a background tile
if ((internal_cycle % 2) == 0)
if ((internal_cycle % 2) == 1)
{
// calculate the row number of the tiles to be fetched
y_tile = ((int)Math.Floor((float)(scroll_y + LY) / 8)) % 32;
temp_fetch = y_tile * 32 + (x_tile + tile_inc) % 32;
tile_byte = Core.VRAM[0x1800 + (LCDC.Bit(3) ? 1 : 0) * 0x400 + temp_fetch];
}
else
{
read_case = 1;
if (!pre_render)
{
@ -629,7 +644,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 1: // read from tile graphics (0)
if ((internal_cycle % 2) == 0)
if ((internal_cycle % 2) == 1)
{
y_scroll_offset = (scroll_y + LY) % 8;
@ -647,9 +662,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
tile_data[0] = Core.VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2];
}
}
else
{
read_case = 2;
}
break;
@ -657,6 +669,10 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
case 2: // read from tile graphics (1)
if ((internal_cycle % 2) == 0)
{
pre_render_2 = false;
}
else
{
y_scroll_offset = (scroll_y + LY) % 8;
if (LCDC.Bit(4))
@ -680,9 +696,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
tile_data[1] = Core.VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
}
}
else
{
if (pre_render)
{
// here we set up rendering
@ -700,25 +713,19 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 3: // read from sprite data
if ((internal_cycle % 2) == 0)
{
// nothing to do if not fetching
}
else
{
if ((internal_cycle % 2) == 1)
{
read_case = 0;
latch_new_data = true;
}
break;
case 4: // read from window data
if ((window_counter % 2) == 0)
if ((window_counter % 2) == 1)
{
temp_fetch = window_y_tile * 32 + (window_x_tile + window_tile_inc) % 32;
tile_byte = Core.VRAM[0x1800 + (LCDC.Bit(6) ? 1 : 0) * 0x400 + temp_fetch]; ;
}
else
{
tile_byte = Core.VRAM[0x1800 + (LCDC.Bit(6) ? 1 : 0) * 0x400 + temp_fetch];
window_tile_inc++;
read_case = 5;
}
@ -726,15 +733,15 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 5: // read from tile graphics (for the window)
if ((window_counter % 2) == 0)
if ((window_counter % 2) == 1)
{
y_scroll_offset = (window_y_tile_inc) % 8;
if (LCDC.Bit(4))
{
tile_data[0] = Core.VRAM[tile_byte * 16 + y_scroll_offset * 2];
}
else
{
@ -743,19 +750,17 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
{
tile_byte -= 256;
}
tile_data[0] = Core.VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2];
}
}
else
{
read_case = 6;
}
window_counter++;
break;
case 6: // read from tile graphics (for the window)
if ((window_counter % 2) == 0)
if ((window_counter % 2) == 1)
{
y_scroll_offset = (window_y_tile_inc) % 8;
if (LCDC.Bit(4))
@ -779,9 +784,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
tile_data[1] = Core.VRAM[0x1000 + tile_byte * 16 + y_scroll_offset * 2 + 1];
}
}
else
{
if (window_pre_render)
{
// here we set up rendering
@ -804,11 +806,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
break;
case 7: // read from sprite data
if ((window_counter % 2) == 0)
{
// nothing to do if not fetching
}
else
if ((window_counter % 2) == 1)
{
read_case = 4;
latch_new_data = true;
@ -823,11 +821,11 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
if (hbl_countdown > 0)
{
hbl_countdown--;
STAT &= 0xFC;
STAT |= 0x00;
if (hbl_countdown == 0)
{
STAT &= 0xFC;
STAT |= 0x00;
if (STAT.Bit(3)) { HBL_INT = true; }
OAM_access_read = true;
@ -854,6 +852,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
tile_data_latch[0] = tile_data[0];
tile_data_latch[1] = tile_data[1];
}
if (consecutive_sprite > 0) { consecutive_sprite -= 1; }
}
// every in range sprite takes 6 cycles to process
@ -864,22 +864,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
if (going_to_fetch)
{
going_to_fetch = false;
sprite_fetch_counter = first_fetch ? 2 : 0;
first_fetch = false;
if (fetch_sprite_01)
{
sprite_fetch_counter += 2;
fetch_sprite_01 = false;
}
if (fetch_sprite_4)
{
sprite_fetch_counter -= 2;
fetch_sprite_4 = false;
}
int last_eval = 0;
last_eval = 0;
// at this time it is unknown what each cycle does, but we only need to accurately keep track of cycles
for (int i = 0; i < SL_sprites_index; i++)
@ -894,16 +880,24 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
}
}
// if we didn't evaluate all the sprites immediately, 2 more cycles are added to restart it
if (evaled_sprites != (Math.Pow(2,SL_sprites_index) - 1))
// x scroll offsets the penalty table
// there is no penalty if the next sprites to be fetched are within the currentfetch block (8 pixels)
if (first_fetch || (last_eval >= consecutive_sprite))
{
if ((last_eval % 8) == 0) { sprite_fetch_counter += 3; }
else if ((last_eval % 8) == 1) { sprite_fetch_counter += 2; }
else if ((last_eval % 8) == 2) { sprite_fetch_counter += 3; }
else if ((last_eval % 8) == 3) { sprite_fetch_counter += 2; }
else if ((last_eval % 8) == 4) { sprite_fetch_counter += 3; }
else { sprite_fetch_counter += 2; }
if (((last_eval + render_offset) % 8) == 0) { sprite_fetch_counter += 5; }
else if (((last_eval + render_offset) % 8) == 1) { sprite_fetch_counter += 4; }
else if (((last_eval + render_offset) % 8) == 2) { sprite_fetch_counter += 3; }
else if (((last_eval + render_offset) % 8) == 3) { sprite_fetch_counter += 2; }
else if (((last_eval + render_offset) % 8) == 4) { sprite_fetch_counter += 1; }
else if (((last_eval + render_offset) % 8) == 5) { sprite_fetch_counter += 0; }
else if (((last_eval + render_offset) % 8) == 6) { sprite_fetch_counter += 0; }
else if (((last_eval + render_offset) % 8) == 7) { sprite_fetch_counter += 0; }
}
total_counter += sprite_fetch_counter;
consecutive_sprite = last_eval + (8 - (last_eval % 8));
first_fetch = false;
}
else
{

View File

@ -39,7 +39,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
// Interrupt flags
case 0xFF0F:
ret = REG_FF0F;
ret = REG_FF0F_OLD;
break;
// audio regs
@ -316,6 +316,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
Console.Write("GBC Compatibility? ");
Console.WriteLine(value);
GBC_compat = false;
// cpu operation is a function of hardware only
//cpu.is_GBC = GBC_compat;
}
break;

View File

@ -66,8 +66,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
public int tile_byte;
public int sprite_fetch_cycles;
public bool fetch_sprite;
public bool fetch_sprite_01;
public bool fetch_sprite_4;
public bool going_to_fetch;
public bool first_fetch;
public int sprite_fetch_counter;
@ -77,6 +75,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
public int temp_fetch;
public int tile_inc;
public bool pre_render;
public bool pre_render_2;
public byte[] tile_data = new byte[3];
public byte[] tile_data_latch = new byte[3];
public int latch_counter;
@ -94,7 +93,10 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
public int sprite_ordered_index;
public bool blank_frame;
public bool window_latch;
public int consecutive_sprite;
public int last_eval;
public int total_counter;
// windowing state
public int window_counter;
public bool window_pre_render;
@ -218,8 +220,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
ser.Sync("tile_byte", ref tile_byte);
ser.Sync("sprite_fetch_cycles", ref sprite_fetch_cycles);
ser.Sync("fetch_sprite", ref fetch_sprite);
ser.Sync("fetch_sprite_01", ref fetch_sprite_01);
ser.Sync("fetch_sprite_4", ref fetch_sprite_4);
ser.Sync("going_to_fetch", ref going_to_fetch);
ser.Sync("first_fetch", ref first_fetch);
ser.Sync("sprite_fetch_counter", ref sprite_fetch_counter);
@ -229,6 +229,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
ser.Sync("temp_fetch", ref temp_fetch);
ser.Sync("tile_inc", ref tile_inc);
ser.Sync("pre_render", ref pre_render);
ser.Sync("pre_render_2", ref pre_render_2);
ser.Sync("tile_data", ref tile_data, false);
ser.Sync("tile_data_latch", ref tile_data_latch, false);
ser.Sync("latch_counter", ref latch_counter);
@ -246,6 +247,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk
ser.Sync("sprite_ordered_index", ref sprite_ordered_index);
ser.Sync("blank_frame", ref blank_frame);
ser.Sync("window_latch", ref window_latch);
ser.Sync("consecutive_sprite", ref consecutive_sprite);
ser.Sync("last_eval", ref last_eval);
ser.Sync("window_counter", ref window_counter);
ser.Sync("window_pre_render", ref window_pre_render);