package org.spongycastle.crypto.agreement.kdf; import java.io.IOException; import org.spongycastle.asn1.ASN1EncodableVector; import org.spongycastle.asn1.ASN1Encoding; import org.spongycastle.asn1.ASN1ObjectIdentifier; import org.spongycastle.asn1.DERNull; import org.spongycastle.asn1.DEROctetString; import org.spongycastle.asn1.DERSequence; import org.spongycastle.asn1.DERTaggedObject; import org.spongycastle.asn1.x509.AlgorithmIdentifier; import org.spongycastle.crypto.DataLengthException; import org.spongycastle.crypto.DerivationFunction; import org.spongycastle.crypto.DerivationParameters; import org.spongycastle.crypto.Digest; import org.spongycastle.crypto.generators.KDF2BytesGenerator; import org.spongycastle.crypto.params.KDFParameters; /** * X9.63 based key derivation function for ECDH CMS. */ public class ECDHKEKGenerator implements DerivationFunction { private DerivationFunction kdf; private ASN1ObjectIdentifier algorithm; private int keySize; private byte[] z; public ECDHKEKGenerator( Digest digest) { this.kdf = new KDF2BytesGenerator(digest); } public void init(DerivationParameters param) { DHKDFParameters params = (DHKDFParameters)param; this.algorithm = params.getAlgorithm(); this.keySize = params.getKeySize(); this.z = params.getZ(); } public Digest getDigest() { return kdf.getDigest(); } public int generateBytes(byte[] out, int outOff, int len) throws DataLengthException, IllegalArgumentException { // TODO Create an ASN.1 class for this (RFC3278) // ECC-CMS-SharedInfo ASN1EncodableVector v = new ASN1EncodableVector(); v.add(new AlgorithmIdentifier(algorithm, new DERNull())); v.add(new DERTaggedObject(true, 2, new DEROctetString(integerToBytes(keySize)))); try { kdf.init(new KDFParameters(z, new DERSequence(v).getEncoded(ASN1Encoding.DER))); } catch (IOException e) { throw new IllegalArgumentException("unable to initialise kdf: " + e.getMessage()); } return kdf.generateBytes(out, outOff, len); } private byte[] integerToBytes(int keySize) { byte[] val = new byte[4]; val[0] = (byte)(keySize >> 24); val[1] = (byte)(keySize >> 16); val[2] = (byte)(keySize >> 8); val[3] = (byte)keySize; return val; } }