From d9ce4bee6bd0f66d977fdf684640786523a2c70a Mon Sep 17 00:00:00 2001 From: Daniel Cousens Date: Sat, 17 May 2014 11:53:05 +1000 Subject: [PATCH 1/6] ec: use signum() not equals(BigInteger.ZERO) --- src/ec.js | 22 +++++++++++----------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/src/ec.js b/src/ec.js index 854532a..bcc5648 100644 --- a/src/ec.js +++ b/src/ec.js @@ -94,15 +94,15 @@ function pointFpEquals(other) { var u, v; // u = Y2 * Z1 - Y1 * Z2 u = other.y.toBigInteger().multiply(this.z).subtract(this.y.toBigInteger().multiply(other.z)).mod(this.curve.q); - if(!u.equals(BigInteger.ZERO)) return false; + if (u.signum() !== 0) return false; // v = X2 * Z1 - X1 * Z2 v = other.x.toBigInteger().multiply(this.z).subtract(this.x.toBigInteger().multiply(other.z)).mod(this.curve.q); - return v.equals(BigInteger.ZERO); + return v.signum() === 0; } function pointFpIsInfinity() { - if((this.x == null) && (this.y == null)) return true; - return this.z.equals(BigInteger.ZERO) && !this.y.toBigInteger().equals(BigInteger.ZERO); + if ((this.x == null) && (this.y == null)) return true; + return this.z.signum() === 0 && this.y.toBigInteger().signum() !== 0; } function pointFpNegate() { @@ -118,8 +118,8 @@ function pointFpAdd(b) { // v = X2 * Z1 - X1 * Z2 var v = b.x.toBigInteger().multiply(this.z).subtract(this.x.toBigInteger().multiply(b.z)).mod(this.curve.q); - if(BigInteger.ZERO.equals(v)) { - if(BigInteger.ZERO.equals(u)) { + if(v.signum() === 0) { + if(u.signum() === 0) { return this.twice(); // this == b, so double } return this.curve.getInfinity(); // this = -b, so infinity @@ -148,7 +148,7 @@ function pointFpAdd(b) { function pointFpTwice() { if(this.isInfinity()) return this; - if(this.y.toBigInteger().signum() == 0) return this.curve.getInfinity(); + if(this.y.toBigInteger().signum() === 0) return this.curve.getInfinity(); // TODO: optimized handling of constants var THREE = new BigInteger("3"); @@ -161,7 +161,7 @@ function pointFpTwice() { // w = 3 * x1^2 + a * z1^2 var w = x1.square().multiply(THREE); - if(!BigInteger.ZERO.equals(a)) { + if(a.signum() !== 0) { w = w.add(this.z.square().multiply(a)); } w = w.mod(this.curve.q); @@ -179,7 +179,7 @@ function pointFpTwice() { // TODO: modularize the multiplication algorithm function pointFpMultiply(k) { if(this.isInfinity()) return this; - if(k.signum() == 0) return this.curve.getInfinity(); + if(k.signum() === 0) return this.curve.getInfinity() var e = k; var h = e.multiply(new BigInteger("3")); @@ -392,7 +392,7 @@ ECPointFp.prototype.add2D = function (b) { ECPointFp.prototype.twice2D = function () { if (this.isInfinity()) return this; - if (this.y.toBigInteger().signum() == 0) { + if (this.y.toBigInteger().signum() === 0) { // if y1 == 0, then (x1, y1) == (x1, -y1) // and hence this = -this and thus 2(x1, y1) == infinity return this.curve.getInfinity(); @@ -410,7 +410,7 @@ ECPointFp.prototype.twice2D = function () { ECPointFp.prototype.multiply2D = function (k) { if(this.isInfinity()) return this; - if(k.signum() == 0) return this.curve.getInfinity(); + if (k.signum() === 0) return this.curve.getInfinity() var e = k; var h = e.multiply(new BigInteger("3")); From 1e54c521d57618d04460ff7c6783080539dbf465 Mon Sep 17 00:00:00 2001 From: Daniel Cousens Date: Sat, 17 May 2014 11:56:33 +1000 Subject: [PATCH 2/6] ec: extract BigInteger constants --- src/ec.js | 22 +++++++++++----------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/src/ec.js b/src/ec.js index bcc5648..9c7ae1e 100644 --- a/src/ec.js +++ b/src/ec.js @@ -5,6 +5,11 @@ var assert = require('assert') var BigInteger = require('bigi') +// constants +var TWO = BigInteger.valueOf(2) +var THREE = BigInteger.valueOf(3) +var SEVEN = BigInteger.valueOf(7) + function ECFieldElementFp(q,x) { this.x = x; // TODO if(x.compareTo(q) >= 0) error @@ -125,7 +130,6 @@ function pointFpAdd(b) { return this.curve.getInfinity(); // this = -b, so infinity } - var THREE = new BigInteger("3"); var x1 = this.x.toBigInteger(); var y1 = this.y.toBigInteger(); var x2 = b.x.toBigInteger(); @@ -150,8 +154,6 @@ function pointFpTwice() { if(this.isInfinity()) return this; if(this.y.toBigInteger().signum() === 0) return this.curve.getInfinity(); - // TODO: optimized handling of constants - var THREE = new BigInteger("3"); var x1 = this.x.toBigInteger(); var y1 = this.y.toBigInteger(); @@ -182,7 +184,7 @@ function pointFpMultiply(k) { if(k.signum() === 0) return this.curve.getInfinity() var e = k; - var h = e.multiply(new BigInteger("3")); + var h = e.multiply(THREE) var neg = this.negate(); var R = this; @@ -327,8 +329,6 @@ ECPointFp.prototype.getEncoded = function(compressed) { return buffer } -var SEVEN = BigInteger.valueOf(7) - ECPointFp.decodeFrom = function (curve, buffer) { var type = buffer.readUInt8(0) var compressed = type !== 0x04 @@ -398,11 +398,11 @@ ECPointFp.prototype.twice2D = function () { return this.curve.getInfinity(); } - var TWO = this.curve.fromBigInteger(BigInteger.valueOf(2)); - var THREE = this.curve.fromBigInteger(BigInteger.valueOf(3)); - var gamma = this.x.square().multiply(THREE).add(this.curve.a).divide(this.y.multiply(TWO)); + var FpTWO = this.curve.fromBigInteger(TWO); + var FpTHREE = this.curve.fromBigInteger(THREE) + var gamma = this.x.square().multiply(FpTHREE).add(this.curve.a).divide(this.y.multiply(FpTWO)); - var x3 = gamma.square().subtract(this.x.multiply(TWO)); + var x3 = gamma.square().subtract(this.x.multiply(FpTWO)); var y3 = gamma.multiply(this.x.subtract(x3)).subtract(this.y); return new ECPointFp(this.curve, x3, y3); @@ -413,7 +413,7 @@ ECPointFp.prototype.multiply2D = function (k) { if (k.signum() === 0) return this.curve.getInfinity() var e = k; - var h = e.multiply(new BigInteger("3")); + var h = e.multiply(THREE) var neg = this.negate(); var R = this; From 0865f09d20e91dcf358249e0dfc464a84d5d5f65 Mon Sep 17 00:00:00 2001 From: Daniel Cousens Date: Sat, 17 May 2014 12:03:23 +1000 Subject: [PATCH 3/6] ec/dsa: extract P_OVER_FOUR as a curve specific constant This actually resolves a possible bug if multiple ecparams were used (aka different values for P_OVER_FOUR, but only the cached was used). --- src/ec.js | 6 ++++-- src/ecdsa.js | 10 ++++------ 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/src/ec.js b/src/ec.js index 9c7ae1e..d21f073 100644 --- a/src/ec.js +++ b/src/ec.js @@ -343,11 +343,13 @@ ECPointFp.decodeFrom = function (curve, buffer) { var p = curve.getQ() // We precalculate (p + 1) / 4 where p is the field order - var P_OVER_FOUR = p.add(BigInteger.ONE).shiftRight(2) + if (!curve.P_OVER_FOUR) { + curve.P_OVER_FOUR = p.add(BigInteger.ONE).shiftRight(2) + } // Convert x to point var alpha = x.square().multiply(x).add(SEVEN).mod(p) - var beta = alpha.modPow(P_OVER_FOUR, p) + var beta = alpha.modPow(curve.P_OVER_FOUR, p) // If beta is even, but y isn't, or vice versa, then convert it, // otherwise we're done and y == beta. diff --git a/src/ecdsa.js b/src/ecdsa.js index 9993af0..0490523 100644 --- a/src/ecdsa.js +++ b/src/ecdsa.js @@ -6,8 +6,6 @@ var ecparams = sec("secp256k1") var BigInteger = require('bigi') var ECPointFp = require('./ec').ECPointFp -var P_OVER_FOUR = null - function implShamirsTrick(P, k, Q, l) { var m = Math.max(k.bitLength(), l.bitLength()) var Z = P.add2D(Q) @@ -257,9 +255,9 @@ var ecdsa = { var a = curve.getA().toBigInteger() var b = curve.getB().toBigInteger() - // We precalculate (p + 1) / 4 where p is if the field order - if (!P_OVER_FOUR) { - P_OVER_FOUR = p.add(BigInteger.ONE).divide(BigInteger.valueOf(4)) + // We precalculate (p + 1) / 4 where p is the field order + if (!curve.P_OVER_FOUR) { + curve.P_OVER_FOUR = p.add(BigInteger.ONE).shiftRight(2) } // 1.1 Compute x @@ -267,7 +265,7 @@ var ecdsa = { // 1.3 Convert x to point var alpha = x.multiply(x).multiply(x).add(a.multiply(x)).add(b).mod(p) - var beta = alpha.modPow(P_OVER_FOUR, p) + var beta = alpha.modPow(curve.P_OVER_FOUR, p) // If beta is even, but y isn't, or vice versa, then convert it, // otherwise we're done and y == beta. From 7d89ff427c84cb4545bfe1bcadb71ef567329429 Mon Sep 17 00:00:00 2001 From: Daniel Cousens Date: Sat, 17 May 2014 12:04:32 +1000 Subject: [PATCH 4/6] ecdsa: comment cleanup --- src/ecdsa.js | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/src/ecdsa.js b/src/ecdsa.js index 0490523..6d78516 100644 --- a/src/ecdsa.js +++ b/src/ecdsa.js @@ -75,7 +75,7 @@ var ecdsa = { var N_OVER_TWO = n.divide(BigInteger.valueOf(2)) - // Make 's' value 'low' as per bip62 + // enforce low S values, see bip62: 'low s values in signatures' if (s.compareTo(N_OVER_TWO) > 0) { s = n.subtract(s) } @@ -122,7 +122,6 @@ var ecdsa = { } var c = s.modInverse(n) - var u1 = e.multiply(c).mod(n) var u2 = r.multiply(c).mod(n) @@ -271,7 +270,7 @@ var ecdsa = { // otherwise we're done and y == beta. var y = (beta.isEven() ^ isYEven) ? p.subtract(beta) : beta - // 1.4 Check that nR is at infinity + // 1.4 Check that nR isn't at infinity var R = new ECPointFp(curve, curve.fromBigInteger(x), curve.fromBigInteger(y)) R.validate() From 5ec9504ed6883649da791105697ddb1cd5f79696 Mon Sep 17 00:00:00 2001 From: Daniel Cousens Date: Sat, 17 May 2014 12:05:05 +1000 Subject: [PATCH 5/6] ecdsa: opt for shiftRight, pow and square In the given situations, these offer better readability, or in the case of shiftRight, a substantial performance increase. --- src/ec.js | 13 ++++++------- src/ecdsa.js | 4 ++-- test/ecdsa.js | 3 ++- 3 files changed, 10 insertions(+), 10 deletions(-) diff --git a/src/ec.js b/src/ec.js index d21f073..c31ce43 100644 --- a/src/ec.js +++ b/src/ec.js @@ -170,9 +170,9 @@ function pointFpTwice() { // x3 = 2 * y1 * z1 * (w^2 - 8 * x1 * y1^2 * z1) var x3 = w.square().subtract(x1.shiftLeft(3).multiply(y1sqz1)).shiftLeft(1).multiply(y1z1).mod(this.curve.q); // y3 = 4 * y1^2 * z1 * (3 * w * x1 - 2 * y1^2 * z1) - w^3 - var y3 = w.multiply(THREE).multiply(x1).subtract(y1sqz1.shiftLeft(1)).shiftLeft(2).multiply(y1sqz1).subtract(w.square().multiply(w)).mod(this.curve.q); + var y3 = w.multiply(THREE).multiply(x1).subtract(y1sqz1.shiftLeft(1)).shiftLeft(2).multiply(y1sqz1).subtract(w.pow(3)).mod(this.curve.q); // z3 = 8 * (y1 * z1)^3 - var z3 = y1z1.square().multiply(y1z1).shiftLeft(3).mod(this.curve.q); + var z3 = y1z1.pow(3).shiftLeft(3).mod(this.curve.q); return new ECPointFp(this.curve, this.curve.fromBigInteger(x3), this.curve.fromBigInteger(y3), z3); } @@ -348,7 +348,7 @@ ECPointFp.decodeFrom = function (curve, buffer) { } // Convert x to point - var alpha = x.square().multiply(x).add(SEVEN).mod(p) + var alpha = x.pow(3).add(SEVEN).mod(p) var beta = alpha.modPow(curve.P_OVER_FOUR, p) // If beta is even, but y isn't, or vice versa, then convert it, @@ -440,10 +440,9 @@ ECPointFp.prototype.isOnCurve = function () { var y = this.getY().toBigInteger(); var a = this.curve.getA().toBigInteger(); var b = this.curve.getB().toBigInteger(); - var n = this.curve.getQ(); - var lhs = y.multiply(y).mod(n); - var rhs = x.multiply(x).multiply(x) - .add(a.multiply(x)).add(b).mod(n); + var p = this.curve.getQ() + var lhs = y.square().mod(p) + var rhs = x.pow(3).add(a.multiply(x)).add(b).mod(p) return lhs.equals(rhs); }; diff --git a/src/ecdsa.js b/src/ecdsa.js index 6d78516..acf88c3 100644 --- a/src/ecdsa.js +++ b/src/ecdsa.js @@ -73,7 +73,7 @@ var ecdsa = { var s = k.modInverse(n).multiply(e.add(D.multiply(r))).mod(n) assert.notEqual(s.signum(), 0, 'Invalid S value') - var N_OVER_TWO = n.divide(BigInteger.valueOf(2)) + var N_OVER_TWO = n.shiftRight(1) // enforce low S values, see bip62: 'low s values in signatures' if (s.compareTo(N_OVER_TWO) > 0) { @@ -263,7 +263,7 @@ var ecdsa = { var x = isSecondKey ? r.add(n) : r // 1.3 Convert x to point - var alpha = x.multiply(x).multiply(x).add(a.multiply(x)).add(b).mod(p) + var alpha = x.pow(3).add(a.multiply(x)).add(b).mod(p) var beta = alpha.modPow(curve.P_OVER_FOUR, p) // If beta is even, but y isn't, or vice versa, then convert it, diff --git a/test/ecdsa.js b/test/ecdsa.js index aef4f2d..9decb27 100644 --- a/test/ecdsa.js +++ b/test/ecdsa.js @@ -59,7 +59,8 @@ describe('ecdsa', function() { var psig = ecdsa.parseSig(signature) // See BIP62 for more information - assert(psig.s.compareTo(ecparams.getN().divide(BigInteger.valueOf(2))) <= 0) + var N_OVER_TWO = ecparams.getN().shiftRight(1) + assert(psig.s.compareTo(N_OVER_TWO) <= 0) }) }) From a17a5b23d9052db3980a8f7c7df85c132983c891 Mon Sep 17 00:00:00 2001 From: Daniel Cousens Date: Sat, 17 May 2014 14:10:32 +1000 Subject: [PATCH 6/6] ec: fix point decoding for other curves --- src/ec.js | 5 +++-- test/ec.js | 15 +++++++++++++++ 2 files changed, 18 insertions(+), 2 deletions(-) diff --git a/src/ec.js b/src/ec.js index c31ce43..b0f82ac 100644 --- a/src/ec.js +++ b/src/ec.js @@ -8,7 +8,6 @@ var BigInteger = require('bigi') // constants var TWO = BigInteger.valueOf(2) var THREE = BigInteger.valueOf(3) -var SEVEN = BigInteger.valueOf(7) function ECFieldElementFp(q,x) { this.x = x; @@ -340,6 +339,8 @@ ECPointFp.decodeFrom = function (curve, buffer) { assert(type === 0x02 || type === 0x03, 'Invalid sequence tag') var isYEven = (type === 0x02) + var a = curve.getA().toBigInteger() + var b = curve.getB().toBigInteger() var p = curve.getQ() // We precalculate (p + 1) / 4 where p is the field order @@ -348,7 +349,7 @@ ECPointFp.decodeFrom = function (curve, buffer) { } // Convert x to point - var alpha = x.pow(3).add(SEVEN).mod(p) + var alpha = x.pow(3).add(a.multiply(x)).add(b).mod(p) var beta = alpha.modPow(curve.P_OVER_FOUR, p) // If beta is even, but y isn't, or vice versa, then convert it, diff --git a/test/ec.js b/test/ec.js index 9cd74c8..f9aa100 100644 --- a/test/ec.js +++ b/test/ec.js @@ -64,6 +64,21 @@ describe('ec', function() { }) }) }) + + it('supports other curves', function() { + var f = fixtures.valid.ECPointFp[1] + var ecparams2 = sec('secp256r1') + var curve = ecparams2.getCurve() + + var D = BigInteger.ONE + var Q = ecparams2.getG().multiply(D) + + var buffer = Q.getEncoded(true) + var decoded = ECPointFp.decodeFrom(curve, buffer) + + assert(Q.equals(decoded.Q)) + assert(decoded.compressed, true) + }) }) }) })