pub fn compress(state: &mut [u32; 5], block: &[u8]) {
let mut x = [0u32; 16];
for (slot, bytes) in x.iter_mut().zip(block.chunks_exact(4)) {
*slot = u32::from_le_bytes([bytes[0], bytes[1], bytes[2], bytes[3]]);
}
let [mut al, mut bl, mut cl, mut dl, mut el] = *state;
let [mut ar, mut br, mut cr, mut dr, mut er] = *state;
for j in 0..80 {
let tl = al
.wrapping_add(f(j, bl, cl, dl))
.wrapping_add(x[RL[j]])
.wrapping_add(kl(j))
.rotate_left(SL[j])
.wrapping_add(el);
al = el;
el = dl;
dl = cl.rotate_left(10);
cl = bl;
bl = tl;
let tr = ar
.wrapping_add(f(79 - j, br, cr, dr))
.wrapping_add(x[RR[j]])
.wrapping_add(kr(j))
.rotate_left(SR[j])
.wrapping_add(er);
ar = er;
er = dr;
dr = cr.rotate_left(10);
cr = br;
br = tr;
}
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;
}
pub 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),
}
}
pub fn kl(j: usize) -> u32 {
match j {
0..=15 => 0x0000_0000,
16..=31 => 0x5a82_7999,
32..=47 => 0x6ed9_eba1,
48..=63 => 0x8f1b_bcdc,
_ => 0xa953_fd4e,
}
}
pub fn kr(j: usize) -> u32 {
match j {
0..=15 => 0x50a2_8be6,
16..=31 => 0x5c4d_d124,
32..=47 => 0x6d70_3ef3,
48..=63 => 0x7a6d_76e9,
_ => 0x0000_0000,
}
}
#[must_use]
pub(crate) fn ripemd160_words(input: &[u8]) -> [u32; 5] {
let mut state = [
0x6745_2301,
0xefcd_ab89,
0x98ba_dcfe,
0x1032_5476,
0xc3d2_e1f0,
];
let mut blocks = input.chunks_exact(64);
for block in &mut blocks {
compress(&mut state, block);
}
let rem = blocks.remainder();
let bit_len = (input.len() as u64).wrapping_mul(8);
let mut tail = [0u8; 128];
tail[..rem.len()].copy_from_slice(rem);
tail[rem.len()] = 0x80;
let total = if rem.len() < 56 { 64 } else { 128 };
tail[total - 8..total].copy_from_slice(&bit_len.to_le_bytes());
for block in tail[..total].chunks_exact(64) {
compress(&mut state, block);
}
state.map(u32::to_be)
}
pub 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,
];
pub 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,
];
pub 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,
];
pub 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,
];