// Copyright 2015-2016 Brian Smith. // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice appear in all copies. // // THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHORS DISCLAIM ALL WARRANTIES // WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF // MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY // SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES // WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION // OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN // CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. //! Constant-time operations. use crate::{c, error}; /// Returns `Ok(())` if `a == b` and `Err(error::Unspecified)` otherwise. /// The comparison of `a` and `b` is done in constant time with respect to the /// contents of each, but NOT in constant time with respect to the lengths of /// `a` and `b`. pub fn verify_slices_are_equal(a: &[u8], b: &[u8]) -> Result<(), error::Unspecified> { if a.len() != b.len() { return Err(error::Unspecified); } let result = unsafe { OPENSSL_memcmp(a.as_ptr(), b.as_ptr(), a.len()) }; match result { 0 => Ok(()), _ => Err(error::Unspecified), } } prefixed_extern! { fn OPENSSL_memcmp(a: *const u8, b: *const u8, len: c::size_t) -> c::int; } #[cfg(test)] mod tests { use crate::limb::LimbMask; use crate::{bssl, error, rand}; #[test] fn test_constant_time() -> Result<(), error::Unspecified> { prefixed_extern! { fn bssl_constant_time_test_main() -> bssl::Result; } Result::from(unsafe { bssl_constant_time_test_main() }) } #[test] fn constant_time_conditional_memcpy() -> Result<(), error::Unspecified> { let rng = rand::SystemRandom::new(); for _ in 0..100 { let mut out = rand::generate::<[u8; 256]>(&rng)?.expose(); let input = rand::generate::<[u8; 256]>(&rng)?.expose(); // Mask to 16 bits to make zero more likely than it would otherwise be. let b = (rand::generate::<[u8; 1]>(&rng)?.expose()[0] & 0x0f) == 0; let ref_in = input; let ref_out = if b { input } else { out }; prefixed_extern! { fn bssl_constant_time_test_conditional_memcpy(dst: &mut [u8; 256], src: &[u8; 256], b: LimbMask); } unsafe { bssl_constant_time_test_conditional_memcpy( &mut out, &input, if b { LimbMask::True } else { LimbMask::False }, ) } assert_eq!(ref_in, input); assert_eq!(ref_out, out); } Ok(()) } #[test] fn constant_time_conditional_memxor() -> Result<(), error::Unspecified> { let rng = rand::SystemRandom::new(); for _ in 0..256 { let mut out = rand::generate::<[u8; 256]>(&rng)?.expose(); let input = rand::generate::<[u8; 256]>(&rng)?.expose(); // Mask to 16 bits to make zero more likely than it would otherwise be. let b = (rand::generate::<[u8; 1]>(&rng)?.expose()[0] & 0x0f) != 0; let ref_in = input; let mut ref_out = out; if b { ref_out .iter_mut() .zip(ref_in.iter()) .for_each(|(out, input)| { *out ^= input; }); } prefixed_extern! { fn bssl_constant_time_test_conditional_memxor(dst: &mut [u8; 256], src: &[u8; 256], b: LimbMask); } unsafe { bssl_constant_time_test_conditional_memxor( &mut out, &input, if b { LimbMask::True } else { LimbMask::False }, ); } assert_eq!(ref_in, input); assert_eq!(ref_out, out); } Ok(()) } }