os/ossrv/ossrv_pub/boost_apis/boost/lexical_cast.hpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
#ifndef BOOST_LEXICAL_CAST_INCLUDED
sl@0
     2
#define BOOST_LEXICAL_CAST_INCLUDED
sl@0
     3
sl@0
     4
// Boost lexical_cast.hpp header  -------------------------------------------//
sl@0
     5
//
sl@0
     6
// See http://www.boost.org for most recent version including documentation.
sl@0
     7
// See end of this header for rights and permissions.
sl@0
     8
//
sl@0
     9
// what:  lexical_cast custom keyword cast
sl@0
    10
// who:   contributed by Kevlin Henney,
sl@0
    11
//        enhanced with contributions from Terje Slettebų,
sl@0
    12
//        with additional fixes and suggestions from Gennaro Prota,
sl@0
    13
//        Beman Dawes, Dave Abrahams, Daryle Walker, Peter Dimov,
sl@0
    14
//        and other Boosters
sl@0
    15
// when:  November 2000, March 2003, June 2005
sl@0
    16
sl@0
    17
#include <cstddef>
sl@0
    18
#include <string>
sl@0
    19
#include <typeinfo>
sl@0
    20
#include <boost/config.hpp>
sl@0
    21
#include <boost/limits.hpp>
sl@0
    22
#include <boost/throw_exception.hpp>
sl@0
    23
#include <boost/type_traits/is_pointer.hpp>
sl@0
    24
sl@0
    25
#ifdef BOOST_NO_STRINGSTREAM
sl@0
    26
#include <strstream>
sl@0
    27
#else
sl@0
    28
#include <sstream>
sl@0
    29
#endif
sl@0
    30
sl@0
    31
#if defined(BOOST_NO_STRINGSTREAM) || \
sl@0
    32
    defined(BOOST_NO_STD_WSTRING) || \
sl@0
    33
    defined(BOOST_NO_STD_LOCALE) 
sl@0
    34
#define DISABLE_WIDE_CHAR_SUPPORT
sl@0
    35
#endif
sl@0
    36
sl@0
    37
