ripemd/
lib.rs

1//! An implementation of the [RIPEMD] cryptographic hash.
2//!
3//! This crate implements only the modified 1996 versions, not the original
4//! one from 1992.
5//!
6//! Note that RIPEMD-256 provides only the same security as RIPEMD-128,
7//! and RIPEMD-320 provides only the same security as RIPEMD-160.
8//!
9//! # Usage
10//!
11//! ```rust
12//! use hex_literal::hex;
13//! use ripemd::{Ripemd160, Ripemd320, Digest};
14//!
15//! // create a RIPEMD-160 hasher instance
16//! let mut hasher = Ripemd160::new();
17//!
18//! // process input message
19//! hasher.update(b"Hello world!");
20//!
21//! // acquire hash digest in the form of GenericArray,
22//! // which in this case is equivalent to [u8; 20]
23//! let result = hasher.finalize();
24//! assert_eq!(result[..], hex!("7f772647d88750add82d8e1a7a3e5c0902a346a3"));
25//!
26//! // same for RIPEMD-320
27//! let mut hasher = Ripemd320::new();
28//! hasher.update(b"Hello world!");
29//! let result = hasher.finalize();
30//! assert_eq!(&result[..], &hex!("
31//!     f1c1c231d301abcf2d7daae0269ff3e7bc68e623
32//!     ad723aa068d316b056d26b7d1bb6f0cc0f28336d
33//! ")[..]);
34//! ```
35//!
36//! Also see [RustCrypto/hashes] readme.
37//!
38//! [RIPEMD]: https://en.wikipedia.org/wiki/RIPEMD
39//! [RustCrypto/hashes]: https://github.com/RustCrypto/hashes
40
41#![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                // Assumes that `block_len` does not overflow
101                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    /// The OID used for the RIPEMD-160. There are two OIDs defined. The Teletrust one (which is
168    /// used by almost anybody, including BouncyCastle, OpenSSL, GnuTLS, etc. and the ISO one
169    /// (1.0.10118.3.0.50), which seems to be used by nobody.
170    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    /// The OID used for the RIPEMD-160. There are two OIDs defined. The Teletrust one (which is
177    /// used by almost anybody, including BouncyCastle, OpenSSL, GnuTLS, etc. and the ISO one
178    /// (1.0.10118.3.0.49), which seems to be used by Go and nobody else.
179    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}