568 lines
16 KiB
C#
568 lines
16 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.Data;
|
|
using System.Drawing;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Windows.Forms;
|
|
using System.Globalization;
|
|
using BizHawk.Emulation.Consoles.Nintendo;
|
|
using System.Diagnostics;
|
|
|
|
namespace BizHawk.MultiClient
|
|
{
|
|
public partial class NESPPU : Form
|
|
{
|
|
//TODO:
|
|
//If 8/16 sprite mode, mouse over should put 32x64 version of prite
|
|
|
|
//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
|
|
|
|
Bitmap ZoomBoxDefaultImage = new Bitmap(64, 64);
|
|
int defaultWidth; //For saving the default size of the dialog, so the user can restore if desired
|
|
int defaultHeight;
|
|
NES Nes;
|
|
|
|
NES.PPU.DebugCallback Callback = new NES.PPU.DebugCallback();
|
|
|
|
public NESPPU()
|
|
{
|
|
InitializeComponent();
|
|
Closing += (o, e) => SaveConfigSettings();
|
|
Callback.Callback = () => Generate();
|
|
}
|
|
|
|
private void SaveConfigSettings()
|
|
{
|
|
Global.Config.NESPPUWndx = this.Location.X;
|
|
Global.Config.NESPPUWndy = this.Location.Y;
|
|
Global.Config.NESPPURefreshRate = RefreshRate.Value;
|
|
}
|
|
|
|
public void Restart()
|
|
{
|
|
if (!(Global.Emulator is NES)) this.Close();
|
|
if (!this.IsHandleCreated || this.IsDisposed) return;
|
|
Nes = Global.Emulator as NES;
|
|
}
|
|
|
|
private void LoadConfigSettings()
|
|
{
|
|
defaultWidth = Size.Width; //Save these first so that the user can restore to its original size
|
|
defaultHeight = Size.Height;
|
|
|
|
if (Global.Config.NESPPUSaveWindowPosition && Global.Config.NESPPUWndx >= 0 && Global.Config.NESPPUWndy >= 0)
|
|
Location = new Point(Global.Config.NESPPUWndx, Global.Config.NESPPUWndy);
|
|
}
|
|
|
|
private byte GetBit(int address, int bit)
|
|
{
|
|
byte value = Nes.ppu.ppubus_peek(address);
|
|
return (byte)(((value >> (7 - bit)) & 1));
|
|
}
|
|
|
|
unsafe void Generate()
|
|
{
|
|
if (!this.IsHandleCreated || this.IsDisposed) return;
|
|
|
|
|
|
//Pattern Viewer
|
|
for (int x = 0; x < 16; x++)
|
|
{
|
|
PaletteView.bgPalettesPrev[x] = new PaletteViewer.Palette(PaletteView.bgPalettes[x]);
|
|
PaletteView.spritePalettesPrev[x] = new PaletteViewer.Palette(PaletteView.spritePalettes[x]);
|
|
PaletteView.bgPalettes[x].SetValue(Nes.LookupColor(Nes.ppu.PALRAM[PaletteView.bgPalettes[x].address]));
|
|
PaletteView.spritePalettes[x].SetValue(Nes.LookupColor(Nes.ppu.PALRAM[PaletteView.spritePalettes[x].address]));
|
|
}
|
|
if (PaletteView.HasChanged())
|
|
PaletteView.Refresh();
|
|
|
|
|
|
if (Global.Emulator.Frame % RefreshRate.Value == 0)
|
|
{
|
|
//Pattern Viewer
|
|
int b0 = 0;
|
|
int b1 = 0;
|
|
byte value;
|
|
int cvalue;
|
|
int pal;
|
|
|
|
System.Drawing.Imaging.BitmapData bmpdata = PatternView.pattern.LockBits(new Rectangle(new Point(0, 0), PatternView.pattern.Size), System.Drawing.Imaging.ImageLockMode.WriteOnly, System.Drawing.Imaging.PixelFormat.Format32bppArgb);
|
|
int* framebuf = (int*)bmpdata.Scan0.ToPointer();
|
|
for (int z = 0; z < 2; z++)
|
|
{
|
|
if (z == 0)
|
|
pal = PatternView.Pal0;
|
|
else
|
|
pal = PatternView.Pal1;
|
|
|
|
for (int i = 0; i < 16; i++)
|
|
{
|
|
for (int j = 0; j < 16; j++)
|
|
{
|
|
for (int x = 0; x < 8; x++)
|
|
{
|
|
for (int y = 0; y < 8; y++)
|
|
{
|
|
b0 = GetBit((z * 0x1000) + (i * 256) + (j * 16) + y + 0 * 8, x);
|
|
b1 = GetBit((z * 0x1000) + (i * 256) + (j * 16) + y + 1 * 8, x);
|
|
|
|
value = (byte)(b0 + (b1 << 1));
|
|
|
|
cvalue = Nes.LookupColor(Nes.ppu.PALRAM[value + (pal * 4)]);
|
|
|
|
Color color = Color.FromArgb(cvalue);
|
|
|
|
int adr = (x + (j * 8)) + (y + (i * 8)) * (bmpdata.Stride / 4);
|
|
framebuf[adr + (z * 128)] = color.ToArgb();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
PatternView.pattern.UnlockBits(bmpdata);
|
|
PatternView.Refresh();
|
|
|
|
System.Drawing.Imaging.BitmapData bmpdata2 = SpriteView.sprites.LockBits(new Rectangle(new Point(0, 0), SpriteView.sprites.Size), System.Drawing.Imaging.ImageLockMode.WriteOnly, System.Drawing.Imaging.PixelFormat.Format32bppArgb);
|
|
int* framebuf2 = (int*)bmpdata2.Scan0.ToPointer();
|
|
int BaseAddr, TileNum, Attributes, Palette;
|
|
|
|
int pt_add = Nes.ppu.reg_2000.obj_pattern_hi ? 0x1000 : 0;
|
|
bool is8x16 = Nes.ppu.reg_2000.obj_size_16;
|
|
|
|
//Sprite Viewer
|
|
for (int n = 0; n < 4; n++)
|
|
{
|
|
for (int r = 0; r < 16; r++)
|
|
{
|
|
BaseAddr = (r << 2) + (n << 6);
|
|
TileNum = Nes.ppu.OAM[BaseAddr + 1];
|
|
int PatAddr = 0;
|
|
|
|
if (is8x16)
|
|
{
|
|
PatAddr = ((TileNum >> 1) * 0x20);
|
|
PatAddr += (0x1000 * (TileNum & 1));
|
|
}
|
|
else
|
|
{
|
|
PatAddr = TileNum * 0x10;
|
|
PatAddr += pt_add;
|
|
}
|
|
|
|
|
|
Attributes = Nes.ppu.OAM[BaseAddr + 2];
|
|
Palette = Attributes & 0x03;
|
|
//TODO: 8x16 viewing
|
|
for (int x = 0; x < 8; x++)
|
|
{
|
|
for (int y = 0; y < 8; y++)
|
|
{
|
|
b0 = GetBit(PatAddr + y + 0 * 8, x);
|
|
b1 = GetBit(PatAddr + y + 1 * 8, x);
|
|
value = (byte)(b0 + (b1 << 1));
|
|
cvalue = Nes.LookupColor(Nes.ppu.PALRAM[16 + value + (Palette << 2)]);
|
|
Color color = Color.FromArgb(cvalue);
|
|
|
|
int adr = (x + (r * 8 * 2)) + (y + (n * 8 * 3)) * (bmpdata2.Stride / 4);
|
|
framebuf2[adr] = color.ToArgb();
|
|
}
|
|
if (is8x16)
|
|
{
|
|
PatAddr += 0x10;
|
|
for (int y = 0; y < 8; y++)
|
|
{
|
|
b0 = GetBit(PatAddr + y + 0 * 8, x);
|
|
b1 = GetBit(PatAddr + y + 1 * 8, x);
|
|
value = (byte)(b0 + (b1 << 1));
|
|
cvalue = Nes.LookupColor(Nes.ppu.PALRAM[16 + value + (Palette << 2)]);
|
|
Color color = Color.FromArgb(cvalue);
|
|
|
|
int adr = (x + (r * 8 * 2)) + ((y+8) + (n * 8 * 3)) * (bmpdata2.Stride / 4);
|
|
framebuf2[adr] = color.ToArgb();
|
|
}
|
|
PatAddr -= 0x10;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
SpriteView.sprites.UnlockBits(bmpdata2);
|
|
SpriteView.Refresh();
|
|
}
|
|
}
|
|
|
|
public unsafe void UpdateValues()
|
|
{
|
|
if (!this.IsHandleCreated || this.IsDisposed) return;
|
|
if (!(Global.Emulator is NES)) return;
|
|
Nes.ppu.PPUViewCallback = Callback;
|
|
}
|
|
|
|
private void NESPPU_Load(object sender, EventArgs e)
|
|
{
|
|
LoadConfigSettings();
|
|
Nes = Global.Emulator as NES;
|
|
ClearDetails();
|
|
RefreshRate.Value = Global.Config.NESPPURefreshRate;
|
|
}
|
|
|
|
private void ClearDetails()
|
|
{
|
|
DetailsBox.Text = "Details";
|
|
AddressLabel.Text = "";
|
|
ValueLabel.Text = "";
|
|
Value2Label.Text = "";
|
|
Value3Label.Text = "";
|
|
Value4Label.Text = "";
|
|
Value5Label.Text = "";
|
|
ZoomBox.Image = ZoomBoxDefaultImage;
|
|
}
|
|
|
|
private void PaletteView_MouseLeave(object sender, EventArgs e)
|
|
{
|
|
ClearDetails();
|
|
}
|
|
|
|
private void PaletteView_MouseEnter(object sender, EventArgs e)
|
|
{
|
|
DetailsBox.Text = "Details - Palettes";
|
|
}
|
|
|
|
private void PaletteView_MouseMove(object sender, MouseEventArgs e)
|
|
{
|
|
int baseAddr = 0x3F00;
|
|
if (e.Y > 16)
|
|
baseAddr += 16;
|
|
int column = (e.X - PaletteView.Location.X) / 16;
|
|
int addr = column + baseAddr;
|
|
AddressLabel.Text = "Address: 0x" + String.Format("{0:X4}", addr, NumberStyles.HexNumber);
|
|
int val;
|
|
int offset = addr & 0x03;
|
|
|
|
Bitmap bmp = new Bitmap(64, 64);
|
|
Graphics g= Graphics.FromImage(bmp);
|
|
|
|
if (baseAddr == 0x3F00)
|
|
{
|
|
val = Nes.ppu.PALRAM[PaletteView.bgPalettes[column].address];
|
|
ValueLabel.Text = "ID: BG" + (column / 4).ToString();
|
|
g.FillRectangle(new SolidBrush(PaletteView.bgPalettes[column].GetColor()), 0, 0, 64, 64);
|
|
}
|
|
else
|
|
{
|
|
val = Nes.ppu.PALRAM[PaletteView.spritePalettes[column].address];
|
|
ValueLabel.Text = "ID: SPR" + (column / 4).ToString();
|
|
g.FillRectangle(new SolidBrush(PaletteView.spritePalettes[column].GetColor()), 0, 0, 64, 64);
|
|
}
|
|
g.Dispose();
|
|
|
|
Value3Label.Text = "Color: 0x" + String.Format("{0:X2}", val, NumberStyles.HexNumber);
|
|
Value4Label.Text = "Offset: " + offset.ToString();
|
|
ZoomBox.Image = bmp;
|
|
}
|
|
|
|
private void autoloadToolStripMenuItem_Click(object sender, EventArgs e)
|
|
{
|
|
Global.Config.AutoLoadNESPPU ^= true;
|
|
}
|
|
|
|
private void saveWindowPositionToolStripMenuItem_Click(object sender, EventArgs e)
|
|
{
|
|
Global.Config.NESPPUSaveWindowPosition ^= true;
|
|
}
|
|
|
|
private void toolStripDropDownButton1_DropDownOpened(object sender, EventArgs e)
|
|
{
|
|
autoloadToolStripMenuItem.Checked = Global.Config.AutoLoadNESPPU;
|
|
saveWindowPositionToolStripMenuItem.Checked = Global.Config.NESPPUSaveWindowPosition;
|
|
}
|
|
|
|
private void PatternView_Click(object sender, MouseEventArgs e)
|
|
{
|
|
if (e.Button == MouseButtons.Left)
|
|
{
|
|
if (e.X < PatternView.Width / 2)
|
|
{
|
|
PatternView.Pal0++;
|
|
if (PatternView.Pal0 > 7) PatternView.Pal0 = 0;
|
|
}
|
|
else
|
|
{
|
|
PatternView.Pal1++;
|
|
if (PatternView.Pal1 > 7) PatternView.Pal1 = 0;
|
|
}
|
|
UpdateTableLabels();
|
|
}
|
|
HandleDefaultImage(e);
|
|
}
|
|
|
|
private void UpdateTableLabels()
|
|
{
|
|
Table0PaletteLabel.Text = "Palette: " + PatternView.Pal0;
|
|
Table1PaletteLabel.Text = "Palette: " + PatternView.Pal1;
|
|
}
|
|
|
|
private void PatternView_MouseEnter(object sender, EventArgs e)
|
|
{
|
|
DetailsBox.Text = "Details - Patterns";
|
|
}
|
|
|
|
private void PatternView_MouseLeave(object sender, EventArgs e)
|
|
{
|
|
ClearDetails();
|
|
}
|
|
|
|
private void PatternView_MouseMove(object sender, MouseEventArgs e)
|
|
{
|
|
int table = 0;
|
|
int address = 0;
|
|
int tile = 0;
|
|
if (e.X > PatternView.Width / 2)
|
|
table = 1;
|
|
|
|
if (table == 0)
|
|
{
|
|
tile = (e.X - 1) / 8;
|
|
address = tile * 16;
|
|
|
|
}
|
|
else
|
|
{
|
|
tile = (e.X - 128) / 8;
|
|
address = 0x1000 + (tile * 16);
|
|
|
|
}
|
|
|
|
address += (e.Y / 8) * 256;
|
|
tile += (e.Y / 8) * 16;
|
|
string Usage = "Usage: ";
|
|
|
|
if ((Nes.ppu.reg_2000.Value & 0x10) << 4 == ((address >> 4) & 0x100))
|
|
Usage = "BG";
|
|
else if (((Nes.ppu.reg_2000.Value & 0x08) << 5) == ((address >> 4) & 0x100))
|
|
Usage = "SPR";
|
|
|
|
if ((Nes.ppu.reg_2000.Value & 0x20) > 0)
|
|
Usage += " (SPR16)";
|
|
|
|
AddressLabel.Text = "Address: " + String.Format("{0:X4}", address);
|
|
ValueLabel.Text = "Table " + table.ToString();
|
|
Value3Label.Text = "Tile " + String.Format("{0:X2}", tile);
|
|
Value4Label.Text = Usage;
|
|
|
|
ZoomBox.Image = Section(PatternView.pattern, new Rectangle(new Point((e.X / 8) * 8, (e.Y / 8) * 8), new Size(8, 8)), false);
|
|
}
|
|
|
|
static public Bitmap Section(Bitmap srcBitmap, Rectangle section, bool Is8x16)
|
|
{
|
|
// Create the new bitmap and associated graphics object
|
|
Bitmap bmp = new Bitmap(64, 64);
|
|
Graphics g = Graphics.FromImage(bmp);
|
|
|
|
// Draw the specified section of the source bitmap to the new one
|
|
g.InterpolationMode = System.Drawing.Drawing2D.InterpolationMode.NearestNeighbor;
|
|
g.PixelOffsetMode = System.Drawing.Drawing2D.PixelOffsetMode.Half;
|
|
Rectangle rect;
|
|
if (Is8x16)
|
|
rect = new Rectangle(0, 0, 32, 64);
|
|
else
|
|
rect = new Rectangle(0, 0, 64, 64);
|
|
g.DrawImage(srcBitmap, rect, section, GraphicsUnit.Pixel);
|
|
g.Dispose();
|
|
|
|
// Return the bitmap
|
|
return bmp;
|
|
}
|
|
|
|
private void toolStripDropDownButton2_DropDownOpened(object sender, EventArgs e)
|
|
{
|
|
Table0P0.Checked = false;
|
|
Table0P1.Checked = false;
|
|
Table0P2.Checked = false;
|
|
Table0P3.Checked = false;
|
|
Table0P4.Checked = false;
|
|
Table0P5.Checked = false;
|
|
Table0P6.Checked = false;
|
|
Table0P7.Checked = false;
|
|
Table1P0.Checked = false;
|
|
Table1P1.Checked = false;
|
|
Table1P2.Checked = false;
|
|
Table1P3.Checked = false;
|
|
Table1P4.Checked = false;
|
|
Table1P5.Checked = false;
|
|
Table1P6.Checked = false;
|
|
Table1P7.Checked = false;
|
|
|
|
Table0P0.Checked = false;
|
|
|
|
switch (PatternView.Pal0)
|
|
{
|
|
case 0:
|
|
Table0P0.Checked = true;
|
|
break;
|
|
case 1:
|
|
Table0P1.Checked = true;
|
|
break;
|
|
case 2:
|
|
Table0P2.Checked = true;
|
|
break;
|
|
case 3:
|
|
Table0P3.Checked = true;
|
|
break;
|
|
case 4:
|
|
Table0P4.Checked = true;
|
|
break;
|
|
case 5:
|
|
Table0P5.Checked = true;
|
|
break;
|
|
case 6:
|
|
Table0P6.Checked = true;
|
|
break;
|
|
case 7:
|
|
Table0P7.Checked = true;
|
|
break;
|
|
}
|
|
|
|
switch (PatternView.Pal1)
|
|
{
|
|
case 0:
|
|
Table1P0.Checked = true;
|
|
break;
|
|
case 1:
|
|
Table1P1.Checked = true;
|
|
break;
|
|
case 2:
|
|
Table1P2.Checked = true;
|
|
break;
|
|
case 3:
|
|
Table1P3.Checked = true;
|
|
break;
|
|
case 4:
|
|
Table1P4.Checked = true;
|
|
break;
|
|
case 5:
|
|
Table1P5.Checked = true;
|
|
break;
|
|
case 6:
|
|
Table1P6.Checked = true;
|
|
break;
|
|
case 7:
|
|
Table1P7.Checked = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
private void Palette_Click(object sender, EventArgs e)
|
|
{
|
|
if (sender == Table0P0) PatternView.Pal0 = 0;
|
|
if (sender == Table0P1) PatternView.Pal0 = 1;
|
|
if (sender == Table0P2) PatternView.Pal0 = 2;
|
|
if (sender == Table0P3) PatternView.Pal0 = 3;
|
|
if (sender == Table0P4) PatternView.Pal0 = 4;
|
|
if (sender == Table0P5) PatternView.Pal0 = 5;
|
|
if (sender == Table0P6) PatternView.Pal0 = 6;
|
|
if (sender == Table0P7) PatternView.Pal0 = 7;
|
|
|
|
if (sender == Table1P0) PatternView.Pal1 = 0;
|
|
if (sender == Table1P1) PatternView.Pal1 = 1;
|
|
if (sender == Table1P2) PatternView.Pal1 = 2;
|
|
if (sender == Table1P3) PatternView.Pal1 = 3;
|
|
if (sender == Table1P4) PatternView.Pal1 = 4;
|
|
if (sender == Table1P5) PatternView.Pal1 = 5;
|
|
if (sender == Table1P6) PatternView.Pal1 = 6;
|
|
if (sender == Table1P7) PatternView.Pal1 = 7;
|
|
|
|
UpdateTableLabels();
|
|
}
|
|
|
|
private void txtScanline_TextChanged(object sender, EventArgs e)
|
|
{
|
|
int temp = 0;
|
|
if (int.TryParse(txtScanline.Text, out temp))
|
|
{
|
|
Callback.Scanline = temp;
|
|
}
|
|
}
|
|
|
|
private void NESPPU_FormClosed(object sender, FormClosedEventArgs e)
|
|
{
|
|
if (Nes == null) return;
|
|
if (Nes.ppu.PPUViewCallback == Callback)
|
|
Nes.ppu.PPUViewCallback = null;
|
|
}
|
|
|
|
private void SpriteView_MouseEnter(object sender, EventArgs e)
|
|
{
|
|
DetailsBox.Text = "Details - Sprites";
|
|
}
|
|
|
|
private void SpriteView_MouseLeave(object sender, EventArgs e)
|
|
{
|
|
ClearDetails();
|
|
}
|
|
|
|
private void SpriteView_MouseMove(object sender, MouseEventArgs e)
|
|
{
|
|
bool is8x16 = Nes.ppu.reg_2000.obj_size_16;
|
|
int SpriteNumber = ((e.Y / 24) * 16) + (e.X / 16);
|
|
int X = Nes.ppu.OAM[(SpriteNumber * 4) + 3];
|
|
int Y = Nes.ppu.OAM[SpriteNumber * 4];
|
|
int Color = Nes.ppu.OAM[(SpriteNumber * 4) + 2] & 0x03;
|
|
int Attributes = Nes.ppu.OAM[(SpriteNumber * 4) + 2];
|
|
|
|
string flags = "Flags: ";
|
|
int h = GetBit(Attributes, 6);
|
|
int v = GetBit(Attributes, 7);
|
|
int priority = GetBit(Attributes, 5);
|
|
if (h > 0)
|
|
flags += "H ";
|
|
if (v > 0)
|
|
flags += "V ";
|
|
if (priority > 0)
|
|
flags += "Behind";
|
|
else
|
|
flags += "Front";
|
|
|
|
int Tile = Nes.ppu.OAM[SpriteNumber * 1]; ;
|
|
|
|
AddressLabel.Text = "Number: " + String.Format("{0:X2}", SpriteNumber);
|
|
ValueLabel.Text = "X: " + String.Format("{0:X2}", X);
|
|
Value2Label.Text = "Y: " + String.Format("{0:X2}", Y);
|
|
Value3Label.Text = "Tile: " + String.Format("{0:X2}", Tile);
|
|
Value4Label.Text = "Color: " + Color.ToString();
|
|
Value5Label.Text = flags;
|
|
|
|
if (is8x16)
|
|
ZoomBox.Image = Section(SpriteView.sprites, new Rectangle(new Point((e.X / 8) * 8, (e.Y / 24) * 24), new Size(8, 16)), true);
|
|
else
|
|
ZoomBox.Image = Section(SpriteView.sprites, new Rectangle(new Point((e.X / 8) * 8, (e.Y / 8) * 8), new Size(8, 8)), false);
|
|
}
|
|
|
|
private void PaletteView_MouseClick(object sender, MouseEventArgs e)
|
|
{
|
|
HandleDefaultImage(e);
|
|
}
|
|
|
|
private void SpriteView_MouseClick(object sender, MouseEventArgs e)
|
|
{
|
|
HandleDefaultImage(e);
|
|
}
|
|
|
|
private void HandleDefaultImage(MouseEventArgs e)
|
|
{
|
|
if (e.Button == MouseButtons.Right)
|
|
{
|
|
ZoomBoxDefaultImage = ZoomBox.Image as Bitmap;
|
|
}
|
|
}
|
|
|
|
private void NESPPU_MouseClick(object sender, MouseEventArgs e)
|
|
{
|
|
ZoomBox.Image = new Bitmap(64, 64);
|
|
}
|
|
}
|
|
}
|