use crate::{
bytes::Bytes,
types::extra::{LeEqU128, LeEqU16, LeEqU32, LeEqU64, LeEqU8},
FixedI128, FixedI16, FixedI32, FixedI64, FixedI8, FixedU128, FixedU16, FixedU32, FixedU64,
FixedU8,
};
use core::{
fmt::{Display, Formatter, Result as FmtResult},
str::FromStr,
};
#[cfg(feature = "std")]
use std::error::Error;
macro_rules! signed_helpers {
($Single:ident, $Uns:ident) => {
pub mod $Single {
helpers_common! { $Single }
pub const fn from_str(
bytes: &[u8],
radix: u32,
frac_nbits: u32,
) -> Result<($Single, bool), ParseFixedError> {
let bytes = Bytes::new(bytes);
let (neg, abs, mut overflow) = match crate::from_str::$Uns::get_int_frac(
bytes,
radix,
$Single::BITS - frac_nbits,
frac_nbits,
) {
Ok((neg, abs, overflow)) => (neg, abs, overflow),
Err(e) => return Err(e),
};
let bound = if !neg { $Single::MAX } else { $Single::MIN };
if abs > bound.unsigned_abs() {
overflow = true;
}
let abs = if neg { abs.wrapping_neg() } else { abs } as $Single;
Ok((abs, overflow))
}
}
};
}
macro_rules! unsigned_helpers {
(u128) => {
pub mod u128 {
unsigned_helpers_common! { u128, u64, true }
use crate::int256::{self, U256};
use core::num::NonZeroU128;
#[inline]
const fn mul10_overflow(x: u128) -> (u128, u8) {
const LO_MASK: u128 = !(!0 << 64);
let hi = (x >> 64) * 10;
let lo = (x & LO_MASK) * 10;
let (hi_lo, hi_hi) = (hi as u64, (hi >> 64) as u64);
let (lo_lo, lo_hi) = (lo as u64, (lo >> 64) as u64);
let (wrapped, overflow) = hi_lo.overflowing_add(lo_hi);
(
((wrapped as u128) << 64) | (lo_lo as u128),
(hi_hi as u8) + (overflow as u8),
)
}
const fn mul_hi_lo(lhs: u128, rhs: u128) -> (u128, u128) {
const LO: u128 = !(!0 << 64);
let (lhs_hi, lhs_lo) = (lhs >> 64, lhs & LO);
let (rhs_hi, rhs_lo) = (rhs >> 64, rhs & LO);
let lhs_lo_rhs_lo = lhs_lo.wrapping_mul(rhs_lo);
let lhs_hi_rhs_lo = lhs_hi.wrapping_mul(rhs_lo);
let lhs_lo_rhs_hi = lhs_lo.wrapping_mul(rhs_hi);
let lhs_hi_rhs_hi = lhs_hi.wrapping_mul(rhs_hi);
let col01 = lhs_lo_rhs_lo;
let (col01_hi, col01_lo) = (col01 >> 64, col01 & LO);
let partial_col12 = lhs_hi_rhs_lo + col01_hi;
let (col12, carry_col3) = partial_col12.overflowing_add(lhs_lo_rhs_hi);
let (col12_hi, col12_lo) = (col12 >> 64, col12 & LO);
let ans01 = (col12_lo << 64) + col01_lo;
let ans23 = lhs_hi_rhs_hi + col12_hi + if carry_col3 { 1u128 << 64 } else { 0 };
(ans23, ans01)
}
const fn div_tie(
dividend_hi: u128,
dividend_lo: u128,
divisor: NonZeroU128,
) -> (u128, bool) {
let dividend = U256 {
lo: dividend_lo,
hi: dividend_hi,
};
let (quot, rem) = int256::div_rem_u256_u128(dividend, divisor);
(quot.lo, rem == 0)
}
pub const fn dec_to_bin(
(hi, lo): (u128, u128),
nbits: u32,
round: Round,
) -> Option<u128> {
debug_assert!(hi < 10u128.pow(27));
debug_assert!(lo < 10u128.pow(27));
debug_assert!(nbits <= 128);
let fives = 5u128.pow(54);
let denom = fives * 2;
let denom = match NonZeroU128::new(denom) {
None => unreachable!(),
Some(nz) => nz,
};
let (hi_hi, hi_lo) = mul_hi_lo(hi, 10u128.pow(27));
let (val_lo, overflow) = hi_lo.overflowing_add(lo);
let val_hi = if overflow { hi_hi + 1 } else { hi_hi };
let (mut numer_lo, mut numer_hi) = (val_lo, val_hi);
if nbits < (54 - 1) {
let shr = (54 - 1) - nbits;
numer_lo = (numer_lo >> shr) | (numer_hi << (128 - shr));
numer_hi >>= shr;
} else if nbits > (54 - 1) {
let shl = nbits - (54 - 1);
numer_hi = (numer_hi << shl) | (numer_lo >> (128 - shl));
numer_lo <<= shl;
}
match round {
Round::Nearest => {
let (wrapped, overflow) = numer_lo.overflowing_add(fives);
numer_lo = wrapped;
if overflow {
numer_hi += 1;
}
let check_overflow = if nbits == 128 {
numer_hi
} else if nbits == 0 {
numer_lo
} else {
(numer_lo >> nbits) | (numer_hi << (128 - nbits))
};
if check_overflow >= denom.get() {
let half_hi = fives >> (128 - (54 - 1));
let half_lo = fives << (54 - 1);
return if nbits == 0 && val_hi == half_hi && val_lo == half_lo {
Some(0)
} else {
None
};
}
}
Round::Floor => {}
}
let (mut div, tie) = div_tie(numer_hi, numer_lo, denom);
if tie && is_odd(div) {
div -= 1;
}
Some(div)
}
pub const fn parse_is_short(bytes: Bytes) -> ((u128, u128), bool) {
if let Some(rem) = 27usize.checked_sub(bytes.len()) {
let hi = dec_str_int_to_bin(bytes).0 * 10u128.pow(rem as u32);
((hi, 0), true)
} else {
let (begin, end) = bytes.split(27);
let hi = dec_str_int_to_bin(begin).0;
let (is_short, slice, pad) = if let Some(rem) = 54usize.checked_sub(bytes.len())
{
(true, end, 10u128.pow(rem as u32))
} else {
let (mid, _) = end.split(27);
(false, mid, 1)
};
let lo = dec_str_int_to_bin(slice).0 * pad;
((hi, lo), is_short)
}
}
}
};
($Single:ident, $Half:ident, $attempt_half:expr; $Double:ident, $dec:expr, $bin:expr) => {
pub mod $Single {
unsigned_helpers_common! { $Single, $Half, $attempt_half }
#[inline]
const fn mul10_overflow(x: $Single) -> ($Single, u8) {
let prod = (x as $Double) * 10;
(prod as $Single, (prod >> <$Single>::BITS) as u8)
}
pub const fn dec_to_bin(val: $Double, nbits: u32, round: Round) -> Option<$Single> {
debug_assert!(val < $Double::pow(10, $dec));
debug_assert!(nbits <= $bin);
let fives = $Double::pow(5, $dec);
let denom = fives * 2;
let mut numer = val << ($bin - $dec + 1) >> ($bin - nbits);
match round {
Round::Nearest => {
numer += fives;
if numer >> nbits >= denom {
return if nbits == 0 && val == fives << ($dec - 1) {
Some(0)
} else {
None
};
}
}
Round::Floor => {}
}
let (mut div, tie) = (numer / denom, numer % denom == 0);
if tie && crate::from_str::$Double::is_odd(div) {
div -= 1;
}
Some(div as $Single)
}
pub const fn parse_is_short(bytes: Bytes) -> ($Double, bool) {
let (is_short, slice, pad) =
if let Some(rem) = usize::checked_sub($dec, bytes.len()) {
(true, bytes, $Double::pow(10, rem as u32))
} else {
let (short, _) = bytes.split($dec);
(false, short, 1)
};
let val = crate::from_str::$Double::dec_str_int_to_bin(slice).0 * pad;
(val, is_short)
}
}
};
}
macro_rules! unsigned_helpers_common {
($Uns:ident, $Half:ident, $attempt_half:expr) => {
use crate::from_str::{frac_is_half, parse_bounds, unchecked_hex_digit, Parse, Round};
pub const fn from_str(
bytes: &[u8],
radix: u32,
frac_nbits: u32,
) -> Result<($Uns, bool), ParseFixedError> {
let bytes = Bytes::new(bytes);
let (neg, abs, mut overflow) =
match get_int_frac(bytes, radix, $Uns::BITS - frac_nbits, frac_nbits) {
Ok((neg, abs, overflow)) => (neg, abs, overflow),
Err(e) => return Err(e),
};
if neg && abs > 0 {
overflow = true;
}
let abs = if neg { abs.wrapping_neg() } else { abs };
Ok((abs, overflow))
}
helpers_common! { $Uns }
const fn from_byte(b: u8) -> $Uns {
b as $Uns
}
pub const fn is_odd(val: $Uns) -> bool {
val & 1 != 0
}
pub const fn bin_str_int_to_bin(bytes: Bytes) -> ($Uns, bool) {
let max_len = $Uns::BITS as usize;
let (bytes, overflow) = if bytes.len() > max_len {
let (_, last_max_len) = bytes.split(bytes.len() - max_len);
(last_max_len, true)
} else {
(bytes, false)
};
let mut acc = 0;
let mut i = 0;
while i < bytes.len() {
let byte = bytes.get(i);
i += 1;
let i = i - 1;
let _ = i;
acc = (acc << 1) + from_byte(byte - b'0');
}
(acc, overflow)
}
pub const fn bin_str_frac_to_bin(bytes: Bytes, nbits: u32) -> Option<$Uns> {
debug_assert!(!bytes.is_empty());
let dump_bits = $Uns::BITS - nbits;
let mut rem_bits = nbits;
let mut acc = 0;
let mut i = 0;
while i < bytes.len() {
let byte = bytes.get(i);
i += 1;
let i = i - 1;
let val = byte - b'0';
if rem_bits < 1 {
if val != 0 {
if bytes.len() > i + 1 || is_odd(acc) {
acc = match acc.checked_add(1) {
Some(acc) => acc,
None => return None,
};
}
}
if dump_bits != 0 && acc >> nbits != 0 {
return None;
}
return Some(acc);
}
acc = (acc << 1) + from_byte(val);
rem_bits -= 1;
}
Some(acc << rem_bits)
}
pub const fn oct_str_int_to_bin(bytes: Bytes) -> ($Uns, bool) {
let max_len = ($Uns::BITS as usize + 2) / 3;
let (bytes, mut overflow) = if bytes.len() > max_len {
let (_, last_max_len) = bytes.split(bytes.len() - max_len);
(last_max_len, true)
} else {
(bytes, false)
};
let mut acc = from_byte(bytes.get(0) - b'0');
if bytes.len() == max_len {
let first_max_bits = $Uns::BITS - (max_len as u32 - 1) * 3;
let first_max = (from_byte(1) << first_max_bits) - 1;
if acc > first_max {
overflow = true;
}
}
let mut i = 1;
while i < bytes.len() {
let byte = bytes.get(i);
i += 1;
let i = i - 1;
let _ = i;
acc = (acc << 3) + from_byte(byte - b'0');
}
(acc, overflow)
}
pub const fn oct_str_frac_to_bin(bytes: Bytes, nbits: u32) -> Option<$Uns> {
debug_assert!(!bytes.is_empty());
let dump_bits = $Uns::BITS - nbits;
let mut rem_bits = nbits;
let mut acc = 0;
let mut i = 0;
while i < bytes.len() {
let byte = bytes.get(i);
i += 1;
let i = i - 1;
let val = byte - b'0';
if rem_bits < 3 {
acc = (acc << rem_bits) + from_byte(val >> (3 - rem_bits));
let half = 1 << (2 - rem_bits);
if val & half != 0 {
if val & (half - 1) != 0 || bytes.len() > i + 1 || is_odd(acc) {
acc = match acc.checked_add(1) {
Some(acc) => acc,
None => return None,
};
}
}
if dump_bits != 0 && acc >> nbits != 0 {
return None;
}
return Some(acc);
}
acc = (acc << 3) + from_byte(val);
rem_bits -= 3;
}
Some(acc << rem_bits)
}
pub const fn hex_str_int_to_bin(bytes: Bytes) -> ($Uns, bool) {
let max_len = ($Uns::BITS as usize + 3) / 4;
let (bytes, mut overflow) = if bytes.len() > max_len {
let (_, last_max_len) = bytes.split(bytes.len() - max_len);
(last_max_len, true)
} else {
(bytes, false)
};
let mut acc = from_byte(unchecked_hex_digit(bytes.get(0)));
if bytes.len() == max_len {
let first_max_bits = $Uns::BITS - (max_len as u32 - 1) * 4;
let first_max = (from_byte(1) << first_max_bits) - 1;
if acc > first_max {
overflow = true;
}
}
let mut i = 1;
while i < bytes.len() {
let byte = bytes.get(i);
i += 1;
let i = i - 1;
let _ = i;
acc = (acc << 4) + from_byte(unchecked_hex_digit(byte));
}
(acc, overflow)
}
pub const fn hex_str_frac_to_bin(bytes: Bytes, nbits: u32) -> Option<$Uns> {
debug_assert!(!bytes.is_empty());
let dump_bits = $Uns::BITS - nbits;
let mut rem_bits = nbits;
let mut acc = 0;
let mut i = 0;
while i < bytes.len() {
let byte = bytes.get(i);
i += 1;
let i = i - 1;
let val = unchecked_hex_digit(byte);
if rem_bits < 4 {
acc = (acc << rem_bits) + from_byte(val >> (4 - rem_bits));
let half = 1 << (3 - rem_bits);
if val & half != 0 {
if val & (half - 1) != 0 || bytes.len() > i + 1 || is_odd(acc) {
acc = match acc.checked_add(1) {
Some(acc) => acc,
None => return None,
};
}
}
if dump_bits != 0 && acc >> nbits != 0 {
return None;
}
return Some(acc);
}
acc = (acc << 4) + from_byte(val);
rem_bits -= 4;
}
Some(acc << rem_bits)
}
pub const fn dec_str_int_to_bin(bytes: Bytes) -> ($Uns, bool) {
let max_effective_len = $Uns::BITS as usize;
let (bytes, mut overflow) = if bytes.len() > max_effective_len {
let (_, last_max_effective_len) = bytes.split(bytes.len() - max_effective_len);
(last_max_effective_len, true)
} else {
(bytes, false)
};
let mut acc = 0;
let mut i = 0;
while i < bytes.len() {
let byte = bytes.get(i);
i += 1;
let i = i - 1;
let _ = i;
let (prod, mul_overflow) = mul10_overflow(acc);
let (add, add_overflow) = prod.overflowing_add(from_byte(byte - b'0'));
acc = add;
overflow = overflow || mul_overflow != 0 || add_overflow;
}
(acc, overflow)
}
pub const fn dec_str_frac_to_bin(bytes: Bytes, nbits: u32) -> Option<$Uns> {
let (val, is_short) = parse_is_short(bytes);
let one: $Uns = 1;
let dump_bits = $Uns::BITS - nbits;
let round = if is_short {
Round::Nearest
} else {
Round::Floor
};
let floor = match dec_to_bin(val, nbits, round) {
Some(floor) => floor,
None => return None,
};
if is_short {
return Some(floor);
}
let (mut boundary, mut add_5) = if nbits == 0 {
(one << ($Uns::BITS - 1), false)
} else if dump_bits == 0 {
(floor, true)
} else {
((floor << dump_bits) + (one << (dump_bits - 1)), false)
};
let mut tie = true;
let mut i = 0;
while i < bytes.len() {
let byte = bytes.get(i);
i += 1;
let i = i - 1;
let _ = i;
if !add_5 && boundary == 0 {
tie = false;
break;
}
let (prod, mut boundary_digit) = mul10_overflow(boundary);
boundary = prod;
if add_5 {
let (wrapped, overflow) = boundary.overflowing_add(5);
boundary = wrapped;
if overflow {
boundary_digit += 1;
}
add_5 = false;
}
if byte - b'0' < boundary_digit {
return Some(floor);
}
if byte - b'0' > boundary_digit {
tie = false;
break;
}
}
if tie && !is_odd(floor) {
return Some(floor);
}
let next_up = match floor.checked_add(1) {
Some(s) => s,
None => return None,
};
if dump_bits != 0 && next_up >> nbits != 0 {
None
} else {
Some(next_up)
}
}
pub const fn get_int_frac(
bytes: Bytes,
radix: u32,
int_nbits: u32,
frac_nbits: u32,
) -> Result<(bool, $Uns, bool), ParseFixedError> {
let Parse { neg, int, frac } = match parse_bounds(bytes, radix) {
Ok(o) => o,
Err(e) => return Err(e),
};
let (int_val, mut overflow) = get_int(int, radix, int_nbits);
let (frac_val, frac_overflow) = match get_frac(frac, radix, frac_nbits) {
Some(val) => (val, false),
None => (0, true),
};
let mut val = int_val | frac_val;
if frac_overflow || (is_odd(int_val) && frac_nbits == 0 && frac_is_half(frac, radix)) {
let (new_val, new_overflow) = if int_nbits == 0 {
(val, true)
} else {
val.overflowing_add(1 << frac_nbits)
};
if new_overflow {
overflow = true;
}
val = new_val;
}
Ok((neg, val, overflow))
}
pub const fn get_int(int: Bytes, radix: u32, nbits: u32) -> ($Uns, bool) {
const HALF: u32 = $Uns::BITS / 2;
if $attempt_half && nbits <= HALF {
let (half, overflow) = crate::from_str::$Half::get_int(int, radix, nbits);
return ((half as $Uns) << HALF, overflow);
}
if int.is_empty() {
return (0, false);
}
let (mut parsed_int, mut overflow): ($Uns, bool) = match radix {
2 => bin_str_int_to_bin(int),
8 => oct_str_int_to_bin(int),
16 => hex_str_int_to_bin(int),
10 => dec_str_int_to_bin(int),
_ => unreachable!(),
};
let remove_bits = $Uns::BITS - nbits;
if nbits == 0 {
overflow = true;
parsed_int = 0;
} else if remove_bits > 0 {
if (parsed_int >> nbits) != 0 {
overflow = true;
}
parsed_int <<= remove_bits;
}
(parsed_int, overflow)
}
pub const fn get_frac(frac: Bytes, radix: u32, nbits: u32) -> Option<$Uns> {
if $attempt_half && nbits <= $Uns::BITS / 2 {
return match crate::from_str::$Half::get_frac(frac, radix, nbits) {
Some(half) => Some(half as $Uns),
None => None,
};
}
if frac.is_empty() {
return Some(0);
}
match radix {
2 => bin_str_frac_to_bin(frac, nbits),
8 => oct_str_frac_to_bin(frac, nbits),
16 => hex_str_frac_to_bin(frac, nbits),
10 => dec_str_frac_to_bin(frac, nbits),
_ => unreachable!(),
}
}
};
}
macro_rules! helpers_common {
($Single:ident) => {
use crate::{
bytes::Bytes,
from_str::{ParseErrorKind, ParseFixedError},
};
#[inline]
pub const fn from_str_radix(
s: &str,
radix: u32,
frac_nbits: u32,
) -> Result<$Single, ParseFixedError> {
match overflowing_from_str_radix(s, radix, frac_nbits) {
Ok((val, false)) => Ok(val),
Ok((_, true)) => Err(ParseFixedError {
kind: ParseErrorKind::Overflow,
}),
Err(e) => Err(e),
}
}
#[inline]
pub const fn saturating_from_str_radix(
s: &str,
radix: u32,
frac_nbits: u32,
) -> Result<$Single, ParseFixedError> {
match overflowing_from_str_radix(s, radix, frac_nbits) {
Ok((val, false)) => Ok(val),
Ok((_, true)) => {
let bytes = s.as_bytes();
let starts_with_minus = match bytes.first() {
Some(s) => *s == b'-',
None => false,
};
if starts_with_minus {
Ok($Single::MIN)
} else {
Ok($Single::MAX)
}
}
Err(e) => Err(e),
}
}
#[inline]
pub const fn wrapping_from_str_radix(
s: &str,
radix: u32,
frac_nbits: u32,
) -> Result<$Single, ParseFixedError> {
match overflowing_from_str_radix(s, radix, frac_nbits) {
Ok((val, _)) => Ok(val),
Err(e) => Err(e),
}
}
#[inline]
pub const fn overflowing_from_str_radix(
s: &str,
radix: u32,
frac_nbits: u32,
) -> Result<($Single, bool), ParseFixedError> {
from_str(s.as_bytes(), radix, frac_nbits)
}
};
}
const fn unchecked_hex_digit(byte: u8) -> u8 {
(byte & 0x0f) + if byte >= 0x40 { 9 } else { 0 }
}
pub enum Round {
Nearest,
Floor,
}
#[derive(Clone, Copy, Debug)]
struct Parse<'a> {
neg: bool,
int: Bytes<'a>,
frac: Bytes<'a>,
}
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
pub struct ParseFixedError {
kind: ParseErrorKind,
}
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
enum ParseErrorKind {
InvalidDigit,
NoDigits,
TooManyPoints,
Overflow,
}
impl From<ParseErrorKind> for ParseFixedError {
#[inline]
fn from(kind: ParseErrorKind) -> ParseFixedError {
ParseFixedError { kind }
}
}
impl ParseFixedError {
#[inline]
pub(crate) const fn message(&self) -> &str {
use self::ParseErrorKind::*;
match self.kind {
InvalidDigit => "invalid digit found in string",
NoDigits => "string has no digits",
TooManyPoints => "more than one decimal point found in string",
Overflow => "overflow",
}
}
}
impl Display for ParseFixedError {
fn fmt(&self, f: &mut Formatter<'_>) -> FmtResult {
Display::fmt(self.message(), f)
}
}
#[cfg(feature = "std")]
impl Error for ParseFixedError {
fn description(&self) -> &str {
self.message()
}
}
const fn parse_bounds(bytes: Bytes, radix: u32) -> Result<Parse<'_>, ParseFixedError> {
let mut sign: Option<bool> = None;
let mut trimmed_int_start: Option<usize> = None;
let mut point: Option<usize> = None;
let mut trimmed_frac_end: Option<usize> = None;
let mut has_any_digit = false;
let mut index = 0;
while index < bytes.len() {
let byte = bytes.get(index);
index += 1;
let index = index - 1;
match (byte, radix) {
(b'+', _) => {
if sign.is_some() || point.is_some() || has_any_digit {
return Err(ParseFixedError {
kind: ParseErrorKind::InvalidDigit,
});
}
sign = Some(false);
continue;
}
(b'-', _) => {
if sign.is_some() || point.is_some() || has_any_digit {
return Err(ParseFixedError {
kind: ParseErrorKind::InvalidDigit,
});
}
sign = Some(true);
continue;
}
(b'.', _) => {
if point.is_some() {
return Err(ParseFixedError {
kind: ParseErrorKind::TooManyPoints,
});
}
point = Some(index);
trimmed_frac_end = Some(index + 1);
continue;
}
(b'0'..=b'1', 2)
| (b'0'..=b'7', 8)
| (b'0'..=b'9', 10)
| (b'0'..=b'9', 16)
| (b'a'..=b'f', 16)
| (b'A'..=b'F', 16) => {
if trimmed_int_start.is_none() && point.is_none() && byte != b'0' {
trimmed_int_start = Some(index);
}
if trimmed_frac_end.is_some() && byte != b'0' {
trimmed_frac_end = Some(index + 1);
}
has_any_digit = true;
}
_ => {
return Err(ParseFixedError {
kind: ParseErrorKind::InvalidDigit,
})
}
}
}
if !has_any_digit {
return Err(ParseFixedError {
kind: ParseErrorKind::NoDigits,
});
}
let neg = match sign {
Some(s) => s,
None => false,
};
let int = match (trimmed_int_start, point) {
(Some(start), Some(point)) => {
let (up_to_point, _) = bytes.split(point);
let (_, from_start) = up_to_point.split(start);
from_start
}
(Some(start), None) => {
let (_, from_start) = bytes.split(start);
from_start
}
(None, _) => Bytes::new(&[]),
};
let frac = match (point, trimmed_frac_end) {
(Some(point), Some(end)) => {
let (up_to_end, _) = bytes.split(end);
let (_, from_after_point) = up_to_end.split(point + 1);
from_after_point
}
_ => Bytes::new(&[]),
};
Ok(Parse { neg, int, frac })
}
const fn frac_is_half(bytes: Bytes, radix: u32) -> bool {
bytes.len() == 1 && bytes.get(0) - b'0' == (radix as u8) / 2
}
signed_helpers! { i8, u8 }
signed_helpers! { i16, u16 }
signed_helpers! { i32, u32 }
signed_helpers! { i64, u64 }
signed_helpers! { i128, u128 }
unsigned_helpers! { u8, u8, false; u16, 3, 8 }
unsigned_helpers! { u16, u8, true; u32, 6, 16 }
unsigned_helpers! { u32, u16, true; u64, 13, 32 }
unsigned_helpers! { u64, u32, true; u128, 27, 64 }
unsigned_helpers! { u128 }
macro_rules! impl_from_str {
($Fixed:ident, $LeEqU:ident) => {
impl<Frac: $LeEqU> FromStr for $Fixed<Frac> {
type Err = ParseFixedError;
#[inline]
fn from_str(s: &str) -> Result<Self, Self::Err> {
Self::from_str(s)
}
}
};
}
impl_from_str! { FixedI8, LeEqU8 }
impl_from_str! { FixedI16, LeEqU16 }
impl_from_str! { FixedI32, LeEqU32 }
impl_from_str! { FixedI64, LeEqU64 }
impl_from_str! { FixedI128, LeEqU128 }
impl_from_str! { FixedU8, LeEqU8 }
impl_from_str! { FixedU16, LeEqU16 }
impl_from_str! { FixedU32, LeEqU32 }
impl_from_str! { FixedU64, LeEqU64 }
impl_from_str! { FixedU128, LeEqU128 }
#[cfg(test)]
mod tests {
use crate::{
bytes::Bytes,
from_str::{self, parse_bounds, Parse, ParseErrorKind, ParseFixedError, Round},
types::*,
};
use std::{
format,
string::{String, ToString},
};
#[test]
fn overflowing() {
let overflow = ParseFixedError {
kind: ParseErrorKind::Overflow,
};
assert_eq!(
U4F4::overflowing_from_str("15.5"),
Ok((U4F4::from_bits(0xF8), false))
);
assert_eq!(U4F4::from_str("15.5"), Ok(U4F4::from_bits(0xF8)));
assert_eq!(
U4F4::overflowing_from_str("31.5"),
Ok((U4F4::from_bits(0xF8), true))
);
assert_eq!(U4F4::from_str("31.5"), Err(overflow));
assert_eq!(
U4F4::overflowing_from_str("271.5"),
Ok((U4F4::from_bits(0xF8), true))
);
assert_eq!(
U8F0::overflowing_from_str("271"),
Ok((U8F0::from_bits(0x0F), true))
);
let longer_than_8 = format!("{}", (1 << 30) + 15);
assert_eq!(
U8F0::overflowing_from_str(&longer_than_8),
Ok((U8F0::from_bits(0x0F), true))
);
assert_eq!(
U4F4::overflowing_from_str_binary("1111.1000"),
Ok((U4F4::from_bits(0xF8), false))
);
assert_eq!(
U4F4::from_str_binary("1111.1000"),
Ok(U4F4::from_bits(0xF8))
);
assert_eq!(
U4F4::overflowing_from_str_binary("11111.1000"),
Ok((U4F4::from_bits(0xF8), true))
);
assert_eq!(U4F4::from_str_binary("11111.1000"), Err(overflow));
assert_eq!(
U8F0::overflowing_from_str_binary("100001111"),
Ok((U8F0::from_bits(0x0F), true))
);
assert_eq!(
U4F4::overflowing_from_str_octal("17.7"),
Ok((U4F4::from_bits(0xFE), false))
);
assert_eq!(U4F4::from_str_octal("17.7"), Ok(U4F4::from_bits(0xFE)));
assert_eq!(
U4F4::overflowing_from_str_octal("77.7"),
Ok((U4F4::from_bits(0xFE), true))
);
assert_eq!(U4F4::from_str_octal("77.7"), Err(overflow));
assert_eq!(
U4F4::overflowing_from_str_octal("707.7"),
Ok((U4F4::from_bits(0x7E), true))
);
assert_eq!(
U8F0::overflowing_from_str_octal("1307"),
Ok((U8F0::from_bits(0o307), true))
);
assert_eq!(
U6F10::overflowing_from_str_hex("3F.8"),
Ok((U6F10::from_bits(0xFE00), false))
);
assert_eq!(U6F10::from_str_hex("3F.8"), Ok(U6F10::from_bits(0xFE00)));
assert_eq!(
U6F10::overflowing_from_str_hex("FF.8"),
Ok((U6F10::from_bits(0xFE00), true))
);
assert_eq!(U6F10::from_str_hex("FF.8"), Err(overflow));
assert_eq!(
U6F10::overflowing_from_str_hex("F0F.8"),
Ok((U6F10::from_bits(0x3E00), true))
);
assert_eq!(
U16F0::overflowing_from_str_hex("100FF"),
Ok((U16F0::from_bits(0x00FF), true))
);
}
#[test]
fn check_dec_8() {
let two_pow = 8f64.exp2();
let limit = 1000;
for i in 0..limit {
let ans = from_str::u8::dec_to_bin(i, 8, Round::Nearest);
let approx = two_pow * f64::from(i) / f64::from(limit);
let error = (ans.map(f64::from).unwrap_or(two_pow) - approx).abs();
assert!(
error <= 0.5,
"i {} ans {:?} approx {} error {}",
i,
ans,
approx,
error
);
}
}
#[test]
fn check_dec_16() {
let two_pow = 16f64.exp2();
let limit = 1_000_000;
for i in 0..limit {
let ans = from_str::u16::dec_to_bin(i, 16, Round::Nearest);
let approx = two_pow * f64::from(i) / f64::from(limit);
let error = (ans.map(f64::from).unwrap_or(two_pow) - approx).abs();
assert!(
error <= 0.5,
"i {} ans {:?} approx {} error {}",
i,
ans,
approx,
error
);
}
}
#[test]
fn check_dec_32() {
let two_pow = 32f64.exp2();
let limit = 10_000_000_000_000;
for iter in 0..1_000_000 {
for &i in &[
iter,
limit / 4 - 1 - iter,
limit / 4 + iter,
limit / 3 - 1 - iter,
limit / 3 + iter,
limit / 2 - 1 - iter,
limit / 2 + iter,
limit - iter - 1,
] {
let ans = from_str::u32::dec_to_bin(i, 32, Round::Nearest);
let approx = two_pow * i as f64 / limit as f64;
let error = (ans.map(f64::from).unwrap_or(two_pow) - approx).abs();
assert!(
error <= 0.5,
"i {} ans {:?} approx {} error {}",
i,
ans,
approx,
error
);
}
}
}
#[test]
fn check_dec_64() {
let two_pow = 64f64.exp2();
let limit = 1_000_000_000_000_000_000_000_000_000;
for iter in 0..200_000 {
for &i in &[
iter,
limit / 4 - 1 - iter,
limit / 4 + iter,
limit / 3 - 1 - iter,
limit / 3 + iter,
limit / 2 - 1 - iter,
limit / 2 + iter,
limit - iter - 1,
] {
let ans = from_str::u64::dec_to_bin(i, 64, Round::Nearest);
let approx = two_pow * i as f64 / limit as f64;
let error = (ans.map(|x| x as f64).unwrap_or(two_pow) - approx).abs();
assert!(
error <= 0.5,
"i {} ans {:?} approx {} error {}",
i,
ans,
approx,
error
);
}
}
}
#[test]
fn check_dec_128() {
let nines = 10u128.pow(27) - 1;
let zeros = 0;
let too_big = from_str::u128::dec_to_bin((nines, nines), 128, Round::Nearest);
assert_eq!(too_big, None);
let big = from_str::u128::dec_to_bin((nines, zeros), 128, Round::Nearest);
assert_eq!(
big,
Some(340_282_366_920_938_463_463_374_607_091_485_844_535)
);
let small = from_str::u128::dec_to_bin((zeros, nines), 128, Round::Nearest);
assert_eq!(small, Some(340_282_366_921));
let zero = from_str::u128::dec_to_bin((zeros, zeros), 128, Round::Nearest);
assert_eq!(zero, Some(0));
let x = from_str::u128::dec_to_bin(
(
123_456_789_012_345_678_901_234_567,
987_654_321_098_765_432_109_876_543,
),
128,
Round::Nearest,
);
assert_eq!(x, Some(42_010_168_377_579_896_403_540_037_811_203_677_112));
let eights = 888_888_888_888_888_888_888_888_888;
let narrow = from_str::u128::dec_to_bin((eights, zeros), 40, Round::Nearest);
assert_eq!(narrow, Some(977_343_669_134));
}
fn check_parse_bounds_ok(bytes: &[u8], radix: u32, check: (bool, &[u8], &[u8])) {
let bytes = Bytes::new(bytes);
let Parse { neg, int, frac } = parse_bounds(bytes, radix).unwrap();
let int = int.slice();
let frac = frac.slice();
assert_eq!((neg, int, frac), check);
}
fn check_parse_bounds_err(bytes: &[u8], radix: u32, check: ParseErrorKind) {
let bytes = Bytes::new(bytes);
let ParseFixedError { kind } = parse_bounds(bytes, radix).unwrap_err();
assert_eq!(kind, check);
}
#[test]
fn check_parse_bounds() {
check_parse_bounds_ok(b"-12.34", 10, (true, &b"12"[..], &b"34"[..]));
check_parse_bounds_ok(b"012.", 10, (false, &b"12"[..], &b""[..]));
check_parse_bounds_ok(b"+.340", 10, (false, &b""[..], &b"34"[..]));
check_parse_bounds_ok(b"0", 10, (false, &b""[..], &b""[..]));
check_parse_bounds_ok(b"-.C1A0", 16, (true, &b""[..], &b"C1A"[..]));
check_parse_bounds_err(b"0 ", 10, ParseErrorKind::InvalidDigit);
check_parse_bounds_err(b"+-", 10, ParseErrorKind::InvalidDigit);
check_parse_bounds_err(b"+.", 10, ParseErrorKind::NoDigits);
check_parse_bounds_err(b".1.", 10, ParseErrorKind::TooManyPoints);
check_parse_bounds_err(b"1+2", 10, ParseErrorKind::InvalidDigit);
check_parse_bounds_err(b"1-2", 10, ParseErrorKind::InvalidDigit);
}
macro_rules! assert_ok {
($T:ty, $str:expr, $radix:expr, $bits:expr, $overflow:expr) => {
let m = match $radix {
2 => <$T>::overflowing_from_str_binary($str),
8 => <$T>::overflowing_from_str_octal($str),
10 => <$T>::overflowing_from_str($str),
16 => <$T>::overflowing_from_str_hex($str),
_ => unreachable!(),
};
match m {
Ok((f, o)) => {
assert_eq!(f.to_bits(), $bits, "{} -> ({}, {})", $str, f, o);
assert_eq!(o, $overflow, "{} -> ({}, {})", $str, f, o);
}
Err(e) => panic!("could not parse {}: {}", $str, e),
}
};
}
#[test]
fn check_i8_u8_from_str() {
assert_ok!(I0F8, "-1", 10, 0x00, true);
assert_ok!(I0F8, "-0.502", 10, 0x7F, true);
assert_ok!(I0F8, "-0.501", 10, -0x80, false);
assert_ok!(I0F8, "0.498", 10, 0x7F, false);
assert_ok!(I0F8, "0.499", 10, -0x80, true);
assert_ok!(I0F8, "1", 10, 0x00, true);
assert_ok!(I4F4, "-8.04", 10, 0x7F, true);
assert_ok!(I4F4, "-8.03", 10, -0x80, false);
assert_ok!(I4F4, "7.96", 10, 0x7F, false);
assert_ok!(I4F4, "7.97", 10, -0x80, true);
assert_ok!(I8F0, "-128.501", 10, 0x7F, true);
assert_ok!(I8F0, "-128.5", 10, -0x80, false);
assert_ok!(I8F0, "127.499", 10, 0x7F, false);
assert_ok!(I8F0, "127.5", 10, -0x80, true);
assert_ok!(U0F8, "-0", 10, 0x00, false);
assert_ok!(U0F8, "0.498", 10, 0x7F, false);
assert_ok!(U0F8, "0.499", 10, 0x80, false);
assert_ok!(U0F8, "0.998", 10, 0xFF, false);
assert_ok!(U0F8, "0.999", 10, 0x00, true);
assert_ok!(U0F8, "1", 10, 0x00, true);
assert_ok!(U4F4, "7.96", 10, 0x7F, false);
assert_ok!(U4F4, "7.97", 10, 0x80, false);
assert_ok!(U4F4, "15.96", 10, 0xFF, false);
assert_ok!(U4F4, "15.97", 10, 0x00, true);
assert_ok!(U8F0, "127.499", 10, 0x7F, false);
assert_ok!(U8F0, "127.5", 10, 0x80, false);
assert_ok!(U8F0, "255.499", 10, 0xFF, false);
assert_ok!(U8F0, "255.5", 10, 0x00, true);
}
#[test]
fn check_i16_u16_from_str() {
assert_ok!(I0F16, "-1", 10, 0x00, true);
assert_ok!(I0F16, "-0.500008", 10, 0x7FFF, true);
assert_ok!(I0F16, "-0.500007", 10, -0x8000, false);
assert_ok!(I0F16, "+0.499992", 10, 0x7FFF, false);
assert_ok!(I0F16, "+0.499993", 10, -0x8000, true);
assert_ok!(I0F16, "1", 10, 0x0000, true);
assert_ok!(I8F8, "-128.002", 10, 0x7FFF, true);
assert_ok!(I8F8, "-128.001", 10, -0x8000, false);
assert_ok!(I8F8, "+127.998", 10, 0x7FFF, false);
assert_ok!(I8F8, "+127.999", 10, -0x8000, true);
assert_ok!(I16F0, "-32768.500001", 10, 0x7FFF, true);
assert_ok!(I16F0, "-32768.5", 10, -0x8000, false);
assert_ok!(I16F0, "+32767.499999", 10, 0x7FFF, false);
assert_ok!(I16F0, "+32767.5", 10, -0x8000, true);
assert_ok!(U0F16, "-0", 10, 0x0000, false);
assert_ok!(U0F16, "0.499992", 10, 0x7FFF, false);
assert_ok!(U0F16, "0.499993", 10, 0x8000, false);
assert_ok!(U0F16, "0.999992", 10, 0xFFFF, false);
assert_ok!(U0F16, "0.999993", 10, 0x0000, true);
assert_ok!(U0F16, "1", 10, 0x0000, true);
assert_ok!(U8F8, "127.998", 10, 0x7FFF, false);
assert_ok!(U8F8, "127.999", 10, 0x8000, false);
assert_ok!(U8F8, "255.998", 10, 0xFFFF, false);
assert_ok!(U8F8, "255.999", 10, 0x0000, true);
assert_ok!(U16F0, "32767.499999", 10, 0x7FFF, false);
assert_ok!(U16F0, "32767.5", 10, 0x8000, false);
assert_ok!(U16F0, "65535.499999", 10, 0xFFFF, false);
assert_ok!(U16F0, "65535.5", 10, 0x0000, true);
}
#[test]
fn check_i32_u32_from_str() {
assert_ok!(I0F32, "-1", 10, 0x0000_0000, true);
assert_ok!(I0F32, "-0.5000000002", 10, 0x7FFF_FFFF, true);
assert_ok!(I0F32, "-0.5000000001", 10, -0x8000_0000, false);
assert_ok!(I0F32, "0.4999999998", 10, 0x7FFF_FFFF, false);
assert_ok!(I0F32, "0.4999999999", 10, -0x8000_0000, true);
assert_ok!(I0F32, "1", 10, 0x0000_0000, true);
assert_ok!(I16F16, "-32768.000008", 10, 0x7FFF_FFFF, true);
assert_ok!(I16F16, "-32768.000007", 10, -0x8000_0000, false);
assert_ok!(I16F16, "32767.999992", 10, 0x7FFF_FFFF, false);
assert_ok!(I16F16, "32767.999993", 10, -0x8000_0000, true);
assert_ok!(I32F0, "-2147483648.5000000001", 10, 0x7FFF_FFFF, true);
assert_ok!(I32F0, "-2147483648.5", 10, -0x8000_0000, false);
assert_ok!(I32F0, "2147483647.4999999999", 10, 0x7FFF_FFFF, false);
assert_ok!(I32F0, "2147483647.5", 10, -0x8000_0000, true);
assert_ok!(U0F32, "-0", 10, 0x0000_0000, false);
assert_ok!(U0F32, "0.4999999998", 10, 0x7FFF_FFFF, false);
assert_ok!(U0F32, "0.4999999999", 10, 0x8000_0000, false);
assert_ok!(U0F32, "0.9999999998", 10, 0xFFFF_FFFF, false);
assert_ok!(U0F32, "0.9999999999", 10, 0x0000_0000, true);
assert_ok!(U0F32, "1", 10, 0x0000_0000, true);
assert_ok!(U16F16, "32767.999992", 10, 0x7FFF_FFFF, false);
assert_ok!(U16F16, "32767.999993", 10, 0x8000_0000, false);
assert_ok!(U16F16, "65535.999992", 10, 0xFFFF_FFFF, false);
assert_ok!(U16F16, "65535.999993", 10, 0x0000_0000, true);
assert_ok!(U32F0, "2147483647.4999999999", 10, 0x7FFF_FFFF, false);
assert_ok!(U32F0, "2147483647.5", 10, 0x8000_0000, false);
assert_ok!(U32F0, "4294967295.4999999999", 10, 0xFFFF_FFFF, false);
assert_ok!(U32F0, "4294967295.5", 10, 0x0000_0000, true);
}
#[test]
fn check_i64_u64_from_str() {
assert_ok!(I0F64, "-1", 10, 0x0000_0000_0000_0000, true);
assert_ok!(
I0F64,
"-0.50000000000000000003",
10,
0x7FFF_FFFF_FFFF_FFFF,
true
);
assert_ok!(
I0F64,
"-0.50000000000000000002",
10,
-0x8000_0000_0000_0000,
false
);
assert_ok!(
I0F64,
"+0.49999999999999999997",
10,
0x7FFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
I0F64,
"+0.49999999999999999998",
10,
-0x8000_0000_0000_0000,
true
);
assert_ok!(I0F64, "1", 10, 0x0000_0000_0000_0000, true);
assert_ok!(
I32F32,
"-2147483648.0000000002",
10,
0x7FFF_FFFF_FFFF_FFFF,
true
);
assert_ok!(
I32F32,
"-2147483648.0000000001",
10,
-0x8000_0000_0000_0000,
false
);
assert_ok!(
I32F32,
"2147483647.9999999998",
10,
0x7FFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
I32F32,
"2147483647.9999999999",
10,
-0x8000_0000_0000_0000,
true
);
assert_ok!(
I64F0,
"-9223372036854775808.50000000000000000001",
10,
0x7FFF_FFFF_FFFF_FFFF,
true
);
assert_ok!(
I64F0,
"-9223372036854775808.5",
10,
-0x8000_0000_0000_0000,
false
);
assert_ok!(
I64F0,
"9223372036854775807.49999999999999999999",
10,
0x7FFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
I64F0,
"9223372036854775807.5",
10,
-0x8000_0000_0000_0000,
true
);
assert_ok!(U0F64, "-0", 10, 0x0000_0000_0000_0000, false);
assert_ok!(
U0F64,
"0.49999999999999999997",
10,
0x7FFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U0F64,
"0.49999999999999999998",
10,
0x8000_0000_0000_0000,
false
);
assert_ok!(
U0F64,
"0.99999999999999999997",
10,
0xFFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U0F64,
"0.99999999999999999998",
10,
0x0000_0000_0000_0000,
true
);
assert_ok!(U0F64, "1", 10, 0x0000_0000_0000_0000, true);
assert_ok!(
U32F32,
"2147483647.9999999998",
10,
0x7FFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U32F32,
"2147483647.9999999999",
10,
0x8000_0000_0000_0000,
false
);
assert_ok!(
U32F32,
"4294967295.9999999998",
10,
0xFFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U32F32,
"4294967295.9999999999",
10,
0x0000_0000_0000_0000,
true
);
assert_ok!(
U64F0,
"9223372036854775807.49999999999999999999",
10,
0x7FFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U64F0,
"9223372036854775807.5",
10,
0x8000_0000_0000_0000,
false
);
assert_ok!(
U64F0,
"18446744073709551615.49999999999999999999",
10,
0xFFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U64F0,
"18446744073709551615.5",
10,
0x0000_0000_0000_0000,
true
);
}
#[test]
fn check_i128_u128_from_str() {
assert_ok!(
I0F128,
"-1",
10,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
I0F128,
"-0.500000000000000000000000000000000000002",
10,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
true
);
assert_ok!(
I0F128,
"-0.500000000000000000000000000000000000001",
10,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
I0F128,
"0.499999999999999999999999999999999999998",
10,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
I0F128,
"0.499999999999999999999999999999999999999",
10,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
I0F128,
"1",
10,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
I64F64,
"-9223372036854775808.00000000000000000003",
10,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
true
);
assert_ok!(
I64F64,
"-9223372036854775808.00000000000000000002",
10,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
I64F64,
"9223372036854775807.99999999999999999997",
10,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
I64F64,
"9223372036854775807.99999999999999999998",
10,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
I128F0,
"-170141183460469231731687303715884105728.5000000000000000000000000000000000000001",
10,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
true
);
assert_ok!(
I128F0,
"-170141183460469231731687303715884105728.5",
10,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
I128F0,
"170141183460469231731687303715884105727.4999999999999999999999999999999999999999",
10,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
I128F0,
"170141183460469231731687303715884105727.5",
10,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
U0F128,
"-0",
10,
0x0000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
U0F128,
"0.499999999999999999999999999999999999998",
10,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U0F128,
"0.499999999999999999999999999999999999999",
10,
0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
U0F128,
"0.999999999999999999999999999999999999998",
10,
0xFFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U0F128,
"0.999999999999999999999999999999999999999",
10,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
U0F128,
"1",
10,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
U64F64,
"9223372036854775807.99999999999999999997",
10,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U64F64,
"9223372036854775807.99999999999999999998",
10,
0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
U64F64,
"18446744073709551615.99999999999999999997",
10,
0xFFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U64F64,
"18446744073709551615.99999999999999999998",
10,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
U128F0,
"170141183460469231731687303715884105727.4999999999999999999999999999999999999999",
10,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U128F0,
"170141183460469231731687303715884105727.5",
10,
0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
U128F0,
"340282366920938463463374607431768211455.4999999999999999999999999999999999999999",
10,
0xFFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U128F0,
"340282366920938463463374607431768211455.5",
10,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
}
#[test]
fn check_i16_u16_from_str_binary() {
assert_ok!(I0F16, "-1", 2, 0x0000, true);
assert_ok!(I0F16, "-0.100000000000000011", 2, 0x7FFF, true);
assert_ok!(I0F16, "-0.100000000000000010", 2, -0x8000, false);
assert_ok!(I0F16, "-0.011111111111111110", 2, -0x8000, false);
assert_ok!(I0F16, "+0.011111111111111101", 2, 0x7FFF, false);
assert_ok!(I0F16, "+0.011111111111111110", 2, -0x8000, true);
assert_ok!(I0F16, "1", 2, 0x0000, true);
assert_ok!(I8F8, "-10000000.0000000011", 2, 0x7FFF, true);
assert_ok!(I8F8, "-10000000.0000000010", 2, -0x8000, false);
assert_ok!(I8F8, "-01111111.1111111110", 2, -0x8000, false);
assert_ok!(I8F8, "+01111111.1111111101", 2, 0x7FFF, false);
assert_ok!(I8F8, "+01111111.1111111110", 2, -0x8000, true);
assert_ok!(I16F0, "-1000000000000000.11", 2, 0x7FFF, true);
assert_ok!(I16F0, "-1000000000000000.10", 2, -0x8000, false);
assert_ok!(I16F0, "-0111111111111111.10", 2, -0x8000, false);
assert_ok!(I16F0, "+0111111111111111.01", 2, 0x7FFF, false);
assert_ok!(I16F0, "+0111111111111111.10", 2, -0x8000, true);
assert_ok!(U0F16, "-0", 2, 0x0000, false);
assert_ok!(U0F16, "0.011111111111111101", 2, 0x7FFF, false);
assert_ok!(U0F16, "0.011111111111111110", 2, 0x8000, false);
assert_ok!(U0F16, "0.111111111111111101", 2, 0xFFFF, false);
assert_ok!(U0F16, "0.111111111111111110", 2, 0x0000, true);
assert_ok!(U0F16, "1", 2, 0x0000, true);
assert_ok!(U8F8, "01111111.1111111101", 2, 0x7FFF, false);
assert_ok!(U8F8, "01111111.1111111110", 2, 0x8000, false);
assert_ok!(U8F8, "11111111.1111111101", 2, 0xFFFF, false);
assert_ok!(U8F8, "11111111.1111111110", 2, 0x0000, true);
assert_ok!(U16F0, "0111111111111111.01", 2, 0x7FFF, false);
assert_ok!(U16F0, "0111111111111111.10", 2, 0x8000, false);
assert_ok!(U16F0, "1111111111111111.01", 2, 0xFFFF, false);
assert_ok!(U16F0, "1111111111111111.10", 2, 0x0000, true);
}
#[test]
fn check_i16_u16_from_str_octal() {
assert_ok!(I0F16, "-1", 8, 0x0000, true);
assert_ok!(I0F16, "-0.400003", 8, 0x7FFF, true);
assert_ok!(I0F16, "-0.400002", 8, -0x8000, false);
assert_ok!(I0F16, "-0.377776", 8, -0x8000, false);
assert_ok!(I0F16, "+0.377775", 8, 0x7FFF, false);
assert_ok!(I0F16, "+0.377776", 8, -0x8000, true);
assert_ok!(I0F16, "1", 8, 0x0000, true);
assert_ok!(I8F8, "-200.0011", 8, 0x7FFF, true);
assert_ok!(I8F8, "-200.0010", 8, -0x8000, false);
assert_ok!(I8F8, "-177.7770", 8, -0x8000, false);
assert_ok!(I8F8, "+177.7767", 8, 0x7FFF, false);
assert_ok!(I8F8, "+177.7770", 8, -0x8000, true);
assert_ok!(I16F0, "-100000.5", 8, 0x7FFF, true);
assert_ok!(I16F0, "-100000.4", 8, -0x8000, false);
assert_ok!(I16F0, "-077777.4", 8, -0x8000, false);
assert_ok!(I16F0, "+077777.3", 8, 0x7FFF, false);
assert_ok!(I16F0, "+077777.4", 8, -0x8000, true);
assert_ok!(U0F16, "-0", 8, 0x0000, false);
assert_ok!(U0F16, "0.377775", 8, 0x7FFF, false);
assert_ok!(U0F16, "0.377776", 8, 0x8000, false);
assert_ok!(U0F16, "0.777775", 8, 0xFFFF, false);
assert_ok!(U0F16, "0.777776", 8, 0x0000, true);
assert_ok!(U0F16, "1", 8, 0x0000, true);
assert_ok!(U8F8, "177.7767", 8, 0x7FFF, false);
assert_ok!(U8F8, "177.7770", 8, 0x8000, false);
assert_ok!(U8F8, "377.7767", 8, 0xFFFF, false);
assert_ok!(U8F8, "377.7770", 8, 0x0000, true);
assert_ok!(U16F0, "077777.3", 8, 0x7FFF, false);
assert_ok!(U16F0, "077777.4", 8, 0x8000, false);
assert_ok!(U16F0, "177777.3", 8, 0xFFFF, false);
assert_ok!(U16F0, "177777.4", 8, 0x0000, true);
}
#[test]
fn check_i16_u16_from_str_hex() {
assert_ok!(I0F16, "-1", 16, 0x0000, true);
assert_ok!(I0F16, "-0.80009", 16, 0x7FFF, true);
assert_ok!(I0F16, "-0.80008", 16, -0x8000, false);
assert_ok!(I0F16, "-0.7FFF8", 16, -0x8000, false);
assert_ok!(I0F16, "+0.7FFF7", 16, 0x7FFF, false);
assert_ok!(I0F16, "+0.7FFF8", 16, -0x8000, true);
assert_ok!(I0F16, "1", 16, 0x0000, true);
assert_ok!(I8F8, "-80.009", 16, 0x7FFF, true);
assert_ok!(I8F8, "-80.008", 16, -0x8000, false);
assert_ok!(I8F8, "-7F.FF8", 16, -0x8000, false);
assert_ok!(I8F8, "+7F.FF7", 16, 0x7FFF, false);
assert_ok!(I8F8, "+7F.FF8", 16, -0x8000, true);
assert_ok!(I16F0, "-8000.9", 16, 0x7FFF, true);
assert_ok!(I16F0, "-8000.8", 16, -0x8000, false);
assert_ok!(I16F0, "-7FFF.8", 16, -0x8000, false);
assert_ok!(I16F0, "+7FFF.7", 16, 0x7FFF, false);
assert_ok!(I16F0, "+7FFF.8", 16, -0x8000, true);
assert_ok!(U0F16, "-0", 16, 0x0000, false);
assert_ok!(U0F16, "0.7FFF7", 16, 0x7FFF, false);
assert_ok!(U0F16, "0.7FFF8", 16, 0x8000, false);
assert_ok!(U0F16, "0.FFFF7", 16, 0xFFFF, false);
assert_ok!(U0F16, "0.FFFF8", 16, 0x0000, true);
assert_ok!(U0F16, "1", 16, 0x0000, true);
assert_ok!(U8F8, "7F.FF7", 16, 0x7FFF, false);
assert_ok!(U8F8, "7F.FF8", 16, 0x8000, false);
assert_ok!(U8F8, "FF.FF7", 16, 0xFFFF, false);
assert_ok!(U8F8, "FF.FF8", 16, 0x0000, true);
assert_ok!(U16F0, "7FFF.7", 16, 0x7FFF, false);
assert_ok!(U16F0, "7FFF.8", 16, 0x8000, false);
assert_ok!(U16F0, "FFFF.7", 16, 0xFFFF, false);
assert_ok!(U16F0, "FFFF.8", 16, 0x0000, true);
}
#[test]
fn check_i128_u128_from_str_hex() {
assert_ok!(
I0F128,
"-1",
16,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
I0F128,
"-0.800000000000000000000000000000009",
16,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
true
);
assert_ok!(
I0F128,
"-0.800000000000000000000000000000008",
16,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
I0F128,
"-0.7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF8",
16,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
I0F128,
"+0.7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF7",
16,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
I0F128,
"+0.7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF8",
16,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
I0F128,
"1",
16,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
I64F64,
"-8000000000000000.00000000000000009",
16,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
true
);
assert_ok!(
I64F64,
"-8000000000000000.00000000000000008",
16,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
I64F64,
"-7FFFFFFFFFFFFFFF.FFFFFFFFFFFFFFFF8",
16,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
I64F64,
"+7FFFFFFFFFFFFFFF.FFFFFFFFFFFFFFFF7",
16,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
I64F64,
"+7FFFFFFFFFFFFFFF.FFFFFFFFFFFFFFFF8",
16,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
I128F0,
"-80000000000000000000000000000000.9",
16,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
true
);
assert_ok!(
I128F0,
"-80000000000000000000000000000000.8",
16,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
I128F0,
"-7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.8",
16,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
I128F0,
"+7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.7",
16,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
I128F0,
"+7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.8",
16,
-0x8000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
U0F128,
"-0",
16,
0x0000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
U0F128,
"0.7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF7",
16,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U0F128,
"0.7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF8",
16,
0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
U0F128,
"0.FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF7",
16,
0xFFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U0F128,
"0.FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF8",
16,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
U0F128,
"1",
16,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
U64F64,
"7FFFFFFFFFFFFFFF.FFFFFFFFFFFFFFFF7",
16,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U64F64,
"7FFFFFFFFFFFFFFF.FFFFFFFFFFFFFFFF8",
16,
0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
U64F64,
"FFFFFFFFFFFFFFFF.FFFFFFFFFFFFFFFF7",
16,
0xFFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U64F64,
"FFFFFFFFFFFFFFFF.FFFFFFFFFFFFFFFF8",
16,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
assert_ok!(
U128F0,
"7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.7",
16,
0x7FFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U128F0,
"7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.8",
16,
0x8000_0000_0000_0000_0000_0000_0000_0000,
false
);
assert_ok!(
U128F0,
"FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.7",
16,
0xFFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF_FFFF,
false
);
assert_ok!(
U128F0,
"FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.8",
16,
0x0000_0000_0000_0000_0000_0000_0000_0000,
true
);
}
struct Fractions {
zero: String,
gt_0: String,
max: String,
over: String,
}
fn without_last(a: &str) -> &str {
&a[..a.len() - 1]
}
fn make_fraction_strings(max_int: &str, eps_frac: &str) -> Fractions {
let eps_frac_compl: String = eps_frac
.chars()
.map(|digit| (b'0' + b'9' - digit as u8) as char)
.collect();
let zero = String::from("0.") + eps_frac;
let gt_0 = String::from(&*zero) + "000001";
let max = String::from(max_int) + &eps_frac_compl + "999999";
let over = String::from(max_int) + without_last(&eps_frac_compl) + "5";
Fractions {
zero,
gt_0,
max,
over,
}
}
#[test]
fn check_exact_decimal() {
let max_int_0 = String::from("0.");
let max_int_4 = String::from("15.");
let max_int_8 = format!("{}.", !0u8);
let max_int_16 = format!("{}.", !0u16);
let max_int_28 = format!("{}.", !0u32 >> 4);
let max_int_32 = format!("{}.", !0u32);
let max_int_64 = format!("{}.", !0u64);
let max_int_124 = format!("{}.", !0u128 >> 4);
let max_int_128 = format!("{}.", !0u128);
let eps_0 = "5";
let eps_4 = "03125";
let eps_8 = "001953125";
let eps_16 = "00000762939453125";
let eps_28 = "00000000186264514923095703125";
let eps_32 = "000000000116415321826934814453125";
let eps_64 = "00000000000000000002710505431213761085018632002174854278564453125";
let eps_124 = "0000000000000000000000000000000000000235098870164457501593747307\
4444491355637331113544175043017503412556834518909454345703125";
let eps_128 = "0000000000000000000000000000000000000014693679385278593849609206\
71527807097273331945965109401885939632848021574318408966064453125";
let frac_0_8 = make_fraction_strings(&max_int_0, eps_8);
assert_ok!(U0F8, &frac_0_8.zero, 10, 0, false);
assert_ok!(U0F8, &frac_0_8.gt_0, 10, 1, false);
assert_ok!(U0F8, &frac_0_8.max, 10, !0, false);
assert_ok!(U0F8, &frac_0_8.over, 10, 0, true);
let frac_4_4 = make_fraction_strings(&max_int_4, eps_4);
assert_ok!(U4F4, &frac_4_4.zero, 10, 0, false);
assert_ok!(U4F4, &frac_4_4.gt_0, 10, 1, false);
assert_ok!(U4F4, &frac_4_4.max, 10, !0, false);
assert_ok!(U4F4, &frac_4_4.over, 10, 0, true);
let frac_8_0 = make_fraction_strings(&max_int_8, eps_0);
assert_ok!(U8F0, &frac_8_0.zero, 10, 0, false);
assert_ok!(U8F0, &frac_8_0.gt_0, 10, 1, false);
assert_ok!(U8F0, &frac_8_0.max, 10, !0, false);
assert_ok!(U8F0, &frac_8_0.over, 10, 0, true);
let frac_0_32 = make_fraction_strings(&max_int_0, eps_32);
assert_ok!(U0F32, &frac_0_32.zero, 10, 0, false);
assert_ok!(U0F32, &frac_0_32.gt_0, 10, 1, false);
assert_ok!(U0F32, &frac_0_32.max, 10, !0, false);
assert_ok!(U0F32, &frac_0_32.over, 10, 0, true);
let frac_4_28 = make_fraction_strings(&max_int_4, eps_28);
assert_ok!(U4F28, &frac_4_28.zero, 10, 0, false);
assert_ok!(U4F28, &frac_4_28.gt_0, 10, 1, false);
assert_ok!(U4F28, &frac_4_28.max, 10, !0, false);
assert_ok!(U4F28, &frac_4_28.over, 10, 0, true);
let frac_16_16 = make_fraction_strings(&max_int_16, eps_16);
assert_ok!(U16F16, &frac_16_16.zero, 10, 0, false);
assert_ok!(U16F16, &frac_16_16.gt_0, 10, 1, false);
assert_ok!(U16F16, &frac_16_16.max, 10, !0, false);
assert_ok!(U16F16, &frac_16_16.over, 10, 0, true);
let frac_28_4 = make_fraction_strings(&max_int_28, eps_4);
assert_ok!(U28F4, &frac_28_4.zero, 10, 0, false);
assert_ok!(U28F4, &frac_28_4.gt_0, 10, 1, false);
assert_ok!(U28F4, &frac_28_4.max, 10, !0, false);
assert_ok!(U28F4, &frac_28_4.over, 10, 0, true);
let frac_32_0 = make_fraction_strings(&max_int_32, eps_0);
assert_ok!(U32F0, &frac_32_0.zero, 10, 0, false);
assert_ok!(U32F0, &frac_32_0.gt_0, 10, 1, false);
assert_ok!(U32F0, &frac_32_0.max, 10, !0, false);
assert_ok!(U32F0, &frac_32_0.over, 10, 0, true);
let frac_0_128 = make_fraction_strings(&max_int_0, eps_128);
assert_ok!(U0F128, &frac_0_128.zero, 10, 0, false);
assert_ok!(U0F128, &frac_0_128.gt_0, 10, 1, false);
assert_ok!(U0F128, &frac_0_128.max, 10, !0, false);
assert_ok!(U0F128, &frac_0_128.over, 10, 0, true);
let frac_4_124 = make_fraction_strings(&max_int_4, eps_124);
assert_ok!(U4F124, &frac_4_124.zero, 10, 0, false);
assert_ok!(U4F124, &frac_4_124.gt_0, 10, 1, false);
assert_ok!(U4F124, &frac_4_124.max, 10, !0, false);
assert_ok!(U4F124, &frac_4_124.over, 10, 0, true);
let frac_64_64 = make_fraction_strings(&max_int_64, eps_64);
assert_ok!(U64F64, &frac_64_64.zero, 10, 0, false);
assert_ok!(U64F64, &frac_64_64.gt_0, 10, 1, false);
assert_ok!(U64F64, &frac_64_64.max, 10, !0, false);
assert_ok!(U64F64, &frac_64_64.over, 10, 0, true);
let frac_124_4 = make_fraction_strings(&max_int_124, eps_4);
assert_ok!(U124F4, &frac_124_4.zero, 10, 0, false);
assert_ok!(U124F4, &frac_124_4.gt_0, 10, 1, false);
assert_ok!(U124F4, &frac_124_4.max, 10, !0, false);
assert_ok!(U124F4, &frac_124_4.over, 10, 0, true);
let frac_128_0 = make_fraction_strings(&max_int_128, eps_0);
assert_ok!(U128F0, &frac_128_0.zero, 10, 0, false);
assert_ok!(U128F0, &frac_128_0.gt_0, 10, 1, false);
assert_ok!(U128F0, &frac_128_0.max, 10, !0, false);
assert_ok!(U128F0, &frac_128_0.over, 10, 0, true);
assert_ok!(
U4F4,
"0.40624999999999999999999999999999999999999999999999",
10,
0x06,
false
);
assert_ok!(U4F4, "0.40625", 10, 0x06, false);
assert_ok!(
U4F4,
"0.40625000000000000000000000000000000000000000000001",
10,
0x07,
false
);
assert_ok!(U4F4, "0.4375", 10, 0x07, false);
assert_ok!(
U4F4,
"0.46874999999999999999999999999999999999999999999999",
10,
0x07,
false
);
assert_ok!(U4F4, "0.46875", 10, 0x08, false);
assert_ok!(
U4F4,
"0.46875000000000000000000000000000000000000000000001",
10,
0x08,
false
);
assert_ok!(U4F4, "0.5", 10, 0x08, false);
assert_ok!(
U4F4,
"0.53124999999999999999999999999999999999999999999999",
10,
0x08,
false
);
assert_ok!(U4F4, "0.53125", 10, 0x08, false);
assert_ok!(
U4F4,
"0.53125000000000000000000000000000000000000000000001",
10,
0x09,
false
);
assert_ok!(U4F4, "0.5625", 10, 0x09, false);
}
#[test]
fn frac4() {
for u in 0..=255u8 {
let (ifix, ufix) = (I4F4::from_bits(u as i8), U4F4::from_bits(u));
let (ifix_str, ufix_str) = (ifix.to_string(), ufix.to_string());
assert_eq!(I4F4::from_str(&ifix_str).unwrap(), ifix);
assert_eq!(U4F4::from_str(&ufix_str).unwrap(), ufix);
}
}
#[test]
fn frac17() {
for u in 0..(1 << 17) {
let fix = U15F17::from_bits(u) + U15F17::from_num(99);
let fix_pos = I15F17::from_num(fix);
let fix_neg = -fix_pos;
let fix_str = fix.to_string();
let fix_pos_str = fix_pos.to_string();
let fix_neg_str = fix_neg.to_string();
assert_eq!(fix_str, fix_pos_str);
if u != 0 {
assert_eq!(&fix_neg_str[..1], "-");
assert_eq!(&fix_neg_str[1..], fix_pos_str);
}
assert_eq!(U15F17::from_str(&fix_str).unwrap(), fix);
assert_eq!(I15F17::from_str(&fix_pos_str).unwrap(), fix_pos);
assert_eq!(I15F17::from_str(&fix_neg_str).unwrap(), fix_neg);
let fix_str3 = format!("{:.3}", fix);
let fix_pos_str3 = format!("{:.3}", fix_pos);
let fix_neg_str3 = format!("{:.3}", fix_neg);
assert_eq!(fix_str3, fix_pos_str3);
if u != 0 {
assert_eq!(&fix_neg_str3[..1], "-");
assert_eq!(&fix_neg_str3[1..], fix_pos_str3);
}
let max_diff = U15F17::from_bits((5 << 17) / 10000 + 1);
let from_fix_str3 = U15F17::from_str(&fix_str3).unwrap();
assert!(from_fix_str3.dist(fix) <= max_diff);
let from_fix_pos_str3 = I15F17::from_str(&fix_pos_str3).unwrap();
assert!(from_fix_pos_str3.dist(fix_pos) <= max_diff);
let from_fix_neg_str3 = I15F17::from_str(&fix_neg_str3).unwrap();
assert!(from_fix_neg_str3.dist(fix_neg) <= max_diff);
let fix_str9 = format!("{:.9}", fix);
let fix_pos_str9 = format!("{:.9}", fix_pos);
let fix_neg_str9 = format!("{:.9}", fix_neg);
assert_eq!(fix_str9, fix_pos_str9);
if u != 0 {
assert_eq!(&fix_neg_str9[..1], "-");
assert_eq!(&fix_neg_str9[1..], fix_pos_str9);
}
assert_eq!(U15F17::from_str(&fix_str9).unwrap(), fix);
assert_eq!(I15F17::from_str(&fix_pos_str9).unwrap(), fix_pos);
assert_eq!(I15F17::from_str(&fix_neg_str9).unwrap(), fix_neg);
}
}
}