diff options
author | Ben Smith <binjimin@gmail.com> | 2019-03-28 16:42:19 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2019-03-28 16:42:19 -0700 |
commit | d84161cbfe968f8b0e67d945fd3669b71c5e73bb (patch) | |
tree | 87155acaae483890da027f505d0330aca7073258 | |
parent | cef3040884af87b702c5d285c940669b5e5a4236 (diff) | |
download | wabt-d84161cbfe968f8b0e67d945fd3669b71c5e73bb.tar.gz wabt-d84161cbfe968f8b0e67d945fd3669b71c5e73bb.tar.bz2 wabt-d84161cbfe968f8b0e67d945fd3669b71c5e73bb.zip |
Add ParseInt{8,16} functions, for use with SIMD (#1052)
-rw-r--r-- | src/literal.cc | 43 | ||||
-rw-r--r-- | src/literal.h | 8 | ||||
-rw-r--r-- | src/test-literal.cc | 130 | ||||
-rw-r--r-- | src/wast-parser.cc | 50 | ||||
-rw-r--r-- | test/parse/expr/bad-const-v128-i16x8-overflow.txt | 4 | ||||
-rw-r--r-- | test/parse/expr/bad-const-v128-i8x16-overflow.txt | 4 | ||||
-rw-r--r-- | test/parse/expr/bad-const-v128-type-i32-expected.txt | 1 |
7 files changed, 194 insertions, 46 deletions
diff --git a/src/literal.cc b/src/literal.cc index 3560a451..4c367ed9 100644 --- a/src/literal.cc +++ b/src/literal.cc @@ -21,6 +21,8 @@ #include <cmath> #include <cstdlib> #include <cstring> +#include <limits> +#include <type_traits> namespace wabt { @@ -620,10 +622,12 @@ Result ParseInt64(const char* s, return result; } -Result ParseInt32(const char* s, - const char* end, - uint32_t* out, - ParseIntType parse_type) { +template <typename U> +Result ParseInt(const char* s, + const char* end, + U* out, + ParseIntType parse_type) { + typedef typename std::make_signed<U>::type S; uint64_t value; bool has_sign = false; if (*s == '-' || *s == '+') { @@ -638,20 +642,41 @@ Result ParseInt32(const char* s, CHECK_RESULT(ParseUint64(s, end, &value)); if (has_sign) { - // abs(INT32_MIN) == INT32_MAX + 1. - if (value > static_cast<uint64_t>(INT32_MAX) + 1) { + // abs(INTN_MIN) == INTN_MAX + 1. + if (value > static_cast<uint64_t>(std::numeric_limits<S>::max()) + 1) { return Result::Error; } - value = UINT32_MAX - value + 1; + value = std::numeric_limits<U>::max() - value + 1; } else { - if (value > static_cast<uint64_t>(UINT32_MAX)) { + if (value > static_cast<uint64_t>(std::numeric_limits<U>::max())) { return Result::Error; } } - *out = static_cast<uint32_t>(value); + *out = static_cast<U>(value); return Result::Ok; } +Result ParseInt8(const char* s, + const char* end, + uint8_t* out, + ParseIntType parse_type) { + return ParseInt(s, end, out, parse_type); +} + +Result ParseInt16(const char* s, + const char* end, + uint16_t* out, + ParseIntType parse_type) { + return ParseInt(s, end, out, parse_type); +} + +Result ParseInt32(const char* s, + const char* end, + uint32_t* out, + ParseIntType parse_type) { + return ParseInt(s, end, out, parse_type); +} + Result ParseFloat(LiteralType literal_type, const char* s, const char* end, diff --git a/src/literal.h b/src/literal.h index 40b395ed..b1315a1a 100644 --- a/src/literal.h +++ b/src/literal.h @@ -49,6 +49,14 @@ enum class ParseIntType { #define WABT_MAX_DOUBLE_HEX 40 Result ParseHexdigit(char c, uint32_t* out); +Result ParseInt8(const char* s, + const char* end, + uint8_t* out, + ParseIntType parse_type); +Result ParseInt16(const char* s, + const char* end, + uint16_t* out, + ParseIntType parse_type); Result ParseInt32(const char* s, const char* end, uint32_t* out, diff --git a/src/test-literal.cc b/src/test-literal.cc index 269470bd..e094d879 100644 --- a/src/test-literal.cc +++ b/src/test-literal.cc @@ -62,6 +62,18 @@ void AssertIntEquals(T expected, } } +void AssertInt8Equals(uint8_t expected, + const char* s, + ParseIntTypeCombo parse_type = Both) { + AssertIntEquals(expected, s, ParseInt8, parse_type); +} + +void AssertInt16Equals(uint16_t expected, + const char* s, + ParseIntTypeCombo parse_type = Both) { + AssertIntEquals(expected, s, ParseInt16, parse_type); +} + void AssertInt32Equals(uint32_t expected, const char* s, ParseIntTypeCombo parse_type = Both) { @@ -74,6 +86,28 @@ void AssertInt64Equals(uint64_t expected, AssertIntEquals(expected, s, ParseInt64, parse_type); } +void AssertInt8Fails(const char* s) { + const char* const end = s + strlen(s); + uint8_t actual; + ASSERT_EQ(Result::Error, + ParseInt8(s, end, &actual, ParseIntType::SignedAndUnsigned)) + << s; + ASSERT_EQ(Result::Error, + ParseInt8(s, end, &actual, ParseIntType::UnsignedOnly)) + << s; +} + +void AssertInt16Fails(const char* s) { + const char* const end = s + strlen(s); + uint16_t actual; + ASSERT_EQ(Result::Error, + ParseInt16(s, end, &actual, ParseIntType::SignedAndUnsigned)) + << s; + ASSERT_EQ(Result::Error, + ParseInt16(s, end, &actual, ParseIntType::UnsignedOnly)) + << s; +} + void AssertInt32Fails(const char* s) { const char* const end = s + strlen(s); uint32_t actual; @@ -139,6 +173,102 @@ void AssertHexDoubleFails(const char* s) { } // end anonymous namespace +TEST(ParseInt8, Both) { + AssertInt8Equals(0, "0"); + AssertInt8Equals(100, "100"); + AssertInt8Equals(123, "123"); + AssertInt8Equals(127, "127"); + AssertInt8Equals(255, "255"); + AssertInt8Equals(0xca, "0xca"); + AssertInt8Equals(0x7f, "0x7f"); + AssertInt8Equals(0x80, "0x80"); + AssertInt8Equals(0xff, "0xff"); +} + +TEST(ParseInt8, SignedAndUnsigned) { + AssertInt8Equals(128, "-128", SignedAndUnsigned); + AssertInt8Equals(-0x80, "-0x80", SignedAndUnsigned); + AssertInt8Equals(255, "-1", SignedAndUnsigned); + AssertInt8Equals(-1, "-0x1", SignedAndUnsigned); + AssertInt8Equals(1, "+1", SignedAndUnsigned); + AssertInt8Equals(-0x7b, "-0x7B", SignedAndUnsigned); + AssertInt8Equals(0xab, "+0xab", SignedAndUnsigned); +} + +TEST(ParseInt8, Invalid) { + AssertInt8Fails(""); + AssertInt8Fails("-100hello"); + AssertInt8Fails("0XAB"); + AssertInt8Fails("0xga"); + AssertInt8Fails("two"); +} + +TEST(ParseInt8, Underscores) { + AssertInt8Equals(123, "12_3", Both); + AssertInt8Equals(123, "+12_3", SignedAndUnsigned); + AssertInt8Equals(-123, "-1_23", SignedAndUnsigned); + AssertInt8Equals(19, "1______9", Both); + AssertInt8Equals(0xab, "0xa_b", Both); + AssertInt8Equals(0xab, "+0xa_b", SignedAndUnsigned); + AssertInt8Equals(-0x7b, "-0x7_b", SignedAndUnsigned); +} + +TEST(ParseInt8, Overflow) { + AssertInt8Fails("256"); + AssertInt8Fails("-129"); + AssertInt8Fails("0x100"); + AssertInt8Fails("-0x81"); + AssertInt8Fails("1231231231231231231231"); +} + +TEST(ParseInt16, Both) { + AssertInt16Equals(0, "0"); + AssertInt16Equals(1000, "1000"); + AssertInt16Equals(12345, "12345"); + AssertInt16Equals(32767, "32767"); + AssertInt16Equals(65535, "65535"); + AssertInt16Equals(0xcafe, "0xcafe"); + AssertInt16Equals(0x7fff, "0x7fff"); + AssertInt16Equals(0x8000, "0x8000"); + AssertInt16Equals(0xffff, "0xffff"); +} + +TEST(ParseInt16, SignedAndUnsigned) { + AssertInt16Equals(32768, "-32768", SignedAndUnsigned); + AssertInt16Equals(-0x8000, "-0x8000", SignedAndUnsigned); + AssertInt16Equals(65535, "-1", SignedAndUnsigned); + AssertInt16Equals(-1, "-0x1", SignedAndUnsigned); + AssertInt16Equals(1, "+1", SignedAndUnsigned); + AssertInt16Equals(-0x7bcd, "-0x7BCD", SignedAndUnsigned); + AssertInt16Equals(0xabcd, "+0xabcd", SignedAndUnsigned); +} + +TEST(ParseInt16, Invalid) { + AssertInt16Fails(""); + AssertInt16Fails("-100hello"); + AssertInt16Fails("0XABCD"); + AssertInt16Fails("0xgabb"); + AssertInt16Fails("two"); +} + +TEST(ParseInt16, Underscores) { + AssertInt16Equals(12345, "12_345", Both); + AssertInt16Equals(12345, "+12_345", SignedAndUnsigned); + AssertInt16Equals(-12345, "-123_45", SignedAndUnsigned); + AssertInt16Equals(19, "1______9", Both); + AssertInt16Equals(0xabcd, "0xa_b_c_d", Both); + AssertInt16Equals(0xabcd, "+0xa_b_c_d", SignedAndUnsigned); + AssertInt16Equals(-0x7bcd, "-0x7_b_c_d", SignedAndUnsigned); +} + +TEST(ParseInt16, Overflow) { + AssertInt16Fails("65536"); + AssertInt16Fails("-32769"); + AssertInt16Fails("0x10000"); + AssertInt16Fails("-0x8001"); + AssertInt16Fails("1231231231231231231231"); +} + TEST(ParseInt32, Both) { AssertInt32Equals(0, "0"); AssertInt32Equals(1000, "1000"); diff --git a/src/wast-parser.cc b/src/wast-parser.cc index 92beffcc..98c22967 100644 --- a/src/wast-parser.cc +++ b/src/wast-parser.cc @@ -1791,47 +1791,33 @@ Result WastParser::ParseSimdV128Const(Const* const_, TokenType token_type) { // the array of bytes: if (integer) { switch(lane_count) { - case 16: { - int32_t value; - result = ParseInt32(s, end, Bitcast<uint32_t*>(&value), ParseIntType::SignedAndUnsigned); - if (value < static_cast<int32_t>(std::numeric_limits<int8_t>::min()) || - value > static_cast<int32_t>(std::numeric_limits<uint8_t>::max())) { - Error(loc, "literal \"%s\" out-of-bounds of i8", literal.text.c_str()); - return Result::Error; - } - *lane_ptr = static_cast<uint8_t>(value); + case 16: + result = ParseInt8(s, end, reinterpret_cast<uint8_t*>(lane_ptr), + ParseIntType::SignedAndUnsigned); break; - } - case 8: { - int32_t value; - result = ParseInt32(s, end, Bitcast<uint32_t*>(&value), ParseIntType::SignedAndUnsigned); - if (value < static_cast<int32_t>(std::numeric_limits<int16_t>::min()) || - value > static_cast<int32_t>(std::numeric_limits<uint16_t>::max())) { - Error(loc, "literal \"%s\" out-of-bounds of i16", literal.text.c_str()); - return Result::Error; - } - *lane_ptr = static_cast<uint16_t>(value); + case 8: + result = ParseInt16(s, end, reinterpret_cast<uint16_t*>(lane_ptr), + ParseIntType::SignedAndUnsigned); break; - } - case 4: { - result = ParseInt32(s, end, Bitcast<uint32_t*>(lane_ptr), ParseIntType::SignedAndUnsigned); + case 4: + result = ParseInt32(s, end, reinterpret_cast<uint32_t*>(lane_ptr), + ParseIntType::SignedAndUnsigned); break; - } - case 2: { - result = ParseInt64(s, end, Bitcast<uint64_t*>(lane_ptr), ParseIntType::SignedAndUnsigned); + case 2: + result = ParseInt64(s, end, reinterpret_cast<uint64_t*>(lane_ptr), + ParseIntType::SignedAndUnsigned); break; - } } } else { switch(lane_count) { - case 4: { - result = ParseFloat(literal.type, s, end, Bitcast<uint32_t*>(lane_ptr)); + case 4: + result = ParseFloat(literal.type, s, end, + reinterpret_cast<uint32_t*>(lane_ptr)); break; - } - case 2: { - result = ParseDouble(literal.type,s, end, Bitcast<uint64_t*>(lane_ptr)); + case 2: + result = ParseDouble(literal.type, s, end, + reinterpret_cast<uint64_t*>(lane_ptr)); break; - } } } diff --git a/test/parse/expr/bad-const-v128-i16x8-overflow.txt b/test/parse/expr/bad-const-v128-i16x8-overflow.txt index 0db692ff..e0ba0dbc 100644 --- a/test/parse/expr/bad-const-v128-i16x8-overflow.txt +++ b/test/parse/expr/bad-const-v128-i16x8-overflow.txt @@ -6,10 +6,10 @@ (func v128.const i16x8 -32768 -32769 32767 65535 65536 0 0 0) ) (;; STDERR ;;; -out/test/parse/expr/bad-const-v128-i16x8-overflow.txt:5:52: error: literal "65536" out-of-bounds of i16 +out/test/parse/expr/bad-const-v128-i16x8-overflow.txt:5:52: error: invalid literal "65536" (func v128.const i16x8 -32768 -32767 32767 65535 65536 0 0 0) ^^^^^ -out/test/parse/expr/bad-const-v128-i16x8-overflow.txt:6:33: error: literal "-32769" out-of-bounds of i16 +out/test/parse/expr/bad-const-v128-i16x8-overflow.txt:6:33: error: invalid literal "-32769" (func v128.const i16x8 -32768 -32769 32767 65535 65536 0 0 0) ^^^^^^ ;;; STDERR ;;) diff --git a/test/parse/expr/bad-const-v128-i8x16-overflow.txt b/test/parse/expr/bad-const-v128-i8x16-overflow.txt index 4edbd826..d50ebd78 100644 --- a/test/parse/expr/bad-const-v128-i8x16-overflow.txt +++ b/test/parse/expr/bad-const-v128-i8x16-overflow.txt @@ -6,10 +6,10 @@ (func v128.const i8x16 -127 -129 128 255 256 0 0 0 0 0 0 0 0 0 0 0) ) (;; STDERR ;;; -out/test/parse/expr/bad-const-v128-i8x16-overflow.txt:5:44: error: literal "256" out-of-bounds of i8 +out/test/parse/expr/bad-const-v128-i8x16-overflow.txt:5:44: error: invalid literal "256" (func v128.const i8x16 -127 -128 128 255 256 0 0 0 0 0 0 0 0 0 0 0) ^^^ -out/test/parse/expr/bad-const-v128-i8x16-overflow.txt:6:31: error: literal "-129" out-of-bounds of i8 +out/test/parse/expr/bad-const-v128-i8x16-overflow.txt:6:31: error: invalid literal "-129" (func v128.const i8x16 -127 -129 128 255 256 0 0 0 0 0 0 0 0 0 0 0) ^^^^ ;;; STDERR ;;) diff --git a/test/parse/expr/bad-const-v128-type-i32-expected.txt b/test/parse/expr/bad-const-v128-type-i32-expected.txt index 14e13bcb..e990f820 100644 --- a/test/parse/expr/bad-const-v128-type-i32-expected.txt +++ b/test/parse/expr/bad-const-v128-type-i32-expected.txt @@ -5,7 +5,6 @@ (module (func v128.const i64 0x12345678 0x00000000 0x00000000 0xabcd3478)) (;; STDERR ;;; out/test/parse/expr/bad-const-v128-type-i32-expected.txt:4:26: error: Unexpected type at start of simd constant. Expected one of: i8x16, i16x8, i32x4, i64x2, f32x4, f64x2. Found "NAT". - (module (func v128.const 0x12345678 0x00000000 0x00000000 0xabcd3478)) ^^^^^^^^^^ ;;; STDERR ;;) |