1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55
use std::ops::Add; use std::marker::PhantomData; use std::marker::Copy; pub trait FletcherSum<T> : Add<Self> + From<T> + From<<Self as Add>::Output> + Copy { fn default_value() -> Self; fn max_chunk_size() -> usize; fn combine(lower: &Self, upper: &Self) -> Self; fn reduce(self) -> Self; } pub struct Fletcher<T, U> { a: T, b: T, phantom: PhantomData<U>, } impl<T, U> Fletcher<T, U> where T: FletcherSum<U>, U: Copy, { pub fn new() -> Fletcher<T, U> { Fletcher { a: T::default_value(), b: T::default_value(), phantom: PhantomData, } } pub fn update(&mut self, data: &[U]) { let max_chunk_size = T::max_chunk_size(); for chunk in data.chunks(max_chunk_size) { let mut intermediate_a = self.a; let mut intermediate_b = self.b; for byte in chunk { intermediate_a = T::from(intermediate_a + T::from(*byte)); intermediate_b = T::from(intermediate_b + intermediate_a); } self.a = intermediate_a.reduce(); self.b = intermediate_b.reduce(); } self.a = self.a.reduce(); self.b = self.b.reduce(); } pub fn value(&self) -> T { T::combine(&self.a, &self.b) } }