commit 879bd52fff4996cbc81d46bd3f0cfebf41123118
parent 68c8cfef81f432942d867191abcff0f2261100d6
Author: Jared Tobin <jared@jtobin.io>
Date: Sat, 27 Dec 2025 18:26:52 -0330
lib: employ synonyms in curve
Diffstat:
1 file changed, 73 insertions(+), 67 deletions(-)
diff --git a/lib/Numeric/Montgomery/Secp256k1/Curve.hs b/lib/Numeric/Montgomery/Secp256k1/Curve.hs
@@ -1,6 +1,7 @@
{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE MagicHash #-}
{-# LANGUAGE NumericUnderscores #-}
+{-# LANGUAGE PatternSynonyms #-}
{-# LANGUAGE ViewPatterns #-}
{-# LANGUAGE UnboxedSums #-}
{-# LANGUAGE UnboxedTuples #-}
@@ -66,7 +67,7 @@ import qualified Data.Word.Limb as L
import qualified Data.Word.Wide as W
import Data.Word.Wider (Wider(..))
import qualified Data.Word.Wider as WW
-import GHC.Exts (Word(..))
+import GHC.Exts (Word(..), Word#)
import Prelude hiding (or, and, not, sqrt, exp)
-- montgomery arithmetic, specialized to the secp256k1 field prime modulus
@@ -80,7 +81,7 @@ import Prelude hiding (or, and, not, sqrt, exp)
-- 1
-- >>> putStrLn (render one)
-- (4294968273, 0, 0, 0)
-data Montgomery = Montgomery !(# Limb, Limb, Limb, Limb #)
+data Montgomery = Montgomery !Limb4
-- | Render a 'Montgomery' value as a 'String', showing its individual
-- 'Limb's.
@@ -88,7 +89,7 @@ data Montgomery = Montgomery !(# Limb, Limb, Limb, Limb #)
-- >>> putStrLn (render 1)
-- (4294968273, 0, 0, 0)
render :: Montgomery -> String
-render (Montgomery (# Limb a, Limb b, Limb c, Limb d #)) =
+render (Montgomery (L4 a b c d)) =
"(" <> show (W# a) <> ", " <> show (W# b) <> ", "
<> show (W# c) <> ", " <> show (W# d) <> ")"
@@ -116,8 +117,16 @@ instance NFData Montgomery where
-- utilities ------------------------------------------------------------------
+type Limb2 = (# Limb, Limb #)
+
+type Limb4 = (# Limb, Limb, Limb, Limb #)
+
+pattern L4 :: Word# -> Word# -> Word# -> Word# -> Limb4
+pattern L4 w0 w1 w2 w3 = (# Limb w0, Limb w1, Limb w2, Limb w3 #)
+{-# COMPLETE L4 #-}
+
-- Wide wrapping addition, when addend is only a limb.
-wadd_w# :: (# Limb, Limb #) -> Limb -> (# Limb, Limb #)
+wadd_w# :: Limb2 -> Limb -> Limb2
wadd_w# (# x_lo, x_hi #) y_lo =
let !(# s0, c0 #) = L.add_o# x_lo y_lo
!(# s1, _ #) = L.add_o# x_hi c0
@@ -125,7 +134,7 @@ wadd_w# (# x_lo, x_hi #) y_lo =
{-# INLINE wadd_w# #-}
-- Truncate a wide word to a 'Limb'.
-lo :: (# Limb, Limb #) -> Limb
+lo :: Limb2 -> Limb
lo (# l, _ #) = l
{-# INLINE lo #-}
@@ -133,10 +142,8 @@ lo (# l, _ #) = l
-- | Constant-time equality comparison.
eq :: Montgomery -> Montgomery -> C.Choice
-eq
- (Montgomery (# Limb a0, Limb a1, Limb a2, Limb a3 #))
- (Montgomery (# Limb b0, Limb b1, Limb b2, Limb b3 #))
- = C.eq_wider# (# a0, a1, a2, a3 #) (# b0, b1, b2, b3 #)
+eq (Montgomery (L4 a0 a1 a2 a3)) (Montgomery (L4 b0 b1 b2 b3)) =
+ C.eq_wider# (# a0, a1, a2, a3 #) (# b0, b1, b2, b3 #)
{-# INLINE eq #-}
-- | Variable-time equality comparison.
@@ -147,9 +154,9 @@ eq_vartime (Montgomery (Wider -> a)) (Montgomery (Wider -> b)) =
-- innards --------------------------------------------------------------------
redc_inner#
- :: (# Limb, Limb, Limb, Limb #) -- ^ upper limbs
- -> (# Limb, Limb, Limb, Limb #) -- ^ lower limbs
- -> (# (# Limb, Limb, Limb, Limb #), Limb #) -- ^ upper limbs, meta-carry
+ :: Limb4 -- ^ upper limbs
+ -> Limb4 -- ^ lower limbs
+ -> (# Limb4, Limb #) -- ^ upper limbs, meta-carry
redc_inner# (# u0, u1, u2, u3 #) (# l0, l1, l2, l3 #) =
let !(# m0, m1, m2, m3 #) =
(# Limb 0xFFFFFFFEFFFFFC2F##, Limb 0xFFFFFFFFFFFFFFFF##
@@ -184,13 +191,13 @@ redc_inner# (# u0, u1, u2, u3 #) (# l0, l1, l2, l3 #) =
-- | Montgomery reduction.
redc#
- :: (# Limb, Limb, Limb, Limb #) -- ^ lower limbs
- -> (# Limb, Limb, Limb, Limb #) -- ^ upper limbs
- -> (# Limb, Limb, Limb, Limb #) -- ^ result
+ :: Limb4 -- ^ lower limbs
+ -> Limb4 -- ^ upper limbs
+ -> Limb4 -- ^ result
redc# l u =
let -- field prime
- !m = (# Limb 0xFFFFFFFEFFFFFC2F##, Limb 0xFFFFFFFFFFFFFFFF##
- , Limb 0xFFFFFFFFFFFFFFFF##, Limb 0xFFFFFFFFFFFFFFFF## #)
+ !m = L4 0xFFFFFFFEFFFFFC2F## 0xFFFFFFFFFFFFFFFF##
+ 0xFFFFFFFFFFFFFFFF## 0xFFFFFFFFFFFFFFFF##
!(# nu, mc #) = redc_inner# u l
in WW.sub_mod_c# nu mc m m
{-# INLINE redc# #-}
@@ -208,12 +215,12 @@ redc (Montgomery l) (Montgomery u) =
in Montgomery res
retr_inner#
- :: (# Limb, Limb, Limb, Limb #) -- ^ value in montgomery form
- -> (# Limb, Limb, Limb, Limb #) -- ^ retrieved value
+ :: Limb4 -- ^ value in montgomery form
+ -> Limb4 -- ^ retrieved value
retr_inner# (# x0, x1, x2, x3 #) =
let !(# m0, m1, m2, m3 #) =
- (# Limb 0xFFFFFFFEFFFFFC2F##, Limb 0xFFFFFFFFFFFFFFFF##
- , Limb 0xFFFFFFFFFFFFFFFF##, Limb 0xFFFFFFFFFFFFFFFF## #)
+ L4 0xFFFFFFFEFFFFFC2F## 0xFFFFFFFFFFFFFFFF##
+ 0xFFFFFFFFFFFFFFFF## 0xFFFFFFFFFFFFFFFF##
!n = Limb 0xD838091DD2253531##
!u_0 = L.mul_w# x0 n
!(# _, o0 #) = L.mac# u_0 m0 x0 (Limb 0##)
@@ -239,8 +246,8 @@ retr_inner# (# x0, x1, x2, x3 #) =
{-# INLINE retr_inner# #-}
retr#
- :: (# Limb, Limb, Limb, Limb #) -- montgomery form
- -> (# Limb, Limb, Limb, Limb #)
+ :: Limb4 -- montgomery form
+ -> Limb4
retr# f = retr_inner# f
{-# INLINE retr# #-}
@@ -255,13 +262,13 @@ retr (Montgomery f) =
-- | Montgomery multiplication (FIOS), without conditional subtract.
mul_inner#
- :: (# Limb, Limb, Limb, Limb #) -- ^ x
- -> (# Limb, Limb, Limb, Limb #) -- ^ y
- -> (# (# Limb, Limb, Limb, Limb #), Limb #) -- ^ product, meta-carry
+ :: Limb4 -- ^ x
+ -> Limb4 -- ^ y
+ -> (# Limb4, Limb #) -- ^ product, meta-carry
mul_inner# (# x0, x1, x2, x3 #) (# y0, y1, y2, y3 #) =
let !(# m0, m1, m2, m3 #) =
- (# Limb 0xFFFFFFFEFFFFFC2F##, Limb 0xFFFFFFFFFFFFFFFF##
- , Limb 0xFFFFFFFFFFFFFFFF##, Limb 0xFFFFFFFFFFFFFFFF## #)
+ L4 0xFFFFFFFEFFFFFC2F## 0xFFFFFFFFFFFFFFFF##
+ 0xFFFFFFFFFFFFFFFF## 0xFFFFFFFFFFFFFFFF##
!n = Limb 0xD838091DD2253531##
!axy0 = L.mul_c# x0 y0
!u0 = L.mul_w# (lo axy0) n
@@ -335,13 +342,13 @@ mul_inner# (# x0, x1, x2, x3 #) (# y0, y1, y2, y3 #) =
{-# INLINE mul_inner# #-}
mul#
- :: (# Limb, Limb, Limb, Limb #)
- -> (# Limb, Limb, Limb, Limb #)
- -> (# Limb, Limb, Limb, Limb #)
+ :: Limb4
+ -> Limb4
+ -> Limb4
mul# a b =
let -- field prime
- !m = (# Limb 0xFFFFFFFEFFFFFC2F##, Limb 0xFFFFFFFFFFFFFFFF##
- , Limb 0xFFFFFFFFFFFFFFFF##, Limb 0xFFFFFFFFFFFFFFFF## #)
+ !m = L4 0xFFFFFFFEFFFFFC2F## 0xFFFFFFFFFFFFFFFF##
+ 0xFFFFFFFFFFFFFFFF## 0xFFFFFFFFFFFFFFFF##
!(# nu, mc #) = mul_inner# a b
in WW.sub_mod_c# nu mc m m
{-# NOINLINE mul# #-} -- cannot be inlined without exploding comp time
@@ -360,11 +367,10 @@ mul
mul (Montgomery a) (Montgomery b) = Montgomery (mul# a b)
to#
- :: (# Limb, Limb, Limb, Limb #) -- ^ integer
- -> (# Limb, Limb, Limb, Limb #)
+ :: Limb4 -- ^ integer
+ -> Limb4
to# x =
- let -- r^2 mod m
- !r2 = (# Limb 0x000007A2000E90A1##, Limb 0x1##, Limb 0##, Limb 0## #)
+ let !r2 = L4 0x000007A2000E90A1## 0x1## 0## 0## -- r^2 mod m
in mul# x r2
{-# INLINE to# #-}
@@ -379,13 +385,13 @@ from :: Montgomery -> Wider
from = retr
add#
- :: (# Limb, Limb, Limb, Limb #) -- ^ augend
- -> (# Limb, Limb, Limb, Limb #) -- ^ addend
- -> (# Limb, Limb, Limb, Limb #) -- ^ sum
+ :: Limb4 -- ^ augend
+ -> Limb4 -- ^ addend
+ -> Limb4 -- ^ sum
add# a b =
let -- field prime
- !m = (# Limb 0xFFFFFFFEFFFFFC2F##, Limb 0xFFFFFFFFFFFFFFFF##
- , Limb 0xFFFFFFFFFFFFFFFF##, Limb 0xFFFFFFFFFFFFFFFF## #)
+ !m = L4 0xFFFFFFFEFFFFFC2F## 0xFFFFFFFFFFFFFFFF##
+ 0xFFFFFFFFFFFFFFFF## 0xFFFFFFFFFFFFFFFF##
in WW.add_mod# a b m
{-# INLINE add# #-}
@@ -400,13 +406,13 @@ add :: Montgomery -> Montgomery -> Montgomery
add (Montgomery a) (Montgomery b) = Montgomery (add# a b)
sub#
- :: (# Limb, Limb, Limb, Limb #) -- ^ minuend
- -> (# Limb, Limb, Limb, Limb #) -- ^ subtrahend
- -> (# Limb, Limb, Limb, Limb #) -- ^ difference
+ :: Limb4 -- ^ minuend
+ -> Limb4 -- ^ subtrahend
+ -> Limb4 -- ^ difference
sub# a b =
let -- field prime
- !m = (# Limb 0xFFFFFFFEFFFFFC2F##, Limb 0xFFFFFFFFFFFFFFFF##
- , Limb 0xFFFFFFFFFFFFFFFF##, Limb 0xFFFFFFFFFFFFFFFF## #)
+ !m = L4 0xFFFFFFFEFFFFFC2F## 0xFFFFFFFFFFFFFFFF##
+ 0xFFFFFFFFFFFFFFFF## 0xFFFFFFFFFFFFFFFF##
in WW.sub_mod# a b m
{-# INLINE sub# #-}
@@ -421,9 +427,9 @@ sub :: Montgomery -> Montgomery -> Montgomery
sub (Montgomery a) (Montgomery b) = Montgomery (sub# a b)
neg#
- :: (# Limb, Limb, Limb, Limb #) -- ^ argument
- -> (# Limb, Limb, Limb, Limb #) -- ^ modular negation
-neg# a = sub# (# Limb 0##, Limb 0##, Limb 0##, Limb 0## #) a
+ :: Limb4 -- ^ argument
+ -> Limb4 -- ^ modular negation
+neg# a = sub# (L4 0## 0## 0## 0##) a
{-# INLINE neg# #-}
-- | Additive inverse in the Montgomery domain.
@@ -438,7 +444,7 @@ neg# a = sub# (# Limb 0##, Limb 0##, Limb 0##, Limb 0## #) a
neg :: Montgomery -> Montgomery
neg (Montgomery a) = Montgomery (neg# a)
-sqr# :: (# Limb, Limb, Limb, Limb #) -> (# Limb, Limb, Limb, Limb #)
+sqr# :: Limb4 -> Limb4
sqr# a =
let !(# l, h #) = WW.sqr# a
in redc# l h
@@ -457,19 +463,19 @@ sqr (Montgomery a) = Montgomery (mul# a a)
-- | Zero (the additive unit) in the Montgomery domain.
zero :: Montgomery
-zero = Montgomery (# Limb 0##, Limb 0##, Limb 0##, Limb 0## #)
+zero = Montgomery (L4 0## 0## 0## 0##)
-- | One (the multiplicative unit) in the Montgomery domain.
one :: Montgomery
-one = Montgomery (# Limb 0x1000003D1##, Limb 0##, Limb 0##, Limb 0## #)
+one = Montgomery (L4 0x1000003D1## 0## 0## 0##)
-- generated by etc/generate_inv.sh
inv#
- :: (# Limb, Limb, Limb, Limb #)
- -> (# Limb, Limb, Limb, Limb #)
+ :: Limb4
+ -> Limb4
inv# a =
let -- montgomery 'one'
- !t0 = (# Limb 0x1000003D1##, Limb 0##, Limb 0##, Limb 0## #)
+ !t0 = L4 0x1000003D1## 0## 0## 0##
!t1 = sqr# t0
!t2 = mul# a t1
!t3 = sqr# t2
@@ -1009,10 +1015,10 @@ sqrt_vartime (Montgomery n) = case sqrt# n of
-- generated by etc/generate_sqrt.sh
sqrt#
- :: (# Limb, Limb, Limb, Limb #)
- -> (# (# Limb, Limb, Limb, Limb #), C.Choice #)
+ :: Limb4
+ -> (# Limb4, C.Choice #)
sqrt# a =
- let !t0 = (# Limb 0x1000003D1##, Limb 0##, Limb 0##, Limb 0## #)
+ let !t0 = L4 0x1000003D1## 0## 0## 0##
!t1 = sqr# t0
!t2 = sqr# t1
!t3 = sqr# t2
@@ -1530,11 +1536,11 @@ exp :: Montgomery -> Wider -> Montgomery
exp (Montgomery b) (Wider e) = Montgomery (exp# b e)
exp#
- :: (# Limb, Limb, Limb, Limb #)
- -> (# Limb, Limb, Limb, Limb #)
- -> (# Limb, Limb, Limb, Limb #)
+ :: Limb4
+ -> Limb4
+ -> Limb4
exp# b e =
- let !o = (# Limb 0x1000003D1##, Limb 0##, Limb 0##, Limb 0## #)
+ let !o = L4 0x1000003D1## 0## 0## 0##
loop !r !m !ex n = case n of
0 -> r
_ ->
@@ -1546,7 +1552,7 @@ exp# b e =
in loop o b e (256 :: Word)
{-# INLINE exp# #-}
-odd# :: (# Limb, Limb, Limb, Limb #) -> C.Choice
+odd# :: Limb4 -> C.Choice
odd# = WW.odd#
{-# INLINE odd# #-}
@@ -1567,10 +1573,10 @@ odd_vartime (Montgomery m) = C.decide (odd# m)
-- constant-time selection ----------------------------------------------------
select#
- :: (# Limb, Limb, Limb, Limb #) -- ^ a
- -> (# Limb, Limb, Limb, Limb #) -- ^ b
+ :: Limb4 -- ^ a
+ -> Limb4 -- ^ b
-> C.Choice -- ^ c
- -> (# Limb, Limb, Limb, Limb #) -- ^ result
+ -> Limb4 -- ^ result
select# = WW.select#
{-# INLINE select# #-}