mirror of https://github.com/zcash/pasta.git
hashtocurve.sage: more realistic use of Montgomery's trick.
Signed-off-by: Daira Hopwood <daira@jacaranda.org>
This commit is contained in:
parent
96fd2c794e
commit
7df33f4ce4
|
@ -42,9 +42,11 @@ class Cost:
|
|||
self.muls += 1
|
||||
return x / y
|
||||
|
||||
def inv0(self, x):
|
||||
def batch_inv0(self, xs):
|
||||
self.invs += 1
|
||||
return 0 if x == 0 else x^-1
|
||||
self.muls += 3*(len(xs)-1)
|
||||
# This should use Montgomery's trick (with constant-time substitutions to handle zeros).
|
||||
return [0 if x == 0 else x^-1 for x in xs]
|
||||
|
||||
def sqrt(self, x):
|
||||
self.sqrs += 247
|
||||
|
@ -98,23 +100,22 @@ def select_z_nz(s, ifz, ifnz):
|
|||
# This should be constant-time in a real implementation.
|
||||
return ifz if (s == 0) else ifnz
|
||||
|
||||
def map_to_curve_simple_swu(E, Z, u, c):
|
||||
def map_to_curve_simple_swu(E, Z, us, c):
|
||||
# would be precomputed
|
||||
(0, 0, 0, A, B) = E.a_invariants()
|
||||
mBdivA = -B / A
|
||||
BdivZA = B / (Z * A)
|
||||
#print("A = ", A)
|
||||
#print("B = ", B)
|
||||
#print("Z = ", Z)
|
||||
#print("-B/A = ", mBdivA)
|
||||
#print("B/ZA = ", BdivZA)
|
||||
Z2 = Z^2
|
||||
|
||||
# 1. tv1 = inv0(Z^2 * u^4 + Z * u^2)
|
||||
Z2 = c.sqr(Z)
|
||||
u2 = c.sqr(u)
|
||||
u4 = c.sqr(u2)
|
||||
ta = c.mul(Z2, u4) + c.mul(Z, u2)
|
||||
tv1 = c.inv0(ta)
|
||||
# = inv0((Z^2 * u^2 + Z) * u^2)
|
||||
u2s = [c.sqr(u) for u in us]
|
||||
tas = [c.mul((Z2*u2 + Z), u2) for u2 in u2s]
|
||||
tv1s = c.batch_inv0(tas)
|
||||
|
||||
Qs = []
|
||||
for i in range(len(us)):
|
||||
(u, u2, tv1) = (us[i], u2s[i], tv1s[i])
|
||||
|
||||
# 2. x1 = (-B / A) * (1 + tv1)
|
||||
# 3. If tv1 == 0, set x1 = B / (Z * A)
|
||||
|
@ -147,8 +148,9 @@ def map_to_curve_simple_swu(E, Z, u, c):
|
|||
|
||||
# 9. If sgn0(u) != sgn0(y), set y = -y
|
||||
y = select_z_nz((int(u) % 2) - (int(y) % 2), y, -y)
|
||||
Qs.append(E((x, y)))
|
||||
|
||||
return E((x, y))
|
||||
return Qs
|
||||
|
||||
|
||||
# iso_Ep = Isogeny of degree 3 from Elliptic Curve defined by y^2 = x^3 + 10949663248450308183708987909873589833737836120165333298109615750520499732811*x + 1265 over Fp
|
||||
|
@ -198,25 +200,21 @@ def OS2IP(bs):
|
|||
|
||||
def hash_to_curve(msg, DST, uniform=True):
|
||||
c = Cost()
|
||||
u = hash_to_field(msg, DST, 2)
|
||||
us = hash_to_field(msg, DST, 2 if uniform else 1)
|
||||
#print("u = ", u)
|
||||
R = map_to_curve_simple_swu(E_isop, Z_isop, u[0], c)
|
||||
Qs = map_to_curve_simple_swu(E_isop, Z_isop, us, c)
|
||||
|
||||
if uniform:
|
||||
Q1 = map_to_curve_simple_swu(E_isop, Z_isop, u[1], c)
|
||||
|
||||
# We could batch the two inv0 inversions (not done above for simplicity).
|
||||
c.invs -= 1
|
||||
c.muls += 3
|
||||
|
||||
# Complete addition using affine coordinates: I + 2M + 2S
|
||||
# (S for x1^2; compute numerator and denominator of the division for the correct case;
|
||||
# I + M to divide; S + M to compute x and y of the result.)
|
||||
R = R + Q1
|
||||
R = Qs[0] + Qs[1]
|
||||
#print("R = ", R)
|
||||
c.invs += 1
|
||||
c.sqrs += 2
|
||||
c.muls += 2
|
||||
else:
|
||||
R = Qs[0]
|
||||
|
||||
# no cofactor clearing needed since Pallas and Vesta are prime-order
|
||||
(x, y) = R.xy()
|
||||
|
@ -228,5 +226,5 @@ def hash_to_curve(msg, DST, uniform=True):
|
|||
|
||||
iters = 100
|
||||
for i in range(iters):
|
||||
(res, cost) = hash_to_curve(pack(">I", i), "blah", uniform=False)
|
||||
(res, cost) = hash_to_curve(pack(">I", i), "blah", uniform=True)
|
||||
print(res, cost)
|
||||
|
|
Loading…
Reference in New Issue