diff --git a/BizHawk.Client.EmuHawk/tools/NES/NESNameTableViewer.cs b/BizHawk.Client.EmuHawk/tools/NES/NESNameTableViewer.cs index 565588e3f3..777eb1f0da 100644 --- a/BizHawk.Client.EmuHawk/tools/NES/NESNameTableViewer.cs +++ b/BizHawk.Client.EmuHawk/tools/NES/NESNameTableViewer.cs @@ -20,7 +20,7 @@ namespace BizHawk.Client.EmuHawk [RequiredService] private IEmulator _emu { get; set; } - private readonly NES.PPU.DebugCallback _callback = new NES.PPU.DebugCallback(); + int scanline; public NESNameTableViewer() { @@ -32,7 +32,6 @@ namespace BizHawk.Client.EmuHawk Global.Config.NESNameTableRefreshRate = RefreshRate.Value; }; TopMost = Global.Config.NesNameTableSettings.TopMost; - _callback.Callback = () => Generate(); } private void NESNameTableViewer_Load(object sender, EventArgs e) @@ -58,7 +57,7 @@ namespace BizHawk.Client.EmuHawk public void UpdateValues() { - _nes.ppu.NTViewCallback = _callback; + xxx.InstallCallback1(() => Generate(), scanline); } public void FastUpdate() @@ -286,18 +285,14 @@ namespace BizHawk.Client.EmuHawk private void NESNameTableViewer_FormClosed(object sender, FormClosedEventArgs e) { - if (_nes != null && _nes.ppu.NTViewCallback == _callback) - { - _nes.ppu.NTViewCallback = null; - } + xxx.RemoveCallback1(); } private void ScanlineTextbox_TextChanged(object sender, EventArgs e) { - int temp; - if (int.TryParse(txtScanline.Text, out temp)) + if (int.TryParse(txtScanline.Text, out scanline)) { - _callback.Scanline = temp; + xxx.InstallCallback1(() => Generate(), scanline); } } diff --git a/BizHawk.Client.EmuHawk/tools/NES/NESPPU.cs b/BizHawk.Client.EmuHawk/tools/NES/NESPPU.cs index cdc82c0893..8ff2d6711a 100644 --- a/BizHawk.Client.EmuHawk/tools/NES/NESPPU.cs +++ b/BizHawk.Client.EmuHawk/tools/NES/NESPPU.cs @@ -17,11 +17,10 @@ namespace BizHawk.Client.EmuHawk // Speedups // Smarter refreshing? only refresh when things of changed, perhaps peek at the ppu to when the pattern table has changed, or sprites have moved // Maybe 48 individual bitmaps for sprites is faster than the overhead of redrawing all that transparent space - private readonly byte[] _ppuBus = Enumerable.Repeat((byte)0, 0x2000).ToArray(); - private readonly byte[] _ppuBusprev = Enumerable.Repeat((byte)0, 0x2000).ToArray(); - private readonly byte[] _palRam = Enumerable.Repeat((byte)0, 0x20).ToArray(); - private readonly byte[] _palRamPrev = Enumerable.Repeat((byte)0, 0x20).ToArray(); - private readonly NES.PPU.DebugCallback _callback = new NES.PPU.DebugCallback(); + private readonly byte[] _ppuBusprev = new byte[0x3000]; + private readonly byte[] _palRamPrev = new byte[0x20]; + + int scanline; private Bitmap _zoomBoxDefaultImage = new Bitmap(64, 64); private bool _forceChange; @@ -43,7 +42,6 @@ namespace BizHawk.Client.EmuHawk Global.Config.NESPPURefreshRate = RefreshRate.Value; }; TopMost = Global.Config.NesPPUSettings.TopMost; - _callback.Callback = () => Generate(); CalculateFormSize(); } @@ -63,7 +61,7 @@ namespace BizHawk.Client.EmuHawk public void UpdateValues() { - _nes.ppu.PPUViewCallback = _callback; + xxx.InstallCallback2(() => Generate(), scanline); } public void FastUpdate() @@ -87,34 +85,38 @@ namespace BizHawk.Client.EmuHawk } } - private byte GetBit(int address, int bit) + private byte GetBit(byte[] PPUBus, int address, int bit) { - return (byte)((_ppuBus[address] >> (7 - bit)) & 1); + return (byte)((PPUBus[address] >> (7 - bit)) & 1); } - private bool CheckChange() + private bool CheckChange(byte[] PALRAM, byte[] PPUBus) { bool changed = false; - for (var i = 0; i < 0x20; i++) + for (int i = 0; i < 0x20; i++) { - _palRamPrev[i] = _palRam[i]; - _palRam[i] = _nes.ppu.PALRAM[i]; - if (_palRam[i] != _palRamPrev[i]) + if (_palRamPrev[i] != PALRAM[i]) { changed = true; + break; } } - for (var i = 0; i < 0x2000; i++) + if (!changed) { - _ppuBusprev[i] = _ppuBus[i]; - _ppuBus[i] = _nes.ppu.ppubus_peek(i); - if (_ppuBus[i] != _ppuBusprev[i]) + for (int i = 0; i < 0x2000; i++) { - changed = true; + if (_ppuBusprev[i] != PPUBus[i]) + { + changed = true; + break; + } } } + Buffer.BlockCopy(PALRAM, 0, _palRamPrev, 0, 0x20); + Buffer.BlockCopy(PPUBus, 0, _ppuBusprev, 0, 0x3000); + if (_forceChange) { return true; @@ -180,13 +182,14 @@ namespace BizHawk.Client.EmuHawk byte[] PALRAM = xxx.GetPalRam(); int[] FinalPalette = xxx.GetPalette(); byte[] OAM = xxx.GetOam(); + byte[] PPUBus = xxx.GetPPUBus(); int b0; int b1; byte value; int cvalue; - if (CheckChange()) + if (CheckChange(PALRAM, PPUBus)) { _forceChange = false; @@ -204,7 +207,7 @@ namespace BizHawk.Client.EmuHawk PaletteView.Refresh(); } - DrawPatternView(PatternView, _ppuBus, FinalPalette, PALRAM); + DrawPatternView(PatternView, PPUBus, FinalPalette, PALRAM); } var bmpdata2 = SpriteView.sprites.LockBits(new Rectangle(new Point(0, 0), SpriteView.sprites.Size), ImageLockMode.WriteOnly, PixelFormat.Format32bppArgb); @@ -242,8 +245,8 @@ namespace BizHawk.Client.EmuHawk for (int y = 0; y < 8; y++) { int address = patternAddr + y; - b0 = (byte)((_ppuBus[address] >> (7 - x)) & 1); - b1 = (byte)((_ppuBus[address + 8] >> (7 - x)) & 1); + b0 = (byte)((PPUBus[address] >> (7 - x)) & 1); + b1 = (byte)((PPUBus[address + 8] >> (7 - x)) & 1); value = (byte)(b0 + (b1 << 1)); cvalue = FinalPalette[PALRAM[16 + value + (Palette << 2)]]; @@ -257,8 +260,8 @@ namespace BizHawk.Client.EmuHawk for (int y = 0; y < 8; y++) { int address = patternAddr + y; - b0 = (byte)((_ppuBus[address] >> (7 - x)) & 1); - b1 = (byte)((_ppuBus[address + 8] >> (7 - x)) & 1); + b0 = (byte)((PPUBus[address] >> (7 - x)) & 1); + b1 = (byte)((PPUBus[address + 8] >> (7 - x)) & 1); value = (byte)(b0 + (b1 << 1)); cvalue = FinalPalette[PALRAM[16 + value + (Palette << 2)]]; @@ -616,6 +619,7 @@ namespace BizHawk.Client.EmuHawk if (e.Y >= SpriteView.ClientRectangle.Bottom) return; byte[] OAM = xxx.GetOam(); + byte[] PPUBus = xxx.GetPPUBus(); // caching is quicker, but not really correct in this case bool is8x16 = xxx.SPTall; var spriteNumber = ((e.Y / 24) * 16) + (e.X / 16); @@ -625,9 +629,9 @@ namespace BizHawk.Client.EmuHawk var attributes = OAM[(spriteNumber * 4) + 2]; var flags = "Flags: "; - int h = GetBit(attributes, 6); - int v = GetBit(attributes, 7); - int priority = GetBit(attributes, 5); + int h = GetBit(PPUBus, attributes, 6); + int v = GetBit(PPUBus, attributes, 7); + int priority = GetBit(PPUBus, attributes, 5); if (h > 0) { flags += "H "; @@ -818,19 +822,15 @@ namespace BizHawk.Client.EmuHawk private void ScanlineTextbox_TextChanged(object sender, EventArgs e) { - int temp; - if (int.TryParse(txtScanline.Text, out temp)) + if (int.TryParse(txtScanline.Text, out scanline)) { - _callback.Scanline = temp; + xxx.InstallCallback2(() => Generate(), scanline); } } private void NesPPU_FormClosed(object sender, FormClosedEventArgs e) { - if (_nes != null && _nes.ppu.PPUViewCallback == _callback) - { - _nes.ppu.PPUViewCallback = null; - } + xxx.RemoveCallback2(); } #endregion diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/INESPPUViewable.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/INESPPUViewable.cs index 3b815c0217..54b16f93b9 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/INESPPUViewable.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/INESPPUViewable.cs @@ -79,5 +79,27 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES /// /// MemoryDomain GetCHRROM(); + + /// + /// install a callback to run at a particular scanline + /// + /// + /// + void InstallCallback1(Action cb, int sl); + /// + /// install a callback to run at a particular scanline + /// + /// + /// + void InstallCallback2(Action cb, int sl); + + /// + /// remove previously installed callback + /// + void RemoveCallback1(); + /// + /// remove previously installed callback + /// + void RemoveCallback2(); } } diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.INESPPUViewable.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.INESPPUViewable.cs index 5bb897f034..f367aabd14 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.INESPPUViewable.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.INESPPUViewable.cs @@ -86,5 +86,26 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES { return MemoryDomains["CHR VROM"]; } + + + public void InstallCallback1(Action cb, int sl) + { + ppu.NTViewCallback = new PPU.DebugCallback { Callback = cb, Scanline = sl }; + } + + public void InstallCallback2(Action cb, int sl) + { + ppu.PPUViewCallback = new PPU.DebugCallback { Callback = cb, Scanline = sl }; + } + + public void RemoveCallback1() + { + ppu.NTViewCallback = null; + } + + public void RemoveCallback2() + { + ppu.PPUViewCallback = null; + } } }