1use core::fmt;
2use digest::{
3 HashMarker, Output,
4 array::Array,
5 block_api::{
6 AlgorithmName, Block, BlockSizeUser, Buffer, BufferKindUser, Eager, FixedOutputCore,
7 OutputSizeUser, Reset, UpdateCore,
8 },
9 common::hazmat::{DeserializeStateError, SerializableState, SerializedState},
10 typenum::{U20, U28, U64, Unsigned},
11};
12
13#[cfg(feature = "zeroize")]
14use digest::zeroize::{Zeroize, ZeroizeOnDrop};
15
16use crate::consts::{H0, State};
17
18pub fn compress(state: &mut State, blocks: &[[u8; 64]]) {
20 crate::compress::compress(state, blocks);
21}
22
23#[derive(Clone)]
25pub struct Sha1Core {
26 h: State,
27 block_len: u64,
28}
29
30impl HashMarker for Sha1Core {}
31
32impl BlockSizeUser for Sha1Core {
33 type BlockSize = U64;
34}
35
36impl BufferKindUser for Sha1Core {
37 type BufferKind = Eager;
38}
39
40impl OutputSizeUser for Sha1Core {
41 type OutputSize = U20;
42}
43
44impl UpdateCore for Sha1Core {
45 #[inline]
46 fn update_blocks(&mut self, blocks: &[Block<Self>]) {
47 self.block_len += blocks.len() as u64;
48 let blocks = Array::cast_slice_to_core(blocks);
49 compress(&mut self.h, blocks);
50 }
51}
52
53impl FixedOutputCore for Sha1Core {
54 #[inline]
55 fn finalize_fixed_core(&mut self, buffer: &mut Buffer<Self>, out: &mut Output<Self>) {
56 let bs = Self::BlockSize::U64;
57 let bit_len = 8 * (buffer.get_pos() as u64 + bs * self.block_len);
58
59 let mut h = self.h;
60 buffer.len64_padding_be(bit_len, |b| compress(&mut h, &[b.0]));
61 for (chunk, v) in out.chunks_exact_mut(4).zip(h.iter()) {
62 chunk.copy_from_slice(&v.to_be_bytes());
63 }
64 }
65}
66
67impl Default for Sha1Core {
68 #[inline]
69 fn default() -> Self {
70 Self {
71 h: H0,
72 block_len: 0,
73 }
74 }
75}
76
77impl Reset for Sha1Core {
78 #[inline]
79 fn reset(&mut self) {
80 *self = Default::default();
81 }
82}
83
84impl AlgorithmName for Sha1Core {
85 fn write_alg_name(f: &mut fmt::Formatter<'_>) -> fmt::Result {
86 f.write_str("Sha1")
87 }
88}
89
90impl fmt::Debug for Sha1Core {
91 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
92 f.write_str("Sha1Core { ... }")
93 }
94}
95
96impl Drop for Sha1Core {
97 fn drop(&mut self) {
98 #[cfg(feature = "zeroize")]
99 {
100 self.h.zeroize();
101 self.block_len.zeroize();
102 }
103 }
104}
105
106#[cfg(feature = "zeroize")]
107impl ZeroizeOnDrop for Sha1Core {}
108
109impl SerializableState for Sha1Core {
110 type SerializedStateSize = U28;
111
112 fn serialize(&self) -> SerializedState<Self> {
113 let mut serialized_h = SerializedState::<Self>::default();
114
115 for (val, chunk) in self.h.iter().zip(serialized_h.chunks_exact_mut(4)) {
116 chunk.copy_from_slice(&val.to_le_bytes());
117 }
118
119 serialized_h[20..].copy_from_slice(&self.block_len.to_le_bytes());
120 serialized_h
121 }
122
123 fn deserialize(
124 serialized_state: &SerializedState<Self>,
125 ) -> Result<Self, DeserializeStateError> {
126 let (serialized_h, serialized_block_len) = serialized_state.split::<U20>();
127
128 let mut h = State::default();
129 for (val, chunk) in h.iter_mut().zip(serialized_h.chunks_exact(4)) {
130 *val = u32::from_le_bytes(chunk.try_into().unwrap());
131 }
132
133 let block_len = u64::from_le_bytes(*serialized_block_len.as_ref());
134
135 Ok(Self { h, block_len })
136 }
137}