use crate::Hasher;
const H0: [u32; 5] = [0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476, 0xC3D2E1F0];
const KL: [u32; 5] = [0x00000000, 0x5A827999, 0x6ED9EBA1, 0x8F1BBCDC, 0xA953FD4E];
const KR: [u32; 5] = [0x50A28BE6, 0x5C4DD124, 0x6D703EF3, 0x7A6D76E9, 0x00000000];
const RL: [usize; 80] = [
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 7, 4, 13, 1, 10, 6, 15, 3, 12, 0, 9, 5, 2, 14, 11, 8, 3, 10,
14, 4, 9, 15, 8, 1, 2, 7, 0, 6, 13, 11, 5, 12, 1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2, 4, 0, 5, 9, 7,
12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13,
];
const RR: [usize; 80] = [
5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, 6, 11, 3, 7, 0, 13, 5, 10, 14, 15, 8, 12, 4, 9, 1, 2, 15, 5,
1, 3, 7, 14, 6, 9, 11, 8, 12, 2, 10, 0, 4, 13, 8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14, 12, 15, 10, 4,
1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11,
];
const SL: [u32; 80] = [
11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8, 7, 6, 8, 13, 11, 9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12, 11,
13, 6, 7, 14, 9, 13, 15, 14, 8, 13, 6, 5, 12, 7, 5, 11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5, 12, 9, 15,
5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6,
];
const SR: [u32; 80] = [
8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6, 9, 13, 15, 7, 12, 8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11, 9,
7, 15, 11, 8, 6, 6, 14, 12, 13, 5, 14, 13, 13, 7, 5, 15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15, 8, 8, 5,
12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11,
];
#[inline(always)]
fn f(j: usize, x: u32, y: u32, z: u32) -> u32 {
match j {
0..=15 => x ^ y ^ z,
16..=31 => (x & y) | ((!x) & z),
32..=47 => (x | (!y)) ^ z,
48..=63 => (x & z) | (y & (!z)),
_ => x ^ (y | (!z)),
}
}
fn compress(state: &mut [u32; 5], block: &[u8]) {
let mut x = [0u32; 16];
for i in 0..16 {
x[i] = u32::from_le_bytes([block[4 * i], block[4 * i + 1], block[4 * i + 2], block[4 * i + 3]]);
}
let mut al = state[0];
let mut bl = state[1];
let mut cl = state[2];
let mut dl = state[3];
let mut el = state[4];
let mut ar = state[0];
let mut br = state[1];
let mut cr = state[2];
let mut dr = state[3];
let mut er = state[4];
for j in 0..80 {
let round = j / 16;
let t = al
.wrapping_add(f(j, bl, cl, dl))
.wrapping_add(x[RL[j]])
.wrapping_add(KL[round])
.rotate_left(SL[j])
.wrapping_add(el);
al = el;
el = dl;
dl = cl.rotate_left(10);
cl = bl;
bl = t;
let t = ar
.wrapping_add(f(79 - j, br, cr, dr))
.wrapping_add(x[RR[j]])
.wrapping_add(KR[round])
.rotate_left(SR[j])
.wrapping_add(er);
ar = er;
er = dr;
dr = cr.rotate_left(10);
cr = br;
br = t;
}
let t = state[1].wrapping_add(cl).wrapping_add(dr);
state[1] = state[2].wrapping_add(dl).wrapping_add(er);
state[2] = state[3].wrapping_add(el).wrapping_add(ar);
state[3] = state[4].wrapping_add(al).wrapping_add(br);
state[4] = state[0].wrapping_add(bl).wrapping_add(cr);
state[0] = t;
}
#[derive(Clone)]
pub struct Ripemd160 {
state: [u32; 5],
buf: [u8; 64],
buf_len: usize,
total_len: u64,
}
impl Hasher for Ripemd160 {
const OUTPUT_LEN: usize = 20;
const BLOCK_LEN: usize = 64;
fn new() -> Self {
Self {
state: H0,
buf: [0u8; 64],
buf_len: 0,
total_len: 0,
}
}
fn update(&mut self, data: &[u8]) {
let mut pos = 0;
self.total_len += data.len() as u64;
if self.buf_len > 0 {
let need = 64 - self.buf_len;
let take = need.min(data.len());
self.buf[self.buf_len..self.buf_len + take].copy_from_slice(&data[..take]);
self.buf_len += take;
pos = take;
if self.buf_len == 64 {
let block = self.buf;
compress(&mut self.state, &block);
self.buf_len = 0;
}
}
while pos + 64 <= data.len() {
compress(&mut self.state, &data[pos..pos + 64]);
pos += 64;
}
if pos < data.len() {
let remaining = data.len() - pos;
self.buf[..remaining].copy_from_slice(&data[pos..]);
self.buf_len = remaining;
}
}
fn finalize(self) -> Vec<u8> {
let mut out = vec![0u8; 20];
self.finalize_into(&mut out);
out
}
fn finalize_into(mut self, out: &mut [u8]) {
let bit_len = self.total_len * 8;
let mut pad = [0u8; 72];
pad[0] = 0x80;
let pad_len = if self.buf_len < 56 {
56 - self.buf_len
} else {
120 - self.buf_len
};
self.update(&pad[..pad_len]);
self.update(&bit_len.to_le_bytes());
for (i, word) in self.state.iter().enumerate() {
let bytes = word.to_le_bytes();
let start = i * 4;
if start + 4 <= out.len() {
out[start..start + 4].copy_from_slice(&bytes);
} else if start < out.len() {
let end = out.len() - start;
out[start..].copy_from_slice(&bytes[..end]);
}
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::Hasher;
#[test]
fn test_ripemd160_empty() {
let digest = Ripemd160::hash(b"");
let expected: [u8; 20] = [
0x9c, 0x11, 0x85, 0xa5, 0xc5, 0xe9, 0xfc, 0x54, 0x61, 0x28, 0x08, 0x97, 0x7e, 0xe8, 0xf5, 0x48, 0xb2, 0x25,
0x8d, 0x31,
];
assert_eq!(&digest[..], &expected[..]);
}
#[test]
fn test_ripemd160_abc() {
let digest = Ripemd160::hash(b"abc");
let expected: [u8; 20] = [
0x8e, 0xb2, 0x08, 0xf7, 0xe0, 0x5d, 0x98, 0x7a, 0x9b, 0x04, 0x4a, 0x8e, 0x98, 0xc6, 0xb0, 0x87, 0xf1, 0x5a,
0x0b, 0xfc,
];
assert_eq!(&digest[..], &expected[..]);
}
}