// Copyright 2014 The Crashpad Authors. All rights reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "util/stdlib/string_number_conversion.h" #include #include #include #include #include "base/cxx17_backports.h" #include "gtest/gtest.h" #define STRINGIFY(a) STR(a) #define STR(a) #a template struct TestSpecification { const char* string; bool valid; TValueType value; }; // Signed 32-bit test data template ::value && std::is_signed::value && (sizeof(TIntType) == 4)>::type* = nullptr> static constexpr std::array, 61> kTestDataFunc() { return {{ {"", false, 0}, {"0", true, 0}, {"1", true, 1}, {"2147483647", true, std::numeric_limits::max()}, {"2147483648", false, 0}, {"4294967295", false, 0}, {"4294967296", false, 0}, {"-1", true, -1}, {"-2147483648", true, std::numeric_limits::min()}, {"-2147483649", false, 0}, {"00", true, 0}, {"01", true, 1}, {"-01", true, -1}, {"+2", true, 2}, {"0x10", true, 16}, {"-0x10", true, -16}, {"+0x20", true, 32}, {"0xf", true, 15}, {"0xg", false, 0}, {"0x7fffffff", true, std::numeric_limits::max()}, {"0x7FfFfFfF", true, std::numeric_limits::max()}, {"0x80000000", false, 0}, {"0xFFFFFFFF", false, 0}, {"-0x7fffffff", true, -2147483647}, {"-0x80000000", true, std::numeric_limits::min()}, {"-0x80000001", false, 0}, {"-0xffffffff", false, 0}, {"0x100000000", false, 0}, {"0xabcdef", true, 11259375}, {"010", true, 8}, {"-010", true, -8}, {"+020", true, 16}, {"07", true, 7}, {"08", false, 0}, {" 0", false, 0}, {"0 ", false, 0}, {" 0 ", false, 0}, {" 1", false, 0}, {"1 ", false, 0}, {" 1 ", false, 0}, {"a2", false, 0}, {"2a", false, 0}, {"2a2", false, 0}, {".0", false, 0}, {".1", false, 0}, {"-.2", false, 0}, {"+.3", false, 0}, {"1.23", false, 0}, {"-273.15", false, 0}, {"+98.6", false, 0}, {"1e1", false, 0}, {"1E1", false, 0}, {"0x123p4", false, 0}, {"infinity", false, 0}, {"NaN", false, 0}, {"-9223372036854775810", false, 0}, {"-9223372036854775809", false, 0}, {"9223372036854775808", false, 0}, {"9223372036854775809", false, 0}, {"18446744073709551615", false, 0}, {"18446744073709551616", false, 0}, }}; } // Unsigned 32-bit test data template ::value && !std::is_signed::value && (sizeof(TIntType) == 4)>::type* = nullptr> static constexpr std::array, 61> kTestDataFunc() { return {{ {"", false, 0}, {"0", true, 0}, {"1", true, 1}, {"2147483647", true, 2147483647}, {"2147483648", true, 2147483648}, {"4294967295", true, std::numeric_limits::max()}, {"4294967296", false, 0}, {"-1", false, 0}, {"-2147483648", false, 0}, {"-2147483649", false, 0}, {"00", true, 0}, {"01", true, 1}, {"-01", false, 0}, {"+2", true, 2}, {"0x10", true, 16}, {"-0x10", false, 0}, {"+0x20", true, 32}, {"0xf", true, 15}, {"0xg", false, 0}, {"0x7fffffff", true, 0x7fffffff}, {"0x7FfFfFfF", true, 0x7fffffff}, {"0x80000000", true, 0x80000000}, {"0xFFFFFFFF", true, 0xffffffff}, {"-0x7fffffff", false, 0}, {"-0x80000000", false, 0}, {"-0x80000001", false, 0}, {"-0xffffffff", false, 0}, {"0x100000000", false, 0}, {"0xabcdef", true, 11259375}, {"010", true, 8}, {"-010", false, 0}, {"+020", true, 16}, {"07", true, 7}, {"08", false, 0}, {" 0", false, 0}, {"0 ", false, 0}, {" 0 ", false, 0}, {" 1", false, 0}, {"1 ", false, 0}, {" 1 ", false, 0}, {"a2", false, 0}, {"2a", false, 0}, {"2a2", false, 0}, {".0", false, 0}, {".1", false, 0}, {"-.2", false, 0}, {"+.3", false, 0}, {"1.23", false, 0}, {"-273.15", false, 0}, {"+98.6", false, 0}, {"1e1", false, 0}, {"1E1", false, 0}, {"0x123p4", false, 0}, {"infinity", false, 0}, {"NaN", false, 0}, {"-9223372036854775810", false, 0}, {"-9223372036854775809", false, 0}, {"9223372036854775808", false, 0}, {"9223372036854775809", false, 0}, {"18446744073709551615", false, 0}, {"18446744073709551616", false, 0}, }}; } // Signed 64-bit test data template ::value && std::is_signed::value && (sizeof(TIntType) == 8)>::type* = nullptr> static constexpr std::array, 24> kTestDataFunc() { return {{ {"", false, 0}, {"0", true, 0}, {"1", true, 1}, {"2147483647", true, 2147483647}, {"2147483648", true, 2147483648}, {"4294967295", true, 4294967295}, {"4294967296", true, 4294967296}, {"9223372036854775807", true, std::numeric_limits::max()}, {"9223372036854775808", false, 0}, {"18446744073709551615", false, 0}, {"18446744073709551616", false, 0}, {"-1", true, -1}, {"-2147483648", true, INT64_C(-2147483648)}, {"-2147483649", true, INT64_C(-2147483649)}, {"-9223372036854775808", true, std::numeric_limits::min()}, {"-9223372036854775809", false, 0}, {"0x7fffffffffffffff", true, std::numeric_limits::max()}, {"0x8000000000000000", false, 0}, {"0xffffffffffffffff", false, 0}, {"0x10000000000000000", false, 0}, {"-0x7fffffffffffffff", true, -9223372036854775807}, {"-0x8000000000000000", true, std::numeric_limits::min()}, {"-0x8000000000000001", false, 0}, {"0x7Fffffffffffffff", true, std::numeric_limits::max()}, }}; } // Unsigned 64-bit test data template ::value && !std::is_signed::value && (sizeof(TIntType) == 8)>::type* = nullptr> static constexpr std::array, 25> kTestDataFunc() { return {{ {"", false, 0}, {"0", true, 0}, {"1", true, 1}, {"2147483647", true, 2147483647}, {"2147483648", true, 2147483648}, {"4294967295", true, 4294967295}, {"4294967296", true, 4294967296}, {"9223372036854775807", true, 9223372036854775807}, {"9223372036854775808", true, 9223372036854775808u}, {"18446744073709551615", true, std::numeric_limits::max()}, {"18446744073709551616", false, 0}, {"-1", false, 0}, {"-2147483648", false, 0}, {"-2147483649", false, 0}, {"-2147483648", false, 0}, {"-9223372036854775808", false, 0}, {"-9223372036854775809", false, 0}, {"0x7fffffffffffffff", true, 9223372036854775807}, {"0x8000000000000000", true, 9223372036854775808u}, {"0xffffffffffffffff", true, std::numeric_limits::max()}, {"0x10000000000000000", false, 0}, {"-0x7fffffffffffffff", false, 0}, {"-0x8000000000000000", false, 0}, {"-0x8000000000000001", false, 0}, {"0xFfffffffffffffff", true, std::numeric_limits::max()}, }}; } // This string is split to avoid MSVC warning: // "decimal digit terminates octal escape sequence". static constexpr char kEmbeddedNullInputRaw[] = "6\000" "6"; namespace crashpad { namespace test { namespace { TEST(StringNumberConversion, StringToInt) { static_assert(sizeof(int) == 4, "Test only configured for 32-bit int."); static constexpr auto kTestData = kTestDataFunc(); for (size_t index = 0; index < kTestData.size(); ++index) { int value; bool valid = StringToNumber(kTestData[index].string, &value); if (kTestData[index].valid) { EXPECT_TRUE(valid) << "index " << index << ", string " << kTestData[index].string; if (valid) { EXPECT_EQ(value, kTestData[index].value) << "index " << index << ", string " << kTestData[index].string; } } else { EXPECT_FALSE(valid) << "index " << index << ", string " << kTestData[index].string << ", value " << value; } } // Ensure that embedded NUL characters are treated as bad input. The string // is split to avoid MSVC warning: // "decimal digit terminates octal escape sequence". int output; std::string kEmbeddedNullInput(kEmbeddedNullInputRaw, base::size(kEmbeddedNullInputRaw) - 1); EXPECT_FALSE(StringToNumber(kEmbeddedNullInput, &output)); } TEST(StringNumberConversion, StringToUnsignedInt) { static_assert(sizeof(unsigned int) == 4, "Test only configured for 32-bit unsigned int."); static constexpr auto kTestData = kTestDataFunc(); for (size_t index = 0; index < kTestData.size(); ++index) { unsigned int value; bool valid = StringToNumber(kTestData[index].string, &value); if (kTestData[index].valid) { EXPECT_TRUE(valid) << "index " << index << ", string " << kTestData[index].string; if (valid) { EXPECT_EQ(value, kTestData[index].value) << "index " << index << ", string " << kTestData[index].string; } } else { EXPECT_FALSE(valid) << "index " << index << ", string " << kTestData[index].string << ", value " << value; } } // Ensure that embedded NUL characters are treated as bad input. The string // is split to avoid MSVC warning: // "decimal digit terminates octal escape sequence". unsigned int output; std::string kEmbeddedNullInput(kEmbeddedNullInputRaw, base::size(kEmbeddedNullInputRaw) - 1); EXPECT_FALSE(StringToNumber(kEmbeddedNullInput, &output)); } TEST(StringNumberConversion, StringToLong) { static_assert( sizeof(long) == 4 || sizeof(long) == 8, "Test not configured for " STRINGIFY(__SIZEOF_LONG__) "-byte long"); static constexpr auto kTestData = kTestDataFunc(); for (size_t index = 0; index < kTestData.size(); ++index) { long value; bool valid = StringToNumber(kTestData[index].string, &value); if (kTestData[index].valid) { EXPECT_TRUE(valid) << "index " << index << ", string " << kTestData[index].string; if (valid) { EXPECT_EQ(value, kTestData[index].value) << "index " << index << ", string " << kTestData[index].string; } } else { EXPECT_FALSE(valid) << "index " << index << ", string " << kTestData[index].string << ", value " << value; } } } TEST(StringNumberConversion, StringToUnsignedLong) { static_assert( sizeof(long) == 4 || sizeof(long) == 8, "Test not configured for " STRINGIFY(__SIZEOF_LONG__) "-byte long"); static constexpr auto kTestData = kTestDataFunc(); for (size_t index = 0; index < kTestData.size(); ++index) { unsigned long value; bool valid = StringToNumber(kTestData[index].string, &value); if (kTestData[index].valid) { EXPECT_TRUE(valid) << "index " << index << ", string " << kTestData[index].string; if (valid) { EXPECT_EQ(value, kTestData[index].value) << "index " << index << ", string " << kTestData[index].string; } } else { EXPECT_FALSE(valid) << "index " << index << ", string " << kTestData[index].string << ", value " << value; } } } TEST(StringNumberConversion, StringToLongLong) { static_assert(sizeof(long long) == 8, "Test only configured for 64-bit long long."); static constexpr auto kTestData = kTestDataFunc(); for (size_t index = 0; index < kTestData.size(); ++index) { long long value; bool valid = StringToNumber(kTestData[index].string, &value); if (kTestData[index].valid) { EXPECT_TRUE(valid) << "index " << index << ", string " << kTestData[index].string; if (valid) { EXPECT_EQ(value, kTestData[index].value) << "index " << index << ", string " << kTestData[index].string; } } else { EXPECT_FALSE(valid) << "index " << index << ", string " << kTestData[index].string << ", value " << value; } } } TEST(StringNumberConversion, StringToUnsignedLongLong) { static_assert(sizeof(unsigned long long) == 8, "Test only configured for 64-bit unsigned long long."); static constexpr auto kTestData = kTestDataFunc(); for (size_t index = 0; index < kTestData.size(); ++index) { unsigned long long value; bool valid = StringToNumber(kTestData[index].string, &value); if (kTestData[index].valid) { EXPECT_TRUE(valid) << "index " << index << ", string " << kTestData[index].string; if (valid) { EXPECT_EQ(value, kTestData[index].value) << "index " << index << ", string " << kTestData[index].string; } } else { EXPECT_FALSE(valid) << "index " << index << ", string " << kTestData[index].string << ", value " << value; } } } } // namespace } // namespace test } // namespace crashpad