diff --git a/BizHawk.Client.EmuHawk/MainForm.cs b/BizHawk.Client.EmuHawk/MainForm.cs index 8e46666f2a..c348a2e11e 100644 --- a/BizHawk.Client.EmuHawk/MainForm.cs +++ b/BizHawk.Client.EmuHawk/MainForm.cs @@ -1855,9 +1855,9 @@ namespace BizHawk.Client.EmuHawk { ofd.Filter = FormatFilter( "Rom Files", "*.nes;*.fds;*.sms;*.gg;*.sg;*.pce;*.sgx;*.bin;*.smd;*.rom;*.a26;*.a78;*.lnx;*.m3u;*.cue;*.ccd;*.exe;*.gb;*.gbc;*.gba;*.gen;*.md;*.col;.int;*.smc;*.sfc;*.prg;*.d64;*.g64;*.crt;*.sgb;*.xml;*.z64;*.v64;*.n64;*.ws;*.wsc;%ARCH%", - "Music Files", "*.psf;*.sid", + "Music Files", "*.psf;*.sid;*.nsf", "Disc Images", "*.cue;*.ccd;*.m3u", - "NES", "*.nes;*.fds;%ARCH%", + "NES", "*.nes;*.fds;*.nsf;%ARCH%", "Super NES", "*.smc;*.sfc;*.xml;%ARCH%", "Master System", "*.sms;*.gg;*.sg;%ARCH%", "PC Engine", "*.pce;*.sgx;*.cue;*.ccd;%ARCH%", @@ -1885,7 +1885,7 @@ namespace BizHawk.Client.EmuHawk ofd.Filter = FormatFilter( "Rom Files", "*.nes;*.fds;*.sms;*.gg;*.sg;*.gb;*.gbc;*.gba;*.pce;*.sgx;*.bin;*.smd;*.gen;*.md;*.smc;*.sfc;*.a26;*.a78;*.lnx;*.col;*.rom;*.cue;*.ccd;*.sgb;*.z64;*.v64;*.n64;*.ws;*.wsc;*.xml;%ARCH%", "Disc Images", "*.cue;*.ccd;*.m3u", - "NES", "*.nes;*.fds;%ARCH%", + "NES", "*.nes;*.fds;*.nsf;%ARCH%", "Super NES", "*.smc;*.sfc;*.xml;%ARCH%", "Nintendo 64", "*.z64;*.v64;*.n64", "Gameboy", "*.gb;*.gbc;*.sgb;%ARCH%", diff --git a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj index 5ad315464c..c2b585a2ab 100644 --- a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj +++ b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj @@ -602,6 +602,7 @@ Code + @@ -664,6 +665,7 @@ NES.cs + diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Boards/NSFBoard.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Boards/NSFBoard.cs new file mode 100644 index 0000000000..4bd79ffb84 --- /dev/null +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Boards/NSFBoard.cs @@ -0,0 +1,369 @@ + +using System; +using BizHawk.Common; + +//NSF ROM and general approaches are taken from FCEUX. however, i've improvised/simplified/broken things so the rom is doing some pointless stuff now. + +//check nsfspec.txt for more on why FDS is weird. lets try not following FCEUX too much there. + +//TODO - add a sleep mode to the cpu and patch the rom program to use it? +//some NSF players know when a song ends. + +namespace BizHawk.Emulation.Cores.Nintendo.NES +{ + [NES.INESBoardImplCancel] + public sealed class NSFBoard : NES.NESBoardBase + { + //configuration + internal NSFFormat nsf; + byte[] InitBankSwitches = new byte[8]; + byte[] FakePRG = new byte[32768]; + bool BankSwitched; + + //------------------------------ + //state + IntBuffer prg_banks_4k = new IntBuffer(8); + + /// + /// whether vectors are currently patched. they should not be patched when running init/play routines because data from the ends of banks might get used + /// + bool Patch_Vectors; + + int CurrentSong; + bool ResetSignal; + int ButtonState; + + public override bool Configure(NES.EDetectionOrigin origin) + { + Cart.wram_size = 8; + + return true; + } + + public override void Dispose() + { + prg_banks_4k.Dispose(); + base.Dispose(); + } + + public void InitNSF(NSFFormat nsf) + { + this.nsf = nsf; + + //patch the NSF rom with the init and play addresses + NSFROM[0x12] = (byte)(nsf.InitAddress); + NSFROM[0x13] = (byte)(nsf.InitAddress >> 8); + NSFROM[0x19] = (byte)(nsf.PlayAddress); + NSFROM[0x1A] = (byte)(nsf.PlayAddress >> 8); + + //complicated anlysis straight from FCEUX + //apparently, it converts a non-bankswitched configuration into a bankswitched configuration + //since the non-bankswitched configuration is seemingly almost pointless. + //I'm not too sure how we would really get a non-bankswitched file, using the code below. + //It would need to be loaded below 0x8000 I think? + BankSwitched = false; + for (int i = 0; i < 8; i++) + { + InitBankSwitches[i] = nsf.BankswitchInitValues[i]; + if (InitBankSwitches[i] != 0) + BankSwitched = true; + } + + if (!BankSwitched) + { + if ((nsf.LoadAddress & 0x7000) >= 0x7000) + { + //"Ice Climber, and other F000 base address tunes need this" + BankSwitched = true; + } + else + { + byte bankCounter = 0; + for (int x = (nsf.LoadAddress >> 12) & 0x7; x < 8; x++) + { + InitBankSwitches[x] = bankCounter; + bankCounter++; + } + BankSwitched = false; + } + } + + for (int i = 0; i < 8; i++) + if (InitBankSwitches[i] != 0) + BankSwitched = true; + + if (!BankSwitched) + { + throw new Exception("Test"); + //setup FakePRG by copying in + } + + ReplayInit(); + CurrentSong = nsf.StartingSong; + } + + void ReplayInit() + { + ResetSignal = true; + Patch_Vectors = true; + } + + public override void NESSoftReset() + { + ReplayInit(); + } + + public override void SyncState(Serializer ser) + { + base.SyncState(ser); + } + + public override void WriteEXP(int addr, byte value) + { + switch (addr) + { + case 0x1FF6: + case 0x1FF7: + //if (!(NSFHeader.SoundChip & 4)) return; //FDS + break; + case 0x1FF8: + case 0x1FF9: + case 0x1FFA: + case 0x1FFB: + case 0x1FFC: + case 0x1FFD: + case 0x1FFE: + case 0x1FFF: + if (!BankSwitched) break; + addr -= 0x1FF8; + prg_banks_4k[addr] = value; + break; + } + } + + + public override void WriteReg2xxx(int addr, byte value) + { + switch (addr) + { + case 0x3FF3: Patch_Vectors = true; break; + case 0x3FF4: Patch_Vectors = false; break; + case 0x3FF5: Patch_Vectors = true; break; + default: + base.WriteReg2xxx(addr, value); + break; + } + } + + public override byte PeekReg2xxx(int addr) + { + if (addr < 0x3FF0) + return NSFROM[addr - 0x3800]; + else return base.PeekReg2xxx(addr); + } + + public override byte ReadReg2xxx(int addr) + { + if (addr < 0x3800) + return base.ReadReg2xxx(addr); + else if (addr >= 0x3FF0) + { + if (addr == 0x3FF0) + { + byte ret = 0; + if (ResetSignal) ret = 1; + ResetSignal = false; + return ret; + } + else if (addr == 0x3FF1) + { + //kevtris's reset process seems not to work. dunno what all is going on in there + + //our own innovation, should work OK.. + NES.apu.NESSoftReset(); + + //mostly fceux's guidance + NES.WriteMemory(0x4015, 0); + for (int i = 0; i < 14; i++) + NES.WriteMemory((ushort)(0x4000 + i), 0); + NES.WriteMemory(0x4015, 0x0F); + + //clearing APU misc stuff, maybe not needed with soft reset above + //NES.WriteMemory(0x4017, 0xC0); + //NES.WriteMemory(0x4017, 0xC0); + //NES.WriteMemory(0x4017, 0x40); + + //important to NSF standard for ram to be cleared, otherwise replayers are confused on account of not initializing memory themselves + var ram = NES.ram; + var wram = this.WRAM; + int wram_size = wram.Length; + for (int i = 0; i < 0x800; i++) + ram[i] = 0; + for (int i = 0; i < wram_size; i++) + wram[i] = 0; + + //store specified initial bank state + if (BankSwitched) + for (int i = 0; i < 8; i++) + WriteEXP(0x5FF8 + i - 0x4000, InitBankSwitches[i]); + + return (byte)(CurrentSong - 1); + } + else if (addr == 0x3FF3) return 0; //always return NTSC I guess + else return base.ReadReg2xxx(addr); + } + else if (addr - 0x3800 < NSFROM.Length) return NSFROM[addr - 0x3800]; + else return base.ReadReg2xxx(addr); + } + + //; @NMIVector + //00:XX00:8D F4 3F STA $3FF4 = #$00 ; clear NMI_2 (the value of A is unimportant) + //00:XX03:A2 FF LDX #$FF + //00:XX05:9A TXS ; stack pointer is initialized + //00:XX06:AD F0 3F LDA $3FF0 = #$00 ; read a flag that says whether we need to run init + //00:XX09:F0 09 BEQ $8014 ; If we dont need init, go to @PastInit + //00:XX0B:AD F1 3F LDA $3FF1 = #$00 ; reading this value causes a reset + //00:XX0E:AE F3 3F LDX $3FF3 = #$00 ; reads the PAL flag + //00:XX11:20 00 00 JSR $0000 ; JSR to INIT routine + //; @PastInit + //00:XX14:A9 00 LDA #$00 + //00:XX16:AA TAX + //00:XX17:A8 TAY ; X and Y are cleared + //00:XX18:20 00 00 JSR $0000 ; JSR to PLAY routine + //00:XX1B:8D F5 3F STA $3FF5 = #$FF ; set NMI_2 flag + //00:XX1E:90 FE BCC $XX1E ; infinite loop.. when the song is over? + //; @ResetVector + //00:XX20:8D F3 3F STA $3FF3 = #$00 ; set NMI_1 flag (the value of A is unimportant); since the rom boots here, this was needed for the initial NMI. but we also get it from having the reset signal set, so.. + //00:XX23:18 CLC + //00:XX24:90 FE BCC $XX24 ;infinite loop to wait for first NMI + + const ushort NMI_VECTOR = 0x3800; + const ushort RESET_VECTOR = 0x3820; + + //for reasons unknown, this is a little extra long + byte[] NSFROM = new byte[0x30 + 6] + { + //0x00 - NMI + 0x8D,0xF4,0x3F, //Stop play routine NMIs. + 0xA2,0xFF,0x9A, //Initialize the stack pointer. + 0xAD,0xF0,0x3F, //See if we need to init. + 0xF0,0x09, //If 0, go to play routine playing. + + 0xAD,0xF1,0x3F, //Confirm and load A + 0xAE,0xF3,0x3F, //Load X with PAL/NTSC byte + + //0x11 + 0x20,0x00,0x00, //JSR to init routine (WILL BE PATCHED) + + 0xA9,0x00, + 0xAA, + 0xA8, + + //0x18 + 0x20,0x00,0x00, //JSR to play routine (WILL BE PATCHED) + 0x8D,0xF5,0x3F, //Start play routine NMIs. + 0x90,0xFE, //Loopie time. + + // 0x20 + 0x8D,0xF3,0x3F, //Init init NMIs + 0x18, + 0x90,0xFE, //Loopie time. + + //0x26 + 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, + //0x30 + 0x00,0x00,0x00,0x00,0x00,0x00 + }; + + public override void AtVsyncNMI() + { + if(Patch_Vectors) + NES.cpu.NMI = true; + + //strobe pad + NES.WriteMemory(0x4016, 1); + NES.WriteMemory(0x4016, 0); + + //read pad and create rising edge button signals so we dont trigger events as quickly as we hold the button down + int currButtons = 0; + for (int i = 0; i < 8; i++) + { + currButtons <<= 1; + currButtons |= (NES.ReadMemory(0x4016) & 1); + } + int justDown = (~ButtonState) & currButtons; + Bit a = (justDown >> 7) & 1; + Bit b = (justDown >> 6) & 1; + Bit sel = (justDown >> 5) & 1; + Bit start = (justDown >> 4) & 1; + Bit up = (justDown >> 3) & 1; + Bit down = (justDown >> 2) & 1; + Bit left = (justDown >> 1) & 1; + Bit right = (justDown >> 0) & 1; + ButtonState = currButtons; + + //RIGHT: next song + //LEFT: prev song + //A: restart song + + bool reset = false; + if (right) + { + CurrentSong++; + reset = true; + } + if (left) + { + CurrentSong--; + reset = true; + } + + if (a) + reset = true; + + if (reset) + { + ReplayInit(); + } + } + + public override byte ReadPPU(int addr) + { + return 0; + } + + public override byte ReadWRAM(int addr) + { + return base.ReadWRAM(addr); + } + + public override byte ReadPRG(int addr) + { + //patch in vector reading + if (Patch_Vectors) + { + if (addr == 0x7FFA) return (byte)(NMI_VECTOR & 0xFF); + else if (addr == 0x7FFB) return (byte)((NMI_VECTOR >> 8) & 0xFF); + else if (addr == 0x7FFC) return (byte)(RESET_VECTOR & 0xFF); + else if (addr == 0x7FFD) { return (byte)((RESET_VECTOR >> 8) & 0xFF); } + return NES.DB; + } + else + { + int bank_4k = addr >> 12; + int ofs = addr & ((1 << 12) - 1); + bank_4k = prg_banks_4k[bank_4k]; + addr = (bank_4k << 12) | ofs; + + if (BankSwitched) + { + //rom data began at 0x80 of the NSF file + addr += 0x80; + + return ROM[addr]; + } + else return NES.DB; + } + } + } +} diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.BoardSystem.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.BoardSystem.cs index 9f178bdcc5..35293dccdf 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.BoardSystem.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.BoardSystem.cs @@ -35,11 +35,15 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES void AddressPPU(int addr); byte ReadWRAM(int addr); byte ReadEXP(int addr); + byte ReadReg2xxx(int addr); + byte PeekReg2xxx(int addr); void WritePRG(int addr, byte value); void WritePPU(int addr, byte value); void WriteWRAM(int addr, byte value); void WriteEXP(int addr, byte value); + void WriteReg2xxx(int addr, byte value); void NESSoftReset(); + void AtVsyncNMI(); byte[] SaveRam { get; } byte[] WRAM { get; set; } byte[] VRAM { get; set; } @@ -83,6 +87,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES public abstract bool Configure(NES.EDetectionOrigin origin); public virtual void ClockPPU() { } public virtual void ClockCPU() { } + public virtual void AtVsyncNMI() { } public CartInfo Cart { get { return NES.cart; } } public NES NES { get; set; } @@ -207,6 +212,21 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES return NES.DB; } + public virtual byte ReadReg2xxx(int addr) + { + return NES.ppu.ReadReg(addr & 7); + } + + public virtual byte PeekReg2xxx(int addr) + { + return NES.ppu.PeekReg(addr & 7); + } + + public virtual void WriteReg2xxx(int addr, byte value) + { + NES.ppu.WriteReg(addr & 7, value); + } + public virtual void WritePPU(int addr, byte value) { if (addr < 0x2000) @@ -338,7 +358,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES void BoardSystemHardReset() { INESBoard newboard; - // fds has a unique activation setup + // FDS and NSF have a unique activation setup if (Board is FDS) { var newfds = new FDS(); @@ -347,6 +367,13 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES newfds.SetDiskImage(oldfds.GetDiskImage()); newboard = newfds; } + else if (Board is NSFBoard) + { + var newnsf = new NSFBoard(); + var oldnsf = Board as NSFBoard; + newnsf.InitNSF(oldnsf.nsf); + newboard = newnsf; + } else { newboard = CreateBoardInstance(Board.GetType()); diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.Core.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.Core.cs index d200b43c61..db8f390432 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.Core.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.Core.cs @@ -17,7 +17,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES int cpu_accumulate; //cpu timekeeper public PPU ppu; public APU apu; - byte[] ram; + public byte[] ram; NESWatch[] sysbus_watch = new NESWatch[65536]; public byte[] CIRAM; //AKA nametables string game_name = string.Empty; //friendly name exposed to user and used as filename base @@ -505,7 +505,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES } else if (addr < 0x4000) { - ret = ppu.PeekReg(addr & 7); + ret = Board.PeekReg2xxx(addr); } else if (addr < 0x4020) { @@ -546,7 +546,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES } else if (addr < 0x4000) { - ret = ppu.ReadReg(addr & 7); + ret = Board.ReadReg2xxx(addr); } else if (addr < 0x4020) { @@ -605,7 +605,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES } else if (addr < 0x4000) { - ppu.WriteReg(addr & 7, value); + Board.WriteReg2xxx(addr,value); } else if (addr < 0x4020) { diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.cs index bca6de5ca1..7ef1f07271 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.cs @@ -346,7 +346,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES public enum EDetectionOrigin { - None, BootGodDB, GameDB, INES, UNIF, FDS + None, BootGodDB, GameDB, INES, UNIF, FDS, NSF } StringWriter LoadReport; @@ -406,6 +406,31 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES hash_sha1_several.Add(hash_sha1); LoadWriteLine("headerless rom hash: {0}", hash_sha1); } + else if(file.Take(5).SequenceEqual(System.Text.Encoding.ASCII.GetBytes("NESM\x1A"))) + { + origin = EDetectionOrigin.NSF; + LoadWriteLine("Loading as NSF"); + var nsf = new NSFFormat(); + nsf.WrapByteArray(file); + + cart = new CartInfo(); + var nsfboard = new NSFBoard(); + nsfboard.Create(this); + nsfboard.ROM = rom; + nsfboard.InitNSF( nsf); + nsfboard.InitialRegisterValues = InitialMapperRegisterValues; + nsfboard.Configure(origin); + nsfboard.WRAM = new byte[cart.wram_size * 1024]; + Board = nsfboard; + Board.PostConfigure(); + + Console.WriteLine("Using NTSC display type for NSF for now"); + _display_type = Common.DisplayType.NTSC; + + HardReset(); + + return; + } else if (file.Take(4).SequenceEqual(System.Text.Encoding.ASCII.GetBytes("FDS\x1A")) || file.Take(4).SequenceEqual(System.Text.Encoding.ASCII.GetBytes("\x01*NI"))) { diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NSFFormat.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NSFFormat.cs new file mode 100644 index 0000000000..fc761dc043 --- /dev/null +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NSFFormat.cs @@ -0,0 +1,84 @@ +using System; +using System.Linq; +using System.IO; +using System.Collections.Generic; + +using BizHawk.Common; +using BizHawk.Common.BufferExtensions; +using BizHawk.Common.IOExtensions; + +using BizHawk.Emulation.Common; + +namespace BizHawk.Emulation.Cores.Nintendo.NES +{ + //http://kevtris.org/nes/nsfspec.txt + //http://en.wikipedia.org/wiki/NES_Sound_Format + public class NSFFormat + { + public byte[] NSFData; + + public byte Version; + public byte TotalSongs; + + /// + /// 1-indexed. 0 is an invalid value, I guess + /// + public byte StartingSong; + + public ushort LoadAddress; + + public ushort InitAddress; + + public ushort PlayAddress; + + public string SongName; + + public string ArtistName; + + public string CopyrightHolder; + + public ushort SpeedNTSC; + + public byte[] BankswitchInitValues = new byte[8]; + + public ushort SpeedPAL; + + public bool IsNTSC; + + public bool IsPAL; + + [Flags] + public enum eExtraChips + { + None = 0, VRC6 = 1, VRC7 = 2, FDS = 4, MMC5 = 8, Namco106 = 16, FME7 = 32 + } + + public eExtraChips ExtraChips; + + public void WrapByteArray(byte[] data) + { + NSFData = data; + + var ms = new MemoryStream(data); + var br = new BinaryReader(ms); + br.BaseStream.Position += 5; + + Version = br.ReadByte(); + TotalSongs = br.ReadByte(); + StartingSong = br.ReadByte(); + LoadAddress = br.ReadUInt16(); + InitAddress = br.ReadUInt16(); + PlayAddress = br.ReadUInt16(); + SongName = br.ReadStringFixedAscii(32); + ArtistName = br.ReadStringFixedAscii(32); + CopyrightHolder = br.ReadStringFixedAscii(32); + SpeedNTSC = br.ReadUInt16(); + br.Read(BankswitchInitValues, 0, 8); + SpeedPAL = br.ReadUInt16(); + byte temp = br.ReadByte(); + if ((temp & 2) != 0) IsNTSC = IsPAL = true; + else if ((temp & 1) != 0) IsPAL = true; else IsNTSC = true; + ExtraChips = (eExtraChips)br.ReadByte(); + } + } +} diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.run.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.run.cs index 2dfc15cc7d..3ccaed7137 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.run.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.run.cs @@ -124,6 +124,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES bool nmi_destiny = reg_2000.vblank_nmi_gen && Reg2002_vblank_active; runppu(3); if (nmi_destiny) TriggerNMI(); + nes.Board.AtVsyncNMI(); runppu(postNMIlines * kLineTime - delay); //this seems to run just before the dummy scanline begins diff --git a/BizHawk.Emulation.Cores/FileID.cs b/BizHawk.Emulation.Cores/FileID.cs index d5699fdfb5..6581ac70ef 100644 --- a/BizHawk.Emulation.Cores/FileID.cs +++ b/BizHawk.Emulation.Cores/FileID.cs @@ -32,7 +32,7 @@ namespace BizHawk.Emulation.Cores Saturn, MegaCD, PCE, SGX, TurboCD, - INES, FDS, UNIF, + INES, FDS, UNIF, NSF, SFC, N64, GB, GBC, GBA, NDS, COL, @@ -256,7 +256,8 @@ namespace BizHawk.Emulation.Cores static class SimpleMagics { public static SimpleMagicRecord INES = new SimpleMagicRecord { Offset = 0, Key = "NES" }; - public static SimpleMagicRecord UNIF = new SimpleMagicRecord { Offset = 0, Key = "UNIF" }; + public static SimpleMagicRecord UNIF = new SimpleMagicRecord { Offset = 0, Key = "UNIF" }; + public static SimpleMagicRecord NSF = new SimpleMagicRecord { Offset = 0, Key = "NESM\x1A" }; public static SimpleMagicRecord FDS_HEADERLESS = new SimpleMagicRecord { Offset = 0, Key = "\x01*NINTENDO-HVC*" }; public static SimpleMagicRecord FDS_HEADER = new SimpleMagicRecord { Offset = 0, Key = "FDS\x1A" };