This commit is contained in:
goyuken 2014-12-17 02:33:05 +00:00
parent 50704247fa
commit eb4aa94546
4 changed files with 83 additions and 45 deletions

View File

@ -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);
}
}

View File

@ -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

View File

@ -79,5 +79,27 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES
/// </summary>
/// <returns></returns>
MemoryDomain GetCHRROM();
/// <summary>
/// install a callback to run at a particular scanline
/// </summary>
/// <param name="cb"></param>
/// <param name="sl"></param>
void InstallCallback1(Action cb, int sl);
/// <summary>
/// install a callback to run at a particular scanline
/// </summary>
/// <param name="cb"></param>
/// <param name="sl"></param>
void InstallCallback2(Action cb, int sl);
/// <summary>
/// remove previously installed callback
/// </summary>
void RemoveCallback1();
/// <summary>
/// remove previously installed callback
/// </summary>
void RemoveCallback2();
}
}

View File

@ -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;
}
}
}