use lib::slice;
use super::rounding::RoundingKind;
pub static mut NAN_STRING: &[u8] = b"NaN";
pub static mut INF_STRING: &[u8] = b"inf";
pub static mut INFINITY_STRING: &[u8] = b"infinity";
#[no_mangle]
pub static mut EXPONENT_DEFAULT_CHAR: u8 = b'e';
#[cfg(feature ="radix")]
#[no_mangle]
pub static mut EXPONENT_BACKUP_CHAR: u8 = b'^';
#[no_mangle]
pub unsafe extern fn get_nan_string(ptr: *mut *const u8, size: *mut usize)
-> i32
{
if ptr.is_null() || size.is_null() {
-1
} else {
*ptr = NAN_STRING.as_ptr();
*size = NAN_STRING.len();
0
}
}
#[no_mangle]
pub unsafe extern fn set_nan_string(ptr: *const u8, size: usize)
-> i32
{
if ptr.is_null() {
-1
} else {
NAN_STRING = slice::from_raw_parts(ptr, size);
0
}
}
#[no_mangle]
pub unsafe extern fn get_inf_string(ptr: *mut *const u8, size: *mut usize)
-> i32
{
if ptr.is_null() || size.is_null() {
-1
} else {
*ptr = INF_STRING.as_ptr();
*size = INF_STRING.len();
0
}
}
#[no_mangle]
pub unsafe extern fn set_inf_string(ptr: *const u8, size: usize)
-> i32
{
if ptr.is_null() {
-1
} else {
INF_STRING = slice::from_raw_parts(ptr, size);
0
}
}
#[no_mangle]
pub unsafe extern fn get_infinity_string(ptr: *mut *const u8, size: *mut usize)
-> i32
{
if ptr.is_null() || size.is_null() {
-1
} else {
*ptr = INFINITY_STRING.as_ptr();
*size = INFINITY_STRING.len();
0
}
}
#[no_mangle]
pub unsafe extern fn set_infinity_string(ptr: *const u8, size: usize)
-> i32
{
if ptr.is_null() {
-1
} else {
INFINITY_STRING = slice::from_raw_parts(ptr, size);
0
}
}
cfg_if! {
if #[cfg(feature = "radix")] {
pub const MAX_I8_SIZE: usize = 16;
pub const MAX_I16_SIZE: usize = 32;
pub const MAX_I32_SIZE: usize = 64;
pub const MAX_I64_SIZE: usize = 128;
pub const MAX_I128_SIZE: usize = 256;
pub const MAX_U8_SIZE: usize = 16;
pub const MAX_U16_SIZE: usize = 32;
pub const MAX_U32_SIZE: usize = 64;
pub const MAX_U64_SIZE: usize = 128;
pub const MAX_U128_SIZE: usize = 256;
pub const MAX_F32_SIZE: usize = 256;
pub const MAX_F64_SIZE: usize = 256;
} else {
pub const MAX_I8_SIZE: usize = 4;
pub const MAX_I16_SIZE: usize = 6;
pub const MAX_I32_SIZE: usize = 11;
pub const MAX_I64_SIZE: usize = 20;
pub const MAX_I128_SIZE: usize = 40;
pub const MAX_U8_SIZE: usize = 3;
pub const MAX_U16_SIZE: usize = 5;
pub const MAX_U32_SIZE: usize = 10;
pub const MAX_U64_SIZE: usize = 20;
pub const MAX_U128_SIZE: usize = 39;
pub const MAX_F32_SIZE: usize = 64;
pub const MAX_F64_SIZE: usize = 64;
}}
cfg_if! {
if #[cfg(target_pointer_width = "16")] {
pub const MAX_ISIZE_SIZE: usize = MAX_I16_SIZE;
pub const MAX_USIZE_SIZE: usize = MAX_U16_SIZE;
} else if #[cfg(target_pointer_width = "32")] {
pub const MAX_ISIZE_SIZE: usize = MAX_I32_SIZE;
pub const MAX_USIZE_SIZE: usize = MAX_U32_SIZE;
} else if #[cfg(target_pointer_width = "64")] {
pub const MAX_ISIZE_SIZE: usize = MAX_I64_SIZE;
pub const MAX_USIZE_SIZE: usize = MAX_U64_SIZE;
}}
pub const BUFFER_SIZE: usize = MAX_F64_SIZE;
#[no_mangle]
pub static MAX_I8_SIZE_FFI: usize = MAX_I8_SIZE;
#[no_mangle]
pub static MAX_I16_SIZE_FFI: usize = MAX_I16_SIZE;
#[no_mangle]
pub static MAX_I32_SIZE_FFI: usize = MAX_I32_SIZE;
#[no_mangle]
pub static MAX_I64_SIZE_FFI: usize = MAX_I64_SIZE;
#[no_mangle]
pub static MAX_I128_SIZE_FFI: usize = MAX_I128_SIZE;
#[no_mangle]
pub static MAX_ISIZE_SIZE_FFI: usize = MAX_ISIZE_SIZE;
#[no_mangle]
pub static MAX_U8_SIZE_FFI: usize = MAX_U8_SIZE;
#[no_mangle]
pub static MAX_U16_SIZE_FFI: usize = MAX_U16_SIZE;
#[no_mangle]
pub static MAX_U32_SIZE_FFI: usize = MAX_U32_SIZE;
#[no_mangle]
pub static MAX_U64_SIZE_FFI: usize = MAX_U64_SIZE;
#[no_mangle]
pub static MAX_U128_SIZE_FFI: usize = MAX_U128_SIZE;
#[no_mangle]
pub static MAX_USIZE_SIZE_FFI: usize = MAX_USIZE_SIZE;
#[no_mangle]
pub static MAX_F32_SIZE_FFI: usize = MAX_F32_SIZE;
#[no_mangle]
pub static MAX_F64_SIZE_FFI: usize = MAX_F64_SIZE;
#[no_mangle]
pub static BUFFER_SIZE_FFI: usize = BUFFER_SIZE;
#[no_mangle]
pub static mut FLOAT_ROUNDING: RoundingKind = RoundingKind::NearestTieEven;
#[inline]
#[allow(unused_variables)]
pub(crate) fn exponent_notation_char(radix: u32) -> u8 {
unsafe {
#[cfg(not(feature ="radix"))] {
EXPONENT_DEFAULT_CHAR
}
#[cfg(feature ="radix")] {
if radix >= 15 { EXPONENT_BACKUP_CHAR } else { EXPONENT_DEFAULT_CHAR }
}
}
}
#[cfg(test)]
mod tests {
use atof::*;
use ftoa::*;
use util::*;
use util::test::*;
use super::*;
#[cfg(feature ="radix")]
#[test]
fn exponent_notation_char_test() {
unsafe {
assert_eq!(exponent_notation_char(2), EXPONENT_DEFAULT_CHAR);
assert_eq!(exponent_notation_char(8), EXPONENT_DEFAULT_CHAR);
assert_eq!(exponent_notation_char(10), EXPONENT_DEFAULT_CHAR);
assert_eq!(exponent_notation_char(15), EXPONENT_BACKUP_CHAR);
assert_eq!(exponent_notation_char(16), EXPONENT_BACKUP_CHAR);
assert_eq!(exponent_notation_char(32), EXPONENT_BACKUP_CHAR);
}
}
#[test]
#[ignore]
fn special_bytes_test() {
unsafe {
let mut buffer = new_buffer();
assert!(try_atof32_slice(b"NaN").value.is_nan());
assert!(try_atof32_slice(b"nan").value.is_nan());
assert!(try_atof32_slice(b"NAN").value.is_nan());
assert!(try_atof32_slice(b"inf").value.is_infinite());
assert!(try_atof32_slice(b"INF").value.is_infinite());
assert!(try_atof32_slice(b"Infinity").value.is_infinite());
assert_eq!(f64toa_slice(f64::NAN, &mut buffer), b"NaN");
assert_eq!(f64toa_slice(f64::INFINITY, &mut buffer), b"inf");
NAN_STRING = b"nan";
INF_STRING = b"Infinity";
assert!(try_atof32_slice(b"inf").error.code == ErrorCode::InvalidDigit);
assert!(try_atof32_slice(b"Infinity").value.is_infinite());
assert_eq!(f64toa_slice(f64::NAN, &mut buffer), b"nan");
assert_eq!(f64toa_slice(f64::INFINITY, &mut buffer), b"Infinity");
NAN_STRING = b"NaN";
INF_STRING = b"inf";
}
}
#[cfg(feature = "correct")]
#[test]
#[ignore]
fn special_rounding_test() {
unsafe {
FLOAT_ROUNDING = RoundingKind::NearestTieEven;
assert_eq!(try_atof64_slice(b"-9007199254740993").value, -9007199254740992.0);
assert_eq!(try_atof64_slice(b"-9007199254740995").value, -9007199254740996.0);
assert_eq!(try_atof64_slice(b"9007199254740993").value, 9007199254740992.0);
assert_eq!(try_atof64_slice(b"9007199254740995").value, 9007199254740996.0);
FLOAT_ROUNDING = RoundingKind::NearestTieAwayZero;
assert_eq!(try_atof64_slice(b"-9007199254740993").value, -9007199254740994.0);
assert_eq!(try_atof64_slice(b"-9007199254740995").value, -9007199254740996.0);
assert_eq!(try_atof64_slice(b"9007199254740993").value, 9007199254740994.0);
assert_eq!(try_atof64_slice(b"9007199254740995").value, 9007199254740996.0);
FLOAT_ROUNDING = RoundingKind::TowardPositiveInfinity;
assert_eq!(try_atof64_slice(b"-9007199254740993").value, -9007199254740992.0);
assert_eq!(try_atof64_slice(b"-9007199254740995").value, -9007199254740994.0);
assert_eq!(try_atof64_slice(b"9007199254740993").value, 9007199254740994.0);
assert_eq!(try_atof64_slice(b"9007199254740995").value, 9007199254740996.0);
FLOAT_ROUNDING = RoundingKind::TowardNegativeInfinity;
assert_eq!(try_atof64_slice(b"-9007199254740993").value, -9007199254740994.0);
assert_eq!(try_atof64_slice(b"-9007199254740995").value, -9007199254740996.0);
assert_eq!(try_atof64_slice(b"9007199254740993").value, 9007199254740992.0);
assert_eq!(try_atof64_slice(b"9007199254740995").value, 9007199254740994.0);
FLOAT_ROUNDING = RoundingKind::TowardZero;
assert_eq!(try_atof64_slice(b"-9007199254740993").value, -9007199254740992.0);
assert_eq!(try_atof64_slice(b"-9007199254740995").value, -9007199254740994.0);
assert_eq!(try_atof64_slice(b"9007199254740993").value, 9007199254740992.0);
assert_eq!(try_atof64_slice(b"9007199254740995").value, 9007199254740994.0);
FLOAT_ROUNDING = RoundingKind::NearestTieEven;
}
}
#[cfg(all(feature = "correct", feature = "radix"))]
#[test]
#[ignore]
fn special_rounding_binary_test() {
unsafe {
FLOAT_ROUNDING = RoundingKind::NearestTieEven;
assert_eq!(try_atof64_radix_slice(2, b"-100000000000000000000000000000000000000000000000000001").value, -9007199254740992.0);
assert_eq!(try_atof64_radix_slice(2, b"-100000000000000000000000000000000000000000000000000011").value, -9007199254740996.0);
assert_eq!(try_atof64_radix_slice(2, b"100000000000000000000000000000000000000000000000000001").value, 9007199254740992.0);
assert_eq!(try_atof64_radix_slice(2, b"100000000000000000000000000000000000000000000000000011").value, 9007199254740996.0);
FLOAT_ROUNDING = RoundingKind::NearestTieAwayZero;
assert_eq!(try_atof64_radix_slice(2, b"-100000000000000000000000000000000000000000000000000001").value, -9007199254740994.0);
assert_eq!(try_atof64_radix_slice(2, b"-100000000000000000000000000000000000000000000000000011").value, -9007199254740996.0);
assert_eq!(try_atof64_radix_slice(2, b"100000000000000000000000000000000000000000000000000001").value, 9007199254740994.0);
assert_eq!(try_atof64_radix_slice(2, b"100000000000000000000000000000000000000000000000000011").value, 9007199254740996.0);
FLOAT_ROUNDING = RoundingKind::TowardPositiveInfinity;
assert_eq!(try_atof64_radix_slice(2, b"-100000000000000000000000000000000000000000000000000001").value, -9007199254740992.0);
assert_eq!(try_atof64_radix_slice(2, b"-100000000000000000000000000000000000000000000000000011").value, -9007199254740994.0);
assert_eq!(try_atof64_radix_slice(2, b"100000000000000000000000000000000000000000000000000001").value, 9007199254740994.0);
assert_eq!(try_atof64_radix_slice(2, b"100000000000000000000000000000000000000000000000000011").value, 9007199254740996.0);
FLOAT_ROUNDING = RoundingKind::TowardNegativeInfinity;
assert_eq!(try_atof64_radix_slice(2, b"-100000000000000000000000000000000000000000000000000001").value, -9007199254740994.0);
assert_eq!(try_atof64_radix_slice(2, b"-100000000000000000000000000000000000000000000000000011").value, -9007199254740996.0);
assert_eq!(try_atof64_radix_slice(2, b"100000000000000000000000000000000000000000000000000001").value, 9007199254740992.0);
assert_eq!(try_atof64_radix_slice(2, b"100000000000000000000000000000000000000000000000000011").value, 9007199254740994.0);
FLOAT_ROUNDING = RoundingKind::TowardZero;
assert_eq!(try_atof64_radix_slice(2, b"-100000000000000000000000000000000000000000000000000001").value, -9007199254740992.0);
assert_eq!(try_atof64_radix_slice(2, b"-100000000000000000000000000000000000000000000000000011").value, -9007199254740994.0);
assert_eq!(try_atof64_radix_slice(2, b"100000000000000000000000000000000000000000000000000001").value, 9007199254740992.0);
assert_eq!(try_atof64_radix_slice(2, b"100000000000000000000000000000000000000000000000000011").value, 9007199254740994.0);
FLOAT_ROUNDING = RoundingKind::NearestTieEven;
}
}
}