use crate::{Error, Result};
use crate::crypto::aead::{Context, CipherOp};
use crate::crypto::mem::Protected;
use crate::types::{AEADAlgorithm, SymmetricAlgorithm};
use openssl::cipher::Cipher;
use openssl::cipher_ctx::CipherCtx;
struct OpenSslContext {
ctx: CipherCtx,
digest_size: usize,
}
impl Context for OpenSslContext {
fn encrypt_seal(&mut self, dst: &mut [u8], src: &[u8]) -> Result<()> {
debug_assert_eq!(dst.len(), src.len() + self.digest_size());
let size = unsafe { self.ctx.cipher_update_unchecked(src, Some(dst))? };
unsafe { self.ctx.cipher_final_unchecked(&mut dst[size..])? };
self.ctx.tag(&mut dst[src.len()..])?;
Ok(())
}
fn decrypt_verify(&mut self, dst: &mut [u8], src: &[u8]) -> Result<()> {
debug_assert!(src.len() >= self.digest_size());
debug_assert_eq!(dst.len() + self.digest_size(), src.len());
let l = self.digest_size();
let ciphertext = &src[..src.len().saturating_sub(l)];
let tag = &src[src.len().saturating_sub(l)..];
let size = unsafe {
self.ctx.cipher_update_unchecked(ciphertext, Some(dst))?
};
self.ctx.set_tag(tag)?;
unsafe { self.ctx.cipher_final_unchecked(&mut dst[size..])? };
Ok(())
}
fn digest_size(&self) -> usize {
self.digest_size
}
}
impl crate::seal::Sealed for OpenSslContext {}
impl crate::crypto::backend::interface::Aead for super::Backend {
fn supports_algo(algo: AEADAlgorithm) -> bool {
(match algo {
AEADAlgorithm::EAX => false,
AEADAlgorithm::OCB => cfg!(not(osslconf = "OPENSSL_NO_OCB")),
AEADAlgorithm::GCM => true,
AEADAlgorithm::Private(_) |
AEADAlgorithm::Unknown(_) => false,
}) && {
use std::sync::OnceLock;
static MODES: [OnceLock<bool>; 3] = [const { OnceLock::new() }; 3];
MODES.get(u8::from(algo) as usize - 1)
.map(|cell| cell.get_or_init(|| {
let symm = SymmetricAlgorithm::AES128;
let key = Protected::from([0; 16]);
let nonce = &key[..algo.nonce_size().unwrap_or(16)];
Self::context(algo, symm, &key, &[], nonce,
crate::crypto::aead::CipherOp::Encrypt)
.is_ok()
}))
.cloned()
.unwrap_or(false)
}
}
fn supports_algo_with_symmetric(algo: AEADAlgorithm,
symm: SymmetricAlgorithm)
-> bool
{
match algo {
AEADAlgorithm::EAX => false,
AEADAlgorithm::OCB => match symm {
SymmetricAlgorithm::AES128 |
SymmetricAlgorithm::AES192 |
SymmetricAlgorithm::AES256 => true,
_ => false,
},
AEADAlgorithm::GCM => match symm {
SymmetricAlgorithm::AES128 |
SymmetricAlgorithm::AES192 |
SymmetricAlgorithm::AES256 => true,
_ => false,
},
AEADAlgorithm::Private(_) |
AEADAlgorithm::Unknown(_) => false,
}
}
fn context(
algo: AEADAlgorithm,
sym_algo: SymmetricAlgorithm,
key: &Protected,
aad: &[u8],
nonce: &[u8],
op: CipherOp,
) -> Result<Box<dyn Context>> {
match algo {
#[cfg(not(osslconf = "OPENSSL_NO_OCB"))]
AEADAlgorithm::OCB => {
let cipher = match sym_algo {
SymmetricAlgorithm::AES128 => Cipher::aes_128_ocb(),
SymmetricAlgorithm::AES192 => Cipher::aes_192_ocb(),
SymmetricAlgorithm::AES256 => Cipher::aes_256_ocb(),
_ => return Err(Error::UnsupportedSymmetricAlgorithm(sym_algo).into()),
};
let mut ctx = CipherCtx::new()?;
match op {
CipherOp::Encrypt =>
ctx.encrypt_init(Some(cipher), Some(key), None)?,
CipherOp::Decrypt =>
ctx.decrypt_init(Some(cipher), Some(key), None)?,
}
ctx.set_iv_length(algo.nonce_size()?)?;
match op {
CipherOp::Encrypt =>
ctx.encrypt_init(None, None, Some(nonce))?,
CipherOp::Decrypt =>
ctx.decrypt_init(None, None, Some(nonce))?,
}
ctx.set_padding(false);
ctx.cipher_update(aad, None)?;
Ok(Box::new(OpenSslContext {
ctx,
digest_size: algo.digest_size()?,
}))
},
AEADAlgorithm::GCM => {
let cipher = match sym_algo {
SymmetricAlgorithm::AES128 => Cipher::aes_128_gcm(),
SymmetricAlgorithm::AES192 => Cipher::aes_192_gcm(),
SymmetricAlgorithm::AES256 => Cipher::aes_256_gcm(),
_ => return Err(Error::UnsupportedSymmetricAlgorithm(sym_algo).into()),
};
let mut ctx = CipherCtx::new()?;
match op {
CipherOp::Encrypt =>
ctx.encrypt_init(Some(cipher), Some(key), None)?,
CipherOp::Decrypt =>
ctx.decrypt_init(Some(cipher), Some(key), None)?,
}
ctx.set_iv_length(algo.nonce_size()?)?;
match op {
CipherOp::Encrypt =>
ctx.encrypt_init(None, None, Some(nonce))?,
CipherOp::Decrypt =>
ctx.decrypt_init(None, None, Some(nonce))?,
}
ctx.set_padding(false);
ctx.cipher_update(aad, None)?;
Ok(Box::new(OpenSslContext {
ctx,
digest_size: algo.digest_size()?,
}))
},
_ => Err(Error::UnsupportedAEADAlgorithm(algo).into()),
}
}
}