#[inline]
pub fn increment_bytes(bytes: &mut [u8]) {
let mut carry: u16 = 1;
for b in bytes {
carry += *b as u16;
*b = (carry & 0xff) as u8;
carry >>= 8;
}
}
pub fn sodium_increment(bytes: &mut [u8]) {
increment_bytes(bytes)
}
#[inline]
pub(crate) fn xor_buf(out: &mut [u8], in_: &[u8]) {
let len = std::cmp::min(out.len(), in_.len());
for i in 0..len {
out[i] ^= in_[i];
}
}
#[inline]
pub(crate) fn load_u64_le(bytes: &[u8]) -> u64 {
(bytes[0] as u64)
| ((bytes[1] as u64) << 8)
| ((bytes[2] as u64) << 16)
| ((bytes[3] as u64) << 24)
| ((bytes[4] as u64) << 32)
| ((bytes[5] as u64) << 40)
| ((bytes[6] as u64) << 48)
| ((bytes[7] as u64) << 56)
}
#[inline]
pub(crate) fn load_u32_le(bytes: &[u8]) -> u32 {
(bytes[0] as u32)
| ((bytes[1] as u32) << 8)
| ((bytes[2] as u32) << 16)
| ((bytes[3] as u32) << 24)
}
#[inline]
#[cfg_attr(all(feature = "simd_backend", feature = "nightly"), allow(dead_code))]
pub(crate) fn rotr64(x: u64, b: u32) -> u64 {
x.rotate_right(b)
}
#[inline]
pub(crate) fn pad16(n: usize) -> usize {
(0x10 - (n % 16)) & 0xf
}
#[cfg(test)]
mod tests {
use rand::TryRng;
use super::*;
#[test]
fn test_increment_bytes() {
let mut b = [0];
increment_bytes(&mut b);
assert_eq!(b, [1]);
increment_bytes(&mut b);
assert_eq!(b, [2]);
let mut b = [0xff];
increment_bytes(&mut b);
assert_eq!(b, [0]);
increment_bytes(&mut b);
assert_eq!(b, [1]);
let mut b = [0xff, 0];
increment_bytes(&mut b);
assert_eq!(b, [0, 1]);
increment_bytes(&mut b);
assert_eq!(b, [1, 1]);
increment_bytes(&mut b);
assert_eq!(b, [2, 1]);
}
#[test]
fn test_xor_buf() {
let mut a = [0];
let b = [0];
xor_buf(&mut a, &b);
assert_eq!([0], a);
let mut a = [1];
let b = [0];
xor_buf(&mut a, &b);
assert_eq!([1], a);
let mut a = [1, 1, 1];
let b = [0];
xor_buf(&mut a, &b);
assert_eq!([1, 1, 1], a);
let mut a = [1, 1, 1];
let b = [0];
xor_buf(&mut a, &b);
assert_eq!([1, 1, 1], a);
let mut a = [1, 1, 1];
let b = [0, 1, 1];
xor_buf(&mut a, &b);
assert_eq!([1, 0, 0], a);
}
#[test]
fn test_sodium_increment() {
use libsodium_sys::sodium_increment as so_sodium_increment;
use rand::rngs::SysRng;
use crate::rng::copy_randombytes;
for _ in 0..20 {
let rand_usize = (SysRng.try_next_u32().unwrap() % 1000) as usize;
let mut data = vec![0u8; rand_usize];
copy_randombytes(&mut data);
let mut data_copy = data.clone();
sodium_increment(&mut data);
unsafe { so_sodium_increment(data_copy.as_mut_ptr(), data_copy.len()) };
assert_eq!(data, data_copy);
}
}
#[test]
fn test_pad16() {
assert_eq!(pad16(0), 0);
assert_eq!(pad16(1), 15);
assert_eq!(pad16(2), 14);
assert_eq!(pad16(15), 1);
assert_eq!(pad16(16), 0);
assert_eq!(pad16(17), 15);
assert_eq!(pad16(32), 0);
assert_eq!(pad16(33), 15);
}
}