diff --git a/src/BizHawk.Common/checksums/SHA1Checksum.cs b/src/BizHawk.Common/checksums/SHA1Checksum.cs
index 88c2b46b02..ba14e9a45c 100644
--- a/src/BizHawk.Common/checksums/SHA1Checksum.cs
+++ b/src/BizHawk.Common/checksums/SHA1Checksum.cs
@@ -7,40 +7,6 @@ using BizHawk.Common.BufferExtensions;
namespace BizHawk.Common
{
- public interface ISHA1
- {
- byte[] ComputeHash(byte[] buffer);
- }
-
- public sealed class NETSHA1 : ISHA1
- {
- private readonly SHA1 _sha1Impl;
-
- public NETSHA1()
- {
- _sha1Impl = SHA1.Create();
- Debug.Assert(_sha1Impl.CanReuseTransform && _sha1Impl.HashSize is SHA1Checksum.EXPECTED_LENGTH);
- }
-
- public byte[] ComputeHash(byte[] buffer)
- => _sha1Impl.ComputeHash(buffer);
- }
-
- public sealed class FastSHA1 : ISHA1
- {
- public unsafe byte[] ComputeHash(byte[] buffer)
- {
- // Set SHA1 start state
- var state = stackalloc uint[] { 0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476, 0xC3D2E1F0 };
- // This will use dedicated SHA instructions, which perform 4x faster than a generic implementation
- LibBizHash.BizCalcSha1((IntPtr)state, buffer, buffer.Length);
- // The copy seems wasteful, but pinning the state down actually has a bigger performance impact
- var ret = new byte[20];
- Marshal.Copy((IntPtr)state, ret, 0, 20);
- return ret;
- }
- }
-
/// uses implementation from BCL
///
///
@@ -70,27 +36,49 @@ namespace BizHawk.Common
return impl.GetHashAndReset();
}
#else
- private static ISHA1? _sha1Impl;
+ private static unsafe byte[] UnmanagedImpl(byte[] buffer)
+ {
+ // Set SHA1 start state
+ var state = stackalloc uint[] { 0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476, 0xC3D2E1F0 };
+ // This will use dedicated SHA instructions, which perform 4x faster than a generic implementation
+ LibBizHash.BizCalcSha1((IntPtr) state, buffer, buffer.Length);
+ // The copy seems wasteful, but pinning the state down actually has a bigger performance impact
+ var ret = new byte[20];
+ Marshal.Copy((IntPtr) state, ret, 0, 20);
+ return ret;
+ }
- private static ISHA1 SHA1Impl
+ private static SHA1? _sha1Impl;
+
+ private static SHA1 SHA1Impl
{
get
{
if (_sha1Impl == null)
{
- _sha1Impl = LibBizHash.BizSupportsShaInstructions()
- ? new FastSHA1()
- : new NETSHA1();
+ _sha1Impl = SHA1.Create();
+ Debug.Assert(_sha1Impl.CanReuseTransform && _sha1Impl.HashSize is EXPECTED_LENGTH);
}
return _sha1Impl;
}
}
public static byte[] Compute(byte[] data)
- => SHA1Impl.ComputeHash(data);
+ => LibBizHash.BizSupportsShaInstructions()
+ ? UnmanagedImpl(data)
+ : SHA1Impl.ComputeHash(data);
public static byte[] ComputeConcat(byte[] dataA, byte[] dataB)
{
+ if (LibBizHash.BizSupportsShaInstructions())
+ {
+ var concat = new byte[dataA.Length + dataB.Length];
+ Array.Copy(sourceArray: dataA, destinationArray: concat, length: dataA.Length);
+ Array.Copy(
+ sourceArray: dataB, sourceIndex: 0,
+ destinationArray: concat, destinationIndex: dataA.Length, length: dataB.Length);
+ return UnmanagedImpl(concat);
+ }
using var impl = IncrementalHash.CreateHash(HashAlgorithmName.SHA1);
impl.AppendData(dataA);
impl.AppendData(dataB);