use super::{hint::unlikely, keccak256_impl as keccak256};
use crate::{B256, KECCAK256_EMPTY};
use std::mem::MaybeUninit;
pub(super) const MAX_INPUT_LEN: usize =
128 - size_of::<B256>() - size_of::<u8>() - size_of::<usize>();
const COUNT: usize = 1 << 17; static CACHE: fixed_cache::Cache<Key, B256, BuildHasher, CacheConfig> =
fixed_cache::static_cache!(Key, B256, COUNT, BuildHasher::new());
struct CacheConfig {}
impl fixed_cache::CacheConfig for CacheConfig {
const STATS: bool = false;
const EPOCHS: bool = false;
}
pub(super) fn compute(input: &[u8], imp: impl FnOnce(&[u8]) -> B256) -> B256 {
if unlikely(input.is_empty() | (input.len() > MAX_INPUT_LEN)) {
return if input.is_empty() { KECCAK256_EMPTY } else { keccak256(input) };
}
CACHE.get_or_insert_with_ref(input, imp, |input| {
let mut data = [MaybeUninit::uninit(); MAX_INPUT_LEN];
unsafe {
std::ptr::copy_nonoverlapping(input.as_ptr(), data.as_mut_ptr().cast(), input.len())
};
Key { len: input.len() as u8, data }
})
}
type BuildHasher = std::hash::BuildHasherDefault<Hasher>;
#[derive(Default)]
struct Hasher(u64);
impl std::hash::Hasher for Hasher {
#[inline]
fn finish(&self) -> u64 {
self.0
}
#[inline]
fn write(&mut self, bytes: &[u8]) {
unsafe { core::hint::assert_unchecked(bytes.len() <= MAX_INPUT_LEN) };
if bytes.len() <= 16 {
super::hint::cold_path();
}
self.0 = rapidhash::v3::rapidhash_v3_micro_inline::<false, false>(
bytes,
const { &rapidhash::v3::RapidSecrets::seed(0) },
);
}
#[inline]
fn write_usize(&mut self, i: usize) {
debug_assert!(i <= MAX_INPUT_LEN, "{i} > {MAX_INPUT_LEN}")
}
#[cfg(feature = "nightly")]
#[inline]
fn write_length_prefix(&mut self, len: usize) {
debug_assert!(len <= MAX_INPUT_LEN, "{len} > {MAX_INPUT_LEN}")
}
}
#[derive(Clone, Copy)]
struct Key {
len: u8,
data: [MaybeUninit<u8>; MAX_INPUT_LEN],
}
impl PartialEq for Key {
#[inline]
fn eq(&self, other: &Self) -> bool {
self.get() == other.get()
}
}
impl Eq for Key {}
impl std::borrow::Borrow<[u8]> for Key {
#[inline]
fn borrow(&self) -> &[u8] {
self.get()
}
}
impl std::hash::Hash for Key {
#[inline]
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
state.write(self.get());
}
}
impl Key {
#[inline]
const fn get(&self) -> &[u8] {
unsafe { std::slice::from_raw_parts(self.data.as_ptr().cast(), self.len as usize) }
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn sizes() {
assert_eq!(size_of::<Key>(), MAX_INPUT_LEN + 1);
assert_eq!(size_of::<fixed_cache::Bucket<(Key, B256)>>(), 128);
}
#[test]
fn caching() {
let mut count: usize = 0;
let mut compute = |input| {
compute(input, |x| {
count += 1;
keccak256(x)
})
};
let input = b"Hello World!";
let input2 = b"Hello World! 2";
let a = compute(input);
let b = compute(input);
let c = compute(input);
assert_eq!(a, b);
assert_eq!(a, c);
let d = compute(input2);
let e = compute(input2);
assert_ne!(a, d);
assert_eq!(d, e);
assert_eq!(count, 2);
}
}