mod rev;
use crate::bitnum::Modnum;
use crate::checksum::{CheckBuilderErr, Checksum, Digest, LinearCheck, parse_hex};
use crate::endian::{Endian, SignedInt, Signedness, WordSpec};
use crate::keyval::KeyValIter;
pub(crate) use rev::find_largest_mod;
pub use rev::reverse_modsum;
use std::fmt::Display;
use std::str::FromStr;
#[derive(Debug, Clone)]
pub struct ModSumBuilder<S: Modnum> {
width: Option<usize>,
modulus: Option<S>,
init: Option<S>,
input_endian: Option<Endian>,
output_endian: Option<Endian>,
signedness: Option<Signedness>,
negated: Option<bool>,
wordsize: Option<usize>,
check: Option<S>,
name: Option<String>,
}
impl<S: Modnum> ModSumBuilder<S> {
pub fn width(&mut self, w: usize) -> &mut Self {
self.width = Some(w);
self
}
pub fn modulus(&mut self, m: S) -> &mut Self {
self.modulus = Some(m);
self
}
pub fn init(&mut self, i: S) -> &mut Self {
self.init = Some(i);
self
}
pub fn inendian(&mut self, e: Endian) -> &mut Self {
self.input_endian = Some(e);
self
}
pub fn wordsize(&mut self, n: usize) -> &mut Self {
self.wordsize = Some(n);
self
}
pub fn outendian(&mut self, e: Endian) -> &mut Self {
self.output_endian = Some(e);
self
}
pub fn signedness(&mut self, s: Signedness) -> &mut Self {
self.signedness = Some(s);
self
}
pub fn negated(&mut self, n: bool) -> &mut Self {
self.negated = Some(n);
self
}
pub fn check(&mut self, c: S) -> &mut Self {
self.check = Some(c);
self
}
pub fn name(&mut self, n: &str) -> &mut Self {
self.name = Some(String::from(n));
self
}
pub fn build(&self) -> Result<ModSum<S>, CheckBuilderErr> {
let width = self
.width
.ok_or(CheckBuilderErr::MissingParameter("width"))?;
if width > 64 {
return Err(CheckBuilderErr::ValueOutOfRange("width"));
}
let mut modulus = self.modulus.unwrap_or_else(S::zero);
if modulus == S::zero() && width < modulus.bits() {
modulus = S::one() << width
};
let mut init = self.init.unwrap_or_else(S::zero);
if modulus != S::zero() {
init = init % modulus
};
let wordsize = self.wordsize.unwrap_or(8);
if wordsize == 0 || wordsize % 8 != 0 || wordsize > 64 {
return Err(CheckBuilderErr::ValueOutOfRange("wordsize"));
}
let negated = self.negated.unwrap_or(false);
let wordspec = WordSpec {
input_endian: self.input_endian.unwrap_or(Endian::Big),
wordsize,
output_endian: self.output_endian.unwrap_or(Endian::Big),
signedness: self.signedness.unwrap_or(Signedness::Unsigned),
};
let s = ModSum {
width,
modulus,
init,
negated,
wordspec,
name: self.name.clone(),
};
match self.check {
Some(c) => {
let mut sum = s.init();
for &x in b"123456789" {
sum = s.dig_word(sum, SignedInt::pos(x as u64));
}
s.finalize(sum);
if sum == c {
return Ok(s);
}
Err(CheckBuilderErr::CheckFail)
}
None => Ok(s),
}
}
}
#[derive(Debug, PartialEq, Eq)]
pub struct ModSum<S: Modnum> {
width: usize,
modulus: S,
init: S,
negated: bool,
wordspec: WordSpec,
name: Option<String>,
}
impl<S: Modnum> ModSum<S> {
pub fn with_options() -> ModSumBuilder<S> {
ModSumBuilder {
width: None,
modulus: None,
init: None,
input_endian: None,
output_endian: None,
signedness: None,
negated: None,
wordsize: None,
check: None,
name: None,
}
}
}
impl<Sum: Modnum> Display for ModSum<Sum> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match &self.name {
Some(n) => write!(f, "{}", n),
None => {
write!(
f,
"modsum width={} modulus={:#x} init={:#x} negated={} signedness={}",
self.width, self.modulus, self.init, self.negated, self.wordspec.signedness
)?;
if self.wordspec.word_bytes() != 1 || self.wordspec.input_endian != Endian::Little {
write!(
f,
" in_endian={} wordsize={}",
self.wordspec.input_endian, self.wordspec.wordsize,
)?;
};
if self.width > 8 {
write!(f, " out_endian={}", self.wordspec.output_endian)?;
}
Ok(())
}
}
}
}
impl<Sum: Modnum> FromStr for ModSumBuilder<Sum> {
fn from_str(s: &str) -> Result<ModSumBuilder<Sum>, CheckBuilderErr> {
let mut sum = ModSum::<Sum>::with_options();
for x in KeyValIter::new(s) {
let (current_key, current_val) = match x {
Err(key) => return Err(CheckBuilderErr::MalformedString(key)),
Ok(s) => s,
};
let crc_op = match current_key.as_str() {
"width" => usize::from_str(¤t_val).ok().map(|x| sum.width(x)),
"modulus" => Some(sum.modulus(parse_hex::<Sum>(¤t_val, "modulus")?)),
"init" => Some(sum.init(parse_hex::<Sum>(¤t_val, "init")?)),
"in_endian" => Endian::from_str(¤t_val).ok().map(|x| sum.inendian(x)),
"wordsize" => usize::from_str(¤t_val).ok().map(|x| sum.wordsize(x)),
"out_endian" => Endian::from_str(¤t_val)
.ok()
.map(|x| sum.outendian(x)),
"signedness" => Signedness::from_str(¤t_val)
.ok()
.map(|x| sum.signedness(x)),
"negated" => bool::from_str(¤t_val).ok().map(|x| sum.negated(x)),
"name" => Some(sum.name(¤t_val)),
_ => return Err(CheckBuilderErr::UnknownKey(current_key)),
};
match crc_op {
Some(c) => sum = c.clone(),
None => return Err(CheckBuilderErr::MalformedString(current_key)),
}
}
Ok(sum)
}
type Err = CheckBuilderErr;
}
impl<Sum: Modnum> FromStr for ModSum<Sum> {
fn from_str(s: &str) -> Result<ModSum<Sum>, CheckBuilderErr> {
ModSumBuilder::from_str(s)?.build()
}
type Err = CheckBuilderErr;
}
impl<S: Modnum> Digest for ModSum<S> {
type Sum = S;
fn init(&self) -> Self::Sum {
if self.negated {
self.init.neg_mod(&self.modulus)
} else {
self.init
}
}
fn dig_word(&self, sum: Self::Sum, word: SignedInt<u64>) -> Self::Sum {
let modword = S::mod_from_signed(word.negate_if(self.negated), &self.modulus);
sum.add_mod(&modword, &self.modulus)
}
fn finalize(&self, sum: Self::Sum) -> Self::Sum {
sum
}
fn to_bytes(&self, s: Self::Sum) -> Vec<u8> {
self.wordspec.output_to_bytes(s, self.width)
}
fn checksum_from_bytes(&self, bytes: &[u8]) -> Option<Self::Sum> {
Checksum::from_bytes(bytes, self.wordspec.output_endian, self.width)
}
fn wordspec(&self) -> WordSpec {
self.wordspec
}
}
impl<S: Modnum> LinearCheck for ModSum<S> {
type Shift = ();
fn init_shift(&self) -> Self::Shift {}
fn inc_shift(&self, _: Self::Shift) -> Self::Shift {}
fn shift(&self, sum: Self::Sum, _: &Self::Shift) -> Self::Sum {
sum
}
fn shift_n(&self, _: usize) -> Self::Shift {}
fn add(&self, sum_a: Self::Sum, sum_b: &Self::Sum) -> Self::Sum {
sum_a.add_mod(sum_b, &self.modulus)
}
fn negate(&self, sum: Self::Sum) -> Self::Sum {
sum.neg_mod(&self.modulus)
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::checksum::tests::{test_prop, test_shifts};
use crate::checksum::{Relativity, const_sum};
#[test]
fn modsum_8() {
let s = ModSum::<u8>::with_options()
.width(8)
.check(0xdd)
.build()
.unwrap();
test_shifts(&s);
test_prop(&s);
let s = ModSum::<u16>::with_options()
.width(8)
.check(0xdd)
.build()
.unwrap();
test_shifts(&s);
test_prop(&s);
}
#[test]
fn mod_17() {
let s = ModSum::<u8>::with_options()
.width(5)
.modulus(17)
.check(1)
.build()
.unwrap();
test_shifts(&s);
test_prop(&s);
}
#[test]
fn ethsum() {
let chk = ModSum::<u16>::with_options()
.width(16)
.init(0xff00)
.modulus(0xffff)
.check(0xde)
.build()
.unwrap();
test_shifts(&chk);
test_prop(&chk);
let many_255: Vec<_> = std::iter::repeat(0xffu8).take(0x101).collect();
assert_eq!(chk.digest(many_255.as_slice()).unwrap(), 0xff00);
let x = Vec::from("implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR ");
let y = Vec::from("This program comes with ABSOLUTELY NO WARRANTY; for details type");
let merchantibility = chk.digest(b" MERCHANTABILITY".as_ref()).unwrap();
let ith_absolutely_ = chk.digest(b"with ABSOLUTELY ".as_ref()).unwrap();
assert_eq!(
chk.find_segments(
&[x, y],
&[Some(merchantibility), Some(ith_absolutely_)].map(const_sum),
Relativity::Start
),
vec![(vec![19], vec![34])]
);
}
#[test]
fn signed_sum() {
let s = ModSum::<u16>::with_options()
.width(16)
.modulus(0)
.signedness(Signedness::Signed)
.build()
.unwrap();
let x = vec![0x80, 0x80];
assert_eq!(s.digest(x.as_slice()).unwrap(), 0xff00);
}
}