diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBHawk/GBHawk.IEmulator.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBHawk/GBHawk.IEmulator.cs index 292917e523..d4418fb71f 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBHawk/GBHawk.IEmulator.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBHawk/GBHawk.IEmulator.cs @@ -330,6 +330,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk public void SetIntRegs(byte r) { + if (((REG_FF0F & 4) == 4) && ((r & 4) == 0)) { timer.IRQ_block = true; } REG_FF0F = r; } diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBHawk/Timer.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBHawk/Timer.cs index 565832c829..ac07687500 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBHawk/Timer.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/GBHawk/Timer.cs @@ -5,6 +5,21 @@ using BizHawk.Common.NumberExtensions; namespace BizHawk.Emulation.Cores.Nintendo.GBHawk { // Timer Emulation + // NOTES: + // + // For GB, it looks like divider reg should start at 0 on reset. however in this case, in order to pass Gambatte timer tests, + // reads from timer when timer is about to rollover need to return the next value + // for GBC, a starting value of 0xFFFF passes all tests. GBA is not explicotlu tested but for now is set to 0xFFFF as well. + // + // Some additional glitches happen on GBC, but they are non-deterministic and not emulated here + // + // TODO: On GBA models, there is a race condition when enabling with a change in bit check + // that would result in a state change that is not consistent in all models, see tac_set_disabled.gbc + // + // TODO: On GBA only, there is a glitch where if the current timer control is 7 and the written value is 7 and + // there is a coincident timer increment, there will be an additional increment along with this write. + // not sure it effects all models or of exact details, see test tac_set_timer_disabled.gbc + public class Timer { public GBHawk Core { get; set; } @@ -15,6 +30,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk public byte timer_old; public byte timer_control; public byte pending_reload; + public bool IRQ_block; // if the timer IRQ happens on the same cycle as a previous one was cleared, the IRQ is blocked public bool old_state; public bool state; public bool reload_block; @@ -66,14 +82,54 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk // TAC (Timer Control) case 0xFF07: + + bool was_off = (timer_control & 4) == 0; + //Console.WriteLine(timer_control + " " + value + " " + timer + " " + divider_reg); timer_control = (byte)((timer_control & 0xf8) | (value & 0x7)); // only bottom 3 bits function + /* + if (was_off && ((timer_control & 4) > 0) && Core.is_GBC) + { + bool temp_check = false; - // NOTE: On GBA models, there is a race condition when enabling with a change in bit check - // that would result in a state change that is not consistent in all models, see tac_set_disabled.gbc + switch (timer_control & 3) + { + case 0: + temp_check = (divider_reg & 0x1FF) == 0x1FF; + break; + case 1: + temp_check = (divider_reg & 0x7) == 0x7; + break; + case 2: + temp_check = (divider_reg & 0x1F) == 0x1F; + break; + case 3: + temp_check = (divider_reg & 0x7F) == 0x7F; + break; + } - // NOTE: On GBA only, there is a glitch where if the current timer control is 7 and the written value is 7 and - // there is a coincident timer increment, there will be an additional increment along with this write. - // not sure it effects all models or of exact details, see test tac_set_timer_disabled.gbc + if (temp_check) + { + timer_old = timer; + timer++; + Console.WriteLine("glitch"); + // if overflow happens, set the interrupt flag and reload the timer (if applicable) + if (timer < timer_old) + { + if (timer_control.Bit(2)) + { + pending_reload = 4; + reload_block = false; + } + else + { + //TODO: Check if timer still gets reloaded if TAC diabled causes overflow + if (Core.REG_FFFF.Bit(2)) { Core.cpu.FlagI = true; } + Core.REG_FF0F |= 0x04; + } + } + } + } + */ break; } } @@ -104,7 +160,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk // this procedure allows several glitchy timer ticks, since it only measures falling edge of the state // so things like turning the timer off and resetting the divider will tick the timer - // NOTE: Some additional glitches happen on GBC, but they are non-deterministic and not emulated here if (old_state && !state) { timer_old = timer; @@ -120,10 +175,9 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk } else { - //TODO: Check if timer still gets reloaded if TAC diabled causes overflow - if (Core.REG_FFFF.Bit(2)) { Core.cpu.FlagI = true; } - Core.REG_FF0F |= 0x04; - } + pending_reload = 3; + reload_block = false; + } } } @@ -140,20 +194,26 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk next_free_cycle = 4 + Core.cpu.TotalExecutedCycles; // set interrupts - if (Core.REG_FFFF.Bit(2)) { Core.cpu.FlagI = true; } - Core.REG_FF0F |= 0x04; + if (!IRQ_block) + { + if (Core.REG_FFFF.Bit(2)) { Core.cpu.FlagI = true; } + Core.REG_FF0F |= 0x04; + } } } + + IRQ_block = false; } public void Reset() { - divider_reg = 0; + divider_reg = (ushort)(Core.is_GBC ? 0xFFFF : 0); timer_reload = 0; timer = 0; timer_old = 0; timer_control = 0xF8; pending_reload = 0; + IRQ_block = false; old_state = false; state = false; reload_block = false; @@ -168,6 +228,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBHawk ser.Sync(nameof(timer_old), ref timer_old); ser.Sync(nameof(timer_control), ref timer_control); ser.Sync(nameof(pending_reload), ref pending_reload); + ser.Sync(nameof(IRQ_block), ref IRQ_block); ser.Sync(nameof(old_state), ref old_state); ser.Sync(nameof(state), ref state); ser.Sync(nameof(reload_block), ref reload_block);