All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.crypto.tink.subtle.EllipticCurves Maven / Gradle / Ivy

// Copyright 2017 Google Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//      http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
////////////////////////////////////////////////////////////////////////////////

package com.google.crypto.tink.subtle;

import com.google.crypto.tink.internal.BigIntegerEncoding;
import com.google.crypto.tink.internal.EllipticCurvesUtil;
import java.math.BigInteger;
import java.security.GeneralSecurityException;
import java.security.InvalidAlgorithmParameterException;
import java.security.KeyFactory;
import java.security.KeyPair;
import java.security.KeyPairGenerator;
import java.security.NoSuchAlgorithmException;
import java.security.PublicKey;
import java.security.interfaces.ECPrivateKey;
import java.security.interfaces.ECPublicKey;
import java.security.spec.ECParameterSpec;
import java.security.spec.ECPoint;
import java.security.spec.ECPrivateKeySpec;
import java.security.spec.ECPublicKeySpec;
import java.security.spec.EllipticCurve;
import java.security.spec.PKCS8EncodedKeySpec;
import java.security.spec.X509EncodedKeySpec;
import java.util.Arrays;
import javax.crypto.KeyAgreement;

/**
 * Utility functions and enums for elliptic curve crypto, used in ECDSA and ECDH.
 *
 * @since 1.0.0
 */
public final class EllipticCurves {

  /**
   * Point format types UNCOMPRESSED and COMPRESSED are defined in https://www.secg.org/sec1-v2.pdf,
   * Sections 2.3.3 and 2.3.4.
   */
  public enum PointFormatType {
    UNCOMPRESSED,
    COMPRESSED,
    // Like UNCOMPRESSED but without the \x04 prefix. Crunchy uses this format.
    // DO NOT USE unless you are a Crunchy user moving to Tink.
    DO_NOT_USE_CRUNCHY_UNCOMPRESSED,
  }

  /** Elliptic curve types. */
  public enum CurveType {
    NIST_P256,
    NIST_P384,
    NIST_P521,
  }

  /** Ecdsa signature encoding. */
  public enum EcdsaEncoding {
    IEEE_P1363,
    DER,
  }

  public static ECParameterSpec getNistP256Params() {
    return EllipticCurvesUtil.NIST_P256_PARAMS;
  }

  public static ECParameterSpec getNistP384Params() {
    return EllipticCurvesUtil.NIST_P384_PARAMS;
  }

  public static ECParameterSpec getNistP521Params() {
    return EllipticCurvesUtil.NIST_P521_PARAMS;
  }


