using System; using System.Diagnostics; using Org.BouncyCastle.Crypto; using Org.BouncyCastle.Crypto.Digests; using Org.BouncyCastle.Math.Raw; using Org.BouncyCastle.Security; using Org.BouncyCastle.Utilities; namespace Org.BouncyCastle.Math.EC.Rfc8032 { using F = Rfc7748.X448Field; /// /// A low-level implementation of the Ed448 and Ed448ph instantiations of the Edwards-Curve Digital Signature /// Algorithm specified in RFC 8032. /// /// /// The implementation uses the "signed mult-comb" algorithm (for scalar multiplication by a fixed point) from /// Mike Hamburg, "Fast and compact elliptic-curve cryptography". Standard /// projective coordinates are used /// for most point arithmetic. /// public static class Ed448 { // x^2 + y^2 == 1 - 39081 * x^2 * y^2 public enum Algorithm { Ed448 = 0, Ed448ph = 1, } public sealed class PublicPoint { internal readonly uint[] m_data; internal PublicPoint(uint[] data) { m_data = data; } } private const int CoordUints = 14; private const int PointBytes = CoordUints * 4 + 1; private const int ScalarUints = 14; private const int ScalarBytes = ScalarUints * 4 + 1; public static readonly int PrehashSize = 64; public static readonly int PublicKeySize = PointBytes; public static readonly int SecretKeySize = 57; public static readonly int SignatureSize = PointBytes + ScalarBytes; // "SigEd448" private static readonly byte[] Dom4Prefix = { 0x53, 0x69, 0x67, 0x45, 0x64, 0x34, 0x34, 0x38 }; private static readonly uint[] P = { 0xFFFFFFFFU, 0xFFFFFFFFU, 0xFFFFFFFFU, 0xFFFFFFFFU, 0xFFFFFFFFU, 0xFFFFFFFFU, 0xFFFFFFFFU, 0xFFFFFFFEU, 0xFFFFFFFFU, 0xFFFFFFFFU, 0xFFFFFFFFU, 0xFFFFFFFFU, 0xFFFFFFFFU, 0xFFFFFFFFU }; private static readonly uint[] B_x = { 0x070CC05EU, 0x026A82BCU, 0x00938E26U, 0x080E18B0U, 0x0511433BU, 0x0F72AB66U, 0x0412AE1AU, 0x0A3D3A46U, 0x0A6DE324U, 0x00F1767EU, 0x04657047U, 0x036DA9E1U, 0x05A622BFU, 0x0ED221D1U, 0x066BED0DU, 0x04F1970CU }; private static readonly uint[] B_y = { 0x0230FA14U, 0x008795BFU, 0x07C8AD98U, 0x0132C4EDU, 0x09C4FDBDU, 0x01CE67C3U, 0x073AD3FFU, 0x005A0C2DU, 0x07789C1EU, 0x0A398408U, 0x0A73736CU, 0x0C7624BEU, 0x003756C9U, 0x02488762U, 0x016EB6BCU, 0x0693F467U }; // 2^225 * B private static readonly uint[] B225_x = { 0x06909EE2U, 0x01D7605CU, 0x0995EC8AU, 0x0FC4D970U, 0x0CF2B361U, 0x02D82E9DU, 0x01225F55U, 0x007F0EF6U, 0x0AEE9C55U, 0x0A240C13U, 0x05627B54U, 0x0D449D1EU, 0x03A44575U, 0x007164A7U, 0x0BD4BD71U, 0x061A15FDU }; private static readonly uint[] B225_y = { 0x0D3A9FE4U, 0x030696B9U, 0x07E7E326U, 0x068308C7U, 0x0CE0B8C8U, 0x03AC222BU, 0x0304DB8EU, 0x083EE319U, 0x05E5DB0BU, 0x0ECA503BU, 0x0B1C6539U, 0x078A8DCEU, 0x02D256BCU, 0x04A8B05EU, 0x0BD9FD57U, 0x0A1C3CB8U }; private const uint C_d = 39081U; //private const int WnafWidth = 6; private const int WnafWidth225 = 5; private const int WnafWidthBase = 7; // ScalarMultBase supports varying blocks, teeth, spacing so long as their product is in range [449, 479] private const int PrecompBlocks = 5; private const int PrecompTeeth = 5; private const int PrecompSpacing = 18; private const int PrecompRange = PrecompBlocks * PrecompTeeth * PrecompSpacing; // 448 < range < 480 private const int PrecompPoints = 1 << (PrecompTeeth - 1); private const int PrecompMask = PrecompPoints - 1; private static readonly object PrecompLock = new object(); private static PointAffine[] PrecompBaseWnaf = null; private static PointAffine[] PrecompBase225Wnaf = null; private static uint[] PrecompBaseComb = null; private struct PointAffine { internal uint[] x, y; } private struct PointProjective { internal uint[] x, y, z; } // Temp space to avoid allocations in point formulae. private struct PointTemp { internal uint[] r0, r1, r2, r3, r4, r5, r6, r7; } private static byte[] CalculateS(byte[] r, byte[] k, byte[] s) { uint[] t = new uint[ScalarUints * 2]; Scalar448.Decode(r, t); uint[] u = new uint[ScalarUints]; Scalar448.Decode(k, u); uint[] v = new uint[ScalarUints]; Scalar448.Decode(s, v); Nat.MulAddTo(ScalarUints, u, v, t); byte[] result = new byte[ScalarBytes * 2]; Codec.Encode32(t, 0, t.Length, result, 0); return Scalar448.Reduce912(result); } private static bool CheckContextVar(byte[] ctx) { return ctx != null && ctx.Length < 256; } private static int CheckPoint(ref PointAffine p) { uint[] t = F.Create(); uint[] u = F.Create(); uint[] v = F.Create(); F.Sqr(p.x, u); F.Sqr(p.y, v); F.Mul(u, v, t); F.Add(u, v, u); F.Mul(t, C_d, t); F.SubOne(t); F.Add(t, u, t); F.Normalize(t); F.Normalize(v); return F.IsZero(t) & ~F.IsZero(v); } private static int CheckPoint(PointProjective p) { uint[] t = F.Create(); uint[] u = F.Create(); uint[] v = F.Create(); uint[] w = F.Create(); F.Sqr(p.x, u); F.Sqr(p.y, v); F.Sqr(p.z, w); F.Mul(u, v, t); F.Add(u, v, u); F.Mul(u, w, u); F.Sqr(w, w); F.Mul(t, C_d, t); F.Sub(t, w, t); F.Add(t, u, t); F.Normalize(t); F.Normalize(v); F.Normalize(w); return F.IsZero(t) & ~F.IsZero(v) & ~F.IsZero(w); } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static bool CheckPointFullVar(ReadOnlySpan p) { if ((p[PointBytes - 1] & 0x7F) != 0x00) return false; uint y13 = Codec.Decode32(p[52..]); uint t0 = y13; uint t1 = y13 ^ P[13]; for (int i = CoordUints - 2; i > 0; --i) { uint yi = Codec.Decode32(p[(i * 4)..]); // Reject non-canonical encodings (i.e. >= P) if (t1 == 0 && yi > P[i]) return false; t0 |= yi; t1 |= yi ^ P[i]; } uint y0 = Codec.Decode32(p); // Reject 0 and 1 if (t0 == 0 && y0 <= 1U) return false; // Reject P - 1 and non-canonical encodings (i.e. >= P) if (t1 == 0 && y0 >= (P[0] - 1U)) return false; return true; } #else private static bool CheckPointFullVar(byte[] p) { if ((p[PointBytes - 1] & 0x7F) != 0x00) return false; uint y13 = Codec.Decode32(p, 52); uint t0 = y13; uint t1 = y13 ^ P[13]; for (int i = CoordUints - 2; i > 0; --i) { uint yi = Codec.Decode32(p, i * 4); // Reject non-canonical encodings (i.e. >= P) if (t1 == 0 && yi > P[i]) return false; t0 |= yi; t1 |= yi ^ P[i]; } uint y0 = Codec.Decode32(p, 0); // Reject 0 and 1 if (t0 == 0 && y0 <= 1U) return false; // Reject P - 1 and non-canonical encodings (i.e. >= P) if (t1 == 0 && y0 >= (P[0] - 1U)) return false; return true; } #endif private static bool CheckPointOrderVar(ref PointAffine p) { Init(out PointProjective r); ScalarMultOrderVar(ref p, ref r); return NormalizeToNeutralElementVar(ref r); } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static bool CheckPointVar(ReadOnlySpan p) { if ((p[PointBytes - 1] & 0x7F) != 0x00) return false; if (Codec.Decode32(p[52..]) < P[13]) return true; int last = p[28] == 0xFF ? 7 : 0; for (int i = CoordUints - 2; i >= last; --i) { if (Codec.Decode32(p[(i * 4)..]) < P[i]) return true; } return false; } #else private static bool CheckPointVar(byte[] p) { if ((p[PointBytes - 1] & 0x7F) != 0x00) return false; if (Codec.Decode32(p, 52) < P[13]) return true; int last = p[28] == 0xFF ? 7 : 0; for (int i = CoordUints - 2; i >= last; --i) { if (Codec.Decode32(p, i * 4) < P[i]) return true; } return false; } #endif private static byte[] Copy(byte[] buf, int off, int len) { byte[] result = new byte[len]; Array.Copy(buf, off, result, 0, len); return result; } public static IXof CreatePrehash() { return CreateXof(); } private static IXof CreateXof() { return new ShakeDigest(256); } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static bool DecodePointVar(ReadOnlySpan p, bool negate, ref PointAffine r) #else private static bool DecodePointVar(byte[] p, bool negate, ref PointAffine r) #endif { int x_0 = (p[PointBytes - 1] & 0x80) >> 7; F.Decode(p, r.y); uint[] u = F.Create(); uint[] v = F.Create(); F.Sqr(r.y, u); F.Mul(u, C_d, v); F.Negate(u, u); F.AddOne(u); F.AddOne(v); if (!F.SqrtRatioVar(u, v, r.x)) return false; F.Normalize(r.x); if (x_0 == 1 && F.IsZeroVar(r.x)) return false; if (negate ^ (x_0 != (r.x[0] & 1))) { F.Negate(r.x, r.x); F.Normalize(r.x); } return true; } private static void Dom4(IXof d, byte phflag, byte[] ctx) { Debug.Assert(ctx != null); int n = Dom4Prefix.Length; #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER Span t = stackalloc byte[n + 2 + ctx.Length]; Dom4Prefix.CopyTo(t); t[n] = phflag; t[n + 1] = (byte)ctx.Length; ctx.CopyTo(t.Slice(n + 2)); d.BlockUpdate(t); #else byte[] t = new byte[n + 2 + ctx.Length]; Dom4Prefix.CopyTo(t, 0); t[n] = phflag; t[n + 1] = (byte)ctx.Length; ctx.CopyTo(t, n + 2); d.BlockUpdate(t, 0, t.Length); #endif } private static void EncodePoint(ref PointAffine p, byte[] r, int rOff) { F.Encode(p.y, r, rOff); r[rOff + PointBytes - 1] = (byte)((p.x[0] & 1) << 7); } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static void EncodePoint(ref PointAffine p, Span r) { F.Encode(p.y, r); r[PointBytes - 1] = (byte)((p.x[0] & 1) << 7); } #endif public static void EncodePublicPoint(PublicPoint publicPoint, byte[] pk, int pkOff) { F.Encode(publicPoint.m_data, F.Size, pk, pkOff); pk[pkOff + PointBytes - 1] = (byte)((publicPoint.m_data[0] & 1) << 7); } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER public static void EncodePublicPoint(PublicPoint publicPoint, Span pk) { F.Encode(publicPoint.m_data.AsSpan(F.Size), pk); pk[PointBytes - 1] = (byte)((publicPoint.m_data[0] & 1) << 7); } #endif private static int EncodeResult(ref PointProjective p, byte[] r, int rOff) { #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER return EncodeResult(ref p, r.AsSpan(rOff)); #else Init(out PointAffine q); NormalizeToAffine(ref p, ref q); int result = CheckPoint(ref q); EncodePoint(ref q, r, rOff); return result; #endif } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static int EncodeResult(ref PointProjective p, Span r) { Init(out PointAffine q); NormalizeToAffine(ref p, ref q); int result = CheckPoint(ref q); EncodePoint(ref q, r); return result; } #endif private static PublicPoint ExportPoint(ref PointAffine p) { uint[] data = new uint[F.Size * 2]; F.Copy(p.x, 0, data, 0); F.Copy(p.y, 0, data, F.Size); return new PublicPoint(data); } public static void GeneratePrivateKey(SecureRandom random, byte[] k) { if (k.Length != SecretKeySize) throw new ArgumentException(nameof(k)); random.NextBytes(k); } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER public static void GeneratePrivateKey(SecureRandom random, Span k) { if (k.Length != SecretKeySize) throw new ArgumentException(nameof(k)); random.NextBytes(k); } #endif public static void GeneratePublicKey(byte[] sk, int skOff, byte[] pk, int pkOff) { #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER GeneratePublicKey(sk.AsSpan(skOff), pk.AsSpan(pkOff)); #else IXof d = CreateXof(); byte[] h = new byte[ScalarBytes * 2]; d.BlockUpdate(sk, skOff, SecretKeySize); d.OutputFinal(h, 0, h.Length); byte[] s = new byte[ScalarBytes]; PruneScalar(h, 0, s); ScalarMultBaseEncoded(s, pk, pkOff); #endif } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER public static void GeneratePublicKey(ReadOnlySpan sk, Span pk) { IXof d = CreateXof(); Span h = stackalloc byte[ScalarBytes * 2]; d.BlockUpdate(sk[..SecretKeySize]); d.OutputFinal(h); Span s = stackalloc byte[ScalarBytes]; PruneScalar(h, s); ScalarMultBaseEncoded(s, pk); } #endif public static PublicPoint GeneratePublicKey(byte[] sk, int skOff) { #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER return GeneratePublicKey(sk.AsSpan(skOff)); #else IXof d = CreateXof(); byte[] h = new byte[ScalarBytes * 2]; d.BlockUpdate(sk, skOff, SecretKeySize); d.OutputFinal(h, 0, h.Length); byte[] s = new byte[ScalarBytes]; PruneScalar(h, 0, s); Init(out PointProjective p); ScalarMultBase(s, ref p); Init(out PointAffine q); NormalizeToAffine(ref p, ref q); if (0 == CheckPoint(ref q)) throw new InvalidOperationException(); return ExportPoint(ref q); #endif } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER public static PublicPoint GeneratePublicKey(ReadOnlySpan sk) { IXof d = CreateXof(); Span h = stackalloc byte[ScalarBytes * 2]; d.BlockUpdate(sk[..SecretKeySize]); d.OutputFinal(h); Span s = stackalloc byte[ScalarBytes]; PruneScalar(h, s); Init(out PointProjective p); ScalarMultBase(s, ref p); Init(out PointAffine q); NormalizeToAffine(ref p, ref q); if (0 == CheckPoint(ref q)) throw new InvalidOperationException(); return ExportPoint(ref q); } #endif #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static uint GetWindow4(ReadOnlySpan x, int n) #else private static uint GetWindow4(uint[] x, int n) #endif { int w = (int)((uint)n >> 3), b = (n & 7) << 2; return (x[w] >> b) & 15U; } private static void ImplSign(IXof d, byte[] h, byte[] s, byte[] pk, int pkOff, byte[] ctx, byte phflag, byte[] m, int mOff, int mLen, byte[] sig, int sigOff) { Dom4(d, phflag, ctx); d.BlockUpdate(h, ScalarBytes, ScalarBytes); d.BlockUpdate(m, mOff, mLen); d.OutputFinal(h, 0, h.Length); byte[] r = Scalar448.Reduce912(h); byte[] R = new byte[PointBytes]; ScalarMultBaseEncoded(r, R, 0); Dom4(d, phflag, ctx); d.BlockUpdate(R, 0, PointBytes); d.BlockUpdate(pk, pkOff, PointBytes); d.BlockUpdate(m, mOff, mLen); d.OutputFinal(h, 0, h.Length); byte[] k = Scalar448.Reduce912(h); byte[] S = CalculateS(r, k, s); Array.Copy(R, 0, sig, sigOff, PointBytes); Array.Copy(S, 0, sig, sigOff + PointBytes, ScalarBytes); } private static void ImplSign(byte[] sk, int skOff, byte[] ctx, byte phflag, byte[] m, int mOff, int mLen, byte[] sig, int sigOff) { if (!CheckContextVar(ctx)) throw new ArgumentException("ctx"); IXof d = CreateXof(); byte[] h = new byte[ScalarBytes * 2]; d.BlockUpdate(sk, skOff, SecretKeySize); d.OutputFinal(h, 0, h.Length); byte[] s = new byte[ScalarBytes]; PruneScalar(h, 0, s); byte[] pk = new byte[PointBytes]; ScalarMultBaseEncoded(s, pk, 0); ImplSign(d, h, s, pk, 0, ctx, phflag, m, mOff, mLen, sig, sigOff); } private static void ImplSign(byte[] sk, int skOff, byte[] pk, int pkOff, byte[] ctx, byte phflag, byte[] m, int mOff, int mLen, byte[] sig, int sigOff) { if (!CheckContextVar(ctx)) throw new ArgumentException("ctx"); IXof d = CreateXof(); byte[] h = new byte[ScalarBytes * 2]; d.BlockUpdate(sk, skOff, SecretKeySize); d.OutputFinal(h, 0, h.Length); byte[] s = new byte[ScalarBytes]; PruneScalar(h, 0, s); ImplSign(d, h, s, pk, pkOff, ctx, phflag, m, mOff, mLen, sig, sigOff); } private static bool ImplVerify(byte[] sig, int sigOff, byte[] pk, int pkOff, byte[] ctx, byte phflag, byte[] m, int mOff, int mLen) { if (!CheckContextVar(ctx)) throw new ArgumentException("ctx"); #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER Span signature = stackalloc byte[SignatureSize]; signature.CopyFrom(sig.AsSpan(sigOff, SignatureSize)); var R = signature[..PointBytes]; var S = signature[PointBytes..]; Span A = stackalloc byte[PublicKeySize]; A.CopyFrom(pk.AsSpan(pkOff)); if (!CheckPointVar(R)) return false; Span nS = stackalloc uint[ScalarUints]; if (!Scalar448.CheckVar(S, nS)) return false; if (!CheckPointFullVar(A)) return false; Init(out PointAffine pR); if (!DecodePointVar(R, true, ref pR)) return false; Init(out PointAffine pA); if (!DecodePointVar(A, true, ref pA)) return false; IXof d = CreateXof(); Span h = stackalloc byte[ScalarBytes * 2]; Dom4(d, phflag, ctx); d.BlockUpdate(R); d.BlockUpdate(A); d.BlockUpdate(m.AsSpan(mOff, mLen)); d.OutputFinal(h); Span k = stackalloc byte[ScalarBytes]; Scalar448.Reduce912(h, k); Span nA = stackalloc uint[ScalarUints]; Scalar448.Decode(k, nA); Span v0 = stackalloc uint[8]; Span v1 = stackalloc uint[8]; #else byte[] R = Copy(sig, sigOff, PointBytes); byte[] S = Copy(sig, sigOff + PointBytes, ScalarBytes); byte[] A = Copy(pk, pkOff, PublicKeySize); if (!CheckPointVar(R)) return false; uint[] nS = new uint[ScalarUints]; if (!Scalar448.CheckVar(S, nS)) return false; if (!CheckPointFullVar(A)) return false; Init(out PointAffine pR); if (!DecodePointVar(R, true, ref pR)) return false; Init(out PointAffine pA); if (!DecodePointVar(A, true, ref pA)) return false; IXof d = CreateXof(); byte[] h = new byte[ScalarBytes * 2]; Dom4(d, phflag, ctx); d.BlockUpdate(R, 0, PointBytes); d.BlockUpdate(A, 0, PointBytes); d.BlockUpdate(m, mOff, mLen); d.OutputFinal(h, 0, h.Length); byte[] k = Scalar448.Reduce912(h); uint[] nA = new uint[ScalarUints]; Scalar448.Decode(k, nA); uint[] v0 = new uint[8]; uint[] v1 = new uint[8]; #endif Scalar448.ReduceBasisVar(nA, v0, v1); Scalar448.Multiply225Var(nS, v1, nS); Init(out PointProjective pZ); ScalarMultStraus225Var(nS, v0, ref pA, v1, ref pR, ref pZ); return NormalizeToNeutralElementVar(ref pZ); } private static bool ImplVerify(byte[] sig, int sigOff, PublicPoint publicPoint, byte[] ctx, byte phflag, byte[] m, int mOff, int mLen) { if (!CheckContextVar(ctx)) throw new ArgumentException("ctx"); #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER Span signature = stackalloc byte[SignatureSize]; signature.CopyFrom(sig.AsSpan(sigOff, SignatureSize)); var R = signature[..PointBytes]; var S = signature[PointBytes..]; if (!CheckPointVar(R)) return false; Span nS = stackalloc uint[ScalarUints]; if (!Scalar448.CheckVar(S, nS)) return false; Init(out PointAffine pR); if (!DecodePointVar(R, true, ref pR)) return false; Init(out PointAffine pA); F.Negate(publicPoint.m_data, pA.x); F.Copy(publicPoint.m_data.AsSpan(F.Size), pA.y); Span A = stackalloc byte[PublicKeySize]; EncodePublicPoint(publicPoint, A); IXof d = CreateXof(); Span h = stackalloc byte[ScalarBytes * 2]; Dom4(d, phflag, ctx); d.BlockUpdate(R); d.BlockUpdate(A); d.BlockUpdate(m.AsSpan(mOff, mLen)); d.OutputFinal(h); Span k = stackalloc byte[ScalarBytes]; Scalar448.Reduce912(h, k); Span nA = stackalloc uint[ScalarUints]; Scalar448.Decode(k, nA); Span v0 = stackalloc uint[8]; Span v1 = stackalloc uint[8]; #else byte[] R = Copy(sig, sigOff, PointBytes); byte[] S = Copy(sig, sigOff + PointBytes, ScalarBytes); if (!CheckPointVar(R)) return false; uint[] nS = new uint[ScalarUints]; if (!Scalar448.CheckVar(S, nS)) return false; Init(out PointAffine pR); if (!DecodePointVar(R, true, ref pR)) return false; Init(out PointAffine pA); F.Negate(publicPoint.m_data, pA.x); F.Copy(publicPoint.m_data, F.Size, pA.y, 0); byte[] A = new byte[PublicKeySize]; EncodePublicPoint(publicPoint, A, 0); IXof d = CreateXof(); byte[] h = new byte[ScalarBytes * 2]; Dom4(d, phflag, ctx); d.BlockUpdate(R, 0, PointBytes); d.BlockUpdate(A, 0, PointBytes); d.BlockUpdate(m, mOff, mLen); d.OutputFinal(h, 0, h.Length); byte[] k = Scalar448.Reduce912(h); uint[] nA = new uint[ScalarUints]; Scalar448.Decode(k, nA); uint[] v0 = new uint[8]; uint[] v1 = new uint[8]; #endif Scalar448.ReduceBasisVar(nA, v0, v1); Scalar448.Multiply225Var(nS, v1, nS); Init(out PointProjective pZ); ScalarMultStraus225Var(nS, v0, ref pA, v1, ref pR, ref pZ); return NormalizeToNeutralElementVar(ref pZ); } private static void Init(out PointAffine r) { r.x = F.Create(); r.y = F.Create(); } private static void Init(out PointProjective r) { r.x = F.Create(); r.y = F.Create(); r.z = F.Create(); } private static void Init(out PointTemp r) { r.r0 = F.Create(); r.r1 = F.Create(); r.r2 = F.Create(); r.r3 = F.Create(); r.r4 = F.Create(); r.r5 = F.Create(); r.r6 = F.Create(); r.r7 = F.Create(); } private static void InvertZs(PointProjective[] points) { int count = points.Length; uint[] cs = F.CreateTable(count); uint[] u = F.Create(); F.Copy(points[0].z, 0, u, 0); F.Copy(u, 0, cs, 0); int i = 0; while (++i < count) { F.Mul(u, points[i].z, u); F.Copy(u, 0, cs, i * F.Size); } F.InvVar(u, u); --i; uint[] t = F.Create(); while (i > 0) { int j = i--; F.Copy(cs, i * F.Size, t, 0); F.Mul(t, u, t); F.Mul(u, points[j].z, u); F.Copy(t, 0, points[j].z, 0); } F.Copy(u, 0, points[0].z, 0); } private static void NormalizeToAffine(ref PointProjective p, ref PointAffine r) { F.Inv(p.z, r.y); F.Mul(r.y, p.x, r.x); F.Mul(r.y, p.y, r.y); F.Normalize(r.x); F.Normalize(r.y); } private static bool NormalizeToNeutralElementVar(ref PointProjective p) { F.Normalize(p.x); F.Normalize(p.y); F.Normalize(p.z); return F.IsZeroVar(p.x) && !F.IsZeroVar(p.y) && F.AreEqualVar(p.y, p.z); } private static void PointAdd(ref PointAffine p, ref PointProjective r, ref PointTemp t) { uint[] b = t.r1; uint[] c = t.r2; uint[] d = t.r3; uint[] e = t.r4; uint[] f = t.r5; uint[] g = t.r6; uint[] h = t.r7; F.Sqr(r.z, b); F.Mul(p.x, r.x, c); F.Mul(p.y, r.y, d); F.Mul(c, d, e); F.Mul(e, C_d, e); //F.Apm(b, e, f, g); F.Add(b, e, f); F.Sub(b, e, g); F.Add(p.y, p.x, h); F.Add(r.y, r.x, e); F.Mul(h, e, h); //F.Apm(d, c, b, e); F.Add(d, c, b); F.Sub(d, c, e); F.Carry(b); F.Sub(h, b, h); F.Mul(h, r.z, h); F.Mul(e, r.z, e); F.Mul(f, h, r.x); F.Mul(e, g, r.y); F.Mul(f, g, r.z); } private static void PointAdd(ref PointProjective p, ref PointProjective r, ref PointTemp t) { uint[] a = t.r0; uint[] b = t.r1; uint[] c = t.r2; uint[] d = t.r3; uint[] e = t.r4; uint[] f = t.r5; uint[] g = t.r6; uint[] h = t.r7; F.Mul(p.z, r.z, a); F.Sqr(a, b); F.Mul(p.x, r.x, c); F.Mul(p.y, r.y, d); F.Mul(c, d, e); F.Mul(e, C_d, e); //F.Apm(b, e, f, g); F.Add(b, e, f); F.Sub(b, e, g); F.Add(p.y, p.x, h); F.Add(r.y, r.x, e); F.Mul(h, e, h); //F.Apm(d, c, b, e); F.Add(d, c, b); F.Sub(d, c, e); F.Carry(b); F.Sub(h, b, h); F.Mul(h, a, h); F.Mul(e, a, e); F.Mul(f, h, r.x); F.Mul(e, g, r.y); F.Mul(f, g, r.z); } private static void PointAddVar(bool negate, ref PointAffine p, ref PointProjective r, ref PointTemp t) { uint[] b = t.r1; uint[] c = t.r2; uint[] d = t.r3; uint[] e = t.r4; uint[] f = t.r5; uint[] g = t.r6; uint[] h = t.r7; uint[] nb, ne, nf, ng; if (negate) { nb = e; ne = b; nf = g; ng = f; F.Sub(p.y, p.x, h); } else { nb = b; ne = e; nf = f; ng = g; F.Add(p.y, p.x, h); } F.Sqr(r.z, b); F.Mul(p.x, r.x, c); F.Mul(p.y, r.y, d); F.Mul(c, d, e); F.Mul(e, C_d, e); //F.Apm(b, e, nf, ng); F.Add(b, e, nf); F.Sub(b, e, ng); F.Add(r.y, r.x, e); F.Mul(h, e, h); //F.Apm(d, c, nb, ne); F.Add(d, c, nb); F.Sub(d, c, ne); F.Carry(nb); F.Sub(h, b, h); F.Mul(h, r.z, h); F.Mul(e, r.z, e); F.Mul(f, h, r.x); F.Mul(e, g, r.y); F.Mul(f, g, r.z); } private static void PointAddVar(bool negate, ref PointProjective p, ref PointProjective r, ref PointTemp t) { uint[] a = t.r0; uint[] b = t.r1; uint[] c = t.r2; uint[] d = t.r3; uint[] e = t.r4; uint[] f = t.r5; uint[] g = t.r6; uint[] h = t.r7; uint[] nb, ne, nf, ng; if (negate) { nb = e; ne = b; nf = g; ng = f; F.Sub(p.y, p.x, h); } else { nb = b; ne = e; nf = f; ng = g; F.Add(p.y, p.x, h); } F.Mul(p.z, r.z, a); F.Sqr(a, b); F.Mul(p.x, r.x, c); F.Mul(p.y, r.y, d); F.Mul(c, d, e); F.Mul(e, C_d, e); //F.Apm(b, e, nf, ng); F.Add(b, e, nf); F.Sub(b, e, ng); F.Add(r.y, r.x, e); F.Mul(h, e, h); //F.Apm(d, c, nb, ne); F.Add(d, c, nb); F.Sub(d, c, ne); F.Carry(nb); F.Sub(h, b, h); F.Mul(h, a, h); F.Mul(e, a, e); F.Mul(f, h, r.x); F.Mul(e, g, r.y); F.Mul(f, g, r.z); } private static void PointCopy(ref PointAffine p, ref PointProjective r) { F.Copy(p.x, 0, r.x, 0); F.Copy(p.y, 0, r.y, 0); F.One(r.z); } private static void PointCopy(ref PointProjective p, ref PointProjective r) { F.Copy(p.x, 0, r.x, 0); F.Copy(p.y, 0, r.y, 0); F.Copy(p.z, 0, r.z, 0); } private static void PointDouble(ref PointProjective r, ref PointTemp t) { uint[] b = t.r1; uint[] c = t.r2; uint[] d = t.r3; uint[] e = t.r4; uint[] h = t.r7; uint[] j = t.r0; F.Add(r.x, r.y, b); F.Sqr(b, b); F.Sqr(r.x, c); F.Sqr(r.y, d); F.Add(c, d, e); F.Carry(e); F.Sqr(r.z, h); F.Add(h, h, h); F.Carry(h); F.Sub(e, h, j); F.Sub(b, e, b); F.Sub(c, d, c); F.Mul(b, j, r.x); F.Mul(e, c, r.y); F.Mul(e, j, r.z); } private static void PointLookup(int block, int index, ref PointAffine p) { Debug.Assert(0 <= block && block < PrecompBlocks); Debug.Assert(0 <= index && index < PrecompPoints); int off = block * PrecompPoints * 2 * F.Size; for (int i = 0; i < PrecompPoints; ++i) { int cond = ((i ^ index) - 1) >> 31; F.CMov(cond, PrecompBaseComb, off, p.x, 0); off += F.Size; F.CMov(cond, PrecompBaseComb, off, p.y, 0); off += F.Size; } } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static void PointLookup(ReadOnlySpan x, int n, ReadOnlySpan table, ref PointProjective r) { // TODO This method is currently hardcoded to 4-bit windows and 8 precomputed points uint w = GetWindow4(x, n); int sign = (int)(w >> (4 - 1)) ^ 1; int abs = ((int)w ^ -sign) & 7; Debug.Assert(sign == 0 || sign == 1); Debug.Assert(0 <= abs && abs < 8); for (int i = 0; i < 8; ++i) { int cond = ((i ^ abs) - 1) >> 31; F.CMov(cond, table, r.x); table = table[F.Size..]; F.CMov(cond, table, r.y); table = table[F.Size..]; F.CMov(cond, table, r.z); table = table[F.Size..]; } F.CNegate(sign, r.x); } #else private static void PointLookup(uint[] x, int n, uint[] table, ref PointProjective r) { // TODO This method is currently hardcoded to 4-bit windows and 8 precomputed points uint w = GetWindow4(x, n); int sign = (int)(w >> (4 - 1)) ^ 1; int abs = ((int)w ^ -sign) & 7; Debug.Assert(sign == 0 || sign == 1); Debug.Assert(0 <= abs && abs < 8); for (int i = 0, off = 0; i < 8; ++i) { int cond = ((i ^ abs) - 1) >> 31; F.CMov(cond, table, off, r.x, 0); off += F.Size; F.CMov(cond, table, off, r.y, 0); off += F.Size; F.CMov(cond, table, off, r.z, 0); off += F.Size; } F.CNegate(sign, r.x); } #endif private static void PointLookup15(uint[] table, ref PointProjective r) { int off = F.Size * 3 * 7; F.Copy(table, off, r.x, 0); off += F.Size; F.Copy(table, off, r.y, 0); off += F.Size; F.Copy(table, off, r.z, 0); } private static uint[] PointPrecompute(ref PointProjective p, int count, ref PointTemp t) { Debug.Assert(count > 0); Init(out PointProjective q); PointCopy(ref p, ref q); Init(out PointProjective d); PointCopy(ref p, ref d); PointDouble(ref d, ref t); uint[] table = F.CreateTable(count * 3); int off = 0; int i = 0; for (;;) { F.Copy(q.x, 0, table, off); off += F.Size; F.Copy(q.y, 0, table, off); off += F.Size; F.Copy(q.z, 0, table, off); off += F.Size; if (++i == count) break; PointAdd(ref d, ref q, ref t); } return table; } private static void PointPrecompute(ref PointAffine p, PointProjective[] points, int pointsOff, int pointsLen, ref PointTemp t) { Debug.Assert(pointsLen > 0); Init(out PointProjective d); PointCopy(ref p, ref d); PointDouble(ref d, ref t); Init(out points[pointsOff]); PointCopy(ref p, ref points[pointsOff]); for (int i = 1; i < pointsLen; ++i) { Init(out points[pointsOff + i]); PointCopy(ref points[pointsOff + i - 1], ref points[pointsOff + i]); PointAdd(ref d, ref points[pointsOff + i], ref t); } } private static void PointSetNeutral(ref PointProjective p) { F.Zero(p.x); F.One(p.y); F.One(p.z); } public static void Precompute() { lock (PrecompLock) { if (PrecompBaseComb != null) return; Debug.Assert(PrecompRange > 448); Debug.Assert(PrecompRange < 480); int wnafPoints = 1 << (WnafWidthBase - 2); int combPoints = PrecompBlocks * PrecompPoints; int totalPoints = wnafPoints * 2 + combPoints; PointProjective[] points = new PointProjective[totalPoints]; Init(out PointTemp t); Init(out PointAffine B); F.Copy(B_x, 0, B.x, 0); F.Copy(B_y, 0, B.y, 0); PointPrecompute(ref B, points, 0, wnafPoints, ref t); Init(out PointAffine B225); F.Copy(B225_x, 0, B225.x, 0); F.Copy(B225_y, 0, B225.y, 0); PointPrecompute(ref B225, points, wnafPoints, wnafPoints, ref t); Init(out PointProjective p); PointCopy(ref B, ref p); int pointsIndex = wnafPoints * 2; PointProjective[] toothPowers = new PointProjective[PrecompTeeth]; for (int tooth = 0; tooth < PrecompTeeth; ++tooth) { Init(out toothPowers[tooth]); } for (int block = 0; block < PrecompBlocks; ++block) { ref PointProjective sum = ref points[pointsIndex++]; Init(out sum); for (int tooth = 0; tooth < PrecompTeeth; ++tooth) { if (tooth == 0) { PointCopy(ref p, ref sum); } else { PointAdd(ref p, ref sum, ref t); } PointDouble(ref p, ref t); PointCopy(ref p, ref toothPowers[tooth]); if (block + tooth != PrecompBlocks + PrecompTeeth - 2) { for (int spacing = 1; spacing < PrecompSpacing; ++spacing) { PointDouble(ref p, ref t); } } } F.Negate(sum.x, sum.x); for (int tooth = 0; tooth < (PrecompTeeth - 1); ++tooth) { int size = 1 << tooth; for (int j = 0; j < size; ++j, ++pointsIndex) { Init(out points[pointsIndex]); PointCopy(ref points[pointsIndex - size], ref points[pointsIndex]); PointAdd(ref toothPowers[tooth], ref points[pointsIndex], ref t); } } } Debug.Assert(pointsIndex == totalPoints); InvertZs(points); PrecompBaseWnaf = new PointAffine[wnafPoints]; for (int i = 0; i < wnafPoints; ++i) { ref PointProjective q = ref points[i]; ref PointAffine r = ref PrecompBaseWnaf[i]; Init(out r); F.Mul(q.x, q.z, r.x); F.Normalize(r.x); F.Mul(q.y, q.z, r.y); F.Normalize(r.y); } PrecompBase225Wnaf = new PointAffine[wnafPoints]; for (int i = 0; i < wnafPoints; ++i) { ref PointProjective q = ref points[wnafPoints + i]; ref PointAffine r = ref PrecompBase225Wnaf[i]; Init(out r); F.Mul(q.x, q.z, r.x); F.Normalize(r.x); F.Mul(q.y, q.z, r.y); F.Normalize(r.y); } PrecompBaseComb = F.CreateTable(combPoints * 2); int off = 0; for (int i = wnafPoints * 2; i < totalPoints; ++i) { ref PointProjective q = ref points[i]; F.Mul(q.x, q.z, q.x); F.Normalize(q.x); F.Mul(q.y, q.z, q.y); F.Normalize(q.y); F.Copy(q.x, 0, PrecompBaseComb, off); off += F.Size; F.Copy(q.y, 0, PrecompBaseComb, off); off += F.Size; } Debug.Assert(off == PrecompBaseComb.Length); } } private static void PruneScalar(byte[] n, int nOff, byte[] r) { Array.Copy(n, nOff, r, 0, ScalarBytes - 1); r[0] &= 0xFC; r[ScalarBytes - 2] |= 0x80; r[ScalarBytes - 1] = 0x00; } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static void PruneScalar(ReadOnlySpan n, Span r) { n[..(ScalarBytes - 1)].CopyTo(r); r[0] &= 0xFC; r[ScalarBytes - 2] |= 0x80; r[ScalarBytes - 1] = 0x00; } #endif #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static void ScalarMult(ReadOnlySpan k, ref PointProjective p, ref PointProjective r) #else private static void ScalarMult(byte[] k, ref PointProjective p, ref PointProjective r) #endif { #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER Span n = stackalloc uint[ScalarUints + 1]; #else uint[] n = new uint[ScalarUints + 1]; #endif Scalar448.Decode(k, n); Scalar448.ToSignedDigits(449, n, n); // NOTE: Bit 448 is handled explicitly by an initial addition Debug.Assert(n[ScalarUints] == 1U); Init(out PointProjective q); Init(out PointTemp t); uint[] table = PointPrecompute(ref p, 8, ref t); // Replace first 4 doublings (2^4 * P) with 1 addition (P + 15 * P) PointLookup15(table, ref r); PointAdd(ref p, ref r, ref t); int w = 111; for (;;) { PointLookup(n, w, table, ref q); PointAdd(ref q, ref r, ref t); if (--w < 0) break; for (int i = 0; i < 4; ++i) { PointDouble(ref r, ref t); } } } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static void ScalarMultBase(ReadOnlySpan k, ref PointProjective r) #else private static void ScalarMultBase(byte[] k, ref PointProjective r) #endif { // Equivalent (but much slower) //Init(out PointProjective p); //F.Copy(B_x, 0, p.x, 0); //F.Copy(B_y, 0, p.y, 0); //F.One(p.z); //ScalarMult(k, ref p, ref r); Precompute(); #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER Span n = stackalloc uint[ScalarUints + 1]; #else uint[] n = new uint[ScalarUints + 1]; #endif Scalar448.Decode(k, n); Scalar448.ToSignedDigits(PrecompRange, n, n); Init(out PointAffine p); Init(out PointTemp t); PointSetNeutral(ref r); int cOff = PrecompSpacing - 1; for (;;) { int tPos = cOff; for (int block = 0; block < PrecompBlocks; ++block) { uint w = 0; for (int tooth = 0; tooth < PrecompTeeth; ++tooth) { uint tBit = n[tPos >> 5] >> (tPos & 0x1F); w &= ~(1U << tooth); w ^= (tBit << tooth); tPos += PrecompSpacing; } int sign = (int)(w >> (PrecompTeeth - 1)) & 1; int abs = ((int)w ^ -sign) & PrecompMask; Debug.Assert(sign == 0 || sign == 1); Debug.Assert(0 <= abs && abs < PrecompPoints); PointLookup(block, abs, ref p); F.CNegate(sign, p.x); PointAdd(ref p, ref r, ref t); } if (--cOff < 0) break; PointDouble(ref r, ref t); } } private static void ScalarMultBaseEncoded(byte[] k, byte[] r, int rOff) { #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER ScalarMultBaseEncoded(k.AsSpan(), r.AsSpan(rOff)); #else Init(out PointProjective p); ScalarMultBase(k, ref p); if (0 == EncodeResult(ref p, r, rOff)) throw new InvalidOperationException(); #endif } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static void ScalarMultBaseEncoded(ReadOnlySpan k, Span r) { Init(out PointProjective p); ScalarMultBase(k, ref p); if (0 == EncodeResult(ref p, r)) throw new InvalidOperationException(); } #endif internal static void ScalarMultBaseXY(byte[] k, int kOff, uint[] x, uint[] y) { #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER ScalarMultBaseXY(k.AsSpan(kOff), x.AsSpan(), y.AsSpan()); #else byte[] n = new byte[ScalarBytes]; PruneScalar(k, kOff, n); Init(out PointProjective p); ScalarMultBase(n, ref p); if (0 == CheckPoint(p)) throw new InvalidOperationException(); F.Copy(p.x, 0, x, 0); F.Copy(p.y, 0, y, 0); #endif } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER internal static void ScalarMultBaseXY(ReadOnlySpan k, Span x, Span y) { Span n = stackalloc byte[ScalarBytes]; PruneScalar(k, n); Init(out PointProjective p); ScalarMultBase(n, ref p); if (0 == CheckPoint(p)) throw new InvalidOperationException(); F.Copy(p.x, x); F.Copy(p.y, y); } #endif private static void ScalarMultOrderVar(ref PointAffine p, ref PointProjective r) { #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER Span ws_p = stackalloc sbyte[447]; #else sbyte[] ws_p = new sbyte[447]; #endif // NOTE: WnafWidth225 because of the special structure of the order Scalar448.GetOrderWnafVar(WnafWidth225, ws_p); int count = 1 << (WnafWidth225 - 2); PointProjective[] tp = new PointProjective[count]; Init(out PointTemp t); PointPrecompute(ref p, tp, 0, count, ref t); PointSetNeutral(ref r); for (int bit = 446;;) { int wp = ws_p[bit]; if (wp != 0) { int index = (wp >> 1) ^ (wp >> 31); PointAddVar(wp < 0, ref tp[index], ref r, ref t); } if (--bit < 0) break; PointDouble(ref r, ref t); } } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER private static void ScalarMultStraus225Var(ReadOnlySpan nb, ReadOnlySpan np, ref PointAffine p, ReadOnlySpan nq, ref PointAffine q, ref PointProjective r) #else private static void ScalarMultStraus225Var(uint[] nb, uint[] np, ref PointAffine p, uint[] nq, ref PointAffine q, ref PointProjective r) #endif { Debug.Assert(nb.Length == ScalarUints); Debug.Assert(nb[ScalarUints - 1] >> 30 == 0U); Debug.Assert(np.Length == 8); Debug.Assert((int)np[7] >> 31 == (int)np[7]); Debug.Assert(nq.Length == 8); Debug.Assert((int)nq[7] >> 31 == (int)nq[7]); Precompute(); #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER Span ws_b = stackalloc sbyte[450]; Span ws_p = stackalloc sbyte[225]; Span ws_q = stackalloc sbyte[225]; #else sbyte[] ws_b = new sbyte[450]; sbyte[] ws_p = new sbyte[225]; sbyte[] ws_q = new sbyte[225]; #endif Wnaf.GetSignedVar(nb, WnafWidthBase, ws_b); Wnaf.GetSignedVar(np, WnafWidth225, ws_p); Wnaf.GetSignedVar(nq, WnafWidth225, ws_q); int count = 1 << (WnafWidth225 - 2); PointProjective[] tp = new PointProjective[count]; PointProjective[] tq = new PointProjective[count]; Init(out PointTemp t); PointPrecompute(ref p, tp, 0, count, ref t); PointPrecompute(ref q, tq, 0, count, ref t); PointSetNeutral(ref r); int bit = 225; while (--bit >= 0) { if (((int)ws_b[bit] | (int)ws_b[225 + bit] | (int)ws_p[bit] | (int)ws_q[bit]) != 0) break; } for (; bit >= 0; --bit) { int wb = ws_b[bit]; if (wb != 0) { int index = (wb >> 1) ^ (wb >> 31); PointAddVar(wb < 0, ref PrecompBaseWnaf[index], ref r, ref t); } int wb225 = ws_b[225 + bit]; if (wb225 != 0) { int index = (wb225 >> 1) ^ (wb225 >> 31); PointAddVar(wb225 < 0, ref PrecompBase225Wnaf[index], ref r, ref t); } int wp = ws_p[bit]; if (wp != 0) { int index = (wp >> 1) ^ (wp >> 31); PointAddVar(wp < 0, ref tp[index], ref r, ref t); } int wq = ws_q[bit]; if (wq != 0) { int index = (wq >> 1) ^ (wq >> 31); PointAddVar(wq < 0, ref tq[index], ref r, ref t); } PointDouble(ref r, ref t); } // NOTE: Together with the final PointDouble of the loop, this clears the cofactor of 4 PointDouble(ref r, ref t); } public static void Sign(byte[] sk, int skOff, byte[] ctx, byte[] m, int mOff, int mLen, byte[] sig, int sigOff) { byte phflag = 0x00; ImplSign(sk, skOff, ctx, phflag, m, mOff, mLen, sig, sigOff); } public static void Sign(byte[] sk, int skOff, byte[] pk, int pkOff, byte[] ctx, byte[] m, int mOff, int mLen, byte[] sig, int sigOff) { byte phflag = 0x00; ImplSign(sk, skOff, pk, pkOff, ctx, phflag, m, mOff, mLen, sig, sigOff); } public static void SignPrehash(byte[] sk, int skOff, byte[] ctx, byte[] ph, int phOff, byte[] sig, int sigOff) { byte phflag = 0x01; ImplSign(sk, skOff, ctx, phflag, ph, phOff, PrehashSize, sig, sigOff); } public static void SignPrehash(byte[] sk, int skOff, byte[] pk, int pkOff, byte[] ctx, byte[] ph, int phOff, byte[] sig, int sigOff) { byte phflag = 0x01; ImplSign(sk, skOff, pk, pkOff, ctx, phflag, ph, phOff, PrehashSize, sig, sigOff); } public static void SignPrehash(byte[] sk, int skOff, byte[] ctx, IXof ph, byte[] sig, int sigOff) { byte[] m = new byte[PrehashSize]; if (PrehashSize != ph.OutputFinal(m, 0, PrehashSize)) throw new ArgumentException("ph"); byte phflag = 0x01; ImplSign(sk, skOff, ctx, phflag, m, 0, m.Length, sig, sigOff); } public static void SignPrehash(byte[] sk, int skOff, byte[] pk, int pkOff, byte[] ctx, IXof ph, byte[] sig, int sigOff) { byte[] m = new byte[PrehashSize]; if (PrehashSize != ph.OutputFinal(m, 0, PrehashSize)) throw new ArgumentException("ph"); byte phflag = 0x01; ImplSign(sk, skOff, pk, pkOff, ctx, phflag, m, 0, m.Length, sig, sigOff); } public static bool ValidatePublicKeyFull(byte[] pk, int pkOff) { #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER return ValidatePublicKeyFull(pk.AsSpan(pkOff)); #else byte[] A = Copy(pk, pkOff, PublicKeySize); if (!CheckPointFullVar(A)) return false; Init(out PointAffine pA); if (!DecodePointVar(A, false, ref pA)) return false; return CheckPointOrderVar(ref pA); #endif } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER public static bool ValidatePublicKeyFull(ReadOnlySpan pk) { Span A = stackalloc byte[PublicKeySize]; A.CopyFrom(pk); if (!CheckPointFullVar(A)) return false; Init(out PointAffine pA); if (!DecodePointVar(A, false, ref pA)) return false; return CheckPointOrderVar(ref pA); } #endif public static PublicPoint ValidatePublicKeyFullExport(byte[] pk, int pkOff) { #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER return ValidatePublicKeyFullExport(pk.AsSpan(pkOff)); #else byte[] A = Copy(pk, pkOff, PublicKeySize); if (!CheckPointFullVar(A)) return null; Init(out PointAffine pA); if (!DecodePointVar(A, false, ref pA)) return null; if (!CheckPointOrderVar(ref pA)) return null; return ExportPoint(ref pA); #endif } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER public static PublicPoint ValidatePublicKeyFullExport(ReadOnlySpan pk) { Span A = stackalloc byte[PublicKeySize]; A.CopyFrom(pk); if (!CheckPointFullVar(A)) return null; Init(out PointAffine pA); if (!DecodePointVar(A, false, ref pA)) return null; if (!CheckPointOrderVar(ref pA)) return null; return ExportPoint(ref pA); } #endif public static bool ValidatePublicKeyPartial(byte[] pk, int pkOff) { #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER return ValidatePublicKeyPartial(pk.AsSpan(pkOff)); #else byte[] A = Copy(pk, pkOff, PublicKeySize); if (!CheckPointFullVar(A)) return false; Init(out PointAffine pA); return DecodePointVar(A, false, ref pA); #endif } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER public static bool ValidatePublicKeyPartial(ReadOnlySpan pk) { Span A = stackalloc byte[PublicKeySize]; A.CopyFrom(pk); if (!CheckPointFullVar(A)) return false; Init(out PointAffine pA); return DecodePointVar(A, false, ref pA); } #endif public static PublicPoint ValidatePublicKeyPartialExport(byte[] pk, int pkOff) { #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER return ValidatePublicKeyPartialExport(pk.AsSpan(pkOff)); #else byte[] A = Copy(pk, pkOff, PublicKeySize); if (!CheckPointFullVar(A)) return null; Init(out PointAffine pA); if (!DecodePointVar(A, false, ref pA)) return null; return ExportPoint(ref pA); #endif } #if NETCOREAPP2_1_OR_GREATER || NETSTANDARD2_1_OR_GREATER public static PublicPoint ValidatePublicKeyPartialExport(ReadOnlySpan pk) { Span A = stackalloc byte[PublicKeySize]; A.CopyFrom(pk); if (!CheckPointFullVar(A)) return null; Init(out PointAffine pA); if (!DecodePointVar(A, false, ref pA)) return null; return ExportPoint(ref pA); } #endif public static bool Verify(byte[] sig, int sigOff, byte[] pk, int pkOff, byte[] ctx, byte[] m, int mOff, int mLen) { byte phflag = 0x00; return ImplVerify(sig, sigOff, pk, pkOff, ctx, phflag, m, mOff, mLen); } public static bool Verify(byte[] sig, int sigOff, PublicPoint publicPoint, byte[] ctx, byte[] m, int mOff, int mLen) { byte phflag = 0x00; return ImplVerify(sig, sigOff, publicPoint, ctx, phflag, m, mOff, mLen); } public static bool VerifyPrehash(byte[] sig, int sigOff, byte[] pk, int pkOff, byte[] ctx, byte[] ph, int phOff) { byte phflag = 0x01; return ImplVerify(sig, sigOff, pk, pkOff, ctx, phflag, ph, phOff, PrehashSize); } public static bool VerifyPrehash(byte[] sig, int sigOff, PublicPoint publicPoint, byte[] ctx, byte[] ph, int phOff) { byte phflag = 0x01; return ImplVerify(sig, sigOff, publicPoint, ctx, phflag, ph, phOff, PrehashSize); } public static bool VerifyPrehash(byte[] sig, int sigOff, byte[] pk, int pkOff, byte[] ctx, IXof ph) { byte[] m = new byte[PrehashSize]; if (PrehashSize != ph.OutputFinal(m, 0, PrehashSize)) throw new ArgumentException("ph"); byte phflag = 0x01; return ImplVerify(sig, sigOff, pk, pkOff, ctx, phflag, m, 0, m.Length); } public static bool VerifyPrehash(byte[] sig, int sigOff, PublicPoint publicPoint, byte[] ctx, IXof ph) { byte[] m = new byte[PrehashSize]; if (PrehashSize != ph.OutputFinal(m, 0, PrehashSize)) throw new ArgumentException("ph"); byte phflag = 0x01; return ImplVerify(sig, sigOff, publicPoint, ctx, phflag, m, 0, m.Length); } } }