1 2 #ifndef BOOST_MPL_SEQUENCE_TAG_HPP_INCLUDED 3 #define BOOST_MPL_SEQUENCE_TAG_HPP_INCLUDED 4 5 // Copyright Aleksey Gurtovoy 2000-2004 6 // 7 // Distributed under the Boost Software License, Version 1.0. 8 // (See accompanying file LICENSE_1_0.txt or copy at 9 // http://www.boost.org/LICENSE_1_0.txt) 10 // 11 // See http://www.boost.org/libs/mpl for documentation. 12 13 // $Id$ 14 // $Date$ 15 // $Revision$ 16 17 #include <boost/mpl/sequence_tag_fwd.hpp> 18 #include <boost/mpl/aux_/has_tag.hpp> 19 #include <boost/mpl/aux_/has_begin.hpp> 20 #include <boost/mpl/aux_/na_spec.hpp> 21 #include <boost/mpl/aux_/is_msvc_eti_arg.hpp> 22 #include <boost/mpl/aux_/config/eti.hpp> 23 #include <boost/mpl/aux_/yes_no.hpp> 24 #include <boost/mpl/aux_/config/workaround.hpp> 25 26 namespace boost { namespace mpl { 27 28 // agurt, 27/nov/02: have to use a simplistic 'sequence_tag' implementation 29 // on MSVC to avoid dreadful "internal structure overflow" error 30 #if BOOST_WORKAROUND(BOOST_MSVC, < 1300) \ 31 || defined(BOOST_MPL_CFG_NO_HAS_XXX) 32 33 template< 34 typename BOOST_MPL_AUX_NA_PARAM(Sequence) 35 > 36 struct sequence_tag 37 { 38 typedef typename Sequence::tag type; 39 }; 40 41 #elif BOOST_WORKAROUND(BOOST_MSVC, == 1300) 42 43 // agurt, 07/feb/03: workaround for what seems to be MSVC 7.0-specific ETI issue 44 45 namespace aux { 46 47 template< bool > 48 struct sequence_tag_impl 49 { 50 template< typename Sequence > struct result_ 51 { 52 typedef typename Sequence::tag type; 53 }; 54 }; 55 56 template<> 57 struct sequence_tag_impl<false> 58 { 59 template< typename Sequence > struct result_ 60 { 61 typedef int type; 62 }; 63 }; 64 65 } // namespace aux 66 67 template< 68 typename BOOST_MPL_AUX_NA_PARAM(Sequence) 69 > 70 struct sequence_tag 71 : aux::sequence_tag_impl< !aux::is_msvc_eti_arg<Sequence>::value > 72 ::template result_<Sequence> 73 { 74 }; 75 76 #else 77 78 namespace aux { 79 80 template< bool has_tag_, bool has_begin_ > 81 struct sequence_tag_impl 82 { 83 // agurt 24/nov/02: MSVC 6.5 gets confused in 'sequence_tag_impl<true>' 84 // specialization below, if we name it 'result_' here 85 template< typename Sequence > struct result2_; 86 }; 87 88 # define AUX_CLASS_SEQUENCE_TAG_SPEC(has_tag, has_begin, result_type) \ 89 template<> struct sequence_tag_impl<has_tag,has_begin> \ 90 { \ 91 template< typename Sequence > struct result2_ \ 92 { \ 93 typedef result_type type; \ 94 }; \ 95 }; \ 96 /**/ 97 98 AUX_CLASS_SEQUENCE_TAG_SPEC(true, true, typename Sequence::tag) 99 AUX_CLASS_SEQUENCE_TAG_SPEC(true, false, typename Sequence::tag) 100 AUX_CLASS_SEQUENCE_TAG_SPEC(false, true, nested_begin_end_tag) 101 AUX_CLASS_SEQUENCE_TAG_SPEC(false, false, non_sequence_tag) 102 103 # undef AUX_CLASS_SEQUENCE_TAG_SPEC 104 105 } // namespace aux 106 107 template< 108 typename BOOST_MPL_AUX_NA_PARAM(Sequence) 109 > 110 struct sequence_tag 111 : aux::sequence_tag_impl< 112 ::boost::mpl::aux::has_tag<Sequence>::value 113 , ::boost::mpl::aux::has_begin<Sequence>::value 114 >::template result2_<Sequence> 115 { 116 }; 117 118 #endif // BOOST_MSVC 119 120 BOOST_MPL_AUX_NA_SPEC(1, sequence_tag) 121 122 }} 123 124 #endif // BOOST_MPL_SEQUENCE_TAG_HPP_INCLUDED 125