diff --git a/cast.htm b/cast.htm
index 587d08a..a1580c3 100644
--- a/cast.htm
+++ b/cast.htm
@@ -120,10 +120,10 @@ void f( Fruit * fruit ) {
polymorphic_cast
was suggested by Bjarne Stroustrup in "The C++
Programming Language".
polymorphic_downcast
was contributed by Dave Abrahams .
+ "http://www.boost.org/people/dave_abrahams.htm">Dave Abrahams.
An old
numeric_cast
that was contributed by Kevlin Henney is now superseeded by the Boost Numeric Conversion Library
+ "http://www.boost.org/people/kevlin_henney.htm">Kevlin Henney is now superseeded by the Boost Numeric Conversion Library
Revised
diff --git a/include/boost/detail/lcast_precision.hpp b/include/boost/detail/lcast_precision.hpp
index 5bd96fd..d40ca21 100644
--- a/include/boost/detail/lcast_precision.hpp
+++ b/include/boost/detail/lcast_precision.hpp
@@ -21,7 +21,13 @@
#include
#endif
-#ifdef BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS
+#if defined(BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS) || \
+ (defined(BOOST_MSVC) && (BOOST_MSVC<1310))
+
+#define BOOST_LCAST_NO_COMPILE_TIME_PRECISION
+#endif
+
+#ifdef BOOST_LCAST_NO_COMPILE_TIME_PRECISION
#include
#else
#include
@@ -31,7 +37,7 @@ namespace boost { namespace detail {
class lcast_abstract_stub {};
-#ifndef BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS
+#ifndef BOOST_LCAST_NO_COMPILE_TIME_PRECISION
// Calculate an argument to pass to std::ios_base::precision from
// lexical_cast. See alternative implementation for broken standard
// libraries in lcast_get_precision below. Keep them in sync, please.
@@ -77,9 +83,9 @@ struct lcast_precision
);
BOOST_STATIC_ASSERT(!is_specialized_bin ||
- limits::digits + 0UL < ULONG_MAX / 30103UL &&
+ (limits::digits + 0UL < ULONG_MAX / 30103UL &&
precision_bin > limits::digits10 + 0UL &&
- precision_bin <= streamsize_max + 0UL
+ precision_bin <= streamsize_max + 0UL)
);
BOOST_STATIC_CONSTANT(std::streamsize, value =
@@ -92,7 +98,7 @@ struct lcast_precision
template
inline std::streamsize lcast_get_precision(T* = 0)
{
-#if !defined(BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS)
+#ifndef BOOST_LCAST_NO_COMPILE_TIME_PRECISION
return lcast_precision::value;
#else // Follow lcast_precision algorithm at run-time:
diff --git a/include/boost/implicit_cast.hpp b/include/boost/implicit_cast.hpp
old mode 100755
new mode 100644
diff --git a/include/boost/lexical_cast.hpp b/include/boost/lexical_cast.hpp
index 1cd23f7..9fddb70 100644
--- a/include/boost/lexical_cast.hpp
+++ b/include/boost/lexical_cast.hpp
@@ -3,7 +3,7 @@
// Boost lexical_cast.hpp header -------------------------------------------//
//
-// See http://www.boost.org/ for most recent version including documentation.
+// See http://www.boost.org/libs/conversion for documentation.
// See end of this header for rights and permissions.
//
// what: lexical_cast custom keyword cast
@@ -17,17 +17,23 @@
#include
#include
#include
-#include
#include
#include
+#include
#include
#include
#include
#include
#include
+#include
#include
#include
#include
+#include
+
+#ifndef BOOST_NO_STD_LOCALE
+#include
+#endif
#ifdef BOOST_NO_STRINGSTREAM
#include
@@ -45,6 +51,12 @@ namespace boost
{
// exception used to indicate runtime lexical_cast failure
class bad_lexical_cast : public std::bad_cast
+
+#if defined(__BORLANDC__) && BOOST_WORKAROUND( __BORLANDC__, < 0x560 )
+ // under bcc32 5.5.1 bad_cast doesn't derive from exception
+ , public std::exception
+#endif
+
{
public:
bad_lexical_cast() :
@@ -52,9 +64,9 @@ namespace boost
{
}
bad_lexical_cast(
- const std::type_info &source_type,
- const std::type_info &target_type) :
- source(&source_type), target(&target_type)
+ const std::type_info &source_type_arg,
+ const std::type_info &target_type_arg) :
+ source(&source_type_arg), target(&target_type_arg)
{
}
const std::type_info &source_type() const
@@ -86,6 +98,14 @@ namespace boost
typedef char type;
};
+#ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
+ template
+ struct stream_char< std::basic_string >
+ {
+ typedef CharT type;
+ };
+#endif
+
#ifndef DISABLE_WIDE_CHAR_SUPPORT
#ifndef BOOST_NO_INTRINSIC_WCHAR_T
template<>
@@ -107,11 +127,13 @@ namespace boost
typedef wchar_t type;
};
+#ifdef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
template<>
struct stream_char
{
typedef wchar_t type;
};
+#endif
#endif
template
@@ -127,6 +149,44 @@ namespace boost
};
}
+ namespace detail // deduce_char_traits template
+ {
+#ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
+ template
+ struct deduce_char_traits
+ {
+ typedef std::char_traits type;
+ };
+
+ template
+ struct deduce_char_traits< CharT
+ , std::basic_string
+ , Source
+ >
+ {
+ typedef Traits type;
+ };
+
+ template
+ struct deduce_char_traits< CharT
+ , Target
+ , std::basic_string
+ >
+ {
+ typedef Traits type;
+ };
+
+ template
+ struct deduce_char_traits< CharT
+ , std::basic_string
+ , std::basic_string
+ >
+ {
+ typedef Traits type;
+ };
+#endif
+ }
+
namespace detail // lcast_src_length
{
// Return max. length of string representation of Source;
@@ -267,14 +327,15 @@ namespace boost
template
struct lcast_src_length_integral
{
-#if !defined(__BORLANDC__) || __BORLANDC__ >= 0x581
+#ifndef BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS
BOOST_STATIC_CONSTANT(std::size_t, value =
std::numeric_limits::is_signed +
std::numeric_limits::is_specialized + // == 1
std::numeric_limits::digits10 * 2
);
#else
- BOOST_STATIC_CONSTANT(std::size_t, value = 156); // 256bit integers
+ BOOST_STATIC_CONSTANT(std::size_t, value = 156);
+ BOOST_STATIC_ASSERT(sizeof(Source) * CHAR_BIT <= 256);
#endif
};
@@ -307,9 +368,7 @@ namespace boost
#undef BOOST_AUX_LEXICAL_CAST_DEF
#undef BOOST_AUX_LEXICAL_CAST_DEF1
-#ifndef BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS
-// This #if is in sync with lcast_precision
-
+#ifndef BOOST_LCAST_NO_COMPILE_TIME_PRECISION
// Helper for floating point types.
// -1.23456789e-123456
// ^ sign
@@ -376,7 +435,7 @@ namespace boost
};
#endif // #ifndef DISABLE_WIDE_CHAR_SUPPORT
-#endif // #ifndef BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS
+#endif // #ifndef BOOST_LCAST_NO_COMPILE_TIME_PRECISION
}
namespace detail // '0' and '-' constants
@@ -400,14 +459,8 @@ namespace boost
#endif
}
- namespace detail // lexical_streambuf and lexical_streambuf_fake
+ namespace detail // lexical_streambuf_fake
{
- template
- class lexical_streambuf : public std::basic_streambuf
- {
- };
-
- template
struct lexical_streambuf_fake
{
};
@@ -420,61 +473,36 @@ namespace boost
// C4146: unary minus operator applied to unsigned type, result still unsigned
# pragma warning( disable : 4146 )
#endif
-
- inline unsigned int lcast_to_unsigned(int value)
+ template
+ inline
+ BOOST_DEDUCED_TYPENAME make_unsigned::type lcast_to_unsigned(T value)
{
- unsigned int uval = value;
- return value < 0 ? -uval : uval;
+ typedef BOOST_DEDUCED_TYPENAME make_unsigned::type result_type;
+ result_type uvalue = static_cast(value);
+ return value < 0 ? -uvalue : uvalue;
}
-
- inline unsigned long lcast_to_unsigned(long value)
- {
- unsigned long uval = value;
- return value < 0 ? -uval : uval;
- }
-
-#if defined(BOOST_HAS_LONG_LONG)
- inline boost::ulong_long_type lcast_to_unsigned(boost::long_long_type v)
- {
- boost::ulong_long_type uval = v;
- return v < 0 ? -uval : uval;
- }
-#elif defined(BOOST_HAS_MS_INT64)
- inline unsigned __int64 lcast_to_unsigned(__int64 value)
- {
- unsigned __int64 uval = value;
- return value < 0 ? -uval : uval;
- }
-#endif
-
#if (defined _MSC_VER)
-# pragma warning( pop ) // C4146: unary minus operator applied to unsigned type, result still unsigned
+# pragma warning( pop )
#endif
}
namespace detail // lcast_put_unsigned
{
- // I'd personally put lcast_put_unsigned in .cpp file if not
- // boost practice for header-only libraries (Alexander Nasonov).
- template
+ template
CharT* lcast_put_unsigned(T n, CharT* finish)
{
- CharT thousands_sep = 0;
+#ifndef BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS
+ BOOST_STATIC_ASSERT(!std::numeric_limits::is_signed);
+#endif
-#ifdef BOOST_LEXICAL_CAST_ASSUME_C_LOCALE
- char const* grouping = "";
- std::size_t const grouping_size = 0;
-#else
+#ifndef BOOST_LEXICAL_CAST_ASSUME_C_LOCALE
+ // TODO: use BOOST_NO_STD_LOCALE
std::locale loc;
typedef std::numpunct numpunct;
numpunct const& np = BOOST_USE_FACET(numpunct, loc);
std::string const& grouping = np.grouping();
std::string::size_type const grouping_size = grouping.size();
-
- if(grouping_size)
- thousands_sep = np.thousands_sep();
-#endif
-
+ CharT thousands_sep = grouping_size ? np.thousands_sep() : 0;
std::string::size_type group = 0; // current group number
char last_grp_size = grouping[0] <= 0 ? CHAR_MAX : grouping[0];
// a) Since grouping is const, grouping[grouping.size()] returns 0.
@@ -485,9 +513,15 @@ namespace boost
#endif
char left = last_grp_size;
+#endif
+
+ typedef typename Traits::int_type int_type;
+ CharT const czero = lcast_char_constants::zero;
+ int_type const zero = Traits::to_int_type(czero);
do
{
+#ifndef BOOST_LEXICAL_CAST_ASSUME_C_LOCALE
if(left == 0)
{
++group;
@@ -499,14 +533,15 @@ namespace boost
left = last_grp_size;
--finish;
- *finish = thousands_sep;
+ Traits::assign(*finish, thousands_sep);
}
--left;
+#endif
+
--finish;
- int const digit = static_cast(n % 10);
- int const cdigit = digit + lcast_char_constants::zero;
- *finish = static_cast(cdigit);
+ int_type const digit = static_cast(n % 10U);
+ Traits::assign(*finish, Traits::to_char_type(zero + digit));
n /= 10;
} while(n);
@@ -516,7 +551,7 @@ namespace boost
namespace detail // stream wrapper for handling lexical conversions
{
- template
+ template
class lexical_stream
{
private:
@@ -524,6 +559,8 @@ namespace boost
typename stream_char::type,
typename stream_char::type>::type char_type;
+ typedef Traits traits_type;
+
public:
lexical_stream(char_type* = 0, char_type* = 0)
{
@@ -553,9 +590,12 @@ namespace boost
EOF;
#else
- std::char_traits::eof();
+ traits_type::eof();
#endif
}
+
+#ifdef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
+
bool operator>>(std::string &output)
{
#if defined(BOOST_NO_STRINGSTREAM)
@@ -571,13 +611,29 @@ namespace boost
return true;
}
#endif
+
+#else
+ bool operator>>(std::basic_string& output)
+ {
+ stream.str().swap(output);
+ return true;
+ }
+
+ template
+ bool operator>>(std::basic_string& out)
+ {
+ std::basic_string str(stream.str());
+ out.assign(str.begin(), str.end());
+ return true;
+ }
+#endif
private:
#if defined(BOOST_NO_STRINGSTREAM)
std::strstream stream;
#elif defined(BOOST_NO_STD_LOCALE)
std::stringstream stream;
#else
- std::basic_stringstream stream;
+ std::basic_stringstream stream;
#endif
};
}
@@ -586,7 +642,8 @@ namespace boost
{
// String representation of Source has an upper limit.
template< class CharT // a result of widest_char transformation
- , class Base // lexical_streambuf or lexical_streambuf_fake
+ , class Base // lexical_streambuf_fake or basic_streambuf
+ , class Traits // usually char_traits
>
class lexical_stream_limited_src : public Base
{
@@ -600,19 +657,21 @@ namespace boost
static void widen_and_assign(char*p, char ch)
{
- *p = ch;
+ Traits::assign(*p, ch);
}
#ifndef DISABLE_WIDE_CHAR_SUPPORT
static void widen_and_assign(wchar_t* p, char ch)
{
+ // TODO: use BOOST_NO_STD_LOCALE
std::locale loc;
- *p = BOOST_USE_FACET(std::ctype, loc).widen(ch);
+ wchar_t w = BOOST_USE_FACET(std::ctype, loc).widen(ch);
+ Traits::assign(*p, w);
}
static void widen_and_assign(wchar_t* p, wchar_t ch)
{
- *p = ch;
+ Traits::assign(*p, ch);
}
static void widen_and_assign(char*, wchar_t ch); // undefined
@@ -642,8 +701,8 @@ namespace boost
public: // output
- template
- bool operator<<(std::basic_string const& str)
+ template
+ bool operator<<(std::basic_string const& str)
{
start = const_cast(str.data());
finish = start + str.length();
@@ -706,7 +765,7 @@ namespace boost
EOF;
#else
- std::char_traits::eof();
+ Traits::eof();
#endif
}
@@ -722,7 +781,7 @@ namespace boost
#endif
#else
- template
+ template
bool operator>>(std::basic_string& str)
{
str.assign(start, finish);
@@ -731,17 +790,21 @@ namespace boost
#endif
};
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
bool value)
{
- *start = value + lcast_char_constants::zero;
+ typedef typename Traits::int_type int_type;
+ CharT const czero = lcast_char_constants::zero;
+ int_type const zero = Traits::to_int_type(czero);
+ Traits::assign(*start, Traits::to_char_type(zero + value));
finish = start + 1;
return true;
}
- template
- inline bool lexical_stream_limited_src::operator<<(char ch)
+ template
+ inline bool lexical_stream_limited_src::operator<<(
+ char ch)
{
widen_and_assign(start, ch);
finish = start + 1;
@@ -749,8 +812,8 @@ namespace boost
}
#if !defined(DISABLE_WIDE_CHAR_SUPPORT) && !defined(BOOST_NO_INTRINSIC_WCHAR_T)
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
wchar_t ch)
{
widen_and_assign(start, ch);
@@ -759,140 +822,163 @@ namespace boost
}
#endif
- template
- inline bool lexical_stream_limited_src::operator<<(short n)
+ template
+ inline bool lexical_stream_limited_src::operator<<(
+ short n)
{
- start = lcast_put_unsigned(lcast_to_unsigned(n), finish);
+ start = lcast_put_unsigned(lcast_to_unsigned(n), finish);
if(n < 0)
- *--start = lcast_char_constants::minus;
+ {
+ --start;
+ CharT const minus = lcast_char_constants::minus;
+ Traits::assign(*start, minus);
+ }
return true;
}
- template
- inline bool lexical_stream_limited_src::operator<<(int n)
+ template
+ inline bool lexical_stream_limited_src::operator<<(
+ int n)
{
- start = lcast_put_unsigned(lcast_to_unsigned(n), finish);
+ start = lcast_put_unsigned(lcast_to_unsigned(n), finish);
if(n < 0)
- *--start = lcast_char_constants::minus;
+ {
+ --start;
+ CharT const minus = lcast_char_constants::minus;
+ Traits::assign(*start, minus);
+ }
return true;
}
- template
- inline bool lexical_stream_limited_src::operator<<(long n)
+ template
+ inline bool lexical_stream_limited_src::operator<<(
+ long n)
{
- start = lcast_put_unsigned(lcast_to_unsigned(n), finish);
+ start = lcast_put_unsigned(lcast_to_unsigned(n), finish);
if(n < 0)
- *--start = lcast_char_constants::minus;
+ {
+ --start;
+ CharT const minus = lcast_char_constants::minus;
+ Traits::assign(*start, minus);
+ }
return true;
}
#if defined(BOOST_HAS_LONG_LONG)
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
boost::long_long_type n)
{
- start = lcast_put_unsigned(lcast_to_unsigned(n), finish);
+ start = lcast_put_unsigned(lcast_to_unsigned(n), finish);
if(n < 0)
- *--start = lcast_char_constants::minus;
+ {
+ --start;
+ CharT const minus = lcast_char_constants::minus;
+ Traits::assign(*start, minus);
+ }
return true;
}
#elif defined(BOOST_HAS_MS_INT64)
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
__int64 n)
{
- start = lcast_put_unsigned(lcast_to_unsigned(n), finish);
+ start = lcast_put_unsigned(lcast_to_unsigned(n), finish);
if(n < 0)
- *--start = lcast_char_constants::minus;
+ {
+ --start;
+ CharT const minus = lcast_char_constants::minus;
+ Traits::assign(*start, minus);
+ }
return true;
}
#endif
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
unsigned short n)
{
- start = lcast_put_unsigned(+n, finish);
+ start = lcast_put_unsigned(n, finish);
return true;
}
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
unsigned int n)
{
- start = lcast_put_unsigned(n, finish);
+ start = lcast_put_unsigned(n, finish);
return true;
}
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
unsigned long n)
{
- start = lcast_put_unsigned(n, finish);
+ start = lcast_put_unsigned(n, finish);
return true;
}
#if defined(BOOST_HAS_LONG_LONG)
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
boost::ulong_long_type n)
{
- start = lcast_put_unsigned(n, finish);
+ start = lcast_put_unsigned(n, finish);
return true;
}
#elif defined(BOOST_HAS_MS_INT64)
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
unsigned __int64 n)
{
- start = lcast_put_unsigned(n, finish);
+ start = lcast_put_unsigned(n, finish);
return true;
}
#endif
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
float val)
{
return this->lcast_put(val);
}
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
double val)
{
return this->lcast_put(val);
}
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
long double val)
{
return this->lcast_put(val);
}
- template
- inline bool lexical_stream_limited_src::operator<<(
+ template
+ inline bool lexical_stream_limited_src::operator<<(
CharT const* str)
{
start = const_cast(str);
- finish = start + std::char_traits::length(str);
+ finish = start + Traits::length(str);
return true;
}
- template
- inline bool lexical_stream_limited_src::operator>>(
+ template
+ inline bool lexical_stream_limited_src::operator>>(
CharT& output)
{
bool const ok = (finish - start == 1);
if(ok)
- output = *start;
+ Traits::assign(output, *start);
return ok;
}
#ifdef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
- template
- inline bool lexical_stream_limited_src::operator>>(
+ template
+ inline bool lexical_stream_limited_src::operator>>(
std::string& str)
{
str.assign(start, finish);
@@ -900,8 +986,8 @@ namespace boost
}
#ifndef DISABLE_WIDE_CHAR_SUPPORT
- template
- inline bool lexical_stream_limited_src::operator>>(
+ template
+ inline bool lexical_stream_limited_src::operator>>(
std::wstring& str)
{
str.assign(start, finish);
@@ -913,7 +999,7 @@ namespace boost
namespace detail // lcast_streambuf_for_source
{
- // Returns true if optimized stream wrapper uses ostream for formatting.
+ // Returns true if optimized stream wrapper needs ostream for writing.
template
struct lcast_streambuf_for_source
{
@@ -941,7 +1027,7 @@ namespace boost
namespace detail // lcast_streambuf_for_target
{
- // Returns true if optimized stream wrapper use istream for reading.
+ // Returns true if optimized stream wrapper needs istream for reading.
template
struct lcast_streambuf_for_target
{
@@ -1022,17 +1108,20 @@ namespace boost
BOOST_DEDUCED_TYPENAME boost::call_traits::param_type arg,
CharT* buf, std::size_t src_len)
{
+ typedef BOOST_DEDUCED_TYPENAME
+ deduce_char_traits::type traits;
+
typedef BOOST_DEDUCED_TYPENAME boost::mpl::if_c<
lcast_streambuf_for_target::value ||
lcast_streambuf_for_source::value
- , lexical_streambuf
- , lexical_streambuf_fake
+ , std::basic_streambuf
+ , lexical_streambuf_fake
>::type base;
BOOST_DEDUCED_TYPENAME boost::mpl::if_c<
Unlimited
- , detail::lexical_stream
- , detail::lexical_stream_limited_src
+ , detail::lexical_stream
+ , detail::lexical_stream_limited_src
>::type interpreter(buf, buf + src_len);
// The original form, reproduced below, is more elegant
@@ -1079,7 +1168,13 @@ namespace boost
template
Target lexical_cast(Source arg)
{
- detail::lexical_stream interpreter;
+ typedef typename detail::widest_char<
+ BOOST_DEDUCED_TYPENAME detail::stream_char::type
+ , BOOST_DEDUCED_TYPENAME detail::stream_char::type
+ >::type char_type;
+
+ typedef std::char_traits traits;
+ detail::lexical_stream interpreter;
Target result;
if(!(interpreter << arg && interpreter >> result))
@@ -1091,7 +1186,7 @@ namespace boost
}
// Copyright Kevlin Henney, 2000-2005.
-// Copyright Alexander Nasonov, 2006.
+// Copyright Alexander Nasonov, 2006-2007.
//
// Distributed under the Boost Software License, Version 1.0. (See
// accompanying file LICENSE_1_0.txt or copy at
diff --git a/lexical_cast.htm b/lexical_cast.htm
index 2a08541..6e38014 100644
--- a/lexical_cast.htm
+++ b/lexical_cast.htm
@@ -194,7 +194,18 @@ public:
Exception used to indicate runtime lexical_cast
failure.
-
+
+
+
+ #define BOOST_LEXICAL_CAST_ASSUME_C_LOCALE
+
+or,
+
+g++ -DBOOST_LEXICAL_CAST_ASSUME_C_LOCALE ... (gcc on Linux/Unix)
+cl.exe /DBOOST_LEXICAL_CAST_ASSUME_C_LOCALE ... (Visual C++ on Windows)
+
+Eliminate an overhead of std::locale
if your program runs in the "C" locale. If the option is set but a program runs in other locale, lexical_cast
result is unspecified.
+
Q: Why does lexical_cast<int8_t>("127")
throw bad_lexical_cast
?
@@ -219,7 +230,7 @@ public:
[N1973] Kevlin Henney, Beman Dawes, Lexical Conversion Library Proposal for TR2,
N1973 .
[Tuning] Alexander Nasonov, Fine Tuning for lexical_cast,
- Overload #74 ,
+ Overload #74 (PDF ),
August 2006.
diff --git a/lexical_cast_test.cpp b/lexical_cast_test.cpp
index 6ba26a3..04478e0 100644
--- a/lexical_cast_test.cpp
+++ b/lexical_cast_test.cpp
@@ -24,6 +24,9 @@
#include
#include
+#include
+#include
+
#if defined(BOOST_NO_STRINGSTREAM) || \
defined(BOOST_NO_STD_WSTRING) || \
defined(BOOST_NO_STD_LOCALE)
@@ -35,6 +38,16 @@
#define LCAST_TEST_LONGLONG
#endif
+template
+struct my_traits : std::char_traits
+{
+};
+
+template
+struct my_allocator : std::allocator
+{
+};
+
// Test all 65536 values if true:
bool const lcast_test_small_integral_types_completely = false;
@@ -57,15 +70,21 @@ void test_conversion_from_wstring();
void test_conversion_to_wstring();
void test_bad_lexical_cast();
void test_no_whitespace_stripping();
-void test_conversion_from_short();
-void test_conversion_from_ushort();
-void test_conversion_from_int();
-void test_conversion_from_uint();
-void test_conversion_from_long();
-void test_conversion_from_ulong();
+void test_conversion_from_to_short();
+void test_conversion_from_to_ushort();
+void test_conversion_from_to_int();
+void test_conversion_from_to_uint();
+void test_conversion_from_to_long();
+void test_conversion_from_to_ulong();
#ifdef LCAST_TEST_LONGLONG
-void test_conversion_from_longlong();
-void test_conversion_from_ulonglong();
+void test_conversion_from_to_longlong();
+void test_conversion_from_to_ulonglong();
+#endif
+#ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
+void test_traits();
+void test_wtraits();
+void test_allocator();
+void test_wallocator();
#endif
unit_test::test_suite *init_unit_test_suite(int, char *[])
@@ -87,15 +106,21 @@ unit_test::test_suite *init_unit_test_suite(int, char *[])
#endif
suite->add(BOOST_TEST_CASE(test_bad_lexical_cast));
suite->add(BOOST_TEST_CASE(test_no_whitespace_stripping));
- suite->add(BOOST_TEST_CASE(&test_conversion_from_short));
- suite->add(BOOST_TEST_CASE(&test_conversion_from_ushort));
- suite->add(BOOST_TEST_CASE(&test_conversion_from_int));
- suite->add(BOOST_TEST_CASE(&test_conversion_from_uint));
- suite->add(BOOST_TEST_CASE(&test_conversion_from_ulong));
- suite->add(BOOST_TEST_CASE(&test_conversion_from_long));
+ suite->add(BOOST_TEST_CASE(&test_conversion_from_to_short));
+ suite->add(BOOST_TEST_CASE(&test_conversion_from_to_ushort));
+ suite->add(BOOST_TEST_CASE(&test_conversion_from_to_int));
+ suite->add(BOOST_TEST_CASE(&test_conversion_from_to_uint));
+ suite->add(BOOST_TEST_CASE(&test_conversion_from_to_ulong));
+ suite->add(BOOST_TEST_CASE(&test_conversion_from_to_long));
#ifdef LCAST_TEST_LONGLONG
- suite->add(BOOST_TEST_CASE(&test_conversion_from_longlong));
- suite->add(BOOST_TEST_CASE(&test_conversion_from_ulonglong));
+ suite->add(BOOST_TEST_CASE(&test_conversion_from_to_longlong));
+ suite->add(BOOST_TEST_CASE(&test_conversion_from_to_ulonglong));
+ #endif
+ #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
+ suite->add(BOOST_TEST_CASE(&test_traits));
+ suite->add(BOOST_TEST_CASE(&test_wtraits));
+ suite->add(BOOST_TEST_CASE(&test_allocator));
+ suite->add(BOOST_TEST_CASE(&test_wallocator));
#endif
return suite;
@@ -129,10 +154,15 @@ void test_conversion_to_int()
BOOST_CHECK_EQUAL(0, lexical_cast('0'));
BOOST_CHECK_THROW(lexical_cast('A'), bad_lexical_cast);
BOOST_CHECK_EQUAL(1, lexical_cast(1));
+ BOOST_CHECK_EQUAL(1, lexical_cast(1.0));
+
BOOST_CHECK_EQUAL(
(std::numeric_limits::max)(),
lexical_cast((std::numeric_limits::max)()));
- BOOST_CHECK_EQUAL(1, lexical_cast(1.0));
+
+ BOOST_CHECK_EQUAL(
+ (std::numeric_limits::min)(),
+ lexical_cast((std::numeric_limits::min)()));
BOOST_CHECK_THROW(lexical_cast(1.23), bad_lexical_cast);
@@ -407,7 +437,7 @@ void test_conversion_from_integral_to_char(CharT zero)
BOOST_CHECK_THROW(lexical_cast(static_cast(10)), bad_lexical_cast);
- T t = std::numeric_limits::max();
+ T t = (std::numeric_limits::max)();
BOOST_CHECK_THROW(lexical_cast(t), bad_lexical_cast);
}
@@ -427,10 +457,10 @@ void test_conversion_from_integral_to_integral()
BOOST_CHECK(lexical_cast(t) == st);
BOOST_CHECK(lexical_cast(t) == ut);
- t = std::numeric_limits::max();
+ t = (std::numeric_limits::max)();
BOOST_CHECK(lexical_cast(t) == t);
- t = std::numeric_limits::min();
+ t = (std::numeric_limits::min)();
BOOST_CHECK(lexical_cast(t) == t);
}
@@ -442,19 +472,20 @@ void test_conversion_from_integral_to_string(CharT)
T t;
- t = limits::min();
+ t = (limits::min)();
BOOST_CHECK(lexical_cast(t) == to_str(t));
- t = limits::max();
+ t = (limits::max)();
BOOST_CHECK(lexical_cast(t) == to_str(t));
if(limits::digits <= 16 && lcast_test_small_integral_types_completely)
- for(t = 1 + limits::min(); t != limits::max(); ++t)
+ // min and max have already been tested.
+ for(t = 1 + (limits::min)(); t != (limits::max)(); ++t)
BOOST_CHECK(lexical_cast(t) == to_str(t));
else
{
- T const min_val = limits::min();
- T const max_val = limits::max();
+ T const min_val = (limits::min)();
+ T const max_val = (limits::max)();
T const half_max_val = max_val / 2;
T const cnt = lcast_integral_test_counter; // to supress warnings
unsigned int const counter = cnt < half_max_val ? cnt : half_max_val;
@@ -480,20 +511,75 @@ void test_conversion_from_integral_to_string(CharT)
T ten_power = 100;
for(int e = 2; e <= limits::digits10; ++e, ten_power *= 10)
{
- // I believe that (ten_power + 100) never overflows
+ // ten_power + 100 probably never overflows
for(t = ten_power - 100; t != ten_power + 100; ++t)
BOOST_CHECK(lexical_cast(t) == to_str(t));
}
}
}
+template
+void test_conversion_from_string_to_integral(CharT)
+{
+ typedef std::numeric_limits limits;
+
+ T t;
+
+ t = (limits::min)();
+ BOOST_CHECK(lexical_cast(to_str(t)) == t);
+
+ t = (limits::max)();
+ BOOST_CHECK(lexical_cast(to_str(t)) == t);
+
+ if(limits::digits <= 16 && lcast_test_small_integral_types_completely)
+ // min and max have already been tested.
+ for(t = 1 + (limits::min)(); t != (limits::max)(); ++t)
+ BOOST_CHECK(lexical_cast