From abeaa2a106a8c40333672cc04b8a195e5f463b76 Mon Sep 17 00:00:00 2001 From: YoshiRulz Date: Mon, 5 Dec 2022 04:53:20 +1000 Subject: [PATCH] Be less lazy about N64 header detection in byteswapper fixes 82c3b471a, 9660c16a0 --- src/BizHawk.Client.Common/RomGame.cs | 2 +- .../debug/N64RomByteswapToolForm.cs | 17 ++--- .../N64RomByteswapper.cs | 63 +++++++++++++++---- .../Consoles/Nintendo/Ares64/Ares64.cs | 12 +--- 4 files changed, 57 insertions(+), 37 deletions(-) diff --git a/src/BizHawk.Client.Common/RomGame.cs b/src/BizHawk.Client.Common/RomGame.cs index 51be2e127d..4a52eadc68 100644 --- a/src/BizHawk.Client.Common/RomGame.cs +++ b/src/BizHawk.Client.Common/RomGame.cs @@ -96,7 +96,7 @@ namespace BizHawk.Client.Common RomData = DeInterleaveSMD(RomData); } - if (file.Extension is ".n64" or ".v64" or ".z64") N64RomByteswapper.ToZ64Native(RomData); //TODO don't use file extension for N64 rom detection (yes that means detecting all formats before converting to Z64) + if (file.Extension is ".n64" or ".v64" or ".z64") _ = N64RomByteswapper.ToZ64Native(RomData); //TODO don't use file extension for N64 rom detection (yes that means detecting all formats before converting to Z64) // note: this will be taking several hashes, of a potentially large amount of data.. yikes! GameInfo = Database.GetGameInfo(RomData, file.Name); diff --git a/src/BizHawk.Client.EmuHawk/debug/N64RomByteswapToolForm.cs b/src/BizHawk.Client.EmuHawk/debug/N64RomByteswapToolForm.cs index 8e79324c3f..5154b2ba89 100644 --- a/src/BizHawk.Client.EmuHawk/debug/N64RomByteswapToolForm.cs +++ b/src/BizHawk.Client.EmuHawk/debug/N64RomByteswapToolForm.cs @@ -51,19 +51,12 @@ namespace BizHawk.Client.EmuHawk.ForDebugging try { var rom = File.ReadAllBytes(txtBaseFile.Text); - switch (comboFormats.SelectedIndex) // can't have Action> (System.Buffers.SpanAction isn't suitable) or I'd be able to have a tiny switch expr >:( --yoshi + _ = comboFormats.SelectedIndex switch { - case 0: - N64RomByteswapper.ToN64LittleEndian(rom); - break; - case 1: - N64RomByteswapper.ToV64ByteSwapped(rom); - break; - case 2: - default: - N64RomByteswapper.ToZ64Native(rom); - break; - } + 0 => N64RomByteswapper.ToN64LittleEndian(rom), + 1 => N64RomByteswapper.ToV64ByteSwapped(rom), + _ => N64RomByteswapper.ToZ64Native(rom) + }; File.WriteAllBytes(txtTargetFile.Text, rom); this.ModalMessageBox($"wrote {txtTargetFile.Text}\n{SHA1Checksum.ComputePrefixedHex(rom)}"); } diff --git a/src/BizHawk.Emulation.Common/N64RomByteswapper.cs b/src/BizHawk.Emulation.Common/N64RomByteswapper.cs index dec57ae0e7..1849050443 100644 --- a/src/BizHawk.Emulation.Common/N64RomByteswapper.cs +++ b/src/BizHawk.Emulation.Common/N64RomByteswapper.cs @@ -1,4 +1,5 @@ using System; +using System.Runtime.InteropServices; using BizHawk.Common; @@ -8,31 +9,67 @@ namespace BizHawk.Emulation.Common /// http://n64dev.org/romformats.html public static class N64RomByteswapper { + private static readonly byte[] MAGIC_BYTES_LE = { 0x37, 0x80, 0x40, 0x12 }; + /// not actually magic, just always the same in commercial carts? https://n64brew.dev/wiki/ROM_Header works all the same - private static readonly byte[] MAGIC_BYTES = { 0x80, 0x37, 0x12, 0x40 }; + private static readonly byte[] MAGIC_BYTES_NATIVE = { 0x80, 0x37, 0x12, 0x40 }; + + private static readonly byte[] MAGIC_BYTES_SWAPPED = { 0x40, 0x12, 0x37, 0x80 }; /// ensures is in the rare little-endian (.n64) format, mutating it in-place if necessary - public static void ToN64LittleEndian(Span rom) + /// iff was one of the 3 valid formats + public static bool ToN64LittleEndian(Span rom) { - if (rom[0] == MAGIC_BYTES[0]) EndiannessUtils.MutatingByteSwap32(rom); // native (.z64) - else if (rom[0] == MAGIC_BYTES[1]) EndiannessUtils.MutatingShortSwap32(rom); // byte-swapped (.v64) - // else already rare little-endian .n64 + var romMagicBytes = rom.Slice(start: 0, length: 4); + if (romMagicBytes.SequenceEqual(MAGIC_BYTES_NATIVE)) + { + EndiannessUtils.MutatingByteSwap32(rom); + return true; + } + if (romMagicBytes.SequenceEqual(MAGIC_BYTES_SWAPPED)) + { + EndiannessUtils.MutatingShortSwap32(rom); + return true; + } + return romMagicBytes.SequenceEqual(MAGIC_BYTES_LE); } /// ensures is in the byte-swapped (.v64) format, mutating it in-place if necessary - public static void ToV64ByteSwapped(Span rom) + /// iff was one of the 3 valid formats + public static bool ToV64ByteSwapped(Span rom) { - if (rom[0] == MAGIC_BYTES[0]) EndiannessUtils.MutatingByteSwap16(rom); // native (.z64) - else if (rom[0] == MAGIC_BYTES[3]) EndiannessUtils.MutatingShortSwap32(rom); // rare little-endian .n64 - // else already byte-swapped (.v64) + var romMagicBytes = rom.Slice(start: 0, length: 4); + if (romMagicBytes.SequenceEqual(MAGIC_BYTES_NATIVE)) + { + EndiannessUtils.MutatingByteSwap16(rom); + return true; + } + if (romMagicBytes.SequenceEqual(MAGIC_BYTES_SWAPPED)) return true; + if (romMagicBytes.SequenceEqual(MAGIC_BYTES_LE)) + { + EndiannessUtils.MutatingShortSwap32(rom); + return true; + } + return false; } /// ensures is in the native (.z64) format, mutating it in-place if necessary - public static void ToZ64Native(Span rom) + /// iff was one of the 3 valid formats + public static bool ToZ64Native(Span rom) { - if (rom[0] == MAGIC_BYTES[1]) EndiannessUtils.MutatingByteSwap16(rom); // byte-swapped (.v64) - else if (rom[0] == MAGIC_BYTES[3]) EndiannessUtils.MutatingByteSwap32(rom); // rare little-endian .n64 - // else already native (.z64) + var romMagicBytes = rom.Slice(start: 0, length: 4); + if (romMagicBytes.SequenceEqual(MAGIC_BYTES_NATIVE)) return true; + if (romMagicBytes.SequenceEqual(MAGIC_BYTES_SWAPPED)) + { + EndiannessUtils.MutatingByteSwap16(rom); + return true; + } + if (romMagicBytes.SequenceEqual(MAGIC_BYTES_LE)) + { + EndiannessUtils.MutatingByteSwap32(rom); + return true; + } + return false; } } } diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Ares64/Ares64.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Ares64/Ares64.cs index 5cc68a87df..0c8e97ddb0 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Ares64/Ares64.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Ares64/Ares64.cs @@ -69,17 +69,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Nintendo.Ares64 // TODO: this is normally handled frontend side // except XML files don't go through RomGame // (probably should, but needs refactoring) - foreach (var d in lp.Roms.Select(static r => r.RomData)) - { - // magic N64 rom bytes are 0x80, 0x37, 0x12, 0x40 - // this should hopefully only ever detect N64 roms and not other kinds of files sent through here... - if ((d[1] is 0x80 && d[0] is 0x37 && d[3] is 0x12 && d[2] is 0x40) // .v64 byteswapped - || (d[3] is 0x80 && d[2] is 0x37 && d[1] is 0x12 && d[0] is 0x40) // .n64 little-endian - || (d[0] is 0x80 && d[1] is 0x37 && d[2] is 0x12 && d[3] is 0x40)) // .z64 native - { - N64RomByteswapper.ToZ64Native(d); - } - } + foreach (var r in lp.Roms) _ = N64RomByteswapper.ToZ64Native(r.RomData); // no-op if N64 magic bytes not present var gbRoms = lp.Roms.FindAll(r => IsGBRom(r.FileData)).Select(r => r.FileData).ToArray(); var rom = lp.Roms.Find(r => !gbRoms.Contains(r.FileData) && (char)r.RomData[0x3B] is 'N' or 'C')?.RomData;