namespace boost
sl@0
    38
{
sl@0
    39
    // exception used to indicate runtime lexical_cast failure
sl@0
    40
    class bad_lexical_cast : public std::bad_cast
sl@0
    41
    {
sl@0
    42
    public:
sl@0
    43
        bad_lexical_cast() :
sl@0
    44
        source(&typeid(void)), target(&typeid(void))
sl@0
    45
        {
sl@0
    46
        }
sl@0
    47
        bad_lexical_cast(
sl@0
    48
            const std::type_info &source_type,
sl@0
    49
            const std::type_info &target_type) :
sl@0
    50
            source(&source_type), target(&target_type)
sl@0
    51
        {
sl@0
    52
        }
sl@0
    53
        const std::type_info &source_type() const
sl@0
    54
        {
sl@0
    55
            return *source;
sl@0
    56
        }
sl@0
    57
        const std::type_info &target_type() const
sl@0
    58
        {
sl@0
    59
            return *target;
sl@0
    60
        }
sl@0
    61
        virtual const char *what() const throw()
sl@0
    62
        {
sl@0
    63
            return "bad lexical cast: "
sl@0
    64
                   "source type value could not be interpreted as target";
sl@0
    65
        }
sl@0
    66
        virtual ~bad_lexical_cast() throw()
sl@0
    67
        {
sl@0
    68
        }
sl@0
    69
    private:
sl@0
    70
        const std::type_info *source;
sl@0
    71
        const std::type_info *target;
sl@0
    72
    };
sl@0
    73
sl@0
    74
    namespace detail // selectors for choosing stream character type
sl@0
    75
    {
sl@0
    76
        template<typename Type>
sl@0
    77
        struct stream_char
sl@0
    78
        {
sl@0
    79
            typedef char type;
sl@0
    80
        };
sl@0
    81
sl@0
    82
        #ifndef DISABLE_WIDE_CHAR_SUPPORT
sl@0
    83
#if !defined(BOOST_NO_INTRINSIC_WCHAR_T)
sl@0
    84
        template<>
sl@0
    85
        struct stream_char<wchar_t>
sl@0
    86
        {
sl@0
    87
            typedef wchar_t type;
sl@0
    88
        };
sl@0
    89
#endif
sl@0
    90
sl@0
    91
        template<>
sl@0
    92
        struct stream_char<wchar_t *>
sl@0
    93
        {
sl@0
    94
            typedef wchar_t type;
sl@0
    95
        };
sl@0
    96
sl@0
    97
        template<>
sl@0
    98
        struct stream_char<const wchar_t *>
sl@0
    99
        {
sl@0
   100
            typedef wchar_t type;
sl@0
   101
        };
sl@0
   102
sl@0
   103
        template<>
sl@0
   104
        struct stream_char<std::wstring>
sl@0
   105
        {
sl@0
   106
            typedef wchar_t type;
sl@0
   107
        };
sl@0
   108
        #endif
sl@0
   109
sl@0
   110
        template<typename TargetChar, typename SourceChar>
sl@0
   111
        struct widest_char
sl@0
   112
        {
sl@0
   113
            typedef TargetChar type;
sl@0
   114
        };
sl@0
   115
sl@0
   116
        template<>
sl@0
   117
        struct widest_char<char, wchar_t>
sl@0
   118
        {
sl@0
   119
            typedef wchar_t type;
sl@0
   120
        };
sl@0
   121
    }
sl@0
   122
    
sl@0
   123
    namespace detail // stream wrapper for handling lexical conversions
sl@0
   124
    {
sl@0
   125
        template<typename Target, typename Source>
sl@0
   126
        class lexical_stream
sl@0
   127
        {
sl@0
   128
        private:
sl@0
   129
            typedef typename widest_char<
sl@0
   130
                typename stream_char<Target>::type,
sl@0
   131
                typename stream_char<Source>::type>::type char_type;
sl@0
   132
sl@0
   133
        public:
sl@0
   134
            lexical_stream()
sl@0
   135
            {
sl@0
   136
                stream.unsetf(std::ios::skipws);
sl@0
   137
sl@0
   138
                if(std::numeric_limits<Target>::is_specialized)
sl@0
   139
                    stream.precision(std::numeric_limits<Target>::digits10 + 1);
sl@0
   140
                else if(std::numeric_limits<Source>::is_specialized)
sl@0
   141
                    stream.precision(std::numeric_limits<Source>::digits10 + 1);
sl@0
   142
            }
sl@0
   143
            ~lexical_stream()
sl@0
   144
            {
sl@0
   145
                #if defined(BOOST_NO_STRINGSTREAM)
sl@0
   146
                stream.freeze(false);
sl@0
   147
                #endif
sl@0
   148
            }
sl@0
   149
            bool operator<<(const Source &input)
sl@0
   150
            {
sl@0
   151
                return !(stream << input).fail();
sl@0
   152
            }
sl@0
   153
            template<typename InputStreamable>
sl@0
   154
            bool operator>>(InputStreamable &output)
sl@0
   155
            {
sl@0
   156
                return !is_pointer<InputStreamable>::value &&
sl@0
   157
                       stream >> output &&
sl@0
   158
                       stream.get() ==
sl@0
   159
#if defined(__GNUC__) && (__GNUC__<3) && defined(BOOST_NO_STD_WSTRING)
sl@0
   160
// GCC 2.9x lacks std::char_traits<>::eof().
sl@0
   161
// We use BOOST_NO_STD_WSTRING to filter out STLport and libstdc++-v3
sl@0
   162
// configurations, which do provide std::char_traits<>::eof().
sl@0
   163
    
sl@0
   164
                           EOF;
sl@0
   165
#else
sl@0
   166
                           std::char_traits<char_type>::eof();
sl@0
   167
#endif
sl@0
   168
            }
sl@0
   169
            bool operator>>(std::string &output)
sl@0
   170
            {
sl@0
   171
                #if defined(BOOST_NO_STRINGSTREAM)
sl@0
   172
                stream << '\0';
sl@0
   173
                #endif
sl@0
   174
                output = stream.str();
sl@0
   175
                return true;
sl@0
   176
            }
sl@0
   177
            #ifndef DISABLE_WIDE_CHAR_SUPPORT
sl@0
   178
            bool operator>>(std::wstring &output)
sl@0
   179
            {
sl@0
   180
                output = stream.str();
sl@0
   181
                return true;
sl@0
   182
            }
sl@0
   183
            #endif
sl@0
   184
        private:
sl@0
   185
            #if defined(BOOST_NO_STRINGSTREAM)
sl@0
   186
            std::strstream stream;
sl@0
   187
            #elif defined(BOOST_NO_STD_LOCALE)
sl@0
   188
            std::stringstream stream;
sl@0
   189
            #else
sl@0
   190
            std::basic_stringstream<char_type> stream;
sl@0
   191
            #endif
sl@0
   192
        };
sl@0
   193
    }
sl@0
   194
sl@0
   195
    #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
sl@0
   196
sl@0
   197
    // call-by-const reference version
sl@0
   198
sl@0
   199
    namespace detail
sl@0
   200
    {
sl@0
   201
        template<class T>
sl@0
   202
        struct array_to_pointer_decay
sl@0
   203
        {
sl@0
   204
            typedef T type;
sl@0
   205
        };
sl@0
   206
sl@0
   207
        template<class T, std::size_t N>
sl@0
   208
        struct array_to_pointer_decay<T[N]>
sl@0
   209
        {
sl@0
   210
            typedef const T * type;
sl@0
   211
        };
sl@0
   212
    }
sl@0
   213
sl@0
   214
    template<typename Target, typename Source>
sl@0
   215
    Target lexical_cast(const Source &arg)
sl@0
   216
    {
sl@0
   217
        typedef typename detail::array_to_pointer_decay<Source>::type NewSource;
sl@0
   218
sl@0
   219
        detail::lexical_stream<Target, NewSource> interpreter;
sl@0
   220
        Target result ;
sl@0
   221
sl@0
   222
        if(!(interpreter << arg && interpreter >> result))
sl@0
   223
            throw_exception(bad_lexical_cast(typeid(NewSource), typeid(Target)));
sl@0
   224
        return result;
sl@0
   225
    }
sl@0
   226
sl@0
   227
    #else
sl@0
   228
sl@0
   229
    // call-by-value fallback version (deprecated)
sl@0
   230
sl@0
   231
    template<typename Target, typename Source>
sl@0
   232
    Target lexical_cast(Source arg)
sl@0
   233
    {
sl@0
   234
        detail::lexical_stream<Target, Source> interpreter;
sl@0
   235
        Target result;
sl@0
   236
sl@0
   237
        if(!(interpreter << arg && interpreter >> result))
sl@0
   238
            throw_exception(bad_lexical_cast(typeid(Source), typeid(Target)));
sl@0
   239
        return result;
sl@0
   240
    }
sl@0
   241
sl@0
   242
    #endif
sl@0
   243
}
sl@0
   244
sl@0
   245
// Copyright Kevlin Henney, 2000-2005. All rights reserved.
sl@0
   246
//
sl@0
   247
// Distributed under the Boost Software License, Version 1.0. (See
sl@0
   248
// accompanying file LICENSE_1_0.txt or copy at
sl@0
   249
// http://www.boost.org/LICENSE_1_0.txt)
sl@0
   250
sl@0
   251
#undef DISABLE_WIDE_CHAR_SUPPORT
sl@0
   252
#endif