os/ossrv/ossrv_pub/boost_apis/boost/smart_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_SMART_CAST_HPP
sl@0
     2
#define BOOST_SMART_CAST_HPP
sl@0
     3
sl@0
     4
// MS compatible compilers support #pragma once
sl@0
     5
#if defined(_MSC_VER) && (_MSC_VER >= 1020)
sl@0
     6
# pragma once
sl@0
     7
#endif
sl@0
     8
sl@0
     9
/////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8
sl@0
    10
// smart_cast.hpp:
sl@0
    11
sl@0
    12
// (C) Copyright 2002 Robert Ramey - http://www.rrsd.com . 
sl@0
    13
// Use, modification and distribution is subject to the Boost Software
sl@0
    14
// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
sl@0
    15
// http://www.boost.org/LICENSE_1_0.txt)
sl@0
    16
sl@0
    17
//  See http://www.boost.org for updates, documentation, and revision history.
sl@0
    18
sl@0
    19
// casting of pointers and references.  
sl@0
    20
sl@0
    21
// In casting between different C++ classes, there are a number of
sl@0
    22
// rules that have to be kept in mind in deciding whether to use
sl@0
    23
// static_cast or dynamic_cast.  
sl@0
    24
sl@0
    25
// a) dynamic casting can only be applied when one of the types is polymorphic
sl@0
    26
// Otherwise static_cast must be used.
sl@0
    27
// b) only dynamic casting can do runtime error checking
sl@0
    28
// use of static_cast is generally un checked even when compiled for debug
sl@0
    29
// c) static_cast would be considered faster than dynamic_cast.
sl@0
    30
sl@0
    31
// If casting is applied to a template parameter, there is no apriori way
sl@0
    32
// to know which of the two casting methods will be permitted or convenient.
sl@0
    33
sl@0
    34
// smart_cast uses C++ type_traits, and program debug mode to select the
sl@0
    35
// most convenient cast to use.
sl@0
    36
sl@0
    37
#include <exception>
sl@0
    38
#include <typeinfo>
sl@0
    39
sl@0
    40
#include <boost/config.hpp>
sl@0
    41
#include <boost/static_assert.hpp>
sl@0
    42
sl@0
    43
#include <boost/type_traits/is_base_and_derived.hpp>
sl@0
    44
#include <boost/type_traits/is_polymorphic.hpp>
sl@0
    45
#include <boost/type_traits/is_pointer.hpp>
sl@0
    46
#include <boost/type_traits/is_reference.hpp>
sl@0
    47
#include <boost/type_traits/is_same.hpp>
sl@0
    48
#include <boost/type_traits/remove_pointer.hpp>
sl@0
    49
#include <boost/type_traits/remove_reference.hpp>
sl@0
    50
sl@0
    51
#include <boost/mpl/eval_if.hpp>
sl@0
    52
#include <boost/mpl/if.hpp>
sl@0
    53
#include <boost/mpl/or.hpp>
sl@0
    54
#include <boost/mpl/and.hpp>
sl@0
    55
#include <boost/mpl/not.hpp>
sl@0
    56
#include <boost/mpl/identity.hpp>
sl@0
    57
sl@0
    58
