ece/crypto/holder.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
use super::Cryptographer;
use once_cell::sync::OnceCell;
static CRYPTOGRAPHER: OnceCell<&'static dyn Cryptographer> = OnceCell::new();
#[derive(Debug, thiserror::Error)]
#[error("Cryptographer already initialized")]
pub struct SetCryptographerError(());
/// Sets the global object that will be used for cryptographic operations.
///
/// This is a convenience wrapper over [`set_cryptographer`],
/// but takes a `Box<dyn Cryptographer>` instead.
#[cfg(not(feature = "backend-openssl"))]
pub fn set_boxed_cryptographer(c: Box<dyn Cryptographer>) -> Result<(), SetCryptographerError> {
// Just leak the Box. It wouldn't be freed as a `static` anyway, and we
// never allow this to be re-assigned (so it's not a meaningful memory leak).
set_cryptographer(Box::leak(c))
}
/// Sets the global object that will be used for cryptographic operations.
///
/// This function may only be called once in the lifetime of a program.
///
/// Any calls into this crate that perform cryptography prior to calling this
/// function will panic.
pub fn set_cryptographer(c: &'static dyn Cryptographer) -> Result<(), SetCryptographerError> {
CRYPTOGRAPHER.set(c).map_err(|_| SetCryptographerError(()))
}
pub(crate) fn get_cryptographer() -> &'static dyn Cryptographer {
autoinit_crypto();
*CRYPTOGRAPHER
.get()
.expect("`rust-ece` cryptographer not initialized!")
}
#[cfg(feature = "backend-openssl")]
#[inline]
fn autoinit_crypto() {
let _ = set_cryptographer(&super::openssl::OpensslCryptographer);
}
#[cfg(not(feature = "backend-openssl"))]
#[inline]
fn autoinit_crypto() {}