  /**
   * Checks that the point of the public key is on the curve of the public key.
   *
   * 

Warning

* *

Please use {@link #validatePublicKey} if you want to validate a public key to avoid invalid * curve attacks or small subgroup attacks in ECDH. * *

This is a sanity check, because the curve of the public key might be under control of the * adversary. * * @param key must be a key defined over a curve using a prime order field. * @throws GeneralSecurityException if the key is not valid. */ static void checkPublicKey(ECPublicKey key) throws GeneralSecurityException { EllipticCurvesUtil.checkPointOnCurve(key.getW(), key.getParams().getCurve()); } /** Returns whether {@code spec} is a {@link ECParameterSpec} of one of the NIST curves. */ public static boolean isNistEcParameterSpec(ECParameterSpec spec) { return EllipticCurvesUtil.isNistEcParameterSpec(spec); } /** Returns whether {@code one} is the same {@link ECParameterSpec} as {@code two}. */ public static boolean isSameEcParameterSpec(ECParameterSpec one, ECParameterSpec two) { return EllipticCurvesUtil.isSameEcParameterSpec(one, two); } /** * Checks that the public key's params is the same as the private key's params, and the public key * is a valid point on the private key's curve. * * @since 1.1.0 */ public static void validatePublicKey(ECPublicKey publicKey, ECPrivateKey privateKey) throws GeneralSecurityException { validatePublicKeySpec(publicKey, privateKey); EllipticCurvesUtil.checkPointOnCurve(publicKey.getW(), privateKey.getParams().getCurve()); } /** Checks that the public key's params spec is the same as the private key's params spec. */ static void validatePublicKeySpec(ECPublicKey publicKey, ECPrivateKey privateKey) throws GeneralSecurityException { try { ECParameterSpec publicKeySpec = publicKey.getParams(); ECParameterSpec privateKeySpec = privateKey.getParams(); if (!isSameEcParameterSpec(publicKeySpec, privateKeySpec)) { throw new GeneralSecurityException("invalid public key spec"); } } catch (IllegalArgumentException | NullPointerException ex) { // The Java security providers on Android K and Android L might throw these unchecked // exceptions, converting them to a checked one to not crash the JVM. throw new GeneralSecurityException(ex); } } /** * Returns the modulus of the field used by the curve specified in ecParams. * * @param curve must be a prime order elliptic curve * @return the order of the finite field over which curve is defined. */ public static BigInteger getModulus(EllipticCurve curve) throws GeneralSecurityException { return EllipticCurvesUtil.getModulus(curve); } /** * Returns the size of an element of the field over which the curve is defined. * * @param curve must be a prime order elliptic curve * @return the size of an element in bits */ public static int fieldSizeInBits(EllipticCurve curve) throws GeneralSecurityException { return getModulus(curve).subtract(BigInteger.ONE).bitLength(); } /** * Returns the size of an element of the field over which the curve is defined. * * @param curve must be a prime order elliptic curve * @return the size of an element in bytes. */ public static int fieldSizeInBytes(EllipticCurve curve) throws GeneralSecurityException { return (fieldSizeInBits(curve) + 7) / 8; } /** * Computes a square root modulo an odd prime. Timing and exceptions can leak information about * the inputs. Therefore this method must only be used to decompress public keys. * * @param x the square * @param p the prime modulus (the behaviour of the method is undefined if p is not prime). * @return a value s such that s^2 mod p == x mod p * @throws GeneralSecurityException if the square root could not be found. */ private static BigInteger modSqrt(BigInteger x, BigInteger p) throws GeneralSecurityException { if (p.signum() != 1) { throw new InvalidAlgorithmParameterException("p must be positive"); } x = x.mod(p); BigInteger squareRoot = null; // Special case for x == 0. // This check is necessary for Cipolla's algorithm. if (x.equals(BigInteger.ZERO)) { return BigInteger.ZERO; } if (p.testBit(0) && p.testBit(1)) { // Case p % 4 == 3 // q = (p + 1) / 4 BigInteger q = p.add(BigInteger.ONE).shiftRight(2); squareRoot = x.modPow(q, p); } else if (p.testBit(0) && !p.testBit(1)) { // Case p % 4 == 1 // For this case we use Cipolla's algorithm. // This alogorithm is preferrable to Tonelli-Shanks for primes p where p-1 is divisible by // a large power of 2, which is a frequent choice since it simplifies modular reduction. BigInteger a = BigInteger.ONE; BigInteger d = null; BigInteger q1 = p.subtract(BigInteger.ONE).shiftRight(1); int tries = 0; while (true) { d = a.multiply(a).subtract(x).mod(p); // Special case d==0. We need d!=0 below. if (d.equals(BigInteger.ZERO)) { return a; } // Computes the Legendre symbol. Using the Jacobi symbol would be a faster. BigInteger t = d.modPow(q1, p); if (t.add(BigInteger.ONE).equals(p)) { // d is a quadratic non-residue. break; } else if (!t.equals(BigInteger.ONE)) { // p does not divide d. Hence, t != 1 implies that p is not a prime. throw new InvalidAlgorithmParameterException("p is not prime"); } else { a = a.add(BigInteger.ONE); } tries++; // If 128 tries were not enough to find a quadratic non-residue, then it is likely that // p is not prime. To avoid an infinite loop in this case we perform a primality test. // If p is prime then this test will be done with a negligible probability of 2^{-128}. if (tries == 128) { if (!p.isProbablePrime(80)) { throw new InvalidAlgorithmParameterException("p is not prime"); } } } // Since d = a^2 - x is a quadratic non-residue modulo p, we have // a - sqrt(d) == (a + sqrt(d))^p (mod p), // and hence // x == (a + sqrt(d))(a - sqrt(d)) == (a + sqrt(d))^(p+1) (mod p). // Thus if x is square then (a + sqrt(d))^((p+1)/2) (mod p) is a square root of x. BigInteger q = p.add(BigInteger.ONE).shiftRight(1); BigInteger u = a; BigInteger v = BigInteger.ONE; for (int bit = q.bitLength() - 2; bit >= 0; bit--) { // Square u + v sqrt(d) and reduce mod p. BigInteger tmp = u.multiply(v); u = u.multiply(u).add(v.multiply(v).mod(p).multiply(d)).mod(p); v = tmp.add(tmp).mod(p); if (q.testBit(bit)) { // Multiply u + v sqrt(d) by a + sqrt(d) and reduce mod p. tmp = u.multiply(a).add(v.multiply(d)).mod(p); v = a.multiply(v).add(u).mod(p); u = tmp; } } squareRoot = u; } // The methods used to compute the square root only guarantees a correct result if the // preconditions (i.e. p prime and x is a square) are satisfied. Otherwise the value is // undefined. Hence it is important to verify that squareRoot is indeed a square root. if (squareRoot != null && squareRoot.multiply(squareRoot).mod(p).compareTo(x) != 0) { throw new GeneralSecurityException("Could not find a modular square root"); } return squareRoot; } /** * Computes the y coordinate of a point on an elliptic curve. This method can be used to * decompress elliptic curve points. * * @param x the x-coordinate of the point * @param lsb the least significant bit of the y-coordinate of the point. * @param curve this must be an elliptic curve over a prime field using Weierstrass * representation. * @return the y coordinate. * @throws GeneralSecurityException if there is no point with coordinate x on the curve, or if * curve is not supported. */ private static BigInteger computeY(BigInteger x, boolean lsb, EllipticCurve curve) throws GeneralSecurityException { BigInteger p = getModulus(curve); BigInteger a = curve.getA(); BigInteger b = curve.getB(); BigInteger rhs = x.multiply(x).add(a).multiply(x).add(b).mod(p); BigInteger y = modSqrt(rhs, p); if (lsb != y.testBit(0)) { y = p.subtract(y).mod(p); } return y; } /** * Computes the y coordinate of a point on an elliptic curve. * * @deprecated This shouldn't be used directly, use {@link pointDecode} to decompress points. */ @Deprecated public static BigInteger getY(BigInteger x, boolean lsb, EllipticCurve curve) throws GeneralSecurityException { return computeY(x, lsb, curve); } /** * Transforms a big integer to its minimal signed form, i.e., no extra zero byte at the beginning * except single one when the highest bit is set. */ private static byte[] toMinimalSignedNumber(byte[] bs) { // Remove zero prefixes. int start = 0; while (start < bs.length && bs[start] == 0) { start++; } if (start == bs.length) { start = bs.length - 1; } int extraZero = 0; // If the 1st bit is not zero, add 1 zero byte. if ((bs[start] & 0x80) == 0x80) { // Add extra zero. extraZero = 1; } byte[] res = new byte[bs.length - start + extraZero]; System.arraycopy(bs, start, res, extraZero, bs.length - start); return res; } /** * Transforms ECDSA IEEE_P1363 signature encoding to DER encoding. * *

The IEEE_P1363 signature's format is r || s, where r and s are zero-padded and have the same * size in bytes as the order of the curve. For example, for NIST P-256 curve, r and s are * zero-padded to 32 bytes. * *

The DER signature is encoded using ASN.1 (https://tools.ietf.org/html/rfc5480#appendix-A): * ECDSA-Sig-Value :: = SEQUENCE { r INTEGER, s INTEGER }. In particular, the encoding is: 0x30 || * totalLength || 0x02 || r's length || r || 0x02 || s's length || s. * * @param ieee ECDSA's signature in IEEE_P1363 format. * @return ECDSA's signature in DER format. * @throws GeneralSecurityException if ieee's length is zero, greater than 132-byte (corresponding * to NIST P521) or not divisible by 2. */ public static byte[] ecdsaIeee2Der(byte[] ieee) throws GeneralSecurityException { if (ieee.length % 2 != 0 || ieee.length == 0 || ieee.length > 132) { throw new GeneralSecurityException("Invalid IEEE_P1363 encoding"); } byte[] r = toMinimalSignedNumber(Arrays.copyOf(ieee, ieee.length / 2)); byte[] s = toMinimalSignedNumber(Arrays.copyOfRange(ieee, ieee.length / 2, ieee.length)); int offset = 0; int length = 1 + 1 + r.length + 1 + 1 + s.length; byte[] der; if (length >= 128) { der = new byte[length + 3]; der[offset++] = (byte) 0x30; der[offset++] = (byte) (0x80 + 0x01); der[offset++] = (byte) length; } else { der = new byte[length + 2]; der[offset++] = (byte) 0x30; der[offset++] = (byte) length; } der[offset++] = (byte) 0x02; der[offset++] = (byte) r.length; System.arraycopy(r, 0, der, offset, r.length); offset += r.length; der[offset++] = (byte) 0x02; der[offset++] = (byte) s.length; System.arraycopy(s, 0, der, offset, s.length); return der; } /** * Transforms ECDSA DER signature encoding to IEEE_P1363 encoding. * *

The IEEE_P1363 signature's format is r || s, where r and s are zero-padded and have the same * size in bytes as the order of the curve. For example, for NIST P-256 curve, r and s are * zero-padded to 32 bytes. * *

The DER signature is encoded using ASN.1 (https://tools.ietf.org/html/rfc5480#appendix-A): * ECDSA-Sig-Value :: = SEQUENCE { r INTEGER, s INTEGER }. In particular, the encoding is: 0x30 || * totalLength || 0x02 || r's length || r || 0x02 || s's length || s. * * @param der ECDSA's signature in DER encoding. * @param ieeeLength length of ECDSA signature's in IEEE_P1363's format which equals to 2 * (size * of elliptic curve's field in bytes). * @return ECDSA's signature in IEEE_P1363 format. * @throws GeneralSecurityException if the signature is not valid DER encoding. */ public static byte[] ecdsaDer2Ieee(byte[] der, int ieeeLength) throws GeneralSecurityException { if (!isValidDerEncoding(der)) { throw new GeneralSecurityException("Invalid DER encoding"); } byte[] ieee = new byte[ieeeLength]; int length = der[1] & 0xff; int offset = 1 /* 0x30 */ + 1 /* totalLength */; if (length >= 128) { offset++; // Long form length } offset++; // 0x02 int rLength = der[offset++]; int extraZero = 0; if (der[offset] == 0) { extraZero = 1; } System.arraycopy( der, offset + extraZero, ieee, ieeeLength / 2 - rLength + extraZero, rLength - extraZero); offset += rLength /* r byte array */ + 1 /* 0x02 */; int sLength = der[offset++]; extraZero = 0; if (der[offset] == 0) { extraZero = 1; } System.arraycopy( der, offset + extraZero, ieee, ieeeLength - sLength + extraZero, sLength - extraZero); return ieee; } // Validates that the signature is in DER encoding, based on // https://github.com/bitcoin/bips/blob/master/bip-0066.mediawiki. public static boolean isValidDerEncoding(final byte[] sig) { // Format: 0x30 [total-length] 0x02 [R-length] [R] 0x02 [S-length] [S] // * total-length: 1-byte or 2-byte length descriptor of everything that follows. // * R-length: 1-byte length descriptor of the R value that follows. // * R: arbitrary-length big-endian encoded R value. It must use the shortest // possible encoding for a positive integers (which means no null bytes at // the start, except a single one when the next byte has its highest bit set). // * S-length: 1-byte length descriptor of the S value that follows. // * S: arbitrary-length big-endian encoded S value. The same rules apply. if (sig.length < 1 /* 0x30 */ + 1 /* total-length */ + 1 /* 0x02 */ + 1 /* R-length */ + 1 /* R */ + 1 /* 0x02 */ + 1 /* S-length */ + 1 /* S */) { // Signature is too short. return false; } // Checking bytes from left to right. // byte #1: a signature is of type 0x30 (compound). if (sig[0] != 0x30) { return false; } // byte #2 and maybe #3: the total length of the signature. int totalLen = sig[1] & 0xff; int totalLenLen = 1; // the length of the total length field, could be 2-byte. if (totalLen == 129) { // The signature is >= 128 bytes thus total length field is in long-form encoding and occupies // 2 bytes. totalLenLen = 2; // byte #3 is the total length. totalLen = sig[2] & 0xff; if (totalLen < 128) { // Length in long-form encoding must be >= 128. return false; } } else if (totalLen == 128 || totalLen > 129) { // Impossible values for the second byte. return false; } // Make sure the length covers the entire sig. if (totalLen != sig.length - 1 - totalLenLen) { return false; } // Start checking R. // Check whether the R element is an integer. if (sig[1 + totalLenLen] != 0x02) { return false; } // Extract the length of the R element. int rLen = sig[1 /* 0x30 */ + totalLenLen + 1 /* 0x02 */] & 0xff; // Make sure the length of the S element is still inside the signature. if (1 /* 0x30 */ + totalLenLen + 1 /* 0x02 */ + 1 /* rLen */ + rLen + 1 /* 0x02 */ >= sig.length) { return false; } // Zero-length integers are not allowed for R. if (rLen == 0) { return false; } // Negative numbers are not allowed for R. if ((sig[3 + totalLenLen] & 0xff) >= 128) { return false; } // Null bytes at the start of R are not allowed, unless R would // otherwise be interpreted as a negative number. if (rLen > 1 && (sig[3 + totalLenLen] == 0x00) && ((sig[4 + totalLenLen] & 0xff) < 128)) { return false; } // Start checking S. // Check whether the S element is an integer. if (sig[3 + totalLenLen + rLen] != 0x02) { return false; } // Extract the length of the S element. int sLen = sig[1 /* 0x30 */ + totalLenLen + 1 /* 0x02 */ + 1 /* rLen */ + rLen + 1 /* 0x02 */] & 0xff; // Verify that the length of the signature matches the sum of the length of the elements. if (1 /* 0x30 */ + totalLenLen + 1 /* 0x02 */ + 1 /* rLen */ + rLen + 1 /* 0x02 */ + 1 /* sLen */ + sLen != sig.length) { return false; } // Zero-length integers are not allowed for S. if (sLen == 0) { return false; } // Negative numbers are not allowed for S. if ((sig[5 + totalLenLen + rLen] & 0xff) >= 128) { return false; } // Null bytes at the start of S are not allowed, unless S would // otherwise be interpreted as a negative number. if (sLen > 1 && (sig[5 + totalLenLen + rLen] == 0x00) && ((sig[6 + totalLenLen + rLen] & 0xff) < 128)) { return false; } return true; } /** * Returns the encoding size of a point on an elliptic curve. * * @param curve the elliptic curve * @param format the format used to encode the point * @return the size of an encoded point in bytes * @throws GeneralSecurityException if the point format is unknown or if the elliptic curve is not * supported */ public static int encodingSizeInBytes(EllipticCurve curve, PointFormatType format) throws GeneralSecurityException { int coordinateSize = fieldSizeInBytes(curve); switch (format) { case UNCOMPRESSED: return 2 * coordinateSize + 1; case DO_NOT_USE_CRUNCHY_UNCOMPRESSED: return 2 * coordinateSize; case COMPRESSED: return coordinateSize + 1; } throw new GeneralSecurityException("unknown EC point format"); } /** * Decodes an encoded point on an elliptic curve. This method checks that the encoded point is on * the curve. * * @param curve the elliptic curve * @param format the format used to enocde the point * @param encoded the encoded point * @return the point * @throws GeneralSecurityException if the encoded point is invalid or if the curve or format are * not supported. */ public static ECPoint ecPointDecode(EllipticCurve curve, PointFormatType format, byte[] encoded) throws GeneralSecurityException { return pointDecode(curve, format, encoded); } /** * Decodes an encoded point on an elliptic curve. This method checks that the encoded point is on * the curve. * * @param curveType the elliptic curve * @param format the format used to enocde the point * @param encoded the encoded point * @return the point * @throws GeneralSecurityException if the encoded point is invalid or if the curve or format are * not supported. * @since 1.1.0 */ public static ECPoint pointDecode(CurveType curveType, PointFormatType format, byte[] encoded) throws GeneralSecurityException { return pointDecode(getCurveSpec(curveType).getCurve(), format, encoded); } /** * Decodes an encoded point on an elliptic curve. This method checks that the encoded point is on * the curve. * * @param curve the elliptic curve * @param format the format used to enocde the point * @param encoded the encoded point * @return the point * @throws GeneralSecurityException if the encoded point is invalid or if the curve or format are * not supported. * @since 1.1.0 */ public static ECPoint pointDecode(EllipticCurve curve, PointFormatType format, byte[] encoded) throws GeneralSecurityException { int coordinateSize = fieldSizeInBytes(curve); switch (format) { case UNCOMPRESSED: { if (encoded.length != 2 * coordinateSize + 1) { throw new GeneralSecurityException("invalid point size"); } if (encoded[0] != 4) { throw new GeneralSecurityException("invalid point format"); } BigInteger x = new BigInteger(1, Arrays.copyOfRange(encoded, 1, coordinateSize + 1)); BigInteger y = new BigInteger(1, Arrays.copyOfRange(encoded, coordinateSize + 1, encoded.length)); ECPoint point = new ECPoint(x, y); EllipticCurvesUtil.checkPointOnCurve(point, curve); return point; } case DO_NOT_USE_CRUNCHY_UNCOMPRESSED: { if (encoded.length != 2 * coordinateSize) { throw new GeneralSecurityException("invalid point size"); } BigInteger x = new BigInteger(1, Arrays.copyOf(encoded, coordinateSize)); BigInteger y = new BigInteger(1, Arrays.copyOfRange(encoded, coordinateSize, encoded.length)); ECPoint point = new ECPoint(x, y); EllipticCurvesUtil.checkPointOnCurve(point, curve); return point; } case COMPRESSED: { BigInteger p = getModulus(curve); if (encoded.length != coordinateSize + 1) { throw new GeneralSecurityException("compressed point has wrong length"); } boolean lsb; if (encoded[0] == 2) { lsb = false; } else if (encoded[0] == 3) { lsb = true; } else { throw new GeneralSecurityException("invalid format"); } BigInteger x = new BigInteger(1, Arrays.copyOfRange(encoded, 1, encoded.length)); if (x.signum() == -1 || x.compareTo(p) >= 0) { throw new GeneralSecurityException("x is out of range"); } BigInteger y = computeY(x, lsb, curve); return new ECPoint(x, y); } } throw new GeneralSecurityException("invalid format:" + format); } /** * Encodes a point on an elliptic curve. * * @param curveType the elliptic curve * @param format the format for the encoding * @param point the point to encode * @return the encoded key exchange * @throws GeneralSecurityException if the point is not on the curve or if the format is not * supported. * @since 1.1.0 */ public static byte[] pointEncode(CurveType curveType, PointFormatType format, ECPoint point) throws GeneralSecurityException { return pointEncode(getCurveSpec(curveType).getCurve(), format, point); } /** * Encodes a point on an elliptic curve. * * @param curve the elliptic curve * @param format the format for the encoding * @param point the point to encode * @return the encoded key exchange * @throws GeneralSecurityException if the point is not on the curve or if the format is not * supported. * @since 1.1.0 */ public static byte[] pointEncode(EllipticCurve curve, PointFormatType format, ECPoint point) throws GeneralSecurityException { EllipticCurvesUtil.checkPointOnCurve(point, curve); int coordinateSize = fieldSizeInBytes(curve); switch (format) { case UNCOMPRESSED: { byte[] encoded = new byte[2 * coordinateSize + 1]; byte[] x = BigIntegerEncoding.toBigEndianBytes(point.getAffineX()); byte[] y = BigIntegerEncoding.toBigEndianBytes(point.getAffineY()); // Order of System.arraycopy is important because x,y can have leading 0's. System.arraycopy(y, 0, encoded, 1 + 2 * coordinateSize - y.length, y.length); System.arraycopy(x, 0, encoded, 1 + coordinateSize - x.length, x.length); encoded[0] = 4; return encoded; } case DO_NOT_USE_CRUNCHY_UNCOMPRESSED: { byte[] encoded = new byte[2 * coordinateSize]; byte[] x = BigIntegerEncoding.toBigEndianBytes(point.getAffineX()); if (x.length > coordinateSize) { // x has leading 0's, strip them. x = Arrays.copyOfRange(x, x.length - coordinateSize, x.length); } byte[] y = BigIntegerEncoding.toBigEndianBytes(point.getAffineY()); if (y.length > coordinateSize) { // y has leading 0's, strip them. y = Arrays.copyOfRange(y, y.length - coordinateSize, y.length); } System.arraycopy(y, 0, encoded, 2 * coordinateSize - y.length, y.length); System.arraycopy(x, 0, encoded, coordinateSize - x.length, x.length); return encoded; } case COMPRESSED: { byte[] encoded = new byte[coordinateSize + 1]; byte[] x = BigIntegerEncoding.toBigEndianBytes(point.getAffineX()); System.arraycopy(x, 0, encoded, 1 + coordinateSize - x.length, x.length); encoded[0] = (byte) (point.getAffineY().testBit(0) ? 3 : 2); return encoded; } } throw new GeneralSecurityException("invalid format:" + format); } /** * Returns the ECParameterSpec for a named curve. * * @param curve the curve type * @return the ECParameterSpec for the curve. */ public static ECParameterSpec getCurveSpec(CurveType curve) throws NoSuchAlgorithmException { switch (curve) { case NIST_P256: return getNistP256Params(); case NIST_P384: return getNistP384Params(); case NIST_P521: return getNistP521Params(); } throw new NoSuchAlgorithmException("curve not implemented:" + curve); } /** * Returns an {@link ECPublicKey} from {@code x509PublicKey} which is an encoding of a public key, * encoded according to the ASN.1 type SubjectPublicKeyInfo. * *

TODO(b/68672497): test that in Java one can always get this representation by using {@link * ECPublicKey#getEncoded}, regardless of the provider. */ public static ECPublicKey getEcPublicKey(final byte[] x509PublicKey) throws GeneralSecurityException { KeyFactory kf = EngineFactory.KEY_FACTORY.getInstance("EC"); return (ECPublicKey) kf.generatePublic(new X509EncodedKeySpec(x509PublicKey)); } /** * Returns an {@link ECPublicKey} from {@code publicKey} that is a public key in point format * {@code pointFormat} on {@code curve}. */ public static ECPublicKey getEcPublicKey( CurveType curve, PointFormatType pointFormat, final byte[] publicKey) throws GeneralSecurityException { return getEcPublicKey(getCurveSpec(curve), pointFormat, publicKey); } /** * Returns an {@link ECPublicKey} from {@code publicKey} that is a public key in point format * {@code pointFormat} on {@code curve}. */ public static ECPublicKey getEcPublicKey( ECParameterSpec spec, PointFormatType pointFormat, final byte[] publicKey) throws GeneralSecurityException { ECPoint point = pointDecode(spec.getCurve(), pointFormat, publicKey); ECPublicKeySpec pubSpec = new ECPublicKeySpec(point, spec); KeyFactory kf = EngineFactory.KEY_FACTORY.getInstance("EC"); return (ECPublicKey) kf.generatePublic(pubSpec); } /** * Returns an {@code ECPublicKey} from {@code curve} type and {@code x} and {@code y} coordinates. */ public static ECPublicKey getEcPublicKey(CurveType curve, final byte[] x, final byte[] y) throws GeneralSecurityException { ECParameterSpec ecParams = getCurveSpec(curve); BigInteger pubX = new BigInteger(1, x); BigInteger pubY = new BigInteger(1, y); ECPoint w = new ECPoint(pubX, pubY); EllipticCurvesUtil.checkPointOnCurve(w, ecParams.getCurve()); ECPublicKeySpec spec = new ECPublicKeySpec(w, ecParams); KeyFactory kf = EngineFactory.KEY_FACTORY.getInstance("EC"); return (ECPublicKey) kf.generatePublic(spec); } /** * Returns an {@code ECPrivateKey} from {@code pkcs8PrivateKey} which is an encoding of a private * key, encoded according to the ASN.1 type SubjectPublicKeyInfo. * *

TODO(b/68672497): test that in Java one can always get this representation by using {@link * ECPrivateKey#getEncoded}, regardless of the provider. */ public static ECPrivateKey getEcPrivateKey(final byte[] pkcs8PrivateKey) throws GeneralSecurityException { KeyFactory kf = EngineFactory.KEY_FACTORY.getInstance("EC"); return (ECPrivateKey) kf.generatePrivate(new PKCS8EncodedKeySpec(pkcs8PrivateKey)); } /** Returns an {@code ECPrivateKey} from {@code curve} type and {@code keyValue}. */ public static ECPrivateKey getEcPrivateKey(CurveType curve, final byte[] keyValue) throws GeneralSecurityException { ECParameterSpec ecParams = getCurveSpec(curve); BigInteger privValue = BigIntegerEncoding.fromUnsignedBigEndianBytes(keyValue); ECPrivateKeySpec spec = new ECPrivateKeySpec(privValue, ecParams); KeyFactory kf = EngineFactory.KEY_FACTORY.getInstance("EC"); return (ECPrivateKey) kf.generatePrivate(spec); } /** Generates a new key pair for {@code curve}. */ public static KeyPair generateKeyPair(CurveType curve) throws GeneralSecurityException { return generateKeyPair(getCurveSpec(curve)); } /** Generates a new key pair for {@code spec}. */ public static KeyPair generateKeyPair(ECParameterSpec spec) throws GeneralSecurityException { KeyPairGenerator keyGen = EngineFactory.KEY_PAIR_GENERATOR.getInstance("EC"); keyGen.initialize(spec); return keyGen.generateKeyPair(); } /** * Basic validation of the shared secret. * *

Checks that there exists a point on the curve of {@code privateKey} that has {@code secret} * as x coordinate. This validation gives very limited protection against arithmetic errors * or fault attacks. * *

Only visible for testing. */ static void validateSharedSecret(byte[] secret, ECPrivateKey privateKey) throws GeneralSecurityException { EllipticCurve privateKeyCurve = privateKey.getParams().getCurve(); BigInteger x = new BigInteger(1, secret); if (x.signum() == -1 || x.compareTo(getModulus(privateKeyCurve)) >= 0) { throw new GeneralSecurityException("shared secret is out of range"); } // This will throw if x is not a valid coordinate. Object unused = computeY(x, true /* lsb, doesn't matter here */, privateKeyCurve); } /* Generates the DH shared secret using {@code myPrivateKey} and {@code peerPublicKey} */ public static byte[] computeSharedSecret(ECPrivateKey myPrivateKey, ECPublicKey peerPublicKey) throws GeneralSecurityException { validatePublicKeySpec(peerPublicKey, myPrivateKey); return computeSharedSecret(myPrivateKey, peerPublicKey.getW()); } /** * Generates the DH shared secret using {@code myPrivateKey} and {@code publicPoint} * * @since 1.1.0 */ public static byte[] computeSharedSecret(ECPrivateKey myPrivateKey, ECPoint publicPoint) throws GeneralSecurityException { EllipticCurvesUtil.checkPointOnCurve(publicPoint, myPrivateKey.getParams().getCurve()); // Explicitly reconstruct the peer public key using private key's spec. ECParameterSpec privSpec = myPrivateKey.getParams(); ECPublicKeySpec publicKeySpec = new ECPublicKeySpec(publicPoint, privSpec); KeyFactory kf = EngineFactory.KEY_FACTORY.getInstance("EC"); PublicKey publicKey = kf.generatePublic(publicKeySpec); KeyAgreement ka = EngineFactory.KEY_AGREEMENT.getInstance("ECDH"); ka.init(myPrivateKey); try { ka.doPhase(publicKey, /* lastPhase= */ true); byte[] secret = ka.generateSecret(); validateSharedSecret(secret, myPrivateKey); return secret; } catch (IllegalStateException ex) { // Due to CVE-2017-10176 some versions of OpenJDK might throw this unchecked exception, // converting it to a checked one to not crash the JVM. See also b/73760761. throw new GeneralSecurityException(ex); } } private EllipticCurves() {} }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy