Fix bug in hex parsing for negative numbers.
This commit is contained in:
parent
10fa6844c4
commit
9736171089
@ -8,6 +8,7 @@ Bug fixes
|
|||||||
---------
|
---------
|
||||||
|
|
||||||
* Custom syntax now works properly inside binary expressions and with method calls.
|
* Custom syntax now works properly inside binary expressions and with method calls.
|
||||||
|
* Hex numbers with the high-bit set now parse correctly into negative integer numbers.
|
||||||
|
|
||||||
Enhancements
|
Enhancements
|
||||||
------------
|
------------
|
||||||
|
14
src/lib.rs
14
src/lib.rs
@ -98,6 +98,20 @@ pub type INT = i64;
|
|||||||
#[cfg(feature = "only_i32")]
|
#[cfg(feature = "only_i32")]
|
||||||
pub type INT = i32;
|
pub type INT = i32;
|
||||||
|
|
||||||
|
/// The system base integer type. It is defined as [`u64`].
|
||||||
|
///
|
||||||
|
/// If the `only_i32` feature is enabled, this will be [`u32`] instead.
|
||||||
|
#[cfg(not(feature = "only_i32"))]
|
||||||
|
#[allow(non_camel_case_types)]
|
||||||
|
type INT_BASE = u64;
|
||||||
|
/// The system integer base type.
|
||||||
|
/// It is defined as [`u32`] since the `only_i32` feature is used.
|
||||||
|
///
|
||||||
|
/// If the `only_i32` feature is not used, this will be `u64` instead.
|
||||||
|
#[cfg(feature = "only_i32")]
|
||||||
|
#[allow(non_camel_case_types)]
|
||||||
|
type INT_BASE = u32;
|
||||||
|
|
||||||
/// The system floating-point type. It is defined as [`f64`].
|
/// The system floating-point type. It is defined as [`f64`].
|
||||||
/// Not available under `no_float`.
|
/// Not available under `no_float`.
|
||||||
///
|
///
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
#![allow(non_snake_case)]
|
#![allow(non_snake_case)]
|
||||||
|
|
||||||
use crate::plugin::*;
|
use crate::plugin::*;
|
||||||
use crate::{def_package, Position, INT};
|
use crate::{def_package, Position, INT, INT_BASE};
|
||||||
#[cfg(feature = "no_std")]
|
#[cfg(feature = "no_std")]
|
||||||
use std::prelude::v1::*;
|
use std::prelude::v1::*;
|
||||||
|
|
||||||
@ -120,7 +120,9 @@ mod int_functions {
|
|||||||
.into());
|
.into());
|
||||||
}
|
}
|
||||||
|
|
||||||
INT::from_str_radix(string.trim(), radix as u32).map_err(|err| {
|
INT_BASE::from_str_radix(string.trim(), radix as u32)
|
||||||
|
.map(|v| v as INT)
|
||||||
|
.map_err(|err| {
|
||||||
EvalAltResult::ErrorArithmetic(
|
EvalAltResult::ErrorArithmetic(
|
||||||
format!("Error parsing integer number '{}': {}", string, err),
|
format!("Error parsing integer number '{}': {}", string, err),
|
||||||
Position::NONE,
|
Position::NONE,
|
||||||
|
@ -5,7 +5,7 @@ use crate::engine::{
|
|||||||
KEYWORD_FN_PTR_CURRY, KEYWORD_IS_DEF_VAR, KEYWORD_PRINT, KEYWORD_THIS, KEYWORD_TYPE_OF,
|
KEYWORD_FN_PTR_CURRY, KEYWORD_IS_DEF_VAR, KEYWORD_PRINT, KEYWORD_THIS, KEYWORD_TYPE_OF,
|
||||||
};
|
};
|
||||||
use crate::func::native::OnParseTokenCallback;
|
use crate::func::native::OnParseTokenCallback;
|
||||||
use crate::{Engine, LexError, StaticVec, INT};
|
use crate::{Engine, LexError, StaticVec, INT, INT_BASE};
|
||||||
#[cfg(feature = "no_std")]
|
#[cfg(feature = "no_std")]
|
||||||
use std::prelude::v1::*;
|
use std::prelude::v1::*;
|
||||||
use std::{
|
use std::{
|
||||||
@ -1507,7 +1507,8 @@ fn get_next_token_inner(
|
|||||||
.filter(|&&c| c != NUMBER_SEPARATOR)
|
.filter(|&&c| c != NUMBER_SEPARATOR)
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
INT::from_str_radix(&out, radix)
|
INT_BASE::from_str_radix(&out, radix)
|
||||||
|
.map(|v| v as INT)
|
||||||
.map(Token::IntegerConstant)
|
.map(Token::IntegerConstant)
|
||||||
.unwrap_or_else(|_| {
|
.unwrap_or_else(|_| {
|
||||||
Token::LexError(LERR::MalformedNumber(result.into_iter().collect()))
|
Token::LexError(LERR::MalformedNumber(result.into_iter().collect()))
|
||||||
@ -1515,7 +1516,9 @@ fn get_next_token_inner(
|
|||||||
} else {
|
} else {
|
||||||
let out: String =
|
let out: String =
|
||||||
result.iter().filter(|&&c| c != NUMBER_SEPARATOR).collect();
|
result.iter().filter(|&&c| c != NUMBER_SEPARATOR).collect();
|
||||||
let num = INT::from_str(&out).map(Token::IntegerConstant);
|
let num = INT_BASE::from_str(&out)
|
||||||
|
.map(|v| v as INT)
|
||||||
|
.map(Token::IntegerConstant);
|
||||||
|
|
||||||
// If integer parsing is unnecessary, try float instead
|
// If integer parsing is unnecessary, try float instead
|
||||||
#[cfg(not(feature = "no_float"))]
|
#[cfg(not(feature = "no_float"))]
|
||||||
|
@ -27,6 +27,11 @@ fn test_hex_literal() -> Result<(), Box<EvalAltResult>> {
|
|||||||
assert_eq!(engine.eval::<INT>("let x = 0Xf; x")?, 15);
|
assert_eq!(engine.eval::<INT>("let x = 0Xf; x")?, 15);
|
||||||
assert_eq!(engine.eval::<INT>("let x = 0xff; x")?, 255);
|
assert_eq!(engine.eval::<INT>("let x = 0xff; x")?, 255);
|
||||||
|
|
||||||
|
#[cfg(not(feature = "only_i32"))]
|
||||||
|
assert_eq!(engine.eval::<INT>("let x = 0xffffffffffffffff; x")?, -1);
|
||||||
|
#[cfg(feature = "only_i32")]
|
||||||
|
assert_eq!(engine.eval::<INT>("let x = 0xffffffff; x")?, -1);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -51,6 +56,18 @@ fn test_binary_literal() -> Result<(), Box<EvalAltResult>> {
|
|||||||
engine.eval::<INT>("let x = 0b0011_1100_1010_0101; x")?,
|
engine.eval::<INT>("let x = 0b0011_1100_1010_0101; x")?,
|
||||||
15525
|
15525
|
||||||
);
|
);
|
||||||
|
#[cfg(not(feature = "only_i32"))]
|
||||||
|
assert_eq!(
|
||||||
|
engine.eval::<INT>(
|
||||||
|
"let x = 0b11111111_11111111_11111111_11111111_11111111_11111111_11111111_11111111; x"
|
||||||
|
)?,
|
||||||
|
-1
|
||||||
|
);
|
||||||
|
#[cfg(feature = "only_i32")]
|
||||||
|
assert_eq!(
|
||||||
|
engine.eval::<INT>("let x = 0b11111111_11111111_11111111_11111111; x")?,
|
||||||
|
-1
|
||||||
|
);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user