summaryrefslogtreecommitdiff
path: root/ecp.cpp
diff options
context:
space:
mode:
authorJeffrey Walton <noloader@gmail.com>2019-08-09 17:34:14 -0400
committerJeffrey Walton <noloader@gmail.com>2019-08-09 17:34:14 -0400
commit04b2a20c5da517672173a9102b0d3fbda8b44b42 (patch)
treea4041f2f44be84e8634188d130b292f36bd88124 /ecp.cpp
parent7ac579119958f230a7e020577db3be58f923a4f0 (diff)
downloadcryptopp-git-04b2a20c5da517672173a9102b0d3fbda8b44b42.tar.gz
Restore ECP ABI (GH #869)
Placing AdditionFunction as an inner class of ECP broke the ABI. We need to maintain the ABI so distros can patch Crypto++ 8.2.
Diffstat (limited to 'ecp.cpp')
-rw-r--r--ecp.cpp972
1 files changed, 493 insertions, 479 deletions
diff --git a/ecp.cpp b/ecp.cpp
index 036296a7..15007e10 100644
--- a/ecp.cpp
+++ b/ecp.cpp
@@ -46,6 +46,497 @@ inline Integer IdentityToInteger(bool val)
return val ? Integer::One() : Integer::Zero();
}
+struct ProjectivePoint
+{
+ ProjectivePoint() {}
+ ProjectivePoint(const Integer &x, const Integer &y, const Integer &z)
+ : x(x), y(y), z(z) {}
+
+ Integer x, y, z;
+};
+
+/// \brief Addition and Double functions
+/// \sa <A HREF="https://eprint.iacr.org/2015/1060.pdf">Complete
+/// addition formulas for prime order elliptic curves</A>
+struct AdditionFunction
+{
+ explicit AdditionFunction(const ECP::Field& field,
+ const ECP::FieldElement &a, const ECP::FieldElement &b, ECP::Point &r);
+
+ // Double(P)
+ ECP::Point operator()(const ECP::Point& P) const;
+ // Add(P, Q)
+ ECP::Point operator()(const ECP::Point& P, const ECP::Point& Q) const;
+
+protected:
+ /// \brief Parameters and representation for Addition
+ /// \details Addition and Doubling will use different algorithms,
+ /// depending on the <tt>A</tt> coefficient and the representation
+ /// (Affine or Montgomery with precomputation).
+ enum Alpha {
+ /// \brief Coefficient A is 0
+ A_0 = 1,
+ /// \brief Coefficient A is -3
+ A_3 = 2,
+ /// \brief Coefficient A is arbitrary
+ A_Star = 4,
+ /// \brief Representation is Montgomery
+ A_Montgomery = 8
+ };
+
+ const ECP::Field& field;
+ const ECP::FieldElement &a, &b;
+ ECP::Point &R;
+
+ Alpha m_alpha;
+};
+
+#define X p.x
+#define Y p.y
+#define Z p.z
+
+#define X1 p.x
+#define Y1 p.y
+#define Z1 p.z
+
+#define X2 q.x
+#define Y2 q.y
+#define Z2 q.z
+
+#define X3 r.x
+#define Y3 r.y
+#define Z3 r.z
+
+AdditionFunction::AdditionFunction(const ECP::Field& field,
+ const ECP::FieldElement &a, const ECP::FieldElement &b, ECP::Point &r)
+ : field(field), a(a), b(b), R(r), m_alpha(static_cast<Alpha>(0))
+{
+ if (field.IsMontgomeryRepresentation())
+ {
+ m_alpha = A_Montgomery;
+ }
+ else
+ {
+ if (a == 0)
+ {
+ m_alpha = A_0;
+ }
+ else if (a == -3 || (a - field.GetModulus()) == -3)
+ {
+ m_alpha = A_3;
+ }
+ else
+ {
+ m_alpha = A_Star;
+ }
+ }
+}
+
+ECP::Point AdditionFunction::operator()(const ECP::Point& P) const
+{
+ if (m_alpha == A_3)
+ {
+ // Gyrations attempt to maintain constant-timeness
+ // We need either (P.x, P.y, 1) or (0, 1, 0).
+ const Integer x = P.x * IdentityToInteger(!P.identity);
+ const Integer y = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
+ const Integer z = 1 * IdentityToInteger(!P.identity);
+
+ ProjectivePoint p(x, y, z), r;
+
+ ECP::FieldElement t0 = field.Square(X);
+ ECP::FieldElement t1 = field.Square(Y);
+ ECP::FieldElement t2 = field.Square(Z);
+ ECP::FieldElement t3 = field.Multiply(X, Y);
+ t3 = field.Add(t3, t3);
+ Z3 = field.Multiply(X, Z);
+ Z3 = field.Add(Z3, Z3);
+ Y3 = field.Multiply(b, t2);
+ Y3 = field.Subtract(Y3, Z3);
+ X3 = field.Add(Y3, Y3);
+ Y3 = field.Add(X3, Y3);
+ X3 = field.Subtract(t1, Y3);
+ Y3 = field.Add(t1, Y3);
+ Y3 = field.Multiply(X3, Y3);
+ X3 = field.Multiply(X3, t3);
+ t3 = field.Add(t2, t2);
+ t2 = field.Add(t2, t3);
+ Z3 = field.Multiply(b, Z3);
+ Z3 = field.Subtract(Z3, t2);
+ Z3 = field.Subtract(Z3, t0);
+ t3 = field.Add(Z3, Z3);
+ Z3 = field.Add(Z3, t3);
+ t3 = field.Add(t0, t0);
+ t0 = field.Add(t3, t0);
+ t0 = field.Subtract(t0, t2);
+ t0 = field.Multiply(t0, Z3);
+ Y3 = field.Add(Y3, t0);
+ t0 = field.Multiply(Y, Z);
+ t0 = field.Add(t0, t0);
+ Z3 = field.Multiply(t0, Z3);
+ X3 = field.Subtract(X3, Z3);
+ Z3 = field.Multiply(t0, t1);
+ Z3 = field.Add(Z3, Z3);
+ Z3 = field.Add(Z3, Z3);
+
+ const ECP::FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
+ X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
+
+ // More gyrations
+ R.x = X3*Z3.NotZero();
+ R.y = Y3*Z3.NotZero();
+ R.identity = Z3.IsZero();
+
+ return R;
+ }
+ else if (m_alpha == A_0)
+ {
+ const ECP::FieldElement b3 = field.Multiply(b, 3);
+
+ // Gyrations attempt to maintain constant-timeness
+ // We need either (P.x, P.y, 1) or (0, 1, 0).
+ const Integer x = P.x * IdentityToInteger(!P.identity);
+ const Integer y = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
+ const Integer z = 1 * IdentityToInteger(!P.identity);
+
+ ProjectivePoint p(x, y, z), r;
+
+ ECP::FieldElement t0 = field.Square(Y);
+ Z3 = field.Add(t0, t0);
+ Z3 = field.Add(Z3, Z3);
+ Z3 = field.Add(Z3, Z3);
+ ECP::FieldElement t1 = field.Add(Y, Z);
+ ECP::FieldElement t2 = field.Square(Z);
+ t2 = field.Multiply(b3, t2);
+ X3 = field.Multiply(t2, Z3);
+ Y3 = field.Add(t0, t2);
+ Z3 = field.Multiply(t1, Z3);
+ t1 = field.Add(t2, t2);
+ t2 = field.Add(t1, t2);
+ t0 = field.Subtract(t0, t2);
+ Y3 = field.Multiply(t0, Y3);
+ Y3 = field.Add(X3, Y3);
+ t1 = field.Multiply(X, Y);
+ X3 = field.Multiply(t0, t1);
+ X3 = field.Add(X3, X3);
+
+ const ECP::FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
+ X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
+
+ // More gyrations
+ R.x = X3*Z3.NotZero();
+ R.y = Y3*Z3.NotZero();
+ R.identity = Z3.IsZero();
+
+ return R;
+ }
+ else if (m_alpha == A_Star)
+ {
+ const ECP::FieldElement b3 = field.Multiply(b, 3);
+
+ // Gyrations attempt to maintain constant-timeness
+ // We need either (P.x, P.y, 1) or (0, 1, 0).
+ const Integer x = P.x * IdentityToInteger(!P.identity);
+ const Integer y = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
+ const Integer z = 1 * IdentityToInteger(!P.identity);
+
+ ProjectivePoint p(x, y, z), r;
+
+ ECP::FieldElement t0 = field.Square(Y);
+ Z3 = field.Add(t0, t0);
+ Z3 = field.Add(Z3, Z3);
+ Z3 = field.Add(Z3, Z3);
+ ECP::FieldElement t1 = field.Add(Y, Z);
+ ECP::FieldElement t2 = field.Square(Z);
+ t2 = field.Multiply(b3, t2);
+ X3 = field.Multiply(t2, Z3);
+ Y3 = field.Add(t0, t2);
+ Z3 = field.Multiply(t1, Z3);
+ t1 = field.Add(t2, t2);
+ t2 = field.Add(t1, t2);
+ t0 = field.Subtract(t0, t2);
+ Y3 = field.Multiply(t0, Y3);
+ Y3 = field.Add(X3, Y3);
+ t1 = field.Multiply(X, Y);
+ X3 = field.Multiply(t0, t1);
+ X3 = field.Add(X3, X3);
+
+ const ECP::FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
+ X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
+
+ // More gyrations
+ R.x = X3*Z3.NotZero();
+ R.y = Y3*Z3.NotZero();
+ R.identity = Z3.IsZero();
+
+ return R;
+ }
+ else // A_Montgomery
+ {
+ // More gyrations
+ bool identity = !!(P.identity + (P.y == field.Identity()));
+
+ ECP::FieldElement t = field.Square(P.x);
+ t = field.Add(field.Add(field.Double(t), t), a);
+ t = field.Divide(t, field.Double(P.y));
+ ECP::FieldElement x = field.Subtract(field.Subtract(field.Square(t), P.x), P.x);
+ R.y = field.Subtract(field.Multiply(t, field.Subtract(P.x, x)), P.y);
+ R.x.swap(x);
+
+ // More gyrations
+ R.x *= IdentityToInteger(!identity);
+ R.y *= IdentityToInteger(!identity);
+ R.identity = identity;
+
+ return R;
+ }
+}
+
+ECP::Point AdditionFunction::operator()(const ECP::Point& P, const ECP::Point& Q) const
+{
+ if (m_alpha == A_3)
+ {
+ // Gyrations attempt to maintain constant-timeness
+ // We need either (P.x, P.y, 1) or (0, 1, 0).
+ const Integer x1 = P.x * IdentityToInteger(!P.identity);
+ const Integer y1 = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
+ const Integer z1 = 1 * IdentityToInteger(!P.identity);
+
+ const Integer x2 = Q.x * IdentityToInteger(!Q.identity);
+ const Integer y2 = Q.y * IdentityToInteger(!Q.identity) + 1 * IdentityToInteger(Q.identity);
+ const Integer z2 = 1 * IdentityToInteger(!Q.identity);
+
+ ProjectivePoint p(x1, y1, z1), q(x2, y2, z2), r;
+
+ ECP::FieldElement t0 = field.Multiply(X1, X2);
+ ECP::FieldElement t1 = field.Multiply(Y1, Y2);
+ ECP::FieldElement t2 = field.Multiply(Z1, Z2);
+ ECP::FieldElement t3 = field.Add(X1, Y1);
+ ECP::FieldElement t4 = field.Add(X2, Y2);
+ t3 = field.Multiply(t3, t4);
+ t4 = field.Add(t0, t1);
+ t3 = field.Subtract(t3, t4);
+ t4 = field.Add(Y1, Z1);
+ X3 = field.Add(Y2, Z2);
+ t4 = field.Multiply(t4, X3);
+ X3 = field.Add(t1, t2);
+ t4 = field.Subtract(t4, X3);
+ X3 = field.Add(X1, Z1);
+ Y3 = field.Add(X2, Z2);
+ X3 = field.Multiply(X3, Y3);
+ Y3 = field.Add(t0, t2);
+ Y3 = field.Subtract(X3, Y3);
+ Z3 = field.Multiply(b, t2);
+ X3 = field.Subtract(Y3, Z3);
+ Z3 = field.Add(X3, X3);
+ X3 = field.Add(X3, Z3);
+ Z3 = field.Subtract(t1, X3);
+ X3 = field.Add(t1, X3);
+ Y3 = field.Multiply(b, Y3);
+ t1 = field.Add(t2, t2);
+ t2 = field.Add(t1, t2);
+ Y3 = field.Subtract(Y3, t2);
+ Y3 = field.Subtract(Y3, t0);
+ t1 = field.Add(Y3, Y3);
+ Y3 = field.Add(t1, Y3);
+ t1 = field.Add(t0, t0);
+ t0 = field.Add(t1, t0);
+ t0 = field.Subtract(t0, t2);
+ t1 = field.Multiply(t4, Y3);
+ t2 = field.Multiply(t0, Y3);
+ Y3 = field.Multiply(X3, Z3);
+ Y3 = field.Add(Y3, t2);
+ X3 = field.Multiply(t3, X3);
+ X3 = field.Subtract(X3, t1);
+ Z3 = field.Multiply(t4, Z3);
+ t1 = field.Multiply(t3, t0);
+ Z3 = field.Add(Z3, t1);
+
+ const ECP::FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
+ X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
+
+ // More gyrations
+ R.x = X3*Z3.NotZero();
+ R.y = Y3*Z3.NotZero();
+ R.identity = Z3.IsZero();
+
+ return R;
+ }
+ else if (m_alpha == A_0)
+ {
+ const ECP::FieldElement b3 = field.Multiply(b, 3);
+
+ // Gyrations attempt to maintain constant-timeness
+ // We need either (P.x, P.y, 1) or (0, 1, 0).
+ const Integer x1 = P.x * IdentityToInteger(!P.identity);
+ const Integer y1 = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
+ const Integer z1 = 1 * IdentityToInteger(!P.identity);
+
+ const Integer x2 = Q.x * IdentityToInteger(!Q.identity);
+ const Integer y2 = Q.y * IdentityToInteger(!Q.identity) + 1 * IdentityToInteger(Q.identity);
+ const Integer z2 = 1 * IdentityToInteger(!Q.identity);
+
+ ProjectivePoint p(x1, y1, z1), q(x2, y2, z2), r;
+
+ ECP::FieldElement t0 = field.Square(Y);
+ Z3 = field.Add(t0, t0);
+ Z3 = field.Add(Z3, Z3);
+ Z3 = field.Add(Z3, Z3);
+ ECP::FieldElement t1 = field.Add(Y, Z);
+ ECP::FieldElement t2 = field.Square(Z);
+ t2 = field.Multiply(b3, t2);
+ X3 = field.Multiply(t2, Z3);
+ Y3 = field.Add(t0, t2);
+ Z3 = field.Multiply(t1, Z3);
+ t1 = field.Add(t2, t2);
+ t2 = field.Add(t1, t2);
+ t0 = field.Subtract(t0, t2);
+ Y3 = field.Multiply(t0, Y3);
+ Y3 = field.Add(X3, Y3);
+ t1 = field.Multiply(X, Y);
+ X3 = field.Multiply(t0, t1);
+ X3 = field.Add(X3, X3);
+
+ const ECP::FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
+ X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
+
+ // More gyrations
+ R.x = X3*Z3.NotZero();
+ R.y = Y3*Z3.NotZero();
+ R.identity = Z3.IsZero();
+
+ return R;
+ }
+ else if (m_alpha == A_Star)
+ {
+ const ECP::FieldElement b3 = field.Multiply(b, 3);
+
+ // Gyrations attempt to maintain constant-timeness
+ // We need either (P.x, P.y, 1) or (0, 1, 0).
+ const Integer x1 = P.x * IdentityToInteger(!P.identity);
+ const Integer y1 = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
+ const Integer z1 = 1 * IdentityToInteger(!P.identity);
+
+ const Integer x2 = Q.x * IdentityToInteger(!Q.identity);
+ const Integer y2 = Q.y * IdentityToInteger(!Q.identity) + 1 * IdentityToInteger(Q.identity);
+ const Integer z2 = 1 * IdentityToInteger(!Q.identity);
+
+ ProjectivePoint p(x1, y1, z1), q(x2, y2, z2), r;
+
+ ECP::FieldElement t0 = field.Multiply(X1, X2);
+ ECP::FieldElement t1 = field.Multiply(Y1, Y2);
+ ECP::FieldElement t2 = field.Multiply(Z1, Z2);
+ ECP::FieldElement t3 = field.Add(X1, Y1);
+ ECP::FieldElement t4 = field.Add(X2, Y2);
+ t3 = field.Multiply(t3, t4);
+ t4 = field.Add(t0, t1);
+ t3 = field.Subtract(t3, t4);
+ t4 = field.Add(X1, Z1);
+ ECP::FieldElement t5 = field.Add(X2, Z2);
+ t4 = field.Multiply(t4, t5);
+ t5 = field.Add(t0, t2);
+ t4 = field.Subtract(t4, t5);
+ t5 = field.Add(Y1, Z1);
+ X3 = field.Add(Y2, Z2);
+ t5 = field.Multiply(t5, X3);
+ X3 = field.Add(t1, t2);
+ t5 = field.Subtract(t5, X3);
+ Z3 = field.Multiply(a, t4);
+ X3 = field.Multiply(b3, t2);
+ Z3 = field.Add(X3, Z3);
+ X3 = field.Subtract(t1, Z3);
+ Z3 = field.Add(t1, Z3);
+ Y3 = field.Multiply(X3, Z3);
+ t1 = field.Add(t0, t0);
+ t1 = field.Add(t1, t0);
+ t2 = field.Multiply(a, t2);
+ t4 = field.Multiply(b3, t4);
+ t1 = field.Add(t1, t2);
+ t2 = field.Subtract(t0, t2);
+ t2 = field.Multiply(a, t2);
+ t4 = field.Add(t4, t2);
+ t0 = field.Multiply(t1, t4);
+ Y3 = field.Add(Y3, t0);
+ t0 = field.Multiply(t5, t4);
+ X3 = field.Multiply(t3, X3);
+ X3 = field.Subtract(X3, t0);
+ t0 = field.Multiply(t3, t1);
+ Z3 = field.Multiply(t5, Z3);
+ Z3 = field.Add(Z3, t0);
+
+ const ECP::FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
+ X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
+
+ // More gyrations
+ R.x = X3*Z3.NotZero();
+ R.y = Y3*Z3.NotZero();
+ R.identity = Z3.IsZero();
+
+ return R;
+ }
+ else // A_Montgomery
+ {
+ ECP::Point S = R;
+
+ // More gyrations
+ bool return_Q = P.identity;
+ bool return_P = Q.identity;
+ bool double_P = field.Equal(P.x, Q.x) && field.Equal(P.y, Q.y);
+ bool identity = field.Equal(P.x, Q.x) && !field.Equal(P.y, Q.y);
+
+ // This code taken from Double(P) for below
+ identity = !!((double_P * (P.identity + (P.y == field.Identity()))) + identity);
+
+ if (double_P)
+ {
+ // This code taken from Double(P)
+ ECP::FieldElement t = field.Square(P.x);
+ t = field.Add(field.Add(field.Double(t), t), a);
+ t = field.Divide(t, field.Double(P.y));
+ ECP::FieldElement x = field.Subtract(field.Subtract(field.Square(t), P.x), P.x);
+ R.y = field.Subtract(field.Multiply(t, field.Subtract(P.x, x)), P.y);
+ R.x.swap(x);
+ }
+ else
+ {
+ // Original Add(P,Q) code
+ ECP::FieldElement t = field.Subtract(Q.y, P.y);
+ t = field.Divide(t, field.Subtract(Q.x, P.x));
+ ECP::FieldElement x = field.Subtract(field.Subtract(field.Square(t), P.x), Q.x);
+ R.y = field.Subtract(field.Multiply(t, field.Subtract(P.x, x)), P.y);
+ R.x.swap(x);
+ }
+
+ // More gyrations
+ R.x = R.x * IdentityToInteger(!identity);
+ R.y = R.y * IdentityToInteger(!identity);
+ R.identity = identity;
+
+ if (return_Q)
+ return (R = S), Q;
+ else if (return_P)
+ return (R = S), P;
+ else
+ return (S = R), R;
+ }
+}
+
+#undef X
+#undef Y
+#undef Z
+
+#undef X1
+#undef Y1
+#undef Z1
+
+#undef X2
+#undef Y2
+#undef Z2
+
+#undef X3
+#undef Y3
+#undef Z3
+
ANONYMOUS_NAMESPACE_END
NAMESPACE_BEGIN(CryptoPP)
@@ -250,13 +741,13 @@ const ECP::Point& ECP::Inverse(const Point &P) const
const ECP::Point& ECP::Add(const Point &P, const Point &Q) const
{
- AdditionFunction add(*this);
+ AdditionFunction add(GetField(), m_a, m_b, m_R);
return (m_R = add(P, Q));
}
const ECP::Point& ECP::Double(const Point &P) const
{
- AdditionFunction add(*this);
+ AdditionFunction add(GetField(), m_a, m_b, m_R);
return (m_R = add(P));
}
@@ -297,15 +788,6 @@ template <class T, class Iterator> void ParallelInvert(const AbstractRing<T> &ri
}
}
-struct ProjectivePoint
-{
- ProjectivePoint() {}
- ProjectivePoint(const Integer &x, const Integer &y, const Integer &z)
- : x(x), y(y), z(z) {}
-
- Integer x,y,z;
-};
-
class ProjectiveDoubling
{
public:
@@ -481,474 +963,6 @@ ECP::Point ECP::CascadeScalarMultiply(const Point &P, const Integer &k1, const P
return AbstractGroup<Point>::CascadeScalarMultiply(P, k1, Q, k2);
}
-#define X p.x
-#define Y p.y
-#define Z p.z
-
-#define X1 p.x
-#define Y1 p.y
-#define Z1 p.z
-
-#define X2 q.x
-#define Y2 q.y
-#define Z2 q.z
-
-#define X3 r.x
-#define Y3 r.y
-#define Z3 r.z
-
-ECP::AdditionFunction::AdditionFunction(const ECP& ecp)
- : m_ecp(ecp), m_alpha(static_cast<Alpha>(0))
-{
- if (m_ecp.GetField().IsMontgomeryRepresentation())
- {
- m_alpha = A_Montgomery;
- }
- else
- {
- if (m_ecp.m_a == 0)
- {
- m_alpha = A_0;
- }
- else if (m_ecp.m_a == -3 || (m_ecp.m_a - m_ecp.GetField().GetModulus()) == -3)
- {
- m_alpha = A_3;
- }
- else
- {
- m_alpha = A_Star;
- }
- }
-}
-
-ECP::Point ECP::AdditionFunction::operator()(const Point& P) const
-{
- if (m_alpha == A_3)
- {
- const ECP::Field& field = m_ecp.GetField();
- const FieldElement& b = m_ecp.m_b;
- ECP::Point& R = m_ecp.m_R;
-
- // Gyrations attempt to maintain constant-timeness
- // We need either (P.x, P.y, 1) or (0, 1, 0).
- const Integer x = P.x * IdentityToInteger(!P.identity);
- const Integer y = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
- const Integer z = 1 * IdentityToInteger(!P.identity);
-
- ProjectivePoint p(x, y, z), r;
-
- FieldElement t0 = field.Square(X);
- FieldElement t1 = field.Square(Y);
- FieldElement t2 = field.Square(Z);
- FieldElement t3 = field.Multiply(X,Y);
- t3 = field.Add(t3,t3);
- Z3 = field.Multiply(X,Z);
- Z3 = field.Add(Z3,Z3);
- Y3 = field.Multiply(b,t2);
- Y3 = field.Subtract(Y3,Z3);
- X3 = field.Add(Y3,Y3);
- Y3 = field.Add(X3,Y3);
- X3 = field.Subtract(t1,Y3);
- Y3 = field.Add(t1,Y3);
- Y3 = field.Multiply(X3,Y3);
- X3 = field.Multiply(X3,t3);
- t3 = field.Add(t2,t2);
- t2 = field.Add(t2,t3);
- Z3 = field.Multiply(b,Z3);
- Z3 = field.Subtract(Z3,t2);
- Z3 = field.Subtract(Z3,t0);
- t3 = field.Add(Z3,Z3);
- Z3 = field.Add(Z3,t3);
- t3 = field.Add(t0,t0);
- t0 = field.Add(t3,t0);
- t0 = field.Subtract(t0,t2);
- t0 = field.Multiply(t0,Z3);
- Y3 = field.Add(Y3,t0);
- t0 = field.Multiply(Y,Z);
- t0 = field.Add(t0,t0);
- Z3 = field.Multiply(t0,Z3);
- X3 = field.Subtract(X3,Z3);
- Z3 = field.Multiply(t0,t1);
- Z3 = field.Add(Z3,Z3);
- Z3 = field.Add(Z3,Z3);
-
- const FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
- X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
-
- // More gyrations
- R.x = X3*Z3.NotZero();
- R.y = Y3*Z3.NotZero();
- R.identity = Z3.IsZero();
-
- return R;
- }
- else if (m_alpha == A_0)
- {
- const ECP::Field& field = m_ecp.GetField();
- const FieldElement b3 = field.Multiply(m_ecp.m_b, 3);
- ECP::Point& R = m_ecp.m_R;
-
- // Gyrations attempt to maintain constant-timeness
- // We need either (P.x, P.y, 1) or (0, 1, 0).
- const Integer x = P.x * IdentityToInteger(!P.identity);
- const Integer y = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
- const Integer z = 1 * IdentityToInteger(!P.identity);
-
- ProjectivePoint p(x, y, z), r;
-
- FieldElement t0 = field.Square(Y);
- Z3 = field.Add(t0,t0);
- Z3 = field.Add(Z3,Z3);
- Z3 = field.Add(Z3,Z3);
- FieldElement t1 = field.Add(Y,Z);
- FieldElement t2 = field.Square(Z);
- t2 = field.Multiply(b3,t2);
- X3 = field.Multiply(t2,Z3);
- Y3 = field.Add(t0,t2);
- Z3 = field.Multiply(t1,Z3);
- t1 = field.Add(t2,t2);
- t2 = field.Add(t1,t2);
- t0 = field.Subtract(t0,t2);
- Y3 = field.Multiply(t0,Y3);
- Y3 = field.Add(X3,Y3);
- t1 = field.Multiply(X,Y);
- X3 = field.Multiply(t0,t1);
- X3 = field.Add(X3,X3);
-
- const FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
- X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
-
- // More gyrations
- R.x = X3*Z3.NotZero();
- R.y = Y3*Z3.NotZero();
- R.identity = Z3.IsZero();
-
- return R;
- }
- else if (m_alpha == A_Star)
- {
- const ECP::Field& field = m_ecp.GetField();
- const FieldElement b3 = field.Multiply(m_ecp.m_b, 3);
- ECP::Point& R = m_ecp.m_R;
-
- // Gyrations attempt to maintain constant-timeness
- // We need either (P.x, P.y, 1) or (0, 1, 0).
- const Integer x = P.x * IdentityToInteger(!P.identity);
- const Integer y = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
- const Integer z = 1 * IdentityToInteger(!P.identity);
-
- ProjectivePoint p(x, y, z), r;
-
- FieldElement t0 = field.Square(Y);
- Z3 = field.Add(t0,t0);
- Z3 = field.Add(Z3,Z3);
- Z3 = field.Add(Z3,Z3);
- FieldElement t1 = field.Add(Y,Z);
- FieldElement t2 = field.Square(Z);
- t2 = field.Multiply(b3,t2);
- X3 = field.Multiply(t2,Z3);
- Y3 = field.Add(t0,t2);
- Z3 = field.Multiply(t1,Z3);
- t1 = field.Add(t2,t2);
- t2 = field.Add(t1,t2);
- t0 = field.Subtract(t0,t2);
- Y3 = field.Multiply(t0,Y3);
- Y3 = field.Add(X3,Y3);
- t1 = field.Multiply(X,Y);
- X3 = field.Multiply(t0,t1);
- X3 = field.Add(X3,X3);
-
- const FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
- X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
-
- // More gyrations
- R.x = X3*Z3.NotZero();
- R.y = Y3*Z3.NotZero();
- R.identity = Z3.IsZero();
-
- return R;
- }
- else // A_Montgomery
- {
- ECP::Point& R = m_ecp.m_R;
- const ECP::Field& field = m_ecp.GetField();
- const FieldElement& a = m_ecp.m_a;
-
- // More gyrations
- bool identity = static_cast<bool>(P.identity + (P.y==field.Identity()));
-
- FieldElement t = field.Square(P.x);
- t = field.Add(field.Add(field.Double(t), t), a);
- t = field.Divide(t, field.Double(P.y));
- FieldElement x = field.Subtract(field.Subtract(field.Square(t), P.x), P.x);
- R.y = field.Subtract(field.Multiply(t, field.Subtract(P.x, x)), P.y);
- R.x.swap(x);
-
- // More gyrations
- R.x *= IdentityToInteger(!identity);
- R.y *= IdentityToInteger(!identity);
- R.identity = identity;
-
- return R;
- }
-}
-
-ECP::Point ECP::AdditionFunction::operator()(const Point& P, const Point& Q) const
-{
- if (m_alpha == A_3)
- {
- const ECP::Field& field = m_ecp.GetField();
- const FieldElement& b = m_ecp.m_b;
- ECP::Point& R = m_ecp.m_R;
-
- // Gyrations attempt to maintain constant-timeness
- // We need either (P.x, P.y, 1) or (0, 1, 0).
- const Integer x1 = P.x * IdentityToInteger(!P.identity);
- const Integer y1 = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
- const Integer z1 = 1 * IdentityToInteger(!P.identity);
-
- const Integer x2 = Q.x * IdentityToInteger(!Q.identity);
- const Integer y2 = Q.y * IdentityToInteger(!Q.identity) + 1 * IdentityToInteger(Q.identity);
- const Integer z2 = 1 * IdentityToInteger(!Q.identity);
-
- ProjectivePoint p(x1, y1, z1), q(x2, y2, z2), r;
-
- FieldElement t0 = field.Multiply(X1,X2);
- FieldElement t1 = field.Multiply(Y1,Y2);
- FieldElement t2 = field.Multiply(Z1,Z2);
- FieldElement t3 = field.Add(X1,Y1);
- FieldElement t4 = field.Add(X2,Y2);
- t3 = field.Multiply(t3,t4);
- t4 = field.Add(t0,t1);
- t3 = field.Subtract(t3,t4);
- t4 = field.Add(Y1,Z1);
- X3 = field.Add(Y2,Z2);
- t4 = field.Multiply(t4,X3);
- X3 = field.Add(t1,t2);
- t4 = field.Subtract(t4,X3);
- X3 = field.Add(X1,Z1);
- Y3 = field.Add(X2,Z2);
- X3 = field.Multiply(X3,Y3);
- Y3 = field.Add(t0,t2);
- Y3 = field.Subtract(X3,Y3);
- Z3 = field.Multiply(b,t2);
- X3 = field.Subtract(Y3,Z3);
- Z3 = field.Add(X3,X3);
- X3 = field.Add(X3,Z3);
- Z3 = field.Subtract(t1,X3);
- X3 = field.Add(t1,X3);
- Y3 = field.Multiply(b,Y3);
- t1 = field.Add(t2,t2);
- t2 = field.Add(t1,t2);
- Y3 = field.Subtract(Y3,t2);
- Y3 = field.Subtract(Y3,t0);
- t1 = field.Add(Y3,Y3);
- Y3 = field.Add(t1,Y3);
- t1 = field.Add(t0,t0);
- t0 = field.Add(t1,t0);
- t0 = field.Subtract(t0,t2);
- t1 = field.Multiply(t4,Y3);
- t2 = field.Multiply(t0,Y3);
- Y3 = field.Multiply(X3,Z3);
- Y3 = field.Add(Y3,t2);
- X3 = field.Multiply(t3,X3);
- X3 = field.Subtract(X3,t1);
- Z3 = field.Multiply(t4,Z3);
- t1 = field.Multiply(t3,t0);
- Z3 = field.Add(Z3,t1);
-
- const FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
- X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
-
- // More gyrations
- R.x = X3*Z3.NotZero();
- R.y = Y3*Z3.NotZero();
- R.identity = Z3.IsZero();
-
- return R;
- }
- else if (m_alpha == A_0)
- {
- const ECP::Field& field = m_ecp.GetField();
- const FieldElement b3 = field.Multiply(m_ecp.m_b, 3);
- ECP::Point& R = m_ecp.m_R;
-
- // Gyrations attempt to maintain constant-timeness
- // We need either (P.x, P.y, 1) or (0, 1, 0).
- const Integer x1 = P.x * IdentityToInteger(!P.identity);
- const Integer y1 = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
- const Integer z1 = 1 * IdentityToInteger(!P.identity);
-
- const Integer x2 = Q.x * IdentityToInteger(!Q.identity);
- const Integer y2 = Q.y * IdentityToInteger(!Q.identity) + 1 * IdentityToInteger(Q.identity);
- const Integer z2 = 1 * IdentityToInteger(!Q.identity);
-
- ProjectivePoint p(x1, y1, z1), q(x2, y2, z2), r;
-
- FieldElement t0 = field.Square(Y);
- Z3 = field.Add(t0,t0);
- Z3 = field.Add(Z3,Z3);
- Z3 = field.Add(Z3,Z3);
- FieldElement t1 = field.Add(Y,Z);
- FieldElement t2 = field.Square(Z);
- t2 = field.Multiply(b3,t2);
- X3 = field.Multiply(t2,Z3);
- Y3 = field.Add(t0,t2);
- Z3 = field.Multiply(t1,Z3);
- t1 = field.Add(t2,t2);
- t2 = field.Add(t1,t2);
- t0 = field.Subtract(t0,t2);
- Y3 = field.Multiply(t0,Y3);
- Y3 = field.Add(X3,Y3);
- t1 = field.Multiply(X,Y);
- X3 = field.Multiply(t0,t1);
- X3 = field.Add(X3,X3);
-
- const FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
- X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
-
- // More gyrations
- R.x = X3*Z3.NotZero();
- R.y = Y3*Z3.NotZero();
- R.identity = Z3.IsZero();
-
- return R;
- }
- else if (m_alpha == A_Star)
- {
- const ECP::Field& field = m_ecp.GetField();
- const FieldElement &a = m_ecp.m_a;
- const FieldElement b3 = field.Multiply(m_ecp.m_b, 3);
- ECP::Point& R = m_ecp.m_R;
-
- // Gyrations attempt to maintain constant-timeness
- // We need either (P.x, P.y, 1) or (0, 1, 0).
- const Integer x1 = P.x * IdentityToInteger(!P.identity);
- const Integer y1 = P.y * IdentityToInteger(!P.identity) + 1 * IdentityToInteger(P.identity);
- const Integer z1 = 1 * IdentityToInteger(!P.identity);
-
- const Integer x2 = Q.x * IdentityToInteger(!Q.identity);
- const Integer y2 = Q.y * IdentityToInteger(!Q.identity) + 1 * IdentityToInteger(Q.identity);
- const Integer z2 = 1 * IdentityToInteger(!Q.identity);
-
- ProjectivePoint p(x1, y1, z1), q(x2, y2, z2), r;
-
- FieldElement t0 = field.Multiply(X1,X2);
- FieldElement t1 = field.Multiply(Y1,Y2);
- FieldElement t2 = field.Multiply(Z1,Z2);
- FieldElement t3 = field.Add(X1,Y1);
- FieldElement t4 = field.Add(X2,Y2);
- t3 = field.Multiply(t3,t4);
- t4 = field.Add(t0,t1);
- t3 = field.Subtract(t3,t4);
- t4 = field.Add(X1,Z1);
- FieldElement t5 = field.Add(X2,Z2);
- t4 = field.Multiply(t4,t5);
- t5 = field.Add(t0,t2);
- t4 = field.Subtract(t4,t5);
- t5 = field.Add(Y1,Z1);
- X3 = field.Add(Y2,Z2);
- t5 = field.Multiply(t5,X3);
- X3 = field.Add(t1,t2);
- t5 = field.Subtract(t5,X3);
- Z3 = field.Multiply(a,t4);
- X3 = field.Multiply(b3,t2);
- Z3 = field.Add(X3,Z3);
- X3 = field.Subtract(t1,Z3);
- Z3 = field.Add(t1,Z3);
- Y3 = field.Multiply(X3,Z3);
- t1 = field.Add(t0,t0);
- t1 = field.Add(t1,t0);
- t2 = field.Multiply(a,t2);
- t4 = field.Multiply(b3,t4);
- t1 = field.Add(t1,t2);
- t2 = field.Subtract(t0,t2);
- t2 = field.Multiply(a,t2);
- t4 = field.Add(t4,t2);
- t0 = field.Multiply(t1,t4);
- Y3 = field.Add(Y3,t0);
- t0 = field.Multiply(t5,t4);
- X3 = field.Multiply(t3,X3);
- X3 = field.Subtract(X3,t0);
- t0 = field.Multiply(t3,t1);
- Z3 = field.Multiply(t5,Z3);
- Z3 = field.Add(Z3,t0);
-
- const FieldElement inv = field.MultiplicativeInverse(Z3.IsZero() ? Integer::One() : Z3);
- X3 = field.Multiply(X3, inv); Y3 = field.Multiply(Y3, inv);
-
- // More gyrations
- R.x = X3*Z3.NotZero();
- R.y = Y3*Z3.NotZero();
- R.identity = Z3.IsZero();
-
- return R;
- }
- else // A_Montgomery
- {
- const ECP::Field& field = m_ecp.GetField();
- const FieldElement& a = m_ecp.m_a;
- ECP::Point& R = m_ecp.m_R, S = m_ecp.m_R;
-
- // More gyrations
- bool return_Q = P.identity;
- bool return_P = Q.identity;
- bool double_P = field.Equal(P.x, Q.x) && field.Equal(P.y, Q.y);
- bool identity = field.Equal(P.x, Q.x) && !field.Equal(P.y, Q.y);
-
- // This code taken from Double(P) for below
- identity = static_cast<bool>((double_P * (P.identity + (P.y == field.Identity()))) + identity);
-
- if (double_P)
- {
- // This code taken from Double(P)
- FieldElement t = field.Square(P.x);
- t = field.Add(field.Add(field.Double(t), t), a);
- t = field.Divide(t, field.Double(P.y));
- FieldElement x = field.Subtract(field.Subtract(field.Square(t), P.x), P.x);
- R.y = field.Subtract(field.Multiply(t, field.Subtract(P.x, x)), P.y);
- R.x.swap(x);
- }
- else
- {
- // Original Add(P,Q) code
- FieldElement t = field.Subtract(Q.y, P.y);
- t = field.Divide(t, field.Subtract(Q.x, P.x));
- FieldElement x = field.Subtract(field.Subtract(field.Square(t), P.x), Q.x);
- R.y = field.Subtract(field.Multiply(t, field.Subtract(P.x, x)), P.y);
- R.x.swap(x);
- }
-
- // More gyrations
- R.x = R.x * IdentityToInteger(!identity);
- R.y = R.y * IdentityToInteger(!identity);
- R.identity = identity;
-
- if (return_Q)
- return (R = S), Q;
- else if (return_P)
- return (R = S), P;
- else
- return (S = R), R;
- }
-}
-
-#undef X
-#undef Y
-#undef Z
-
-#undef X1
-#undef Y1
-#undef Z1
-
-#undef X2
-#undef Y2
-#undef Z2
-
-#undef X3
-#undef Y3
-#undef Z3
-
NAMESPACE_END
#endif