From 92dfdaec0fb8824b749ffb708296df1275c1a2d1 Mon Sep 17 00:00:00 2001 From: alyosha-tas Date: Sat, 31 Jul 2021 22:32:10 -0400 Subject: [PATCH] NESHawk: fix MMC3 clear timing, fixes Klax --- .../Nintendo/NES/Boards/MMC3_family/MMC3.cs | 28 ++++++++++++++----- 1 file changed, 21 insertions(+), 7 deletions(-) diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Boards/MMC3_family/MMC3.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Boards/MMC3_family/MMC3.cs index ec11545c8d..5477a9af28 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Boards/MMC3_family/MMC3.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Boards/MMC3_family/MMC3.cs @@ -5,6 +5,7 @@ using BizHawk.Common; using BizHawk.Common.NumberExtensions; +using System; namespace BizHawk.Emulation.Cores.Nintendo.NES { @@ -23,6 +24,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES public bool irq_pending, irq_enable, irq_reload_flag; public bool wram_enable, wram_write_protect; + public bool just_cleared_pending, just_cleared; + //it really seems like these should be the same but i cant seem to unify them. //theres no sense in delaying the IRQ, so its logic must be tied to the separator. //the hint, of course, is that the countdown value is the same. @@ -64,6 +67,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES protected NesBoardBase board; public MMC3(NesBoardBase board, int num_prg_banks) { + just_cleared = just_cleared_pending = false; + MirrorMask = 1; this.board = board; if (board.Cart.Chips.Contains("MMC3A")) MMC3Type = EMMC3Type.MMC3A; @@ -150,6 +155,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES ser.Sync(nameof(wram_enable), ref wram_enable); ser.Sync(nameof(wram_write_protect), ref wram_write_protect); ser.Sync(nameof(cmd), ref cmd); + ser.Sync(nameof(just_cleared), ref just_cleared); + ser.Sync(nameof(just_cleared_pending), ref just_cleared_pending); Sync(); } @@ -180,7 +187,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES Sync(); break; case 0x2000: //$A000 - //mirroring + //mirroring mirror = (byte)(value & MirrorMask); board.SetMirrorType(MirrorType); break; @@ -194,9 +201,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES irq_reload = value; break; case 0x4001: //$C001 - IRQ Clear - irq_counter = 0; - if (oldIrqType) - irq_reload_flag = true; + // does not take immediate effect (fixes Klax) + just_cleared_pending = true; break; case 0x6000: //$E000 - IRQ Acknowledge / Disable irq_enable = false; @@ -260,6 +266,16 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES ClockIRQ(); } } + + if (just_cleared) + { + irq_counter = 0; + if (oldIrqType) + irq_reload_flag = true; + } + + just_cleared = just_cleared_pending; + just_cleared_pending = false; } public virtual int Get_PRGBank_8K(int addr) @@ -300,8 +316,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES } a12_old = a12; - } - + } } } @@ -311,7 +326,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES public MMC3 mmc3; public int extra_vrom; - public override void AddressPpu(int addr) { mmc3.AddressPPU(addr);