use nom::bytes::streaming::take;
use nom::combinator::map_parser;
use nom::error::{make_error, ErrorKind, ParseError};
use nom::{IResult, Needed, Parser};
use nom::{InputIter, InputTake};
use nom::{InputLength, ToUsize};
#[deprecated(since = "3.0.1", note = "please use `be_var_u64` instead")]
pub fn bytes_to_u64(s: &[u8]) -> Result<u64, &'static str> {
let mut u: u64 = 0;
if s.is_empty() {
return Err("empty");
};
if s.len() > 8 {
return Err("overflow");
}
for &c in s {
let u1 = u << 8;
u = u1 | (c as u64);
}
Ok(u)
}
#[inline]
pub fn be_var_u64<'a, E: ParseError<&'a [u8]>>(input: &'a [u8]) -> IResult<&'a [u8], u64, E> {
if input.is_empty() {
return Err(nom::Err::Incomplete(Needed::new(1)));
}
if input.len() > 8 {
return Err(nom::Err::Error(make_error(input, ErrorKind::TooLarge)));
}
let mut res = 0u64;
for byte in input {
res = (res << 8) + *byte as u64;
}
Ok((&b""[..], res))
}
#[inline]
pub fn le_var_u64<'a, E: ParseError<&'a [u8]>>(input: &'a [u8]) -> IResult<&'a [u8], u64, E> {
if input.is_empty() {
return Err(nom::Err::Incomplete(Needed::new(1)));
}
if input.len() > 8 {
return Err(nom::Err::Error(make_error(input, ErrorKind::TooLarge)));
}
let mut res = 0u64;
for byte in input.iter().rev() {
res = (res << 8) + *byte as u64;
}
Ok((&b""[..], res))
}
#[inline]
pub fn parse_hex_to_u64<S>(i: &[u8], size: S) -> IResult<&[u8], u64>
where
S: ToUsize + Copy,
{
map_parser(take(size.to_usize()), be_var_u64)(i)
}
pub fn upgrade_error<I, O, E1: ParseError<I>, E2: ParseError<I>, F>(
mut f: F,
) -> impl FnMut(I) -> IResult<I, O, E2>
where
F: FnMut(I) -> IResult<I, O, E1>,
E2: From<E1>,
{
move |i| f(i).map_err(nom::Err::convert)
}
pub fn pure<I, O, E: ParseError<I>>(val: O) -> impl Fn(I) -> IResult<I, O, E>
where
O: Clone,
{
move |input: I| Ok((input, val.clone()))
}
pub fn flat_take<I, C, O, E: ParseError<I>, F>(
len: C,
mut parser: F,
) -> impl FnMut(I) -> IResult<I, O, E>
where
I: InputTake + InputLength + InputIter,
C: ToUsize + Copy,
F: Parser<I, O, E>,
{
move |input: I| {
let (input, o1) = take(len.to_usize())(input)?;
let (_, o2) = parser.parse(o1)?;
Ok((input, o2))
}
}
pub fn flat_takec<I, O, E: ParseError<I>, C, F>(input: I, len: C, parser: F) -> IResult<I, O, E>
where
C: ToUsize + Copy,
F: Parser<I, O, E>,
I: InputTake + InputLength + InputIter,
O: InputLength,
{
flat_take(len, parser)(input)
}
pub fn cond_else<I, O, E: ParseError<I>, C, F, G>(
cond: C,
mut first: F,
mut second: G,
) -> impl FnMut(I) -> IResult<I, O, E>
where
C: Fn() -> bool,
F: Parser<I, O, E>,
G: Parser<I, O, E>,
{
move |input: I| {
if cond() {
first.parse(input)
} else {
second.parse(input)
}
}
}
pub const fn align_n2(x: usize, n: usize) -> usize {
(x + (n - 1)) & !(n - 1)
}
pub const fn align32(x: usize) -> usize {
(x + 3) & !3
}
#[cfg(test)]
mod tests {
use super::{align32, be_var_u64, cond_else, flat_take, pure};
use nom::bytes::streaming::take;
use nom::number::streaming::{be_u16, be_u32, be_u8};
use nom::{Err, IResult, Needed};
#[test]
fn test_be_var_u64() {
let res: IResult<&[u8], u64> = be_var_u64(b"\x12\x34\x56");
let (_, v) = res.expect("be_var_u64 failed");
assert_eq!(v, 0x123456);
}
#[test]
fn test_flat_take() {
let input = &[0x00, 0x01, 0xff];
let res: IResult<&[u8], u16> = flat_take(2u8, be_u16)(input);
assert_eq!(res, Ok((&input[2..], 0x0001)));
let res: IResult<&[u8], u16> = flat_take(3u8, be_u16)(input);
assert_eq!(res, Ok((&b""[..], 0x0001)));
let res: IResult<&[u8], u32> = flat_take(2u8, be_u32)(input);
assert_eq!(res, Err(Err::Incomplete(Needed::new(2))));
}
#[test]
fn test_flat_take_str() {
let input = "abcdef";
let res: IResult<&str, &str> = flat_take(2u8, take(2u8))(input);
assert_eq!(res, Ok(("cdef", "ab")));
let res: IResult<&str, &str> = flat_take(3u8, take(2u8))(input);
assert_eq!(res, Ok(("def", "ab")));
let res: IResult<&str, &str> = flat_take(2u8, take(4u8))(input);
assert_eq!(res, Err(Err::Incomplete(Needed::Unknown)));
}
#[test]
fn test_cond_else() {
let input = &[0x01][..];
let empty = &b""[..];
let a = 1;
fn parse_u8(i: &[u8]) -> IResult<&[u8], u8> {
be_u8(i)
}
assert_eq!(
cond_else(|| a == 1, parse_u8, pure(0x02))(input),
Ok((empty, 0x01))
);
assert_eq!(
cond_else(|| a == 1, parse_u8, pure(0x02))(input),
Ok((empty, 0x01))
);
assert_eq!(
cond_else(|| a == 2, parse_u8, pure(0x02))(input),
Ok((input, 0x02))
);
assert_eq!(
cond_else(|| a == 1, pure(0x02), parse_u8)(input),
Ok((input, 0x02))
);
let res: IResult<&[u8], u8> = cond_else(|| a == 1, parse_u8, parse_u8)(input);
assert_eq!(res, Ok((empty, 0x01)));
}
#[test]
fn test_align32() {
assert_eq!(align32(3), 4);
assert_eq!(align32(4), 4);
assert_eq!(align32(5), 8);
assert_eq!(align32(5usize), 8);
}
}