From 483eb2f56657e8e7f419ab1a4fab8dce9ade8609 Mon Sep 17 00:00:00 2001 From: Daniel Baumann Date: Sat, 27 Apr 2024 20:24:20 +0200 Subject: Adding upstream version 14.2.21. Signed-off-by: Daniel Baumann --- src/boost/libs/convert/test/callable.cpp | 114 ++++++ src/boost/libs/convert/test/encryption.cpp | 44 +++ src/boost/libs/convert/test/fallbacks.cpp | 81 ++++ src/boost/libs/convert/test/has_member.cpp | 104 ++++++ src/boost/libs/convert/test/is_converter.cpp | 86 +++++ src/boost/libs/convert/test/jamfile.v2 | 41 +++ src/boost/libs/convert/test/lcast_converter.cpp | 77 ++++ src/boost/libs/convert/test/performance.cpp | 317 ++++++++++++++++ src/boost/libs/convert/test/performance_spirit.cpp | 112 ++++++ src/boost/libs/convert/test/prepare.hpp | 59 +++ src/boost/libs/convert/test/printf_converter.cpp | 47 +++ src/boost/libs/convert/test/sfinae.cpp | 159 ++++++++ src/boost/libs/convert/test/spirit_converter.cpp | 87 +++++ src/boost/libs/convert/test/str_to_int.cpp | 156 ++++++++ src/boost/libs/convert/test/stream_converter.cpp | 381 +++++++++++++++++++ src/boost/libs/convert/test/strtol_converter.cpp | 407 +++++++++++++++++++++ src/boost/libs/convert/test/test.hpp | 187 ++++++++++ src/boost/libs/convert/test/user_type.cpp | 101 +++++ 18 files changed, 2560 insertions(+) create mode 100644 src/boost/libs/convert/test/callable.cpp create mode 100644 src/boost/libs/convert/test/encryption.cpp create mode 100644 src/boost/libs/convert/test/fallbacks.cpp create mode 100644 src/boost/libs/convert/test/has_member.cpp create mode 100644 src/boost/libs/convert/test/is_converter.cpp create mode 100644 src/boost/libs/convert/test/jamfile.v2 create mode 100644 src/boost/libs/convert/test/lcast_converter.cpp create mode 100644 src/boost/libs/convert/test/performance.cpp create mode 100644 src/boost/libs/convert/test/performance_spirit.cpp create mode 100644 src/boost/libs/convert/test/prepare.hpp create mode 100644 src/boost/libs/convert/test/printf_converter.cpp create mode 100644 src/boost/libs/convert/test/sfinae.cpp create mode 100644 src/boost/libs/convert/test/spirit_converter.cpp create mode 100644 src/boost/libs/convert/test/str_to_int.cpp create mode 100644 src/boost/libs/convert/test/stream_converter.cpp create mode 100644 src/boost/libs/convert/test/strtol_converter.cpp create mode 100644 src/boost/libs/convert/test/test.hpp create mode 100644 src/boost/libs/convert/test/user_type.cpp (limited to 'src/boost/libs/convert/test') diff --git a/src/boost/libs/convert/test/callable.cpp b/src/boost/libs/convert/test/callable.cpp new file mode 100644 index 00000000..b8f36146 --- /dev/null +++ b/src/boost/libs/convert/test/callable.cpp @@ -0,0 +1,114 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include +#include +#include + +using std::string; +using boost::convert; +using boost::lexical_cast; + +//[callable_example1 +void plain_old_func(string const& value_in, boost::optional& value_out) +//] +{ + try + { + value_out = lexical_cast(value_in); + } + catch (...) + { + } +} + +template +void +convert_all(TypeIn const&, boost::optional&) +{ +} + +template +void +assign(boost::optional& value_out, Type const& value_in) +{ + value_out = value_in; +} + +struct converter1 +{ + template + void + operator()(TypeIn const&, boost::optional&) const + { + } +}; + +//[callable_example4 +struct take_double { void operator()(double, boost::optional&) const {}}; +struct take_int { void operator()(int, boost::optional&) const {}}; +//] + +//[callable_example6 +struct double_only +{ + // Declared for all types. + template void operator()(TypeIn, boost::optional&) const; +}; + +// Defined only for certain types. +template<> void double_only::operator()(double, boost::optional&) const {} +//] + +int +main(int, char const* []) +{ + typedef boost::function&)> boost_func; + + char const* const str = "-12"; + + // Testing old-function-based converter. + //[callable_example2 + int v01 = convert(str, plain_old_func).value_or(-1); + //] + // Testing boost::function-based converter. + int v02 = convert(str, boost_func(plain_old_func)).value_or(-1); + // Testing crazy boost::bind-based converter. + //[callable_example3 + int v03 = convert(str, + boost::bind(assign, _2, + boost::bind(lexical_cast, _1))).value_or(-1); + //] + BOOST_TEST(v01 == -12); + BOOST_TEST(v02 == -12); + BOOST_TEST(v03 == -12); + + convert(str, convert_all); + convert(11, convert_all); + convert< int>(str, converter1()); + convert(11, converter1()); + convert(11.23, take_double()); + convert(11, take_int()); + //[callable_example5 + convert(11, take_double()); // Compiler applies int-to-double promotion to call the converter. + convert(11.23, take_int()); // Compiler applies double-to-int implicit truncation. + //] + //[callable_example7 + convert(11.23, double_only()); // Fine. +// convert(11, double_only()); // Fails: undefined reference to double_only::operator() + //] + + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/encryption.cpp b/src/boost/libs/convert/test/encryption.cpp new file mode 100644 index 00000000..684c6d50 --- /dev/null +++ b/src/boost/libs/convert/test/encryption.cpp @@ -0,0 +1,44 @@ +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include + +static +bool +my_cypher(std::string const& value_in, boost::optional& value_out) +{ + size_t const cypher = 'A' - '1'; + + value_out = value_in; + + for (std::string::iterator it = value_out->begin(); it != value_out->end(); ++it) + (*it < 'A') ? (*it += cypher) : (*it -= cypher); + + return true; +} + +int +main(int, char const* []) +{ + //////////////////////////////////////////////////////////////////////////// + // Testing custom converter. + //////////////////////////////////////////////////////////////////////////// + + std::string encrypted = boost::convert("ABC", my_cypher).value(); + std::string decrypted = boost::convert(encrypted, my_cypher).value(); + + BOOST_TEST(encrypted == "123"); + BOOST_TEST(decrypted == "ABC"); + + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/fallbacks.cpp b/src/boost/libs/convert/test/fallbacks.cpp new file mode 100644 index 00000000..b0192e9b --- /dev/null +++ b/src/boost/libs/convert/test/fallbacks.cpp @@ -0,0 +1,81 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include +#include +#include + +namespace { namespace local +{ + bool called_functor_int; + bool called_functor_double; + bool called_functor_foo; + bool called_function_int; + bool called_function_long; +}} + +struct functor_int { int operator() () const { local:: called_functor_int = true; return INT_MAX; }}; +struct functor_double { double operator() () const { local::called_functor_double = true; return INT_MAX; }}; +struct functor_foo { int func (int) const { local:: called_functor_foo = true; return INT_MAX; }}; + +int function_int () { local:: called_function_int = true; return INT_MAX; } +long function_long () { local::called_function_long = true; return INT_MAX; } + +int +main(int, char const* []) +{ + boost::cnv::cstream cnv; + functor_foo foo; + + int i01 = boost::convert("uhm", cnv).value_or_eval(functor_int()); + int i02 = boost::convert("uhm", cnv).value_or_eval(functor_double()); + int i03 = boost::convert("uhm", cnv).value_or_eval(boost::bind(&functor_foo::func, foo, 0)); + int i04 = boost::convert("uhm", cnv).value_or_eval(function_int); + int i05 = boost::convert("uhm", cnv).value_or_eval(function_long); + + BOOST_TEST(local:: called_functor_int && i01 == INT_MAX); + BOOST_TEST(local::called_functor_double && i02 == INT_MAX); + BOOST_TEST(local:: called_functor_foo && i03 == INT_MAX); + BOOST_TEST(local:: called_function_int && i04 == INT_MAX); + BOOST_TEST(local:: called_function_long && i05 == INT_MAX); + + local:: called_functor_int = false; + local::called_functor_double = false; + local:: called_functor_foo = false; + local:: called_function_int = false; + local:: called_function_long = false; + + boost::convert("uhm", cnv, functor_int()); + boost::convert("uhm", cnv, functor_double()); + boost::convert("uhm", cnv, boost::bind(&functor_foo::func, foo, 0)); + boost::convert("uhm", cnv, function_int); + boost::convert("uhm", cnv, function_long); + + BOOST_TEST(local:: called_functor_int && i01 == INT_MAX); + BOOST_TEST(local::called_functor_double && i02 == INT_MAX); + BOOST_TEST(local:: called_functor_foo && i03 == INT_MAX); + BOOST_TEST(local:: called_function_int && i04 == INT_MAX); + BOOST_TEST(local:: called_function_long && i05 == INT_MAX); + + try + { + boost::convert("uhm", cnv, boost::throw_on_failure); + BOOST_TEST(0); + } + catch (boost::bad_optional_access const&) + { + } + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/has_member.cpp b/src/boost/libs/convert/test/has_member.cpp new file mode 100644 index 00000000..088d047a --- /dev/null +++ b/src/boost/libs/convert/test/has_member.cpp @@ -0,0 +1,104 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include +#include + +//[has_member_declaration +namespace { namespace local +{ + BOOST_DECLARE_HAS_MEMBER(has_begin, begin); + BOOST_DECLARE_HAS_MEMBER(has_funop, operator()); +}} +//] +//[has_member_classes_tested +namespace { namespace local +{ + struct test01 { int begin; }; + struct test02 { char* begin() { return 0; } }; + struct test22 { void operator()() {} }; +}} +//] + +namespace { namespace local +{ + struct test03 + { + void begin(int) {} + int begin(int, int) { return 0; } + }; + struct test04 + { + virtual ~test04() {} + + private: virtual char* begin() { return 0; } + }; + struct test05 + { + virtual char* begin() { return 0; } + virtual ~test05() {} + }; + struct test06 : public test04 {}; + struct test07 : private test04 {}; + struct test08 : public test05 {}; + struct test09 : private test05 {}; + + struct test11 { int no_begin; }; + struct test12 { char* no_begin() { return 0; } }; +}} + +namespace { namespace local +{ + struct test21 { void zoo () {} }; + struct test23 { void operator() () const {} }; + struct test24 { int operator() (int) { return 0; } }; + struct test25 { int operator() (int) const { return 0; } }; + struct test26 { int operator() (int) const { return 0; } void operator() () const {} }; +}} + +int +main(int, char const* []) +{ + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_begin >::value == true); + + BOOST_TEST(local::has_begin::value == false); + BOOST_TEST(local::has_begin::value == false); + BOOST_TEST(local::has_begin::value == false); + + //[has_member_usage + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_begin::value == true); + BOOST_TEST(local::has_funop::value == true); + //] + + BOOST_TEST(local::has_funop::value == false); + BOOST_TEST(local::has_funop::value == true); + BOOST_TEST(local::has_funop::value == true); + BOOST_TEST(local::has_funop::value == true); + BOOST_TEST(local::has_funop::value == true); + BOOST_TEST(local::has_funop::value == true); + + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/is_converter.cpp b/src/boost/libs/convert/test/is_converter.cpp new file mode 100644 index 00000000..f528a33c --- /dev/null +++ b/src/boost/libs/convert/test/is_converter.cpp @@ -0,0 +1,86 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include + +using std::string; +using std::wstring; + +namespace { namespace local +{ + typedef void not_converter01 (string const&); + typedef void not_converter02 (int const&, string const&); + typedef void not_converter03 (); + typedef void converter01 (int const&, boost::optional&); + typedef void converter02 (double const&, boost::optional&); + + struct converter11 + { + void operator()(int const&, boost::optional&) {} + void operator()(double const&, boost::optional&) {} + void operator()(string const&, boost::optional&) {} + }; + struct converter12 + { + template void operator()(TypeIn const&, boost::optional&) {} + }; + struct converter13 + { + void operator()(int const&, boost::optional&) {} + }; + struct converter14 + { + void operator()(int const&, boost::optional&) const {} + }; + struct not_converter11 + { + }; + struct not_converter12 + { + void operator()() {} + }; + struct not_converter13 + { + void operator()(int const&, string const&) {} + }; +}} + +int +main(int, char const* []) +{ + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST( (boost::cnv::is_cnv::value)); + BOOST_TEST(!(boost::cnv::is_cnv::value)); + BOOST_TEST(!(boost::cnv::is_cnv::value)); + + BOOST_TEST(!(boost::cnv::is_cnv::value)); + BOOST_TEST(!(boost::cnv::is_cnv::value)); + BOOST_TEST(!(boost::cnv::is_cnv::value)); + BOOST_TEST(!(boost::cnv::is_cnv::value)); + BOOST_TEST(!(boost::cnv::is_cnv::value)); + + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/jamfile.v2 b/src/boost/libs/convert/test/jamfile.v2 new file mode 100644 index 00000000..9a60a3f5 --- /dev/null +++ b/src/boost/libs/convert/test/jamfile.v2 @@ -0,0 +1,41 @@ +# Convert Jamfile +# Copyright (c) Vladimir Batov 2009-2014 +# Distributed under the Boost Software License, Version 1.0. +# See copy at http://www.boost.org/LICENSE_1_0.txt. + +# bring in the rules for testing +import testing ; + +project convert_test + : requirements + on + icpc:"-std=c++11" + clang:"-std=c++11" + gcc:all + gcc:"-O3 -std=c++0x -Wno-unused-variable -Wno-unused-local-typedefs -Wno-long-long" + msvc:all + msvc:"/wd4996 /wd4512 /wd4610 /wd4510 /wd4127 /wd4701 /wd4127 /wd4305 /wd4244 /wd4714 /wd4189" + msvc:on + msvc:_CRT_SECURE_NO_DEPRECATE + msvc:_SCL_SECURE_NO_DEPRECATE + msvc:_SCL_SECURE_NO_WARNINGS + msvc:_CRT_SECURE_NO_WARNINGS + ../include + ; + +exe convert_test_performance : performance.cpp /boost/timer//boost_timer ; +exe convert_test_performance_spirit : performance_spirit.cpp ; + +run callable.cpp : : : : convert_test_callable ; +run fallbacks.cpp : : : : convert_test_fallbacks ; +run spirit_converter.cpp : : : : convert_test_spirit_converter ; +run stream_converter.cpp : : : : convert_test_stream_converter ; +run printf_converter.cpp : : : : convert_test_printf_converter ; +run strtol_converter.cpp : : : : convert_test_strtol_converter ; +run lcast_converter.cpp : : : : convert_test_lcast_converter ; +run encryption.cpp : : : : convert_test_encryption ; +run user_type.cpp : : : : convert_test_user_type ; +run str_to_int.cpp : : : : convert_test_str_to_int ; +run sfinae.cpp : : : : convert_test_sfinae ; +run has_member.cpp : : : : convert_test_has_member ; + diff --git a/src/boost/libs/convert/test/lcast_converter.cpp b/src/boost/libs/convert/test/lcast_converter.cpp new file mode 100644 index 00000000..8cd14c44 --- /dev/null +++ b/src/boost/libs/convert/test/lcast_converter.cpp @@ -0,0 +1,77 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include + +using std::string; +using boost::convert; + +struct boost::cnv::by_default : boost::cnv::lexical_cast {}; + +static +void +test_invalid() +{ + char const* str[] = { "not", "1 2", " 33", "44 ", "0x11", "7 + 5" }; + int const size = sizeof(str) / sizeof(str[0]); + + for (int k = 0; k < size; ++k) + { + boost::optional const res = convert(str[k]); + bool const failed = !res; + + if (!failed) + { + printf("test::cnv::invalid() failed for: <%s><%d>\n", str[k], res.value()); + } + BOOST_TEST(failed); + } +} + +static +void +test_dbl_to_str() +{ +// printf("100.00 %s\n", convert( 99.999).value().c_str()); +// printf( "99.95 %s\n", convert( 99.949).value().c_str()); +// printf("-99.95 %s\n", convert(-99.949).value().c_str()); +// printf( "99.9 %s\n", convert( 99.949).value().c_str()); +// printf( "1.00 %s\n", convert( 0.999).value().c_str()); +// printf( "-1.00 %s\n", convert( -0.999).value().c_str()); +// printf( "0.95 %s\n", convert( 0.949).value().c_str()); +// printf( "-0.95 %s\n", convert( -0.949).value().c_str()); +// printf( "1.9 %s\n", convert( 1.949).value().c_str()); +// printf( "-1.9 %s\n", convert( -1.949).value().c_str()); +} + +int +main(int, char const* []) +{ + string const not_int_str = "not an int"; + string const std_str = "-11"; + char const* const c_str = "-12"; + int const v00 = convert(not_int_str).value_or(-1); + int const v01 = convert( std_str).value_or(-1); + int const v02 = convert( c_str).value_or(-1); + + BOOST_TEST(v00 == -1); // Failed conversion. No throw + BOOST_TEST(v01 == -11); + BOOST_TEST(v02 == -12); + + test_invalid(); + test_dbl_to_str(); + + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/performance.cpp b/src/boost/libs/convert/test/performance.cpp new file mode 100644 index 00000000..ac208097 --- /dev/null +++ b/src/boost/libs/convert/test/performance.cpp @@ -0,0 +1,317 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include "./prepare.hpp" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +using std::string; +using boost::convert; + +namespace cnv = boost::cnv; +namespace arg = boost::cnv::parameter; + +namespace { namespace local +{ + template + struct array + { + typedef boost::array type; + }; + template static typename array::type const& get(); + + static int const num_cycles = 1000000; + int sum = 0; + + struct timer : public boost::timer::cpu_timer + { + typedef timer this_type; + typedef boost::timer::cpu_timer base_type; + + double value() const + { + boost::timer::cpu_times times = base_type::elapsed(); + int const use_sum = (sum % 2) ? 0 : (sum % 2); BOOST_TEST(use_sum == 0); + + return double(times.user + times.system) / 1000000000 + use_sum; + } + }; + template< typename Type, typename Cnv> static double str_to (Cnv const&); + template static double to_str (Cnv const&); + + template<> + local::array::type const& + get() + { + static array::type ints; + static bool filled; + + if (!filled) + { + boost::random::mt19937 gen (::time(0)); + boost::random::uniform_int_distribution<> dist (INT_MIN, INT_MAX); // INT_MAX(32) = 2,147,483,647 + + for (size_t k = 0; k < ints.size(); ++k) + ints[k] = dist(gen); + + filled = true; + } + return ints; + } + template<> + array::type const& + get() + { + static array::type ints; + static bool filled; + + if (!filled) + { + boost::random::mt19937 gen (::time(0)); + boost::random::uniform_int_distribution<> dist (INT_MIN, INT_MAX); // INT_MAX(32) = 2147483647 + + for (size_t k = 0; k < ints.size(); ++k) + ints[k] = dist(gen); + + filled = true; + } + return ints; + } + template<> + array::type const& + get() + { + static array::type dbls; + static bool filled; + + if (!filled) + { + boost::random::mt19937 gen (::time(0)); + boost::random::uniform_int_distribution<> dist (INT_MIN, INT_MAX); // INT_MAX(32) = 2147483647 + + for (size_t k = 0; k < dbls.size(); ++k) + dbls[k] = double(dist(gen)) + 0.7654321; + + filled = true; + } + return dbls; + } +}} + +struct raw_str_to_int_spirit +{ + int operator()(char const* str) const + { + char const* beg = str; + char const* end = beg + strlen(str); + int result; + + if (boost::spirit::qi::parse(beg, end, boost::spirit::int_, result)) + if (beg == end) // ensure the whole string was parsed + return result; + + return (BOOST_ASSERT(0), result); + } +}; + +struct raw_str_to_int_lxcast +{ + int operator()(char const* str) const + { + return boost::lexical_cast(str); + } +}; + +template +double +raw_str_to(Converter const& cnv) +{ + local::strings strings = local::get_strs(); // Create strings on the stack + int const size = strings.size(); + local::timer timer; + + for (int t = 0; t < local::num_cycles; ++t) + for (int k = 0; k < size; ++k) + local::sum += cnv(strings[k].c_str()); + + return timer.value(); +} + +template +double +local::str_to(Converter const& try_converter) +{ + local::strings strings = local::get_strs(); // Create strings on the stack + int const size = strings.size(); + local::timer timer; + + for (int t = 0; t < local::num_cycles; ++t) + for (int k = 0; k < size; ++k) + local::sum += boost::convert(strings[k].c_str(), try_converter).value(); + + return timer.value(); +} + +template +double +local::to_str(Converter const& try_converter) +{ + typedef typename local::array::type collection; + + collection values = local::get(); + int const size = values.size(); + local::timer timer; + + for (int t = 0; t < local::num_cycles; ++t) + for (int k = 0; k < size; ++k) + local::sum += *boost::convert(Type(values[k]), try_converter).value().begin(); + + return timer.value(); +} + +template +double +performance_str_to_type(Converter const& try_converter) +{ + char const* input[] = { "no", "up", "dn" }; + local::timer timer; + + for (int k = 0; k < local::num_cycles; ++k) + { + change chg = boost::convert(input[k % 3], try_converter).value(); + int res = chg.value(); + + BOOST_TEST(res == k % 3); + + local::sum += res; // Make sure chg is not optimized out + } + return timer.value(); +} + +template +double +performance_type_to_str(Converter const& try_converter) +{ + boost::array input = {{ change::no, change::up, change::dn }}; + boost::array results = {{ "no", "up", "dn" }}; + local::timer timer; + + for (int k = 0; k < local::num_cycles; ++k) + { + string res = boost::convert(input[k % 3], try_converter).value(); + + BOOST_TEST(res == results[k % 3]); + + local::sum += res[0]; // Make sure res is not optimized out + } + return timer.value(); +} + +template +void +performance_comparative(Raw const& raw, Cnv const& cnv, char const* txt) +{ + int const num_tries = 5; + double cnv_time = 0; + double raw_time = 0; + + for (int k = 0; k < num_tries; ++k) cnv_time += local::str_to(cnv); + for (int k = 0; k < num_tries; ++k) raw_time += raw_str_to(raw); + + cnv_time /= num_tries; + raw_time /= num_tries; + + double change = 100 * (1 - cnv_time / raw_time); + + printf("str-to-int: %s raw/cnv=%.2f/%.2f seconds (%.2f%%).\n", txt, raw_time, cnv_time, change); +} + +int +main(int, char const* []) +{ + printf("Started performance tests...\n"); + + printf("str-to-int: spirit/strtol/lcast/scanf/stream=%7.2f/%7.2f/%7.2f/%7.2f/%7.2f seconds.\n", + local::str_to(boost::cnv::spirit()), + local::str_to(boost::cnv::strtol()), + local::str_to(boost::cnv::lexical_cast()), + local::str_to(boost::cnv::printf()), + local::str_to(boost::cnv::cstream())); + printf("str-to-lng: spirit/strtol/lcast/scanf/stream=%7.2f/%7.2f/%7.2f/%7.2f/%7.2f seconds.\n", + local::str_to(boost::cnv::spirit()), + local::str_to(boost::cnv::strtol()), + local::str_to(boost::cnv::lexical_cast()), + local::str_to(boost::cnv::printf()), + local::str_to(boost::cnv::cstream())); + printf("str-to-dbl: spirit/strtol/lcast/scanf/stream=%7.2f/%7.2f/%7.2f/%7.2f/%7.2f seconds.\n", + local::str_to(boost::cnv::spirit()), + local::str_to(boost::cnv::strtol()), + local::str_to(boost::cnv::lexical_cast()), + local::str_to(boost::cnv::printf()), + local::str_to(boost::cnv::cstream())); + + printf("int-to-str: spirit/strtol/lcast/prntf/stream=%7.2f/%7.2f/%7.2f/%7.2f/%7.2f seconds.\n", + local::to_str(boost::cnv::spirit()), + local::to_str(boost::cnv::strtol()), + local::to_str(boost::cnv::lexical_cast()), + local::to_str(boost::cnv::printf()), + local::to_str(boost::cnv::cstream())); + printf("lng-to-str: spirit/strtol/lcast/prntf/stream=%7.2f/%7.2f/%7.2f/%7.2f/%7.2f seconds.\n", + local::to_str(boost::cnv::spirit()), + local::to_str(boost::cnv::strtol()), + local::to_str(boost::cnv::lexical_cast()), + local::to_str(boost::cnv::printf()), + local::to_str(boost::cnv::cstream())); + printf("dbl-to-str: spirit/strtol/lcast/prntf/stream=%7.2f/%7.2f/%7.2f/%7.2f/%7.2f seconds.\n", + local::to_str(boost::cnv::spirit()), + local::to_str(boost::cnv::strtol()(arg::precision = 6)), + local::to_str(boost::cnv::lexical_cast()), + local::to_str(boost::cnv::printf()(arg::precision = 6)), + local::to_str(boost::cnv::cstream()(arg::precision = 6))); + + printf("str-to-user-type: lcast/stream/strtol=%.2f/%.2f/%.2f seconds.\n", + performance_str_to_type(boost::cnv::lexical_cast()), + performance_str_to_type(boost::cnv::cstream()), + performance_str_to_type(boost::cnv::strtol())); + printf("user-type-to-str: lcast/stream/strtol=%.2f/%.2f/%.2f seconds.\n", + performance_type_to_str(boost::cnv::lexical_cast()), + performance_type_to_str(boost::cnv::cstream()), + performance_type_to_str(boost::cnv::strtol())); + + //[small_string_results + printf("strtol int-to std::string/small-string: %.2f/%.2f seconds.\n", + local::to_str(boost::cnv::strtol()), + local::to_str< my_string, int>(boost::cnv::strtol())); + printf("spirit int-to std::string/small-string: %.2f/%.2f seconds.\n", + local::to_str(boost::cnv::spirit()), + local::to_str< my_string, int>(boost::cnv::spirit())); + printf("stream int-to std::string/small-string: %.2f/%.2f seconds.\n", + local::to_str(boost::cnv::cstream()), + local::to_str< my_string, int>(boost::cnv::cstream())); + //] + performance_comparative(raw_str_to_int_spirit(), boost::cnv::spirit(), "spirit"); + performance_comparative(raw_str_to_int_lxcast(), boost::cnv::lexical_cast(), "lxcast"); + + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/performance_spirit.cpp b/src/boost/libs/convert/test/performance_spirit.cpp new file mode 100644 index 00000000..4878b07d --- /dev/null +++ b/src/boost/libs/convert/test/performance_spirit.cpp @@ -0,0 +1,112 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +// This code has been adapted from libs/spirit/optimization/qi/int_parser.cpp. +// This code uses the performance testing framework from libs/spirit/optimization/measure.cpp. +// See these mentioned files for the copyright notice. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include +#include +#include "./prepare.hpp" + +//#define main() old_str_to_int_test_spirit() +//#include +#include +#include +#include +#include +#include + +namespace +{ + namespace local + { + struct base : test::base + { + base() : strings_(local::get_strs()) {} + + // Test strings are created as part of the object, i.e. on the stack to make sure + // they are easily accessed. + local::strings strings_; + }; + } + struct raw_lxcast_str_to_int_test : local::base + { + void benchmark() + { + for (size_t i = 0; i < strings_.size(); ++i) + this->val += boost::lexical_cast(strings_[i].c_str()); + } + }; + struct cnv_lxcast_str_to_int_test : local::base + { + void benchmark() + { + for (size_t i = 0; i < strings_.size(); ++i) + this->val += boost::convert(strings_[i].c_str(), cnv).value(); + } + boost::cnv::lexical_cast cnv; + }; + struct raw_spirit_str_to_int_test : local::base + { + static int parse(char const* str) + { + char const* beg = str; + char const* end = beg + strlen(str); + int n; + + if (boost::spirit::qi::parse(beg, end, boost::spirit::qi::int_, n)) + if (beg == end) + return n; + + return (BOOST_ASSERT(0), 0); + } + void benchmark() + { + for (size_t i = 0; i < strings_.size(); ++i) + this->val += parse(strings_[i].c_str()); + } + }; + struct cnv_spirit_str_to_int_test : local::base + { + void benchmark() + { + for (size_t i = 0; i < strings_.size(); ++i) + this->val += boost::convert(strings_[i].c_str(), cnv).value(); + } + boost::cnv::spirit cnv; + }; +} + +int +main(int, char const* []) +{ + // This code has been adapted from libs/spirit/optimization/qi/int_parser.cpp. + // This code uses the performance testing framework from libs/spirit/optimization/measure.cpp. + // See these mentioned files for the copyright notice. + + BOOST_SPIRIT_TEST_BENCHMARK( + 10000000, // This is the maximum repetitions to execute + (raw_lxcast_str_to_int_test) + (cnv_lxcast_str_to_int_test) + (raw_spirit_str_to_int_test) + (cnv_spirit_str_to_int_test) + ) + + // This is ultimately responsible for preventing all the test code + // from being optimized away. Change this to return 0 and you + // unplug the whole test's life support system. + return test::live_code != 0; +} + +#endif diff --git a/src/boost/libs/convert/test/prepare.hpp b/src/boost/libs/convert/test/prepare.hpp new file mode 100644 index 00000000..15c4d015 --- /dev/null +++ b/src/boost/libs/convert/test/prepare.hpp @@ -0,0 +1,59 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#ifndef BOOST_CONVERT_TEST_PREPARE_HPP +#define BOOST_CONVERT_TEST_PREPARE_HPP + +#include +#include +#include + +// boostinspect:nounnamed +namespace { namespace local +{ + // C1. 18 = 9 positive + 9 negative numbers with the number of digits from 1 to 9. + // Even though INT_MAX(32) = 2147483647, i.e. 10 digits (not to mention long int) + // we only test up to 9 digits as Spirit does not handle more than 9. + + typedef boost::array strings; //C1 + /////////////////////////////////////////////////////////////////////////// + // Generate a random number string with N digits + std::string + gen_int(int digits, bool negative) + { + std::string result; + + if (negative) // Prepend a '-' + result += '-'; + + result += '1' + (std::rand()%9); // The first digit cannot be '0' + + for (int i = 1; i < digits; ++i) // Generate the remaining digits + result += '0' + (std::rand()%10); + return result; + } + + local::strings const& + get_strs() + { + static local::strings strings; + static bool filled; + static bool negative = true; + + if (!filled) + { + // Seed the random generator + std::srand(std::time(0)); + + for (size_t k = 0; k < strings.size(); ++k) + strings[k] = local::gen_int(k/2 + 1, negative = !negative).c_str(); + + filled = true; + } + return strings; + } +}} + +#endif // BOOST_CONVERT_TEST_PREPARE_HPP diff --git a/src/boost/libs/convert/test/printf_converter.cpp b/src/boost/libs/convert/test/printf_converter.cpp new file mode 100644 index 00000000..a56c95b5 --- /dev/null +++ b/src/boost/libs/convert/test/printf_converter.cpp @@ -0,0 +1,47 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include + +using std::string; +using boost::convert; + +namespace arg = boost::cnv::parameter; + +int +main(int, char const* []) +{ + boost::cnv::printf cnv; + + string const not_int_str = "not an int"; + string const std_str = "-11"; + char const* const c_str = "-12"; + + BOOST_TEST( -1 == convert(not_int_str, cnv).value_or(-1)); + BOOST_TEST(-11 == convert(std_str, cnv).value_or(-1)); + BOOST_TEST(-12 == convert(c_str, cnv).value_or(-1)); + + BOOST_TEST("255" == convert(255, cnv(arg::base = boost::cnv::base::dec)).value()); + BOOST_TEST( "ff" == convert(255, cnv(arg::base = boost::cnv::base::hex)).value()); + BOOST_TEST("377" == convert(255, cnv(arg::base = boost::cnv::base::oct)).value()); + + string s01 = convert(12.3456, cnv(arg::precision = 6)).value(); + string s02 = convert(12.3456, cnv(arg::precision = 3)).value(); + + BOOST_TEST(s01 == "12.345600"); + BOOST_TEST(s02 == "12.346"); + + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/sfinae.cpp b/src/boost/libs/convert/test/sfinae.cpp new file mode 100644 index 00000000..5236a4d6 --- /dev/null +++ b/src/boost/libs/convert/test/sfinae.cpp @@ -0,0 +1,159 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include +#include + +//[is_callable_declaration +namespace { namespace local +{ + BOOST_DECLARE_IS_CALLABLE(can_call_funop, operator()); + BOOST_DECLARE_IS_CALLABLE(can_call_func, func); +}} +//] + +//[is_callable_classes_tested +namespace { namespace callable +{ + struct test1 { int operator()(double, std::string) { return 0; }}; + struct test2 { void operator()(double, std::string) {}}; + struct test3 { void operator()(int) {}}; + struct test4 { std::string operator()(int) const { return std::string(); }}; + struct test5 { std::string operator()(int, std::string const& =std::string()) const { return std::string(); }}; + struct test6 { template std::string operator()(T) const { return std::string(); }}; + struct test7 { template T operator()(T) const { return T(); }}; + + struct test11 { int func(double, std::string) { return 0; }}; + struct test12 { void func(double, std::string) {}}; + struct test13 { void func(int) {}}; + struct test14 { std::string func(int) const { return std::string(); }}; + struct test15 { std::string func(int, std::string const& =std::string()) const { return std::string(); }}; + struct test16 { template std::string func(T) const { return std::string(); }}; + struct test17 { template T func(T) const { return T(); }}; +}} +//] + +static +void +test_is_callable() +{ + // C1. Unfortunately, passing 'double' where 'int' is expected returns 'true'. + // The same as the following (which successfully compiles): + // void fun(int) {} + // fun(double(1)); + + //[is_callable_usage1 + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == false)); + BOOST_TEST((local::can_call_funop::value == false)); + + BOOST_TEST((local::can_call_funop::value == false)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + //] + BOOST_TEST((local::can_call_funop::value == true)); //C1 + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == false)); + + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == false)); + BOOST_TEST((local::can_call_funop::value == false)); + + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + + BOOST_TEST((local::can_call_funop::value == false)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + BOOST_TEST((local::can_call_funop::value == true)); + + //[is_callable_usage2 + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == false)); + BOOST_TEST((local::can_call_func::value == false)); + + BOOST_TEST((local::can_call_func::value == false)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + //] + BOOST_TEST((local::can_call_func::value == true)); //C1 + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == false)); + + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == false)); + BOOST_TEST((local::can_call_func::value == false)); + + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + + BOOST_TEST((local::can_call_func::value == false)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); + BOOST_TEST((local::can_call_func::value == true)); +} + +int +main(int, char const* []) +{ + //[is_callable_usage + //] + + test_is_callable(); + + BOOST_TEST(boost::cnv::is_string::value == false); + BOOST_TEST(boost::cnv::is_string::value == true); + BOOST_TEST(boost::cnv::is_string::value == true); + BOOST_TEST(boost::cnv::is_string::value == true); + BOOST_TEST(boost::cnv::is_string::value == false); + + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/spirit_converter.cpp b/src/boost/libs/convert/test/spirit_converter.cpp new file mode 100644 index 00000000..bbbc5b97 --- /dev/null +++ b/src/boost/libs/convert/test/spirit_converter.cpp @@ -0,0 +1,87 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include +#include + +using std::string; +using std::wstring; +using boost::convert; + +namespace cnv = boost::cnv; +namespace arg = boost::cnv::parameter; + +struct boost::cnv::by_default : boost::cnv::spirit {}; + +int +main(int, char const* []) +{ + char const* const c_stri ("12345"); + char const* const c_strd ("123.45"); + wchar_t const* const c_wstri (L"12345"); + wchar_t const* const c_wstrd (L"123.45"); + std::string const std_stri (c_stri); + std::string const std_strd (c_strd); + std::wstring const std_wstri (c_wstri); + std::wstring const std_wstrd (c_wstrd); + my_string const my_stri (c_stri, c_stri + strlen(c_stri)); + my_string const my_strd (c_strd, c_strd + strlen(c_strd)); + + boost::cnv::spirit cnv; + + BOOST_TEST( 12345 == convert< int>( c_stri).value()); + BOOST_TEST( 12345 == convert< int>( c_wstri).value()); + BOOST_TEST( 12345 == convert< int>( std_stri).value()); + BOOST_TEST( 12345 == convert< int>(std_wstri).value()); + BOOST_TEST( 12345 == convert< int>( my_stri).value()); + BOOST_TEST( 12345 == convert(c_stri).value()); + BOOST_TEST( 12345 == convert( c_stri).value()); + BOOST_TEST( 12345 == convert( c_wstri).value()); + BOOST_TEST( 12345 == convert( std_stri).value()); + BOOST_TEST( 12345 == convert(std_wstri).value()); + BOOST_TEST( 12345 == convert( my_stri).value()); + BOOST_TEST( 12345 == convert(c_stri).value()); + BOOST_TEST( 12345 == convert(c_stri).value()); + BOOST_TEST( 12345 == convert(c_stri).value()); + BOOST_TEST(123.45 == convert< double>( c_strd).value()); + BOOST_TEST(123.45 == convert< double>( c_wstrd).value()); +// BOOST_TEST(123.45 == convert< double>( std_strd).value()); +// BOOST_TEST(123.45 == convert< double>(std_wstrd).value()); + BOOST_TEST(123.45 == convert< double>( my_strd).value()); + + BOOST_TEST(!convert< int>("uhm")); + BOOST_TEST(!convert< int>(L"uhm")); + BOOST_TEST(!convert("12.uhm")); + BOOST_TEST(!convert("L12.uhm")); + + BOOST_TEST( "1234" == convert(1234).value()); + BOOST_TEST( "1234" == convert(1234u).value()); + BOOST_TEST( "1234" == convert(1234ll).value()); + BOOST_TEST( "1234" == convert(1234ull).value()); + BOOST_TEST( L"1234" == convert(1234).value()); + BOOST_TEST( "12xxx" == convert(12, cnv(arg::width = 5) + (arg::fill = 'x') + (arg::adjust = cnv::adjust::left)).value()); + BOOST_TEST(L"12xxx" == convert(12, cnv(arg::width = 5) + (arg::fill = 'x') + (arg::adjust = cnv::adjust::left)).value()); + BOOST_TEST( "x12xx" == convert(12, cnv(arg::adjust = cnv::adjust::center)).value()); + BOOST_TEST(L"x12xx" == convert(12, cnv(arg::adjust = cnv::adjust::center)).value()); + +// BOOST_TEST("12.34" == convert(12.34).value()); +// printf("%s\n", convert(12.34).value().c_str()); + + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/str_to_int.cpp b/src/boost/libs/convert/test/str_to_int.cpp new file mode 100644 index 00000000..42951c0d --- /dev/null +++ b/src/boost/libs/convert/test/str_to_int.cpp @@ -0,0 +1,156 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include +#include +#include +#include +#include + +using std::string; + +template +void +int_to_str(Converter const& cnv) +{ + string const not_int_str = "not an int"; + string const std_str = "-11"; + char const* const c_str = "-12"; + char const array_str[] = "-15"; + + //////////////////////////////////////////////////////////////////////////// + // Testing int-to-string conversion with various string + // containers as the fallback values. + //////////////////////////////////////////////////////////////////////////// + + string s01 = boost::convert< string>(-1, cnv).value_or(std_str); + string s02 = boost::convert< string>(-2, cnv).value_or(c_str); + string s05 = boost::convert< string>(-5, cnv).value_or(array_str); + boost::optional< string> rs01 = boost::convert< string>(-1, cnv); + boost::optional< string> rs02 = boost::convert< string>(-2, cnv); + boost::optional< string> rs05 = boost::convert< string>(-5, cnv); + + BOOST_TEST(s01 == "-1"); BOOST_TEST(rs01 && rs01.value() == "-1"); + BOOST_TEST(s02 == "-2"); BOOST_TEST(rs02 && rs02.value() == "-2"); + BOOST_TEST(s05 == "-5"); BOOST_TEST(rs05 && rs05.value() == "-5"); + + string s11 = boost::convert< string>(-1, cnv).value(); + boost::optional< string> rs11 = boost::convert< string>(-1, cnv); + + BOOST_TEST( s11 == "-1"); + BOOST_TEST(rs11 && rs11.value() == "-1"); +} + +template +void +str_to_int(Converter const& cnv) +{ + string const not_int_str = "not an int"; + string const std_str = "-11"; + char const* const c_str = "-123"; + char const array_str[] = "3456"; + + //////////////////////////////////////////////////////////////////////////// + // Testing various kinds of string containers. + // Testing implicit conversion directly to TypeOut (int). + // Testing with the fallback value value provided. + // On failure returns the provided fallback value and DOES NOT THROW. + //////////////////////////////////////////////////////////////////////////// + + int const a00 = boost::convert(not_int_str, cnv).value_or(-1); + int const a01 = boost::convert(std_str, cnv).value_or(-1); + int const a02 = boost::convert(c_str, cnv).value_or(-1); + int const a05 = boost::convert(array_str, cnv).value_or(-1); + + BOOST_TEST(a00 == -1); // Failed conversion + BOOST_TEST(a01 == -11); + BOOST_TEST(a02 == -123); + BOOST_TEST(a05 == 3456); + + //////////////////////////////////////////////////////////////////////////// + // Testing "optional" + //////////////////////////////////////////////////////////////////////////// + + boost::optional const r00 = boost::convert(not_int_str, cnv); + boost::optional const r01 = boost::convert(std_str, cnv); + boost::optional const r02 = boost::convert(c_str, cnv); + boost::optional const r05 = boost::convert(array_str, cnv); + + BOOST_TEST(!r00); // Failed conversion + BOOST_TEST( r01 && r01.value() == -11); + BOOST_TEST( r02 && r02.value() == -123); + BOOST_TEST( r05 && r05.value() == 3456); + + //////////////////////////////////////////////////////////////////////////// + // Testing value() on invalid result. + //////////////////////////////////////////////////////////////////////////// + + try + { + boost::convert(not_int_str, cnv).value(); + BOOST_TEST(!"failed to throw"); + } + catch (std::exception&) + { + // Expected behavior: received 'boost::convert failed' exception. All well. + } + //////////////////////////////////////////////////////////////////////////// + // Testing value() on valid result. + //////////////////////////////////////////////////////////////////////////// + + int a21 = boost::convert(std_str, cnv).value(); + int a22 = boost::convert(c_str, cnv).value(); + int a25 = boost::convert(array_str, cnv).value(); + + BOOST_TEST(a21 == -11); + BOOST_TEST(a22 == -123); + BOOST_TEST(a25 == 3456); + + //////////////////////////////////////////////////////////////////////////// + // Testing empty string. + //////////////////////////////////////////////////////////////////////////// + + int a31 = boost::convert(std::string(), cnv).value_or(-1); + int a32 = boost::convert(std::string(""), cnv).value_or(-1); + int a33 = boost::convert("", cnv).value_or(-1); + + BOOST_ASSERT(a31 == -1); + BOOST_ASSERT(a32 == -1); + BOOST_ASSERT(a33 == -1); +} + +int +main(int, char const* []) +{ + boost::cnv::lexical_cast lxcast_cnv; + boost::cnv::cstream stream_cnv; + boost::cnv::strtol strtol_cnv; + boost::cnv::printf printf_cnv; + boost::cnv::spirit spirit_cnv; + + str_to_int(lxcast_cnv); + str_to_int(stream_cnv); + str_to_int(strtol_cnv); + str_to_int(printf_cnv); + str_to_int(spirit_cnv); + + int_to_str(lxcast_cnv); + int_to_str(stream_cnv); + int_to_str(strtol_cnv); + int_to_str(printf_cnv); + int_to_str(spirit_cnv); + + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/stream_converter.cpp b/src/boost/libs/convert/test/stream_converter.cpp new file mode 100644 index 00000000..fcab03da --- /dev/null +++ b/src/boost/libs/convert/test/stream_converter.cpp @@ -0,0 +1,381 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include +#include +#include +#include + +//[stream_using +using std::string; +using std::wstring; +using boost::convert; +//] +//[stream_cnv_namespace_shortcut +namespace cnv = boost::cnv; +namespace arg = boost::cnv::parameter; +//] + +static +void +test_dbl_to_str() +{ + boost::cnv::cstream cnv; + + cnv(std::fixed); + + BOOST_TEST(convert( 99.999, cnv(arg::precision = 2)).value_or("bad") == "100.00"); + BOOST_TEST(convert( 99.949, cnv(arg::precision = 2)).value_or("bad") == "99.95"); + BOOST_TEST(convert(-99.949, cnv(arg::precision = 2)).value_or("bad") == "-99.95"); + BOOST_TEST(convert( 99.949, cnv(arg::precision = 1)).value_or("bad") == "99.9"); + BOOST_TEST(convert( 0.999, cnv(arg::precision = 2)).value_or("bad") == "1.00"); + BOOST_TEST(convert( -0.999, cnv(arg::precision = 2)).value_or("bad") == "-1.00"); + BOOST_TEST(convert( 0.949, cnv(arg::precision = 2)).value_or("bad") == "0.95"); + BOOST_TEST(convert( -0.949, cnv(arg::precision = 2)).value_or("bad") == "-0.95"); + BOOST_TEST(convert( 1.949, cnv(arg::precision = 1)).value_or("bad") == "1.9"); + BOOST_TEST(convert( -1.949, cnv(arg::precision = 1)).value_or("bad") == "-1.9"); +} + +static +void +test_numbase() +{ + //[stream_numbase_example1 + /*`The following example demonstrates the deployment of `std::dec`, `std::oct` `std::hex` + manipulators: + */ + boost::cnv::cstream ccnv; + + BOOST_TEST(convert( "11", ccnv(std::hex)).value_or(0) == 17); // 11(16) = 17(10) + BOOST_TEST(convert( "11", ccnv(std::oct)).value_or(0) == 9); // 11(8) = 9(10) + BOOST_TEST(convert( "11", ccnv(std::dec)).value_or(0) == 11); + + BOOST_TEST(convert( 18, ccnv(std::hex)).value_or("bad") == "12"); // 18(10) = 12(16) + BOOST_TEST(convert( 10, ccnv(std::oct)).value_or("bad") == "12"); // 10(10) = 12(8) + BOOST_TEST(convert( 12, ccnv(std::dec)).value_or("bad") == "12"); + BOOST_TEST(convert(255, ccnv(arg::base = boost::cnv::base::oct)).value_or("bad") == "377"); + BOOST_TEST(convert(255, ccnv(arg::base = boost::cnv::base::hex)).value_or("bad") == "ff"); + BOOST_TEST(convert(255, ccnv(arg::base = boost::cnv::base::dec)).value_or("bad") == "255"); + + ccnv(std::showbase); + + BOOST_TEST(convert(18, ccnv(std::hex)).value_or("bad") == "0x12"); + BOOST_TEST(convert(10, ccnv(std::oct)).value_or("bad") == "012"); + + ccnv(std::uppercase); + + BOOST_TEST(convert(18, ccnv(std::hex)).value_or("bad") == "0X12"); + //] + //[stream_numbase_example2 + BOOST_TEST(convert("11", ccnv(arg::base = cnv::base::hex)).value_or(0) == 17); + BOOST_TEST(convert("11", ccnv(arg::base = cnv::base::oct)).value_or(0) == 9); + BOOST_TEST(convert("11", ccnv(arg::base = cnv::base::dec)).value_or(0) == 11); + //] + //[wide_stream_numeric_base + boost::cnv::wstream wcnv; + + BOOST_TEST(convert(L"11", wcnv(std::hex)).value_or(0) == 17); // 11(16) = 17(10) + BOOST_TEST(convert(L"11", wcnv(std::oct)).value_or(0) == 9); // 11(8) = 9(10) + BOOST_TEST(convert(L"11", wcnv(std::dec)).value_or(0) == 11); + + BOOST_TEST(convert(254, wcnv(arg::base = cnv::base::dec)).value_or(L"bad") == L"254"); + BOOST_TEST(convert(254, wcnv(arg::base = cnv::base::hex)).value_or(L"bad") == L"fe"); + BOOST_TEST(convert(254, wcnv(arg::base = cnv::base::oct)).value_or(L"bad") == L"376"); + //] +} + +static +void +test_boolalpha() +{ + boost::cnv::cstream cnv; + //[stream_boolalpha_example + BOOST_TEST(convert( true, cnv(std::boolalpha)).value_or("bad") == "true"); + BOOST_TEST(convert(false, cnv(std::boolalpha)).value_or("bad") == "false"); + + BOOST_TEST(convert( "true", cnv(std::boolalpha)).value_or(false) == true); + BOOST_TEST(convert("false", cnv(std::boolalpha)).value_or( true) == false); + + BOOST_TEST(convert( true, cnv(std::noboolalpha)).value_or("bad") == "1"); + BOOST_TEST(convert(false, cnv(std::noboolalpha)).value_or("bad") == "0"); + + BOOST_TEST(convert("1", cnv(std::noboolalpha)).value_or(false) == true); + BOOST_TEST(convert("0", cnv(std::noboolalpha)).value_or( true) == false); + //] +} + +static +void +test_skipws_char() +{ + //[stream_skipws_example + boost::cnv::cstream ccnv; + char const* const cstr_good = " 123"; + char const* const cstr_bad = " 123 "; // std::skipws only affects leading spaces. + + ccnv(std::skipws); // Ignore leading whitespaces +// ccnv(arg::skipws = true); // Ignore leading whitespaces. Alternative interface + + BOOST_TEST(convert(cstr_good, ccnv).value_or(0) == 123); + BOOST_TEST(convert(" 123", ccnv).value_or("bad") == "123"); + + BOOST_TEST(!convert(cstr_bad, ccnv)); + + ccnv(std::noskipws); // Do not ignore leading whitespaces +// ccnv(arg::skipws = false); // Do not ignore leading whitespaces. Alternative interface + + // All conversions fail. + BOOST_TEST(!convert(cstr_good, ccnv)); + BOOST_TEST(!convert( cstr_bad, ccnv)); + //] +} + +static +void +test_skipws_wchar() +{ + //[wide_stream_skipws + boost::cnv::wstream wcnv; + + wcnv(std::noskipws); // Do not ignore leading whitespaces + + BOOST_TEST( convert( L"123", wcnv).value_or(0) == 123); + BOOST_TEST(!convert( L" 123", wcnv)); + BOOST_TEST(!convert(L" 123 ", wcnv)); + + wcnv(std::skipws); // Ignore leading whitespaces +// wcnv(arg::skipws = true); // Ignore leading whitespaces. Alternative interface + + BOOST_TEST( convert( L" 123", wcnv).value_or(0) == 123); + BOOST_TEST(!convert(L" 123 ", wcnv)); + //] +} + +static +void +test_width() +{ + //[stream_width_example + boost::cnv::cstream cnv; + + boost::optional s01 = convert(12, cnv(std::setw(4))); + boost::optional s02 = convert(12, cnv(std::setw(5))(std::setfill('*'))); + boost::optional s03 = convert(12, cnv(std::setw(5))(std::setfill('*'))(std::left)); + + BOOST_TEST(s01 && s01.value() == " 12"); // Field width = 4. + BOOST_TEST(s02 && s02.value() == "***12"); // Field width = 5, filler = '*'. + BOOST_TEST(s03 && s03.value() == "12***"); // Field width = 5, filler = '*', left adjustment + + /*`It needs to be remembered that `boost::cnv::stream` converter uses `std::stream` as its underlying + conversion engine. Consequently, formatting-related behavior are driven by the `std::stream`. Namely, + after every operation is performed, the ['default field width is restored]. The values of + the fill character and the adjustment remain unchanged until they are modified explicitly. + */ + + // The fill and adjustment remain '*' and 'left'. + boost::optional s11 = convert(12, cnv(arg::width = 4)); + boost::optional s12 = convert(12, cnv(arg::width = 5) + (arg::fill = ' ') + (arg::adjust = cnv::adjust::right)); + + BOOST_TEST(s11 && s11.value() == "12**"); // Field width was set to 4. + BOOST_TEST(s12 && s12.value() == " 12"); // Field width was set to 5 with the ' ' filler. + //] +} + +static +void +test_manipulators() +{ + boost::cnv::cstream ccnv; + boost::cnv::wstream wcnv; + + int const hex_v01 = boost::convert("FF", ccnv(std::hex)).value_or(0); + int const hex_v02 = boost::convert(L"F", wcnv(std::hex)).value_or(0); + int const hex_v03 = boost::convert("FF", ccnv(std::dec)).value_or(-5); + int const hex_v04 = boost::convert(L"F", wcnv(std::dec)).value_or(-6); + + BOOST_TEST(hex_v01 == 255); // "FF" + BOOST_TEST(hex_v02 == 15); // L"F" + BOOST_TEST(hex_v03 == -5); // Failed conversion + BOOST_TEST(hex_v04 == -6); // Failed conversion + + ccnv(std::noshowbase)(std::nouppercase)(std::oct); + + BOOST_TEST(boost::convert(255, ccnv).value_or("bad") == "377"); + BOOST_TEST(boost::convert( 15, ccnv).value_or("bad") == "17"); + + ccnv(std::showbase); + + BOOST_TEST(boost::convert(255, ccnv).value_or("bad") == "0377"); + BOOST_TEST(boost::convert( 15, ccnv).value_or("bad") == "017"); + + ccnv(std::uppercase)(std::hex); + + BOOST_TEST(boost::convert(255, ccnv).value_or("bad") == "0XFF"); + BOOST_TEST(boost::convert( 15, ccnv).value_or("bad") == "0XF"); + + ccnv(std::noshowbase)(std::nouppercase)(std::oct); + + BOOST_TEST(boost::convert(255, ccnv).value_or("bad") == "377"); + BOOST_TEST(boost::convert( 15, ccnv).value_or("bad") == "17"); + + ccnv(std::showbase)(arg::uppercase = true)(arg::base = cnv::base::hex); + + BOOST_TEST(boost::convert(255, ccnv).value_or("bad") == "0XFF"); + BOOST_TEST(boost::convert( 15, ccnv).value_or("bad") == "0XF"); +} + +void +test_locale_example() +{ + //[stream_locale_example1 + boost::cnv::cstream cnv; + std::locale rus_locale; + std::locale eng_locale; + + char const* eng_locale_name = test::cnv::is_msc ? "English_United States.1251" : "en_US.UTF-8"; + char const* rus_locale_name = test::cnv::is_msc ? "Russian_Russia.1251" : "ru_RU.UTF-8"; + char const* rus_expected = test::cnv::is_msc ? "1,235e-002" : "1,235e-02"; + char const* eng_expected = test::cnv::is_msc ? "1.235e-002" : "1.235e-02"; + char const* dbl_expected = test::cnv::is_msc ? "1.2345E-002" : "1.2345E-02"; + +// cnv(std::setprecision(4))(std::uppercase)(std::scientific); + cnv(arg::precision = 4) + (arg::uppercase = true) + (arg::notation = cnv::notation::scientific); + + double double_v01 = convert(dbl_expected, cnv).value_or(0); + string double_s02 = convert(double_v01, cnv).value_or("bad"); + + BOOST_TEST(dbl_expected == double_s02); + + try { rus_locale = std::locale(rus_locale_name); } + catch (...) { printf("Bad locale %s.\n", rus_locale_name); exit(1); } + + try { eng_locale = std::locale(eng_locale_name); } + catch (...) { printf("Bad locale %s.\n", eng_locale_name); exit(1); } + +// cnv(std::setprecision(3))(std::nouppercase); + cnv(arg::precision = 3)(arg::uppercase = false); + + string double_rus = convert(double_v01, cnv(rus_locale)).value_or("bad double_rus"); + string double_eng = convert(double_v01, cnv(eng_locale)).value_or("bad double_eng"); + + BOOST_TEST(double_rus == rus_expected); + BOOST_TEST(double_eng == eng_expected); + //] +} + +void +test_locale(double v, boost::cnv::cstream const& cnv, char const* expected) +{ + boost::optional res = convert(v, cnv); + std::string str = res ? *res : "conversion failed"; + + BOOST_TEST(res); + BOOST_TEST(str == expected); + + if (str != expected) + printf("%s [%d]: result=<%s>, expected=<%s>\n", __FILE__, __LINE__, str.c_str(), expected); +} + +static +void +test_locale() +{ + boost::cnv::cstream cnv; + std::locale rus_locale; + std::locale eng_locale; + bool eng_ignore = false; + bool rus_ignore = false; + char const* eng_locale_name = test::cnv::is_msc ? "English_United States.1251" : "en_US.UTF-8"; + char const* rus_locale_name = test::cnv::is_msc ? "Russian_Russia.1251" : "ru_RU.UTF-8"; + char const* eng_expected = test::cnv::is_old_msc ? "1.235e-002" : "1.235e-02"; + char const* rus_expected = test::cnv::is_old_msc ? "1,235e-002" : "1,235e-02"; + char const* dbl_expected = test::cnv::is_old_msc ? "1.2345E-002" : "1.2345E-02"; + + cnv(arg::precision = 4) + (arg::uppercase = true) + (arg::notation = cnv::notation::scientific); + + double const double_v01 = convert(dbl_expected, cnv).value_or(0); + string const double_s02 = convert(double_v01, cnv).value_or("bad"); + + BOOST_TEST(double_v01 != 0); + BOOST_TEST(dbl_expected == double_s02); + + if (dbl_expected != double_s02) + printf("%s [%d]: <%s> != <%s>\n", __FILE__, __LINE__, dbl_expected, double_s02.c_str()); + + try { eng_locale = std::locale(eng_locale_name); } + catch (...) { printf("Bad locale %s. Ignored.\n", eng_locale_name); eng_ignore = true; } + + try { rus_locale = std::locale(rus_locale_name); } + catch (...) { printf("Bad locale %s. Ignored.\n", rus_locale_name); rus_ignore = true; } + +// cnv(std::setprecision(3))(std::nouppercase); + cnv(arg::precision = 3)(arg::uppercase = false); + + if (!eng_ignore) test_locale(double_v01, cnv(eng_locale), eng_expected); + if (!rus_ignore) test_locale(double_v01, cnv(rus_locale), rus_expected); +} + +static +void +test_user_str() +{ + //[stream_my_string + boost::cnv::cstream cnv; + my_string my_str("123"); + + cnv(std::setprecision(2))(std::fixed); + + BOOST_TEST(convert(my_str, cnv).value_or(0) == 123); + + BOOST_TEST(convert( 99.999, cnv).value_or("bad") == "100.00"); + BOOST_TEST(convert( 99.949, cnv).value_or("bad") == "99.95"); + BOOST_TEST(convert(-99.949, cnv).value_or("bad") == "-99.95"); + //] +} + +int +main(int, char const* []) +{ + try + { + // QNX fails to handle std::skipws for wchat_t. + // Excluding from tests so that I do not have to stare on the yellow box (regression failure) + /*********************/ test_skipws_char(); + if (!test::cnv::is_qnx) test_skipws_wchar(); + + test_numbase(); + test_boolalpha(); + test_width(); + test_manipulators(); + test_locale(); + test_dbl_to_str(); + test_user_str(); + } + catch(boost::bad_optional_access const&) + { + BOOST_TEST(!"Caught boost::bad_optional_access exception"); + } + catch(...) + { + BOOST_TEST(!"Caught an unknown exception"); + } + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/strtol_converter.cpp b/src/boost/libs/convert/test/strtol_converter.cpp new file mode 100644 index 00000000..69e33e34 --- /dev/null +++ b/src/boost/libs/convert/test/strtol_converter.cpp @@ -0,0 +1,407 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +//[strtol_basic_deployment_header +#include +#include + +using std::string; +using std::wstring; +using boost::convert; + +struct boost::cnv::by_default : boost::cnv::strtol {}; +//] + +static +void +test_str_to_uint() +{ + string const bad_str = "not an int"; + string const neg_str = "-11"; + string const std_str = "11"; + char const* const c_str = "12"; + boost::string_view v_str = boost::string_view(c_str, 1); + unsigned int const imax = (std::numeric_limits::max)(); + unsigned long int const lmax = (std::numeric_limits::max)(); + std::string const imax_str = boost::lexical_cast(imax); + std::string const lmax_str = boost::lexical_cast(lmax); + + BOOST_TEST( 0 == convert(bad_str).value_or(0)); + BOOST_TEST( 0 == convert(neg_str).value_or(0)); + BOOST_TEST( 0 == convert(neg_str).value_or(0)); + BOOST_TEST( 0 == convert(neg_str).value_or(0)); + BOOST_TEST(11 == convert(std_str).value()); + BOOST_TEST(12 == convert( c_str).value()); + BOOST_TEST( 1 == convert( v_str).value_or(0)); + BOOST_TEST(11 == convert(std_str).value()); + BOOST_TEST(12 == convert( c_str).value()); + BOOST_TEST(imax == convert< unsigned int>(imax_str).value()); + BOOST_TEST(lmax == convert(lmax_str).value()); +} + +static +void +test_str_to_int() +{ + //[strtol_basic_deployment + string const bad_str = "not an int"; + string const std_str = "-11"; + char const* const c_str = "-12"; + boost::string_view v_str = boost::string_view(c_str, 2); + + BOOST_TEST( -1 == convert(bad_str).value_or(-1)); + BOOST_TEST(-11 == convert(std_str).value()); + BOOST_TEST(-12 == convert( c_str).value()); + BOOST_TEST( -1 == convert( v_str).value_or(0)); + //] + wstring const bad_wstr = L"not an int"; + wstring const wstd_str = L"-11"; + wchar_t const* const wc_str = L"-12"; + + BOOST_TEST( -1 == convert(bad_wstr).value_or(-1)); + BOOST_TEST(-11 == convert(wstd_str).value()); + BOOST_TEST(-12 == convert( wc_str).value()); +} + +static +void +test_int_to_str() +{ + short const s_int = -123; + int const i_int = -123; + long const l_int = -123; + long long const ll_int = -123; + + BOOST_TEST( "-123" == convert< std::string> ( s_int).value()); + BOOST_TEST( "-123" == convert< std::string> ( i_int).value()); + BOOST_TEST( "-123" == convert< std::string> ( l_int).value()); + BOOST_TEST( "-123" == convert< std::string> (ll_int).value()); + + BOOST_TEST(L"-123" == convert ( s_int).value()); + BOOST_TEST(L"-123" == convert ( i_int).value()); + BOOST_TEST(L"-123" == convert ( l_int).value()); + BOOST_TEST(L"-123" == convert (ll_int).value()); + + int const imin = (std::numeric_limits::min)(); + int const imax = (std::numeric_limits::max)(); + long int const lmin = (std::numeric_limits::min)(); + long int const lmax = (std::numeric_limits::max)(); + long long int const llmin = (std::numeric_limits::min)(); + long long int const llmax = (std::numeric_limits::max)(); + + std::string const imin_str = boost::lexical_cast(imin); + std::string const imax_str = boost::lexical_cast(imax); + std::string const lmin_str = boost::lexical_cast(lmin); + std::string const lmax_str = boost::lexical_cast(lmax); + std::string const llmin_str = boost::lexical_cast(llmin); + std::string const llmax_str = boost::lexical_cast(llmax); + + BOOST_TEST( imin_str == convert( imin).value()); + BOOST_TEST( imax_str == convert( imax).value()); + BOOST_TEST( lmin_str == convert( lmin).value()); + BOOST_TEST( lmax_str == convert( lmax).value()); + BOOST_TEST(llmin_str == convert(llmin).value()); + BOOST_TEST(llmax_str == convert(llmax).value()); +} + +static +void +test_uint_to_str() +{ + unsigned short const us_int = 123; + unsigned int const ui_int = 123; + unsigned long const ul_int = 123; + unsigned long long const ull_int = 123; + + BOOST_TEST( "123" == convert< std::string> ( us_int).value()); + BOOST_TEST( "123" == convert< std::string> ( ui_int).value()); + BOOST_TEST( "123" == convert< std::string> ( ul_int).value()); + BOOST_TEST( "123" == convert< std::string> (ull_int).value()); + + BOOST_TEST(L"123" == convert ( us_int).value()); + BOOST_TEST(L"123" == convert ( ui_int).value()); + BOOST_TEST(L"123" == convert ( ul_int).value()); + BOOST_TEST(L"123" == convert (ull_int).value()); + + unsigned int const uimax = (std::numeric_limits::max)(); + unsigned long int const ulmax = (std::numeric_limits::max)(); + unsigned long long int const ullmax = (std::numeric_limits::max)(); + + std::string const uimax_str = boost::lexical_cast( uimax); + std::string const ulmax_str = boost::lexical_cast( ulmax); + std::string const ullmax_str = boost::lexical_cast(ullmax); + + BOOST_TEST( uimax_str == convert( uimax).value()); + BOOST_TEST( ulmax_str == convert( ulmax).value()); + BOOST_TEST(ullmax_str == convert(ullmax).value()); +} + +//[strtol_numeric_base_header +#include +#include + +using std::string; +using std::wstring; +using boost::convert; + +namespace cnv = boost::cnv; +namespace arg = boost::cnv::parameter; +//] +static +void +test_width() +{ + //[strtol_width + boost::cnv::strtol cnv; + + string s01 = convert( 12, cnv(arg::width = 4)).value(); + string s02 = convert( 12, cnv(arg::width = 5) + (arg::fill = '*')).value(); + string s03 = convert( 12, cnv(arg::width = 5) + (arg::fill = 'x') + (arg::adjust = cnv::adjust::left)).value(); + string s04 = convert(-98, cnv(arg::width = 6) + (arg::fill = 'Z') + (arg::adjust = cnv::adjust::right)).value(); + + string s05 = convert(-12.3451, cnv(arg::precision = 2) + (arg::width = 10) + (arg::fill = '*') + (arg::adjust = cnv::adjust::left)).value(); + string s06 = convert(-12.3450, cnv(arg::adjust = cnv::adjust::right)).value(); + string s07 = convert(-12.3450, cnv(arg::adjust = cnv::adjust::center)).value(); + + BOOST_TEST(s01 == " 12"); + BOOST_TEST(s02 == "***12"); + BOOST_TEST(s03 == "12xxx"); + BOOST_TEST(s04 == "ZZZ-98"); + BOOST_TEST(s05 == "-12.35****"); + BOOST_TEST(s06 == "****-12.35"); + BOOST_TEST(s07 == "**-12.35**"); + //] +} + +static +void +test_base() +{ + //[strtol_numeric_base + boost::cnv::strtol cnv; + + BOOST_TEST( "11111110" == convert< string>(254, cnv(arg::base = cnv::base::bin)).value()); + BOOST_TEST( "254" == convert< string>(254, cnv(arg::base = cnv::base::dec)).value()); + BOOST_TEST( "FE" == convert< string>(254, cnv(arg::base = cnv::base::hex)).value()); + BOOST_TEST( "376" == convert< string>(254, cnv(arg::base = cnv::base::oct)).value()); + //] + //[wide_strtol_numeric_base + BOOST_TEST(L"11111110" == convert(254, cnv(arg::base = cnv::base::bin)).value()); + BOOST_TEST( L"254" == convert(254, cnv(arg::base = cnv::base::dec)).value()); + BOOST_TEST( L"FE" == convert(254, cnv(arg::base = cnv::base::hex)).value()); + BOOST_TEST( L"376" == convert(254, cnv(arg::base = cnv::base::oct)).value()); + //] +} + +static +void +test_upper() +{ +// boost::cnv::strtol cnv; +// char const* c_lcase = "abcde"; +// char const* c_ucase = "ABCDE"; +// wchar_t const* w_lcase = L"abcde"; +// wchar_t const* w_ucase = L"ABCDE"; +// +// BOOST_TEST(c_lcase == convert< string>(c_lcase, cnv(arg::uppercase = false)).value_or("")); +// BOOST_TEST(c_ucase == convert< string>(c_lcase, cnv(arg::uppercase = true)).value_or("")); +// BOOST_TEST(w_ucase == convert(w_lcase, cnv(arg::uppercase = false)).value_or("")); +// BOOST_TEST(w_ucase == convert(w_lcase, cnv(arg::uppercase = true)).value_or("")); +} + +static +void +test_skipws() +{ + //[strtol_skipws + boost::cnv::strtol cnv; + + BOOST_TEST(-1 == convert( " 12", cnv(arg::skipws = false)).value_or(-1)); + BOOST_TEST(12 == convert( " 12", cnv(arg::skipws = true)).value_or(-1)); + //] + //[wide_strtol_skipws + BOOST_TEST(-1 == convert(L" 12", cnv(arg::skipws = false)).value_or(-1)); + BOOST_TEST(12 == convert(L" 12", cnv(arg::skipws = true)).value_or(-1)); + //] +} + +static +void +dbl_to_str_example() +{ + //[strtol_precision + boost::cnv::strtol cnv; + + BOOST_TEST( "12.3" == convert(12.3456, cnv(arg::precision = 1)).value()); + BOOST_TEST( "12.35" == convert(12.3456, cnv(arg::precision = 2)).value()); + BOOST_TEST("12.346" == convert(12.3456, cnv(arg::precision = 3)).value()); + + BOOST_TEST( "-12.3" == convert(-12.3456, cnv(arg::precision = 1)).value()); + BOOST_TEST( "-12.35" == convert(-12.3456, cnv(arg::precision = 2)).value()); + BOOST_TEST("-12.346" == convert(-12.3456, cnv(arg::precision = 3)).value()); + //] +} + +static +std::pair +get_random() +{ + namespace rdm = boost::random; + + static rdm::mt19937 gen (::time(0)); + static rdm::uniform_int_distribution<> precision (0, 6); + static rdm::uniform_int_distribution<> int_part (0, SHRT_MAX); + static rdm::uniform_01 fraction; // uniform double in [0,1) + static bool sign; + + double dbl = (int_part(gen) + fraction(gen)) * ((sign = !sign) ? 1 : -1); + +// printf("%.12f\n", dbl); + + return std::make_pair(dbl, precision(gen)); +} + +static +void +compare(std::pair pair) +{ + boost::cnv::strtol cnv1; + boost::cnv::printf cnv2; + + string s1 = convert(pair.first, cnv1(arg::precision = pair.second)).value(); + string s2 = convert(pair.first, cnv2(arg::precision = pair.second)).value(); + + if (s1 != s2) + printf("dbl=%.12f(%d).strtol/printf=%s/%s.\n", pair.first, pair.second, s1.c_str(), s2.c_str()); +} + +static +void +test_str_to_dbl() +{ + char const* const c_str = "1.23456"; + + BOOST_TEST(1.2 == convert(boost::string_view(c_str, 3)).value_or(0)); + BOOST_TEST(1.23 == convert(boost::string_view(c_str, 4)).value_or(0)); +} + +static +void +test_dbl_to_str() +{ +// double round_up_abs01 = ::rint(-0.5); +// double round_up_abs02 = ::round(-0.5); +// double round_up_abs11 = ::rint(0.5); +// double round_up_abs12 = ::round(0.5); + +// double huge_v = 987654321098765432109.123; +// +// printf("%f\n", huge_v); +// string huge = convert(huge_v, cnv1(arg::precision = 2)).value(); +// printf("%s\n", huge.c_str()); + + int const num_tries = 1000000; + double const dbls[] = { 0.90, 1.0, 1.1, 0.94, 0.96, 1.04, 1.05, 1.06, 9.654, 999.888 }; + int const num_dbls = sizeof(dbls) / sizeof(dbls[0]); + + printf("cnv::strtol::%s: started with %d random numbers...\n", __FUNCTION__, num_tries); + + BOOST_TEST( "0" == convert( 0.0, cnv::strtol()(arg::precision = 0)).value()); + BOOST_TEST( "0.0" == convert( 0.0, cnv::strtol()(arg::precision = 1)).value()); + BOOST_TEST("0.00" == convert( 0.0, cnv::strtol()(arg::precision = 2)).value()); + BOOST_TEST( "1" == convert(0.95, cnv::strtol()(arg::precision = 0)).value()); + BOOST_TEST( "1.0" == convert(0.95, cnv::strtol()(arg::precision = 1)).value()); + BOOST_TEST("0.95" == convert(0.95, cnv::strtol()(arg::precision = 2)).value()); + + for (int k = 0; k < num_tries; ++k) + compare(get_random()); + + for (int k = 0; k < num_dbls; ++k) + for (int precision = 0; precision < 3; ++precision) + { + compare(std::make_pair( dbls[k], precision)); + compare(std::make_pair(-dbls[k], precision)); + } + + printf("cnv::strtol::%s: finished.\n", __FUNCTION__); +} + +static +void +test_user_string() +{ + //[strtol_user_string + boost::cnv::strtol cnv; + + BOOST_TEST( "12" == convert(12, cnv).value()); + BOOST_TEST("0.95" == convert(0.95, cnv(arg::precision = 2)).value()); + //] +} + +static +void +test_user_type() +{ + //[strtol_user_type + boost::cnv::strtol cnv; + change up_chg = change::up; + change dn_chg = change::dn; + + BOOST_TEST(convert(up_chg, cnv, "bad") == "up"); + BOOST_TEST(convert(dn_chg, cnv, "bad") == "dn"); + BOOST_TEST(convert( 12, cnv, "bad") == "12"); + + BOOST_TEST(convert("up", cnv, change::no) == change::up); + BOOST_TEST(convert("dn", cnv, change::no) == change::dn); + BOOST_TEST(convert< int>("12", cnv, -1) == 12); + //] +} + +int +main(int, char const* []) +{ + dbl_to_str_example(); + + test_str_to_int(); + test_str_to_uint(); + test_int_to_str(); + test_uint_to_str(); + test_base(); + test_upper(); + test_skipws(); + test_str_to_dbl(); + test_dbl_to_str(); + test_width(); + test_user_string(); + test_user_type(); + + return boost::report_errors(); +} + +#endif diff --git a/src/boost/libs/convert/test/test.hpp b/src/boost/libs/convert/test/test.hpp new file mode 100644 index 00000000..e2fbbc6e --- /dev/null +++ b/src/boost/libs/convert/test/test.hpp @@ -0,0 +1,187 @@ +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#ifndef BOOST_CONVERT_TEST_HPP +#define BOOST_CONVERT_TEST_HPP + +#include +#include +#include +#include +#include +#include // For strlen, strcmp, memcpy +#include // Is needed for 'memset' +#include +#include + +#if defined(_MSC_VER) +# pragma warning(disable: 4189) // local variable is initialized but not referenced. +# pragma warning(disable: 4127) // conditional expression is constant. +# pragma warning(disable: 4100) // unreferenced formal parameter. +# pragma warning(disable: 4714) // marked as __forceinline not #endif +# pragma warning(disable: 4706) +# pragma warning(disable: 4005) +# pragma warning(disable: 4459) // declaration hides global declaration +# pragma warning(disable: 4456) // declaration hides previous local declaration +#endif + +//[change_declaration +struct change +{ + enum value_type { no, up, dn }; + + change(value_type v =no) : value_(v) {} + bool operator==(change v) const { return value_ == v.value_; } + value_type value() const { return value_; } + + private: value_type value_; +}; +//] +//[change_stream_operators +std::istream& operator>>(std::istream& stream, change& chg) +{ + std::string str; stream >> str; + + /**/ if (str == "up") chg = change::up; + else if (str == "dn") chg = change::dn; + else if (str == "no") chg = change::no; + else stream.setstate(std::ios_base::failbit); + + return stream; +} + +std::ostream& operator<<(std::ostream& stream, change const& chg) +{ + return stream << (chg == change::up ? "up" : chg == change::dn ? "dn" : "no"); +} +//] +//[change_convert_operators +inline void operator>>(change chg, boost::optional& str) +{ + str = chg == change::up ? "up" : chg == change::dn ? "dn" : "no"; +} + +inline void operator>>(std::string const& str, boost::optional& chg) +{ + /**/ if (str == "up") chg = change::up; + else if (str == "dn") chg = change::dn; + else if (str == "no") chg = change::no; +} +//] +//[direction_declaration +struct direction +{ + // Note: the class does NOT have the default constructor. + + enum value_type { up, dn }; + + direction(value_type value) : value_(value) {} + bool operator==(direction that) const { return value_ == that.value_; } + value_type value() const { return value_; } + + private: value_type value_; +}; +//] +//[direction_stream_operators +std::istream& operator>>(std::istream& stream, direction& dir) +{ + std::string str; stream >> str; + + /**/ if (str == "up") dir = direction::up; + else if (str == "dn") dir = direction::dn; + else stream.setstate(std::ios_base::failbit); + + return stream; +} +std::ostream& operator<<(std::ostream& stream, direction const& dir) +{ + return stream << (dir.value() == direction::up ? "up" : "dn"); +} +//] +//[direction_declaration_make_default +namespace boost +{ + template<> inline direction make_default() + { + return direction(direction::up); + } +} +//] +// Quick and dirty small-string implementation for performance tests. +//[my_string_declaration +struct my_string +{ + typedef my_string this_type; + typedef char value_type; + typedef value_type* iterator; + typedef value_type const* const_iterator; + + my_string (); + my_string (const_iterator, const_iterator =0); + + char const* c_str () const { return storage_; } + const_iterator begin () const { return storage_; } + const_iterator end () const { return storage_ + strlen(storage_); } + this_type& operator= (char const*); + + private: + + static size_t const size_ = 12; + char storage_[size_]; +}; +//] +inline +my_string::my_string() +{ + storage_[0] = 0; +} + +inline +my_string::my_string(const_iterator beg, const_iterator end) +{ + std::size_t const sz = end ? (end - beg) : strlen(beg); + + BOOST_ASSERT(sz < size_); + memcpy(storage_, beg, sz); storage_[sz] = 0; +} + +inline +my_string& +my_string::operator=(char const* str) +{ + BOOST_ASSERT(strlen(str) < size_); + strcpy(storage_, str); + return *this; +} + +inline bool operator==(char const* s1, my_string const& s2) { return strcmp(s1, s2.c_str()) == 0; } +inline bool operator==(my_string const& s1, char const* s2) { return strcmp(s2, s1.c_str()) == 0; } + +namespace test +{ + struct cnv + { +#if defined(__QNXNTO__) + static bool const is_qnx = true; +#else + static bool const is_qnx = false; +#endif + +#if defined(_MSC_VER) && _MSC_VER < 1900 + static bool const is_msc = true; + static bool const is_old_msc = true; +#elif defined(_MSC_VER) + static bool const is_msc = true; + static bool const is_old_msc = false; +#elif defined(__MINGW32__) + static bool const is_msc = true; + static bool const is_old_msc = true; +#else + static bool const is_msc = false; + static bool const is_old_msc = false; +#endif + }; +} + +#endif // BOOST_CONVERT_TEST_HPP diff --git a/src/boost/libs/convert/test/user_type.cpp b/src/boost/libs/convert/test/user_type.cpp new file mode 100644 index 00000000..bd2bb545 --- /dev/null +++ b/src/boost/libs/convert/test/user_type.cpp @@ -0,0 +1,101 @@ +// Boost.Convert test and usage example +// Copyright (c) 2009-2016 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#include "./test.hpp" + +#if defined(BOOST_CONVERT_IS_NOT_SUPPORTED) +int main(int, char const* []) { return 0; } +#else + +#include +#include +#include +#include +#include + +using std::string; +using std::wstring; + +static +void +test_user_type() +{ + boost::cnv::cstream cnv; // stringstream-based char converter + + direction const up_dir1 = direction::up; + direction const dn_dir1 = direction::dn; +// string const up_dir0_str = boost::convert(direction::up, cnv).value(); +// string const dn_dir0_str = boost::convert(direction::dn, cnv).value(); + string const up_dir1_str = boost::convert(up_dir1, cnv).value(); + string const dn_dir1_str = boost::convert(dn_dir1, cnv).value(); + direction const up_dir2 = boost::convert(up_dir1_str, cnv).value(); + direction const dn_dir2 = boost::convert(dn_dir1_str, cnv).value(); + direction const up_dir3 = boost::convert(up_dir1_str, cnv).value(); + direction const dn_dir3 = boost::convert(dn_dir1_str, cnv).value(); + direction const dn_dir4 = boost::convert("junk", cnv).value_or(direction::dn); + boost::optional up_dir4 = boost::convert("junk", cnv); + +// BOOST_TEST(up_dir0_str == "up"); +// BOOST_TEST(dn_dir0_str == "dn"); + BOOST_TEST(up_dir1_str == "up"); + BOOST_TEST(dn_dir1_str == "dn"); + BOOST_TEST(up_dir2 == up_dir1); + BOOST_TEST(dn_dir2 == dn_dir1); + BOOST_TEST(up_dir3 == direction::up); + BOOST_TEST(dn_dir3 == direction::dn); + BOOST_TEST(dn_dir4 == direction::dn); + BOOST_TEST(!up_dir4); // Failed conversion +} + +static +void +test_algorithms() +{ +//[algorithm_example6 + boost::array chgs1 = {{ change::no, change::up, change::dn }}; + boost::array chgs2 = {{ change::no, change::up, change::dn }}; + std::vector strs1; + std::vector strs2; + std::vector strs3; + boost::cnv::cstream cnv; + + std::transform(chgs1.begin(), chgs1.end(), std::back_inserter(strs1), + boost::cnv::apply(boost::cref(cnv))); // Deduced TypeIn is 'change' + + std::transform(chgs2.begin(), chgs2.end(), std::back_inserter(strs2), + boost::cnv::apply(boost::cref(cnv))); // Deduced TypeIn is 'change::value_type' + + BOOST_TEST(strs1.size() == 3); + BOOST_TEST(strs1[0] == "no"); + BOOST_TEST(strs1[1] == "up"); + BOOST_TEST(strs1[2] == "dn"); + + BOOST_TEST(strs2.size() == 3); + BOOST_TEST(strs2[0] == "0"); + BOOST_TEST(strs2[1] == "1"); + BOOST_TEST(strs2[2] == "2"); +//] +//[algorithm_example7 + + std::transform(chgs2.begin(), chgs2.end(), std::back_inserter(strs3), + boost::cnv::apply(boost::cref(cnv))); + + BOOST_TEST(strs3.size() == 3); + BOOST_TEST(strs3[0] == "no"); + BOOST_TEST(strs3[1] == "up"); + BOOST_TEST(strs3[2] == "dn"); +//] +} + +int +main(int, char const* []) +{ + test_user_type(); + test_algorithms(); + + return boost::report_errors(); +} + +#endif -- cgit v1.2.3