org.bouncycastle.crypto.hpke.AEAD Maven / Gradle / Ivy
package org.bouncycastle.crypto.hpke;
import org.bouncycastle.crypto.CipherParameters;
import org.bouncycastle.crypto.InvalidCipherTextException;
import org.bouncycastle.crypto.engines.AESEngine;
import org.bouncycastle.crypto.modes.AEADCipher;
import org.bouncycastle.crypto.modes.ChaCha20Poly1305;
import org.bouncycastle.crypto.modes.GCMBlockCipher;
import org.bouncycastle.crypto.params.KeyParameter;
import org.bouncycastle.crypto.params.ParametersWithIV;
import org.bouncycastle.util.Arrays;
import org.bouncycastle.util.Pack;
public class AEAD
{
private final short aeadId;
private final byte[] key;
private final byte[] baseNonce;
private long seq = 0; // todo throw exception if overflow
private AEADCipher cipher;
public AEAD(short aeadId, byte[] key, byte[] baseNonce)
{
this.key = key;
this.baseNonce = baseNonce;
this.aeadId = aeadId;
seq = 0;
switch (aeadId)
{
case HPKE.aead_AES_GCM128:
case HPKE.aead_AES_GCM256:
cipher = new GCMBlockCipher(new AESEngine());
break;
case HPKE.aead_CHACHA20_POLY1305:
cipher = new ChaCha20Poly1305();
break;
case HPKE.aead_EXPORT_ONLY:
break;
}
}
// used by Sender
public byte[] seal(byte[] aad, byte[] pt)
throws InvalidCipherTextException
{
CipherParameters params;
switch (aeadId)
{
case HPKE.aead_AES_GCM128:
case HPKE.aead_AES_GCM256:
case HPKE.aead_CHACHA20_POLY1305:
params = new ParametersWithIV(new KeyParameter(key), ComputeNonce());
break;
case HPKE.aead_EXPORT_ONLY:
default:
throw new IllegalStateException("Export only mode, cannot be used to seal/open");
}
cipher.init(true, params);
cipher.processAADBytes(aad, 0, aad.length);
byte[] ct = new byte[cipher.getOutputSize(pt.length)];
int len = cipher.processBytes(pt, 0, pt.length, ct, 0);
cipher.doFinal(ct, len);
seq++;
return ct;
}
// used by Receiver
public byte[] open(byte[] aad, byte[] ct)
throws InvalidCipherTextException
{
CipherParameters params;
switch (aeadId)
{
case HPKE.aead_AES_GCM128:
case HPKE.aead_AES_GCM256:
case HPKE.aead_CHACHA20_POLY1305:
params = new ParametersWithIV(new KeyParameter(key), ComputeNonce());
break;
case HPKE.aead_EXPORT_ONLY:
default:
throw new IllegalStateException("Export only mode, cannot be used to seal/open");
}
cipher.init(false, params);
cipher.processAADBytes(aad, 0, aad.length);
byte[] pt = new byte[cipher.getOutputSize(ct.length)];
int len = cipher.processBytes(ct, 0, ct.length, pt, 0);
len += cipher.doFinal(pt, len);
seq++;
return pt;
}
private byte[] ComputeNonce()
{
byte[] seq_bytes = Pack.longToBigEndian(seq);
int Nn = baseNonce.length;
byte[] nonce = Arrays.clone(baseNonce);
//xor
for (int i = 0; i < 8; i++)
{
nonce[Nn-8+i] ^= seq_bytes[i];
}
return nonce;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy