annotate DEPENDENCIES/generic/include/boost/spirit/home/phoenix/statement/detail/switch.hpp @ 39:12d422dd6992

Merge
author Chris Cannam
date Wed, 06 Aug 2014 17:44:04 +0100
parents 2665513ce2d3
children
rev   line source
Chris@16 1 /*=============================================================================
Chris@16 2 Copyright (c) 2001-2007 Joel de Guzman
Chris@16 3
Chris@16 4 Distributed under the Boost Software License, Version 1.0. (See accompanying
Chris@16 5 file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
Chris@16 6 ==============================================================================*/
Chris@16 7 #ifndef PHOENIX_STATEMENT_DETAIL_SWITCH_HPP
Chris@16 8 #define PHOENIX_STATEMENT_DETAIL_SWITCH_HPP
Chris@16 9
Chris@16 10 #include <boost/spirit/home/phoenix/core/nothing.hpp>
Chris@16 11 #include <boost/fusion/include/vector.hpp>
Chris@16 12 #include <boost/fusion/include/as_vector.hpp>
Chris@16 13 #include <boost/fusion/include/push_back.hpp>
Chris@16 14 #include <boost/fusion/include/push_front.hpp>
Chris@16 15 #include <boost/fusion/include/begin.hpp>
Chris@16 16 #include <boost/fusion/include/size.hpp>
Chris@16 17 #include <boost/fusion/include/value_of.hpp>
Chris@16 18 #include <boost/fusion/include/is_sequence.hpp>
Chris@16 19 #include <boost/mpl/identity.hpp>
Chris@16 20 #include <boost/mpl/bool.hpp>
Chris@16 21 #include <boost/mpl/eval_if.hpp>
Chris@16 22 #include <boost/mpl/if.hpp>
Chris@16 23
Chris@16 24 namespace boost { namespace phoenix
Chris@16 25 {
Chris@16 26
Chris@16 27 template <typename Actor, typename K, K Value>
Chris@16 28 struct switch_case;
Chris@16 29
Chris@16 30 template <typename Actor>
Chris@16 31 struct default_case;
Chris@16 32
Chris@16 33 namespace detail
Chris@16 34 {
Chris@16 35 template <typename T>
Chris@16 36 struct is_default_case : mpl::bool_<T::is_default> {};
Chris@16 37
Chris@16 38 template <typename A0, typename A1>
Chris@16 39 struct compose_case_a
Chris@16 40 {
Chris@16 41 // here, A0 and A1 are both switch cases
Chris@16 42 typedef typename
Chris@16 43 mpl::if_<
Chris@16 44 is_default_case<A1>
Chris@16 45 , fusion::vector<actor<A1>, actor<A0> >
Chris@16 46 , fusion::vector<actor<A0>, actor<A1> >
Chris@16 47 >::type
Chris@16 48 type;
Chris@16 49
Chris@16 50 static type
Chris@16 51 eval(A0 const& _0, A1 const& _1, mpl::false_)
Chris@16 52 {
Chris@16 53 return type(_0, _1);
Chris@16 54 }
Chris@16 55
Chris@16 56 static type
Chris@16 57 eval(A0 const& _0, A1 const& _1, mpl::true_)
Chris@16 58 {
Chris@16 59 return type(_1, _0);
Chris@16 60 }
Chris@16 61
Chris@16 62 static type
Chris@16 63 eval(A0 const& _0, A1 const& _1)
Chris@16 64 {
Chris@16 65 return eval(_0, _1, is_default_case<A1>());
Chris@16 66 }
Chris@16 67 };
Chris@16 68
Chris@16 69 template <typename Seq, typename Case>
Chris@16 70 struct compose_case_b
Chris@16 71 {
Chris@16 72 typedef typename fusion::result_of::as_vector<
Chris@16 73 typename mpl::eval_if<
Chris@16 74 is_default_case<Case>
Chris@16 75 , fusion::result_of::push_front<Seq const, actor<Case> >
Chris@16 76 , fusion::result_of::push_back<Seq const, actor<Case> >
Chris@16 77 >::type>::type
Chris@16 78 type;
Chris@16 79
Chris@16 80 static type
Chris@16 81 eval(Seq const& seq, Case const& case_, mpl::false_)
Chris@16 82 {
Chris@16 83 return fusion::as_vector(
Chris@16 84 fusion::push_back(seq, actor<Case>(case_)));
Chris@16 85 }
Chris@16 86
Chris@16 87 static type
Chris@16 88 eval(Seq const& seq, Case const& case_, mpl::true_)
Chris@16 89 {
Chris@16 90 return fusion::as_vector(
Chris@16 91 fusion::push_front(seq, actor<Case>(case_)));
Chris@16 92 }
Chris@16 93
Chris@16 94 static type
Chris@16 95 eval(Seq const& seq, Case const& case_)
Chris@16 96 {
Chris@16 97 return eval(seq, case_, is_default_case<Case>());
Chris@16 98 }
Chris@16 99 };
Chris@16 100
Chris@16 101 template <typename Cases>
Chris@16 102 struct ensure_default
Chris@16 103 {
Chris@16 104 typedef
Chris@16 105 is_default_case<
Chris@16 106 typename fusion::result_of::value_of<
Chris@16 107 typename fusion::result_of::begin<Cases>::type
Chris@16 108 >::type
Chris@16 109 >
Chris@16 110 is_default_case_;
Chris@16 111
Chris@16 112 typedef typename
Chris@16 113 mpl::eval_if<
Chris@16 114 is_default_case_
Chris@16 115 , mpl::identity<Cases>
Chris@16 116 , fusion::result_of::push_front<
Chris@16 117 Cases const, actor<default_case<actor<null_actor> > > >
Chris@16 118 >::type
Chris@16 119 type;
Chris@16 120
Chris@16 121 static type
Chris@16 122 eval(Cases const& cases, mpl::false_);
Chris@16 123
Chris@16 124 static type
Chris@16 125 eval(Cases const& cases, mpl::true_)
Chris@16 126 {
Chris@16 127 return cases;
Chris@16 128 }
Chris@16 129
Chris@16 130 static type
Chris@16 131 eval(Cases const& cases)
Chris@16 132 {
Chris@16 133 return eval(cases, is_default_case_());
Chris@16 134 }
Chris@16 135 };
Chris@16 136
Chris@16 137 template <typename Cond, typename Cases>
Chris@16 138 struct switch_composite
Chris@16 139 {
Chris@16 140 BOOST_STATIC_ASSERT(fusion::traits::is_sequence<Cases>::value);
Chris@16 141 typedef ensure_default<Cases> ensure_default_;
Chris@16 142
Chris@16 143 typedef typename
Chris@16 144 fusion::result_of::as_vector<
Chris@16 145 typename fusion::result_of::push_front<
Chris@16 146 typename ensure_default_::type, Cond>::type
Chris@16 147 >::type
Chris@16 148 tuple_type;
Chris@16 149
Chris@16 150 typedef
Chris@16 151 composite<
Chris@16 152 detail::switch_eval<fusion::result_of::size<tuple_type>::value-2>
Chris@16 153 , tuple_type>
Chris@16 154 type;
Chris@16 155
Chris@16 156 static type
Chris@16 157 eval(Cond const& cond, Cases const& cases)
Chris@16 158 {
Chris@16 159 return fusion::as_vector(
Chris@16 160 fusion::push_front(ensure_default_::eval(cases), cond));
Chris@16 161 }
Chris@16 162 };
Chris@16 163
Chris@16 164 template <typename Cond, typename Cases>
Chris@16 165 struct switch_composite_actor
Chris@16 166 {
Chris@16 167 typedef actor<typename switch_composite<Cond, Cases>::type> type;
Chris@16 168 };
Chris@16 169 }
Chris@16 170 }}
Chris@16 171
Chris@16 172 #endif