diff --git a/Assets/gamedb/gamedb_n64.txt b/Assets/gamedb/gamedb_n64.txt
index ecc5fc9b51..4ceb1ec2e0 100644
--- a/Assets/gamedb/gamedb_n64.txt
+++ b/Assets/gamedb/gamedb_n64.txt
@@ -1,3 +1,4 @@
+; these are for the .z64 (no byte swapping) format -- EmuHawk automatically converts .v64 and .n64 to that format before hashing
167C3C433DEC1F1EB921736F7D53FAC8CB45EE31 G 007 - GoldenEye (Europe) N64 Glide_fb_smart=true;Glide64mk2_enable_hacks_for_game=8;Glide64mk2_filtering=1;Glide64mk2_fb_smart=true;Jabo_Clear_Frame=2
2A5DADE32F7FAD6C73C659D2026994632C1B3174 G 007 - GoldenEye (Japan) N64 Glide_fb_smart=true;Glide64mk2_enable_hacks_for_game=8;Glide64mk2_filtering=1;Glide64mk2_fb_smart=true
ABE01E4AEB033B6C0836819F549C791B26CFDE83 G 007 - GoldenEye (USA) N64 RiceEnableHacksForGame=19;RiceZHack=true;RiceFrameBufferOption=1;RiceScreenUpdateSettingHack=4;Glide_fb_smart=true;Glide64mk2_enable_hacks_for_game=8;Glide64mk2_filtering=1;Glide64mk2_fb_smart=true;Jabo_Clear_Frame=2
diff --git a/src/BizHawk.Client.Common/RomGame.cs b/src/BizHawk.Client.Common/RomGame.cs
index e00093f7ce..51be2e127d 100644
--- a/src/BizHawk.Client.Common/RomGame.cs
+++ b/src/BizHawk.Client.Common/RomGame.cs
@@ -96,16 +96,7 @@ namespace BizHawk.Client.Common
RomData = DeInterleaveSMD(RomData);
}
- if (file.Extension == ".z64" || file.Extension == ".n64" || file.Extension == ".v64")
- {
- // Use a simple magic number to detect N64 rom format, then byteswap the ROM to ensure a consistent endianness/order
- RomData = RomData[0] switch
- {
- 0x37 => EndiannessUtils.ByteSwap16(RomData), // V64 format (byte swapped)
- 0x40 => EndiannessUtils.ByteSwap32(RomData), // N64 format (word swapped)
- _ => RomData // Z64 format (no swap), or something unexpected; in either case do nothing
- };
- }
+ 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.Common/lua/LuaHelperLibs/BitLuaLibrary.cs b/src/BizHawk.Client.Common/lua/LuaHelperLibs/BitLuaLibrary.cs
index 76bf02cd12..afd57927b2 100644
--- a/src/BizHawk.Client.Common/lua/LuaHelperLibs/BitLuaLibrary.cs
+++ b/src/BizHawk.Client.Common/lua/LuaHelperLibs/BitLuaLibrary.cs
@@ -1,4 +1,5 @@
using System;
+using System.Buffers.Binary;
using System.ComponentModel;
// ReSharper disable UnusedMember.Global
@@ -99,26 +100,16 @@ namespace BizHawk.Client.Common
[LuaMethodExample("local usbitbyt = bit.byteswap_16( 100 );")]
[LuaMethod("byteswap_16", "Byte swaps 'short', i.e. bit.byteswap_16(0xFF00) would return 0x00FF")]
public static ushort Byteswap16(ushort val)
- {
- return (ushort)((val & 0xFFU) << 8 | (val & 0xFF00U) >> 8);
- }
+ => BinaryPrimitives.ReverseEndianness(val);
[LuaMethodExample("local uibitbyt = bit.byteswap_32( 1000 );")]
[LuaMethod("byteswap_32", "Byte swaps 'dword'")]
public static uint Byteswap32(uint val)
- {
- return (val & 0x000000FFU) << 24 | (val & 0x0000FF00U) << 8 |
- (val & 0x00FF0000U) >> 8 | (val & 0xFF000000U) >> 24;
- }
+ => BinaryPrimitives.ReverseEndianness(val);
[LuaMethodExample("local ulbitbyt = bit.byteswap_64( 10000 );")]
- [LuaMethod("byteswap_64", "Byte swaps 'long'")]
+ [LuaMethod("byteswap_64", "Byte swaps 'long' (NOTE: You may get unexpected results for large numbers! In current Lua engines, all numbers are double-precision floating-point, even if you intended to use integers.)")]
public static ulong Byteswap64(ulong val)
- {
- return (val & 0x00000000000000FFUL) << 56 | (val & 0x000000000000FF00UL) << 40 |
- (val & 0x0000000000FF0000UL) << 24 | (val & 0x00000000FF000000UL) << 8 |
- (val & 0x000000FF00000000UL) >> 8 | (val & 0x0000FF0000000000UL) >> 24 |
- (val & 0x00FF000000000000UL) >> 40 | (val & 0xFF00000000000000UL) >> 56;
- }
+ => BinaryPrimitives.ReverseEndianness(val);
}
}
diff --git a/src/BizHawk.Common/EndiannessUtils.cs b/src/BizHawk.Common/EndiannessUtils.cs
index 2da5288a27..fd64c38d00 100644
--- a/src/BizHawk.Common/EndiannessUtils.cs
+++ b/src/BizHawk.Common/EndiannessUtils.cs
@@ -5,29 +5,10 @@ namespace BizHawk.Common
{
public static unsafe class EndiannessUtils
{
- /// reverses pairs of octets: 0xAABBIIJJPPQQYYZZ <=> 0xBBAAJJIIQQPPZZYY
- public static byte[] ByteSwap16(byte[] a)
- {
- var l = a.Length;
- var copy = new byte[l];
- Array.Copy(a, copy, l);
- MutatingByteSwap16(copy);
- return copy;
- }
-
- /// reverses groups of 4 octets: 0xAABBCCDDWWXXYYZZ <=> 0xDDCCBBAAZZYYXXWW
- public static byte[] ByteSwap32(byte[] a)
- {
- var l = a.Length;
- var copy = new byte[l];
- Array.Copy(a, copy, l);
- MutatingByteSwap32(copy);
- return copy;
- }
-
/// reverses pairs of octets in-place: 0xAABBIIJJPPQQYYZZ <=> 0xBBAAJJIIQQPPZZYY
- public static void MutatingByteSwap16(byte[] a)
+ public static void MutatingByteSwap16(Span a)
{
+#if true //TODO benchmark (both methods work correctly); also there is another method involving shifting the (output copy of the) array over by 1 byte then manually writing every second byte
var l = a.Length;
Debug.Assert(l % 2 == 0);
fixed (byte* p = &a[0]) for (var i = 0; i < l; i += 2)
@@ -36,11 +17,17 @@ namespace BizHawk.Common
p[i] = p[i + 1];
p[i + 1] = b;
}
+#else
+ Debug.Assert(a.Length % 2 == 0);
+ var shorts = MemoryMarshal.Cast(a);
+ for (var i = 0; i < shorts.Length; i++) shorts[i] = BinaryPrimitives.ReverseEndianness(shorts[i]);
+#endif
}
/// reverses groups of 4 octets in-place: 0xAABBCCDDWWXXYYZZ <=> 0xDDCCBBAAZZYYXXWW
- public static void MutatingByteSwap32(byte[] a)
+ public static void MutatingByteSwap32(Span a)
{
+#if true //TODO benchmark (both methods work correctly)
var l = a.Length;
Debug.Assert(l % 4 == 0);
fixed (byte* p = &a[0]) for (var i = 0; i < l; i += 4)
@@ -52,6 +39,11 @@ namespace BizHawk.Common
p[i + 1] = p[i + 2];
p[i + 2] = b;
}
+#else
+ Debug.Assert(a.Length % 4 == 0);
+ var ints = MemoryMarshal.Cast(a);
+ for (var i = 0; i < ints.Length; i++) ints[i] = BinaryPrimitives.ReverseEndianness(ints[i]);
+#endif
}
}
}
diff --git a/src/BizHawk.Emulation.Common/N64RomByteswapper.cs b/src/BizHawk.Emulation.Common/N64RomByteswapper.cs
new file mode 100644
index 0000000000..a802b1a0f0
--- /dev/null
+++ b/src/BizHawk.Emulation.Common/N64RomByteswapper.cs
@@ -0,0 +1,22 @@
+using System;
+
+using BizHawk.Common;
+
+namespace BizHawk.Emulation.Common
+{
+ /// Uses a simple magic number to detect N64 rom format, then byteswaps the ROM to ensure a consistent endianness/order
+ /// http://n64dev.org/romformats.html
+ public static class N64RomByteswapper
+ {
+ /// 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 };
+
+ /// ensures is in the native (.z64) format, mutating it in-place if necessary
+ public static void 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)
+ }
+ }
+}
diff --git a/src/BizHawk.Tests/Common/EndiannessUtils/EndiannessUtilsTests.cs b/src/BizHawk.Tests/Common/EndiannessUtils/EndiannessUtilsTests.cs
index 05038f872f..0d96a33391 100644
--- a/src/BizHawk.Tests/Common/EndiannessUtils/EndiannessUtilsTests.cs
+++ b/src/BizHawk.Tests/Common/EndiannessUtils/EndiannessUtilsTests.cs
@@ -1,4 +1,4 @@
-using System.Linq;
+using System;
using BizHawk.Common;
@@ -14,17 +14,23 @@ namespace BizHawk.Tests.Common
[TestMethod]
public void TestByteSwap16()
{
- var a = new byte[] { 0x23, 0x01, 0x67, 0x45, 0xAB, 0x89, 0xEF, 0xCD };
+ var b = new byte[] { 0x23, 0x01, 0x67, 0x45, 0xAB, 0x89, 0xEF, 0xCD }.AsSpan();
+ var a = b.ToArray().AsSpan();
EndiannessUtils.MutatingByteSwap16(a);
Assert.IsTrue(a.SequenceEqual(expected));
+ EndiannessUtils.MutatingByteSwap16(a);
+ Assert.IsTrue(a.SequenceEqual(b));
}
[TestMethod]
public void TestByteSwap32()
{
- var a = new byte[] { 0x67, 0x45, 0x23, 0x01, 0xEF, 0xCD, 0xAB, 0x89 };
+ var b = new byte[] { 0x67, 0x45, 0x23, 0x01, 0xEF, 0xCD, 0xAB, 0x89 }.AsSpan();
+ var a = b.ToArray().AsSpan();
EndiannessUtils.MutatingByteSwap32(a);
Assert.IsTrue(a.SequenceEqual(expected));
+ EndiannessUtils.MutatingByteSwap32(a);
+ Assert.IsTrue(a.SequenceEqual(b));
}
}
}