namespace boost {
sl@0
    59
namespace smart_cast_impl {
sl@0
    60
sl@0
    61
    template<class T>
sl@0
    62
    struct reference {
sl@0
    63
sl@0
    64
        struct polymorphic {
sl@0
    65
sl@0
    66
            struct linear {
sl@0
    67
                template<class U>
sl@0
    68
                 static T cast(U & u){
sl@0
    69
                    return static_cast<T>(u);
sl@0
    70
                }
sl@0
    71
            };
sl@0
    72
sl@0
    73
            struct cross {
sl@0
    74
                 template<class U>
sl@0
    75
                static T cast(U & u){
sl@0
    76
                    return dynamic_cast<T>(u);
sl@0
    77
                }
sl@0
    78
            };
sl@0
    79
sl@0
    80
            template<class U>
sl@0
    81
            static T cast(U & u){
sl@0
    82
                // if we're in debug mode
sl@0
    83
                #if ! defined(NDEBUG)                               \
sl@0
    84
                || defined(__BORLANDC__) && (__BORLANDC__ <= 0x560) \
sl@0
    85
                || defined(__MWERKS__)
sl@0
    86
                    // do a checked dynamic cast
sl@0
    87
                    return cross::cast(u);
sl@0
    88
                #else
sl@0
    89
                    // borland 5.51 chokes here so we can't use it
sl@0
    90
                    // note: if remove_reference isn't function for these types
sl@0
    91
                    // cross casting will be selected this will work but will
sl@0
    92
                    // not be the most efficient method. This will conflict with
sl@0
    93
                    // the original smart_cast motivation.
sl@0
    94
                    typedef BOOST_DEDUCED_TYPENAME mpl::eval_if<
sl@0
    95
                            BOOST_DEDUCED_TYPENAME mpl::and_<
sl@0
    96
                                mpl::not_<is_base_and_derived<
sl@0
    97
                                    BOOST_DEDUCED_TYPENAME remove_reference<T>::type,
sl@0
    98
                                    U
sl@0
    99
                                > >,
sl@0
   100
                                mpl::not_<is_base_and_derived<
sl@0
   101
                                    U,
sl@0
   102
                                    BOOST_DEDUCED_TYPENAME remove_reference<T>::type
sl@0
   103
                                > >
sl@0
   104
                            >,
sl@0
   105
                            // borland chokes w/o full qualification here
sl@0
   106
                            mpl::identity<cross>,
sl@0
   107
                            mpl::identity<linear>
sl@0
   108
                    >::type typex;
sl@0
   109
                    // typex works around gcc 2.95 issue
sl@0
   110
                    return typex::cast(u);
sl@0
   111
                #endif
sl@0
   112
            }
sl@0
   113
        };
sl@0
   114
sl@0
   115
        struct non_polymorphic {
sl@0
   116
            template<class U>
sl@0
   117
             static T cast(U & u){
sl@0
   118
                return static_cast<T>(u);
sl@0
   119
            }
sl@0
   120
        };
sl@0
   121
        template<class U>
sl@0
   122
        static T cast(U & u){
sl@0
   123
            #if defined(__BORLANDC__)
sl@0
   124
                return mpl::eval_if<
sl@0
   125
                    boost::is_polymorphic<U>,
sl@0
   126
                    mpl::identity<polymorphic>,
sl@0
   127
                    mpl::identity<non_polymorphic>
sl@0
   128
                >::type::cast(u);
sl@0
   129
            #else
sl@0
   130
                typedef BOOST_DEDUCED_TYPENAME mpl::eval_if<
sl@0
   131
                    boost::is_polymorphic<U>,
sl@0
   132
                    mpl::identity<polymorphic>,
sl@0
   133
                    mpl::identity<non_polymorphic>
sl@0
   134
                >::type typex;
sl@0
   135
                return typex::cast(u);
sl@0
   136
            #endif
sl@0
   137
        }
sl@0
   138
    };
sl@0
   139
sl@0
   140
    template<class T>
sl@0
   141
    struct pointer {
sl@0
   142
sl@0
   143
        struct polymorphic {
sl@0
   144
            // unfortunately, this below fails to work for virtual base 
sl@0
   145
            // classes.  need has_virtual_base to do this.
sl@0
   146
            // Subject for further study
sl@0
   147
            #if 0
sl@0
   148
            struct linear {
sl@0
   149
                template<class U>
sl@0
   150
                 static T cast(U * u){
sl@0
   151
                    return static_cast<T>(u);
sl@0
   152
                }
sl@0
   153
            };
sl@0
   154
sl@0
   155
            struct cross {
sl@0
   156
                template<class U>
sl@0
   157
                static T cast(U * u){
sl@0
   158
                    T tmp = dynamic_cast<T>(u);
sl@0
   159
                    #ifndef NDEBUG
sl@0
   160
                        if ( tmp == 0 ) throw std::bad_cast();
sl@0
   161
                    #endif
sl@0
   162
                    return tmp;
sl@0
   163
                }
sl@0
   164
            };
sl@0
   165
sl@0
   166
            template<class U>
sl@0
   167
            static T cast(U * u){
sl@0
   168
                // if we're in debug mode
sl@0
   169
                #if ! defined(NDEBUG) || defined(__BORLANDC__) && (__BORLANDC__ <= 0x560)
sl@0
   170
                    // do a checked dynamic cast
sl@0
   171
                    return cross::cast(u);
sl@0
   172
                #else
sl@0
   173
                    // borland 5.51 chokes here so we can't use it
sl@0
   174
                    // note: if remove_pointer isn't function for these types
sl@0
   175
                    // cross casting will be selected this will work but will
sl@0
   176
                    // not be the most efficient method. This will conflict with
sl@0
   177
                    // the original smart_cast motivation.
sl@0
   178
                    typedef
sl@0
   179
                        BOOST_DEDUCED_TYPENAME mpl::eval_if<
sl@0
   180
                            BOOST_DEDUCED_TYPENAME mpl::and_<
sl@0
   181
                                mpl::not_<is_base_and_derived<
sl@0
   182
                                    BOOST_DEDUCED_TYPENAME remove_pointer<T>::type,
sl@0
   183
                                    U
sl@0
   184
                                > >,
sl@0
   185
                                mpl::not_<is_base_and_derived<
sl@0
   186
                                    U,
sl@0
   187
                                    BOOST_DEDUCED_TYPENAME remove_pointer<T>::type
sl@0
   188
                                > >
sl@0
   189
                            >,
sl@0
   190
                            // borland chokes w/o full qualification here
sl@0
   191
                            mpl::identity<cross>,
sl@0
   192
                            mpl::identity<linear>
sl@0
   193
                        >::type typex;
sl@0
   194
                    return typex::cast(u);
sl@0
   195
                #endif
sl@0
   196
            }
sl@0
   197
            #else
sl@0
   198
            template<class U>
sl@0
   199
            static T cast(U * u){
sl@0
   200
                T tmp = dynamic_cast<T>(u);
sl@0
   201
                #ifndef NDEBUG
sl@0
   202
                    if ( tmp == 0 ) throw std::bad_cast();
sl@0
   203
                #endif
sl@0
   204
                return tmp;
sl@0
   205
            }
sl@0
   206
            #endif
sl@0
   207
        };
sl@0
   208
sl@0
   209
        struct non_polymorphic {
sl@0
   210
            template<class U>
sl@0
   211
             static T cast(U * u){
sl@0
   212
                return static_cast<T>(u);
sl@0
   213
            }
sl@0
   214
        };
sl@0
   215
sl@0
   216
        template<class U>
sl@0
   217
        static T cast(U * u){
sl@0
   218
            #if defined(__BORLANDC__)
sl@0
   219
                return mpl::eval_if<
sl@0
   220
                    boost::is_polymorphic<U>,
sl@0
   221
                    mpl::identity<polymorphic>,
sl@0
   222
                    mpl::identity<non_polymorphic>
sl@0
   223
                >::type::cast(u);
sl@0
   224
            #else
sl@0
   225
                typedef BOOST_DEDUCED_TYPENAME mpl::eval_if<
sl@0
   226
                    boost::is_polymorphic<U>,
sl@0
   227
                    mpl::identity<polymorphic>,
sl@0
   228
                    mpl::identity<non_polymorphic>
sl@0
   229
                >::type typex;
sl@0
   230
                return typex::cast(u);
sl@0
   231
            #endif
sl@0
   232
        }
sl@0
   233
sl@0
   234
    };
sl@0
   235
sl@0
   236
    template<class TPtr>
sl@0
   237
    struct void_pointer {
sl@0
   238
        template<class UPtr>
sl@0
   239
        static TPtr cast(UPtr uptr){
sl@0
   240
            return static_cast<TPtr>(uptr);
sl@0
   241
        }
sl@0
   242
    };
sl@0
   243
sl@0
   244
    template<class T>
sl@0
   245
    struct error {
sl@0
   246
        // if we get here, its because we are using one argument in the
sl@0
   247
        // cast on a system which doesn't support partial template 
sl@0
   248
        // specialization
sl@0
   249
        template<class U>
sl@0
   250
        static T cast(U u){
sl@0
   251
            BOOST_STATIC_ASSERT(sizeof(T)==0);
sl@0
   252
            return * static_cast<T *>(NULL);
sl@0
   253
        }
sl@0
   254
    };
sl@0
   255
sl@0
   256
} // smart_cast_impl
sl@0
   257
sl@0
   258
// this implements:
sl@0
   259
// smart_cast<Target *, Source *>(Source * s)
sl@0
   260
// smart_cast<Target &, Source &>(s)
sl@0
   261
// note that it will fail with
sl@0
   262
// smart_cast<Target &>(s)
sl@0
   263
template<class T, class U>
sl@0
   264
T smart_cast(U u) {
sl@0
   265
    typedef
sl@0
   266
        BOOST_DEDUCED_TYPENAME mpl::eval_if<
sl@0
   267
            BOOST_DEDUCED_TYPENAME mpl::or_<
sl@0
   268
                boost::is_same<void *, U>,
sl@0
   269
                boost::is_same<void *, T>,
sl@0
   270
                boost::is_same<const void *, U>,
sl@0
   271
                boost::is_same<const void *, T>
sl@0
   272
            >,
sl@0
   273
            mpl::identity<smart_cast_impl::void_pointer<T> >,
sl@0
   274
        // else
sl@0
   275
        BOOST_DEDUCED_TYPENAME mpl::eval_if<boost::is_pointer<U>,
sl@0
   276
            mpl::identity<smart_cast_impl::pointer<T> >,
sl@0
   277
        // else
sl@0
   278
        BOOST_DEDUCED_TYPENAME mpl::eval_if<boost::is_reference<U>,
sl@0
   279
            mpl::identity<smart_cast_impl::reference<T> >,
sl@0
   280
        // else
sl@0
   281
            mpl::identity<smart_cast_impl::error<T>
sl@0
   282
        >
sl@0
   283
        >
sl@0
   284
        >
sl@0
   285
        >::type typex;
sl@0
   286
    return typex::cast(u);
sl@0
   287
}
sl@0
   288
sl@0
   289
// this implements:
sl@0
   290
// smart_cast_reference<Target &>(Source & s)
sl@0
   291
template<class T, class U>
sl@0
   292
T smart_cast_reference(U & u) {
sl@0
   293
    return smart_cast_impl::reference<T>::cast(u);
sl@0
   294
}
sl@0
   295
sl@0
   296
} // namespace boost
sl@0
   297
sl@0
   298
#endif // BOOST_SMART_CAST_HPP