use lib::{mem, ptr};
use util::*;
#[cfg(feature = "radix")]
use super::radix::{double_radix, float_radix};
cfg_if! {
if #[cfg(feature = "grisu3")] {
use super::grisu3::{double_decimal, float_decimal};
} else if #[cfg(feature = "ryu")] {
use super::ryu::{double_decimal, float_decimal};
} else {
use super::grisu2::{double_decimal, float_decimal};
}}
pub(crate) trait FloatToString: Float {
unsafe extern "C" fn decimal(self, first: *mut u8) -> *mut u8;
#[cfg(feature = "radix")]
unsafe extern "C" fn radix(self, radix: u32, first: *mut u8) -> *mut u8;
}
impl FloatToString for f32 {
#[inline(always)]
unsafe extern "C" fn decimal(self, first: *mut u8) -> *mut u8 {
float_decimal(self, first)
}
#[inline(always)]
#[cfg(feature = "radix")]
unsafe extern "C" fn radix(self, radix: u32, first: *mut u8) -> *mut u8 {
float_radix(self, radix, first)
}
}
impl FloatToString for f64 {
#[inline(always)]
unsafe extern "C" fn decimal(self, first: *mut u8) -> *mut u8 {
double_decimal(self, first)
}
#[inline(always)]
#[cfg(feature = "radix")]
unsafe extern "C" fn radix(self, radix: u32, first: *mut u8) -> *mut u8 {
double_radix(self, radix, first)
}
}
#[inline]
unsafe fn forward<F: FloatToString>(value: F, radix: u32, first: *mut u8)
-> *mut u8
{
debug_assert_radix!(radix);
#[cfg(not(feature = "radix"))] {
value.decimal(first)
}
#[cfg(feature = "radix")] {
match radix {
10 => value.decimal(first),
_ => value.radix(radix, first),
}
}
}
#[inline]
unsafe fn filter_special<F: FloatToString>(value: F, radix: u32, first: *mut u8)
-> *mut u8
{
debug_assert!(value.is_sign_positive(), "Value cannot be negative.");
debug_assert_radix!(radix);
if value.is_zero() {
ptr::copy_nonoverlapping(b"0.0".as_ptr(), first, 3);
first.add(3)
} else if value.is_nan() {
ptr::copy_nonoverlapping(NAN_STRING.as_ptr(), first, NAN_STRING.len());
first.add(NAN_STRING.len())
} else if value.is_special() {
ptr::copy_nonoverlapping(INF_STRING.as_ptr(), first, INF_STRING.len());
first.add(INF_STRING.len())
} else {
forward(value, radix, first)
}
}
#[inline]
unsafe fn filter_sign<F: FloatToString>(mut value: F, radix: u32, mut first: *mut u8)
-> *mut u8
{
debug_assert_radix!(radix);
if value.is_sign_negative() {
*first= b'-';
value = -value;
first = first.add(1);
}
filter_special(value, radix, first)
}
#[inline]
unsafe fn filter_buffer<F: FloatToString>(value: F, radix: u32, first: *mut u8, last: *mut u8)
-> *mut u8
{
debug_assert!(first <= last, "First must be <= last");
debug_assert_radix!(radix);
let dist = distance(first, last);
if dist == 0 {
first
} else if dist < BUFFER_SIZE {
let mut buffer: [u8; BUFFER_SIZE] = mem::uninitialized();
let p = buffer.as_mut_ptr();
filter_sign(value, radix, p);
let length = distance(p, p.add(BUFFER_SIZE)).min(dist);
ptr::copy_nonoverlapping(p, first, length);
first.add(length)
} else {
filter_sign(value, radix, first)
}
}
macro_rules! generate_unsafe_api {
($name:ident, $t:ty) => (
#[inline]
unsafe fn $name(value: $t, base: u8, first: *mut u8, last: *mut u8) -> *mut u8
{
filter_buffer(value, base.into(), first, last)
}
)
}
generate_unsafe_api!(f32toa_unsafe, f32);
generate_unsafe_api!(f64toa_unsafe, f64);
generate_to_bytes_local!(f32toa_local, f32, f32toa_unsafe);
generate_to_bytes_local!(f64toa_local, f64, f64toa_unsafe);
generate_to_range_api!(f32toa_range, f32, f32toa_local);
generate_to_range_api!(f64toa_range, f64, f64toa_local);
generate_to_slice_api!(f32toa_slice, f32, f32toa_local);
generate_to_slice_api!(f64toa_slice, f64, f64toa_local);
#[cfg(test)]
mod tests {
use util::test::*;
use super::*;
use atof::*;
const F32_DATA : [f32; 31] = [0., 0.1, 1., 1.1, 12., 12.1, 123., 123.1, 1234., 1234.1, 12345., 12345.1, 123456., 123456.1, 1234567., 1234567.1, 12345678., 12345678.1, 123456789., 123456789.1, 123456789.12, 123456789.123, 123456789.1234, 123456789.12345, 1.2345678912345e8, 1.2345e+8, 1.2345e+11, 1.2345e+38, 1.2345e-8, 1.2345e-11, 1.2345e-38];
const F64_DATA: [f64; 33] = [0., 0.1, 1., 1.1, 12., 12.1, 123., 123.1, 1234., 1234.1, 12345., 12345.1, 123456., 123456.1, 1234567., 1234567.1, 12345678., 12345678.1, 123456789., 123456789.1, 123456789.12, 123456789.123, 123456789.1234, 123456789.12345, 1.2345678912345e8, 1.2345e+8, 1.2345e+11, 1.2345e+38, 1.2345e+308, 1.2345e-8, 1.2345e-11, 1.2345e-38, 1.2345e-299];
#[cfg(feature = "radix")]
#[test]
fn f32toa_base2_test() {
let mut buffer = new_buffer();
assert_eq!(as_slice(b"0.0"), f32toa_slice(0.0, 2, &mut buffer));
assert_eq!(as_slice(b"-0.0"), f32toa_slice(-0.0, 2, &mut buffer));
assert_eq!(as_slice(b"1.0"), f32toa_slice(1.0, 2, &mut buffer));
assert_eq!(as_slice(b"10.0"), f32toa_slice(2.0, 2, &mut buffer));
assert_eq!(as_slice(b"1.1"), f32toa_slice(1.5, 2, &mut buffer));
assert_eq!(as_slice(b"1.01"), f32toa_slice(1.25, 2, &mut buffer));
assert_eq!(b"1.001111000000110010", &f32toa_slice(1.2345678901234567890e0, 2, &mut buffer)[..20]);
assert_eq!(b"1100.010110000111111", &f32toa_slice(1.2345678901234567890e1, 2, &mut buffer)[..20]);
assert_eq!(b"1111011.011101001111", &f32toa_slice(1.2345678901234567890e2, 2, &mut buffer)[..20]);
assert_eq!(b"10011010010.10010001", &f32toa_slice(1.2345678901234567890e3, 2, &mut buffer)[..20]);
assert_eq!(b"-1.001111000000110010", &f32toa_slice(-1.2345678901234567890e0, 2, &mut buffer)[..21]);
assert_eq!(b"-1100.010110000111111", &f32toa_slice(-1.2345678901234567890e1, 2, &mut buffer)[..21]);
assert_eq!(b"-1111011.011101001111", &f32toa_slice(-1.2345678901234567890e2, 2, &mut buffer)[..21]);
assert_eq!(b"-10011010010.10010001", &f32toa_slice(-1.2345678901234567890e3, 2, &mut buffer)[..21]);
assert_eq!(as_slice(b"NaN"), f32toa_slice(f32::NAN, 2, &mut buffer));
assert_eq!(as_slice(b"inf"), f32toa_slice(f32::INFINITY, 2, &mut buffer));
assert_eq!(as_slice(b"1.101010000010101111000e-11011"), f32toa_slice(0.000000012345, 2, &mut buffer));
}
#[test]
fn f32toa_base10_test() {
let mut buffer = new_buffer();
assert_eq!(as_slice(b"0.0"), f32toa_slice(0.0, 10, &mut buffer));
assert_eq!(as_slice(b"-0.0"), f32toa_slice(-0.0, 10, &mut buffer));
assert_eq!(as_slice(b"1.0"), f32toa_slice(1.0, 10, &mut buffer));
assert_eq!(as_slice(b"10.0"), f32toa_slice(10.0, 10, &mut buffer));
assert_eq!(as_slice(b"1.234567"), &f32toa_slice(1.2345678901234567890e0, 10, &mut buffer)[..8]);
assert_eq!(as_slice(b"12.34567"), &f32toa_slice(1.2345678901234567890e1, 10, &mut buffer)[..8]);
assert_eq!(as_slice(b"123.4567"), &f32toa_slice(1.2345678901234567890e2, 10, &mut buffer)[..8]);
assert_eq!(as_slice(b"1234.567"), &f32toa_slice(1.2345678901234567890e3, 10, &mut buffer)[..8]);
assert_eq!(as_slice(b"-1.234567"), &f32toa_slice(-1.2345678901234567890e0, 10, &mut buffer)[..9]);
assert_eq!(as_slice(b"-12.34567"), &f32toa_slice(-1.2345678901234567890e1, 10, &mut buffer)[..9]);
assert_eq!(as_slice(b"-123.4567"), &f32toa_slice(-1.2345678901234567890e2, 10, &mut buffer)[..9]);
assert_eq!(as_slice(b"-1234.567"), &f32toa_slice(-1.2345678901234567890e3, 10, &mut buffer)[..9]);
assert_eq!(as_slice(b"NaN"), f32toa_slice(f32::NAN, 10, &mut buffer));
assert_eq!(as_slice(b"inf"), f32toa_slice(f32::INFINITY, 10, &mut buffer));
}
#[test]
fn f32toa_base10_roundtrip_test() {
let mut buffer = new_buffer();
for f in F32_DATA.iter() {
let s = f32toa_slice(*f, 10, &mut buffer);
assert_relative_eq!(atof32_slice(10, s), *f, epsilon=1e-6, max_relative=1e-6);
}
}
#[cfg(feature = "radix")]
#[test]
fn f32toa_basen_roundtrip_test() {
let mut buffer = new_buffer();
for f in F32_DATA.iter() {
for radix in 2..37 {
let s = f32toa_slice(*f, radix, &mut buffer);
assert_relative_eq!(atof32_slice(radix, s), *f, max_relative=2e-5);
}
}
}
#[cfg(feature = "radix")]
#[test]
fn f64toa_base2_test() {
let mut buffer = new_buffer();
assert_eq!(as_slice(b"0.0"), f64toa_slice(0.0, 2, &mut buffer));
assert_eq!(as_slice(b"-0.0"), f64toa_slice(-0.0, 2, &mut buffer));
assert_eq!(as_slice(b"1.0"), f64toa_slice(1.0, 2, &mut buffer));
assert_eq!(as_slice(b"10.0"), f64toa_slice(2.0, 2, &mut buffer));
assert_eq!(as_slice(b"1.00111100000011001010010000101000110001"), &f64toa_slice(1.2345678901234567890e0, 2, &mut buffer)[..40]);
assert_eq!(as_slice(b"1100.01011000011111100110100110010111101"), &f64toa_slice(1.2345678901234567890e1, 2, &mut buffer)[..40]);
assert_eq!(as_slice(b"1111011.01110100111100000001111111101101"), &f64toa_slice(1.2345678901234567890e2, 2, &mut buffer)[..40]);
assert_eq!(as_slice(b"10011010010.1001000101100001001111110100"), &f64toa_slice(1.2345678901234567890e3, 2, &mut buffer)[..40]);
assert_eq!(as_slice(b"-1.00111100000011001010010000101000110001"), &f64toa_slice(-1.2345678901234567890e0, 2, &mut buffer)[..41]);
assert_eq!(as_slice(b"-1100.01011000011111100110100110010111101"), &f64toa_slice(-1.2345678901234567890e1, 2, &mut buffer)[..41]);
assert_eq!(as_slice(b"-1111011.01110100111100000001111111101101"), &f64toa_slice(-1.2345678901234567890e2, 2, &mut buffer)[..41]);
assert_eq!(as_slice(b"-10011010010.1001000101100001001111110100"), &f64toa_slice(-1.2345678901234567890e3, 2, &mut buffer)[..41]);
assert_eq!(as_slice(b"NaN"), f64toa_slice(f64::NAN, 2, &mut buffer));
assert_eq!(as_slice(b"inf"), f64toa_slice(f64::INFINITY, 2, &mut buffer));
}
#[test]
fn f64toa_base10_test() {
let mut buffer = new_buffer();
assert_eq!(as_slice(b"0.0"), f64toa_slice(0.0, 10, &mut buffer));
assert_eq!(as_slice(b"-0.0"), f64toa_slice(-0.0, 10, &mut buffer));
assert_eq!(as_slice(b"1.0"), f64toa_slice(1.0, 10, &mut buffer));
assert_eq!(as_slice(b"10.0"), f64toa_slice(10.0, 10, &mut buffer));
assert_eq!(as_slice(b"1.234567"), &f64toa_slice(1.2345678901234567890e0, 10, &mut buffer)[..8]);
assert_eq!(as_slice(b"12.34567"), &f64toa_slice(1.2345678901234567890e1, 10, &mut buffer)[..8]);
assert_eq!(as_slice(b"123.4567"), &f64toa_slice(1.2345678901234567890e2, 10, &mut buffer)[..8]);
assert_eq!(as_slice(b"1234.567"), &f64toa_slice(1.2345678901234567890e3, 10, &mut buffer)[..8]);
assert_eq!(as_slice(b"-1.234567"), &f64toa_slice(-1.2345678901234567890e0, 10, &mut buffer)[..9]);
assert_eq!(as_slice(b"-12.34567"), &f64toa_slice(-1.2345678901234567890e1, 10, &mut buffer)[..9]);
assert_eq!(as_slice(b"-123.4567"), &f64toa_slice(-1.2345678901234567890e2, 10, &mut buffer)[..9]);
assert_eq!(as_slice(b"-1234.567"), &f64toa_slice(-1.2345678901234567890e3, 10, &mut buffer)[..9]);
assert_eq!(b"NaN".to_vec(), f64toa_slice(f64::NAN, 10, &mut buffer));
assert_eq!(b"inf".to_vec(), f64toa_slice(f64::INFINITY, 10, &mut buffer));
}
#[test]
fn f64toa_base10_roundtrip_test() {
let mut buffer = new_buffer();
for f in F64_DATA.iter() {
let s = f64toa_slice(*f, 10, &mut buffer);
assert_relative_eq!(atof64_slice(10, s), *f, epsilon=1e-12, max_relative=1e-12);
}
}
#[cfg(feature = "radix")]
#[test]
fn f64toa_basen_roundtrip_test() {
let mut buffer = new_buffer();
for f in F64_DATA.iter() {
for radix in 2..37 {
let s = f64toa_slice(*f, radix, &mut buffer);
assert_relative_eq!(atof64_slice(radix, s), *f, max_relative=3e-5);
}
}
}
#[cfg(feature = "correct")]
quickcheck! {
fn f32_quickcheck(f: f32) -> bool {
let mut buffer = new_buffer();
f == atof32_slice(10, f32toa_slice(f, 10, &mut buffer))
}
fn f64_quickcheck(f: f64) -> bool {
let mut buffer = new_buffer();
f == atof64_slice(10, f64toa_slice(f, 10, &mut buffer))
}
}
}