1#[inline]
4pub fn increment_bytes(bytes: &mut [u8]) {
5 let mut carry: u16 = 1;
6 for b in bytes {
7 carry += *b as u16;
8 *b = (carry & 0xff) as u8;
9 carry >>= 8;
10 }
11}
12
13pub fn sodium_increment(bytes: &mut [u8]) {
16 increment_bytes(bytes)
17}
18
19#[inline]
20pub(crate) fn xor_buf(out: &mut [u8], in_: &[u8]) {
21 let len = std::cmp::min(out.len(), in_.len());
22 for i in 0..len {
23 out[i] ^= in_[i];
24 }
25}
26
27#[inline]
28pub(crate) fn load_u64_le(bytes: &[u8]) -> u64 {
29 (bytes[0] as u64)
30 | ((bytes[1] as u64) << 8)
31 | ((bytes[2] as u64) << 16)
32 | ((bytes[3] as u64) << 24)
33 | ((bytes[4] as u64) << 32)
34 | ((bytes[5] as u64) << 40)
35 | ((bytes[6] as u64) << 48)
36 | ((bytes[7] as u64) << 56)
37}
38
39#[inline]
40pub(crate) fn load_u32_le(bytes: &[u8]) -> u32 {
41 (bytes[0] as u32)
42 | ((bytes[1] as u32) << 8)
43 | ((bytes[2] as u32) << 16)
44 | ((bytes[3] as u32) << 24)
45}
46
47#[inline]
48pub(crate) fn rotr64(x: u64, b: u32) -> u64 {
49 x.rotate_right(b)
50}
51
52#[inline]
53pub(crate) fn pad16(n: usize) -> usize {
54 (0x10 - (n % 16)) & 0xf
55}
56
57#[cfg(test)]
58mod tests {
59 use rand::TryRngCore;
60
61 use super::*;
62
63 #[test]
64 fn test_increment_bytes() {
65 let mut b = [0];
66
67 increment_bytes(&mut b);
68 assert_eq!(b, [1]);
69 increment_bytes(&mut b);
70 assert_eq!(b, [2]);
71
72 let mut b = [0xff];
73
74 increment_bytes(&mut b);
75 assert_eq!(b, [0]);
76 increment_bytes(&mut b);
77 assert_eq!(b, [1]);
78
79 let mut b = [0xff, 0];
80
81 increment_bytes(&mut b);
82 assert_eq!(b, [0, 1]);
83 increment_bytes(&mut b);
84 assert_eq!(b, [1, 1]);
85 increment_bytes(&mut b);
86 assert_eq!(b, [2, 1]);
87 }
88
89 #[test]
90 fn test_xor_buf() {
91 let mut a = [0];
92 let b = [0];
93
94 xor_buf(&mut a, &b);
95 assert_eq!([0], a);
96
97 let mut a = [1];
98 let b = [0];
99
100 xor_buf(&mut a, &b);
101 assert_eq!([1], a);
102
103 let mut a = [1, 1, 1];
104 let b = [0];
105
106 xor_buf(&mut a, &b);
107 assert_eq!([1, 1, 1], a);
108
109 let mut a = [1, 1, 1];
110 let b = [0];
111
112 xor_buf(&mut a, &b);
113 assert_eq!([1, 1, 1], a);
114
115 let mut a = [1, 1, 1];
116 let b = [0, 1, 1];
117
118 xor_buf(&mut a, &b);
119 assert_eq!([1, 0, 0], a);
120 }
121
122 #[test]
123 fn test_sodium_increment() {
124 use libsodium_sys::sodium_increment as so_sodium_increment;
125 use rand_core::OsRng;
126
127 use crate::rng::copy_randombytes;
128
129 for _ in 0..20 {
130 let rand_usize = (OsRng.try_next_u32().unwrap() % 1000) as usize;
131 let mut data = vec![0u8; rand_usize];
132 copy_randombytes(&mut data);
133
134 let mut data_copy = data.clone();
135
136 sodium_increment(&mut data);
137
138 unsafe { so_sodium_increment(data_copy.as_mut_ptr(), data_copy.len()) };
139
140 assert_eq!(data, data_copy);
141 }
142 }
143
144 #[test]
145 fn test_pad16() {
146 assert_eq!(pad16(0), 0);
147 assert_eq!(pad16(1), 15);
148 assert_eq!(pad16(2), 14);
149 assert_eq!(pad16(15), 1);
150 assert_eq!(pad16(16), 0);
151 assert_eq!(pad16(17), 15);
152 assert_eq!(pad16(32), 0);
153 assert_eq!(pad16(33), 15);
154 }
155}