1#![no_std]
42#![doc(
43 html_logo_url = "https://raw.githubusercontent.com/RustCrypto/media/6ee8e381/logo.svg",
44 html_favicon_url = "https://raw.githubusercontent.com/RustCrypto/media/6ee8e381/logo.svg"
45)]
46#![forbid(unsafe_code)]
47#![warn(missing_docs, rust_2018_idioms)]
48
49pub use digest::{self, Digest};
50
51use core::fmt;
52#[cfg(feature = "oid")]
53use digest::const_oid::{AssociatedOid, ObjectIdentifier};
54use digest::{
55 block_buffer::Eager,
56 core_api::{
57 AlgorithmName, Block, BlockSizeUser, Buffer, BufferKindUser, CoreWrapper, FixedOutputCore,
58 OutputSizeUser, Reset, UpdateCore,
59 },
60 typenum::{Unsigned, U16, U20, U32, U40, U64},
61 HashMarker, Output,
62};
63
64mod c128;
65mod c160;
66mod c256;
67mod c320;
68
69macro_rules! impl_ripemd {
70 (
71 $name:ident, $wrapped_name:ident, $mod:ident,
72 $alg_width:expr, $doc_name:expr, $output_size:ty $(,)?
73 ) => {
74 #[doc = "Core block-level"]
75 #[doc = $doc_name]
76 #[doc = " hasher state."]
77 #[derive(Clone)]
78 pub struct $name {
79 h: [u32; $mod::DIGEST_BUF_LEN],
80 block_len: u64,
81 }
82
83 impl HashMarker for $name {}
84
85 impl BlockSizeUser for $name {
86 type BlockSize = U64;
87 }
88
89 impl BufferKindUser for $name {
90 type BufferKind = Eager;
91 }
92
93 impl OutputSizeUser for $name {
94 type OutputSize = $output_size;
95 }
96
97 impl UpdateCore for $name {
98 #[inline]
99 fn update_blocks(&mut self, blocks: &[Block<Self>]) {
100 self.block_len += blocks.len() as u64;
102 for block in blocks {
103 $mod::compress(&mut self.h, block.as_ref());
104 }
105 }
106 }
107
108 impl FixedOutputCore for $name {
109 #[inline]
110 fn finalize_fixed_core(&mut self, buffer: &mut Buffer<Self>, out: &mut Output<Self>) {
111 let bs = Self::BlockSize::U64;
112 let bit_len = 8 * (buffer.get_pos() as u64 + bs * self.block_len);
113 let mut h = self.h;
114 buffer.len64_padding_le(bit_len, |block| $mod::compress(&mut h, block.as_ref()));
115
116 for (chunk, v) in out.chunks_exact_mut(4).zip(h.iter()) {
117 chunk.copy_from_slice(&v.to_le_bytes());
118 }
119 }
120 }
121
122 impl Default for $name {
123 #[inline]
124 fn default() -> Self {
125 Self {
126 h: $mod::H0,
127 block_len: 0,
128 }
129 }
130 }
131
132 impl Reset for $name {
133 #[inline]
134 fn reset(&mut self) {
135 *self = Default::default();
136 }
137 }
138
139 impl AlgorithmName for $name {
140 #[inline]
141 fn write_alg_name(f: &mut fmt::Formatter<'_>) -> fmt::Result {
142 f.write_str(concat!("Ripemd", $alg_width))
143 }
144 }
145
146 impl fmt::Debug for $name {
147 #[inline]
148 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
149 f.write_str(concat!("Ripemd", $alg_width, "Core { ... }"))
150 }
151 }
152
153 #[doc = $doc_name]
154 #[doc = " hasher."]
155 pub type $wrapped_name = CoreWrapper<$name>;
156 };
157}
158
159impl_ripemd!(Ripemd128Core, Ripemd128, c128, "128", "RIPEMD-128", U16);
160impl_ripemd!(Ripemd160Core, Ripemd160, c160, "160", "RIPEMD-160", U20);
161impl_ripemd!(Ripemd256Core, Ripemd256, c256, "256", "RIPEMD-256", U32);
162impl_ripemd!(Ripemd320Core, Ripemd320, c320, "320", "RIPEMD-320", U40);
163
164#[cfg(feature = "oid")]
165#[cfg_attr(docsrs, doc(cfg(feature = "oid")))]
166impl AssociatedOid for Ripemd128Core {
167 const OID: ObjectIdentifier = ObjectIdentifier::new_unwrap("1.3.36.3.2.2");
171}
172
173#[cfg(feature = "oid")]
174#[cfg_attr(docsrs, doc(cfg(feature = "oid")))]
175impl AssociatedOid for Ripemd160Core {
176 const OID: ObjectIdentifier = ObjectIdentifier::new_unwrap("1.3.36.3.2.1");
180}
181
182#[cfg(feature = "oid")]
183#[cfg_attr(docsrs, doc(cfg(feature = "oid")))]
184impl AssociatedOid for Ripemd256Core {
185 const OID: ObjectIdentifier = ObjectIdentifier::new_unwrap("1.3.36.3.2.3");
186}