1 | use crate::{ |
2 | errors::StreamCipherError, Block, OverflowError, SeekNum, StreamCipher, StreamCipherCore, |
3 | StreamCipherSeek, StreamCipherSeekCore, |
4 | }; |
5 | use crypto_common::{ |
6 | typenum::{IsLess, Le, NonZero, Unsigned, U256}, |
7 | BlockSizeUser, Iv, IvSizeUser, Key, KeyInit, KeyIvInit, KeySizeUser, |
8 | }; |
9 | use inout::InOutBuf; |
10 | #[cfg (feature = "zeroize" )] |
11 | use zeroize::{Zeroize, ZeroizeOnDrop}; |
12 | |
13 | /// Wrapper around [`StreamCipherCore`] implementations. |
14 | /// |
15 | /// It handles data buffering and implements the slice-based traits. |
16 | #[derive (Clone, Default)] |
17 | pub struct StreamCipherCoreWrapper<T: BlockSizeUser> |
18 | where |
19 | T::BlockSize: IsLess<U256>, |
20 | Le<T::BlockSize, U256>: NonZero, |
21 | { |
22 | core: T, |
23 | buffer: Block<T>, |
24 | pos: u8, |
25 | } |
26 | |
27 | impl<T: StreamCipherCore> StreamCipherCoreWrapper<T> |
28 | where |
29 | T::BlockSize: IsLess<U256>, |
30 | Le<T::BlockSize, U256>: NonZero, |
31 | { |
32 | /// Return reference to the core type. |
33 | pub fn get_core(&self) -> &T { |
34 | &self.core |
35 | } |
36 | |
37 | /// Return reference to the core type. |
38 | pub fn from_core(core: T) -> Self { |
39 | Self { |
40 | core, |
41 | buffer: Default::default(), |
42 | pos: 0, |
43 | } |
44 | } |
45 | |
46 | /// Return current cursor position. |
47 | #[inline ] |
48 | fn get_pos(&self) -> usize { |
49 | let pos = self.pos as usize; |
50 | if T::BlockSize::USIZE == 0 { |
51 | panic!("Block size can not be equal to zero" ); |
52 | } |
53 | if pos >= T::BlockSize::USIZE { |
54 | debug_assert!(false); |
55 | // SAFETY: `pos` is set only to values smaller than block size |
56 | unsafe { core::hint::unreachable_unchecked() } |
57 | } |
58 | self.pos as usize |
59 | } |
60 | |
61 | /// Return size of the internal buffer in bytes. |
62 | #[inline ] |
63 | fn size(&self) -> usize { |
64 | T::BlockSize::USIZE |
65 | } |
66 | |
67 | #[inline ] |
68 | fn set_pos_unchecked(&mut self, pos: usize) { |
69 | debug_assert!(pos < T::BlockSize::USIZE); |
70 | self.pos = pos as u8; |
71 | } |
72 | |
73 | /// Return number of remaining bytes in the internal buffer. |
74 | #[inline ] |
75 | fn remaining(&self) -> usize { |
76 | self.size() - self.get_pos() |
77 | } |
78 | |
79 | fn check_remaining(&self, dlen: usize) -> Result<(), StreamCipherError> { |
80 | let rem_blocks = match self.core.remaining_blocks() { |
81 | Some(v) => v, |
82 | None => return Ok(()), |
83 | }; |
84 | |
85 | let bytes = if self.pos == 0 { |
86 | dlen |
87 | } else { |
88 | let rem = self.remaining(); |
89 | if dlen > rem { |
90 | dlen - rem |
91 | } else { |
92 | return Ok(()); |
93 | } |
94 | }; |
95 | let bs = T::BlockSize::USIZE; |
96 | let blocks = if bytes % bs == 0 { |
97 | bytes / bs |
98 | } else { |
99 | bytes / bs + 1 |
100 | }; |
101 | if blocks > rem_blocks { |
102 | Err(StreamCipherError) |
103 | } else { |
104 | Ok(()) |
105 | } |
106 | } |
107 | } |
108 | |
109 | impl<T: StreamCipherCore> StreamCipher for StreamCipherCoreWrapper<T> |
110 | where |
111 | T::BlockSize: IsLess<U256>, |
112 | Le<T::BlockSize, U256>: NonZero, |
113 | { |
114 | #[inline ] |
115 | fn try_apply_keystream_inout( |
116 | &mut self, |
117 | mut data: InOutBuf<'_, '_, u8>, |
118 | ) -> Result<(), StreamCipherError> { |
119 | self.check_remaining(data.len())?; |
120 | |
121 | let pos = self.get_pos(); |
122 | if pos != 0 { |
123 | let rem = &self.buffer[pos..]; |
124 | let n = data.len(); |
125 | if n < rem.len() { |
126 | data.xor_in2out(&rem[..n]); |
127 | self.set_pos_unchecked(pos + n); |
128 | return Ok(()); |
129 | } |
130 | let (mut left, right) = data.split_at(rem.len()); |
131 | data = right; |
132 | left.xor_in2out(rem); |
133 | } |
134 | |
135 | let (blocks, mut leftover) = data.into_chunks(); |
136 | self.core.apply_keystream_blocks_inout(blocks); |
137 | |
138 | let n = leftover.len(); |
139 | if n != 0 { |
140 | self.core.write_keystream_block(&mut self.buffer); |
141 | leftover.xor_in2out(&self.buffer[..n]); |
142 | } |
143 | self.set_pos_unchecked(n); |
144 | |
145 | Ok(()) |
146 | } |
147 | } |
148 | |
149 | impl<T: StreamCipherSeekCore> StreamCipherSeek for StreamCipherCoreWrapper<T> |
150 | where |
151 | T::BlockSize: IsLess<U256>, |
152 | Le<T::BlockSize, U256>: NonZero, |
153 | { |
154 | fn try_current_pos<SN: SeekNum>(&self) -> Result<SN, OverflowError> { |
155 | let Self { core: &T, pos: &u8, .. } = self; |
156 | SN::from_block_byte(block:core.get_block_pos(), *pos, T::BlockSize::U8) |
157 | } |
158 | |
159 | fn try_seek<SN: SeekNum>(&mut self, new_pos: SN) -> Result<(), StreamCipherError> { |
160 | let Self { core: &mut T, buffer: &mut GenericArray::BlockSize>, pos: &mut u8 } = self; |
161 | let (block_pos: ::Counter, byte_pos: u8) = new_pos.into_block_byte(T::BlockSize::U8)?; |
162 | core.set_block_pos(block_pos); |
163 | if byte_pos != 0 { |
164 | self.core.write_keystream_block(buffer); |
165 | } |
166 | *pos = byte_pos; |
167 | Ok(()) |
168 | } |
169 | } |
170 | |
171 | // Note: ideally we would only implement the InitInner trait and everything |
172 | // else would be handled by blanket impls, but unfortunately it will |
173 | // not work properly without mutually exclusive traits, see: |
174 | // https://github.com/rust-lang/rfcs/issues/1053 |
175 | |
176 | impl<T: KeySizeUser + BlockSizeUser> KeySizeUser for StreamCipherCoreWrapper<T> |
177 | where |
178 | T::BlockSize: IsLess<U256>, |
179 | Le<T::BlockSize, U256>: NonZero, |
180 | { |
181 | type KeySize = T::KeySize; |
182 | } |
183 | |
184 | impl<T: IvSizeUser + BlockSizeUser> IvSizeUser for StreamCipherCoreWrapper<T> |
185 | where |
186 | T::BlockSize: IsLess<U256>, |
187 | Le<T::BlockSize, U256>: NonZero, |
188 | { |
189 | type IvSize = T::IvSize; |
190 | } |
191 | |
192 | impl<T: KeyIvInit + BlockSizeUser> KeyIvInit for StreamCipherCoreWrapper<T> |
193 | where |
194 | T::BlockSize: IsLess<U256>, |
195 | Le<T::BlockSize, U256>: NonZero, |
196 | { |
197 | #[inline ] |
198 | fn new(key: &Key<Self>, iv: &Iv<Self>) -> Self { |
199 | Self { |
200 | core: T::new(key, iv), |
201 | buffer: Default::default(), |
202 | pos: 0, |
203 | } |
204 | } |
205 | } |
206 | |
207 | impl<T: KeyInit + BlockSizeUser> KeyInit for StreamCipherCoreWrapper<T> |
208 | where |
209 | T::BlockSize: IsLess<U256>, |
210 | Le<T::BlockSize, U256>: NonZero, |
211 | { |
212 | #[inline ] |
213 | fn new(key: &Key<Self>) -> Self { |
214 | Self { |
215 | core: T::new(key), |
216 | buffer: Default::default(), |
217 | pos: 0, |
218 | } |
219 | } |
220 | } |
221 | |
222 | #[cfg (feature = "zeroize" )] |
223 | impl<T> Drop for StreamCipherCoreWrapper<T> |
224 | where |
225 | T: BlockSizeUser, |
226 | T::BlockSize: IsLess<U256>, |
227 | Le<T::BlockSize, U256>: NonZero, |
228 | { |
229 | fn drop(&mut self) { |
230 | self.buffer.zeroize(); |
231 | self.pos.zeroize(); |
232 | } |
233 | } |
234 | |
235 | #[cfg (feature = "zeroize" )] |
236 | impl<T> ZeroizeOnDrop for StreamCipherCoreWrapper<T> |
237 | where |
238 | T: BlockSizeUser + ZeroizeOnDrop, |
239 | T::BlockSize: IsLess<U256>, |
240 | Le<T::BlockSize, U256>: NonZero, |
241 | { |
242 | } |
243 | |