1 /*=============================================================================
2 Copyright (c) 2001-2009 Joel de Guzman
3 Copyright (c) 2005-2006 Dan Marsden
4 Copyright (c) 2009-2011 Christopher Schmidt
6 Distributed under the Boost Software License, Version 1.0. (See accompanying
7 file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
8 ==============================================================================*/
10 #ifndef BOOST_FUSION_ADAPTED_STRUCT_DETAIL_ADAPT_BASE_HPP
11 #define BOOST_FUSION_ADAPTED_STRUCT_DETAIL_ADAPT_BASE_HPP
13 #include <boost/config.hpp>
14 #include <boost/fusion/support/tag_of_fwd.hpp>
16 #include <boost/preprocessor/empty.hpp>
17 #include <boost/preprocessor/stringize.hpp>
18 #include <boost/preprocessor/control/if.hpp>
19 #include <boost/preprocessor/seq/size.hpp>
20 #include <boost/preprocessor/seq/for_each.hpp>
21 #include <boost/preprocessor/seq/for_each_i.hpp>
22 #include <boost/preprocessor/seq/enum.hpp>
23 #include <boost/preprocessor/seq/seq.hpp>
24 #include <boost/preprocessor/tuple/eat.hpp>
25 #include <boost/preprocessor/tuple/elem.hpp>
26 #include <boost/preprocessor/arithmetic/dec.hpp>
27 #include <boost/mpl/bool.hpp>
28 #include <boost/mpl/tag.hpp>
29 #include <boost/mpl/eval_if.hpp>
30 #include <boost/mpl/identity.hpp>
31 #include <boost/type_traits/add_const.hpp>
33 #define BOOST_FUSION_ADAPT_STRUCT_UNPACK_NAME_TEMPLATE_PARAMS(SEQ) \
34 BOOST_PP_SEQ_HEAD(SEQ)<BOOST_PP_SEQ_ENUM(BOOST_PP_SEQ_TAIL(SEQ))> \
37 #define BOOST_FUSION_ADAPT_STRUCT_UNPACK_NAME(SEQ) \
39 BOOST_PP_SEQ_HEAD(SEQ), \
40 BOOST_FUSION_ADAPT_STRUCT_UNPACK_NAME_TEMPLATE_PARAMS, \
41 BOOST_PP_SEQ_HEAD)(BOOST_PP_SEQ_TAIL(SEQ))
43 #define BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS_IMPL_C(R, _, ELEM) \
45 #define BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS_IMPL(SEQ) \
47 BOOST_PP_SEQ_FOR_EACH( \
48 BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS_IMPL_C, \
50 BOOST_PP_SEQ_TAIL(SEQ)))
51 #define BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS(SEQ) \
53 BOOST_PP_SEQ_HEAD(SEQ), \
54 BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS_IMPL, \
55 BOOST_PP_TUPLE_EAT(1))(SEQ)
57 #ifdef BOOST_NO_PARTIAL_SPECIALIZATION_IMPLICIT_DEFAULT_ARGS
58 # define BOOST_FUSION_ADAPT_STRUCT_TAG_OF_SPECIALIZATION( \
59 MODIFIER, TEMPLATE_PARAMS_SEQ, NAME_SEQ, TAG) \
62 BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS(TEMPLATE_PARAMS_SEQ) \
65 BOOST_FUSION_ADAPT_STRUCT_UNPACK_NAME(NAME_SEQ) MODIFIER \
72 # define BOOST_FUSION_ADAPT_STRUCT_TAG_OF_SPECIALIZATION( \
73 MODIFIER, TEMPLATE_PARAMS_SEQ, NAME_SEQ, TAG) \
76 BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS(TEMPLATE_PARAMS_SEQ) \
78 struct tag_of<BOOST_FUSION_ADAPT_STRUCT_UNPACK_NAME(NAME_SEQ) MODIFIER> \
84 #define BOOST_FUSION_ADAPT_STRUCT_BASE_UNPACK_AND_CALL(R,DATA,I,ATTRIBUTE) \
85 BOOST_PP_TUPLE_ELEM(3,0,DATA)( \
86 BOOST_PP_TUPLE_ELEM(3,1,DATA), \
87 BOOST_PP_TUPLE_ELEM(3,2,DATA), \
92 # define BOOST_FUSION_ADAPT_STRUCT_MSVC_REDEFINE_TEMPLATE_PARAM(R,_,ELEM) \
94 # define BOOST_FUSION_ADAPT_STRUCT_MSVC_REDEFINE_TEMPLATE_PARAMS_IMPL(SEQ) \
95 BOOST_PP_SEQ_FOR_EACH( \
96 BOOST_FUSION_ADAPT_STRUCT_MSVC_REDEFINE_TEMPLATE_PARAM, \
98 BOOST_PP_SEQ_TAIL(SEQ))
99 # define BOOST_FUSION_ADAPT_STRUCT_MSVC_REDEFINE_TEMPLATE_PARAMS(SEQ) \
101 BOOST_PP_SEQ_HEAD(SEQ), \
102 BOOST_FUSION_ADAPT_STRUCT_MSVC_REDEFINE_TEMPLATE_PARAMS_IMPL, \
103 BOOST_PP_TUPLE_EAT(1))(SEQ)
105 # define BOOST_FUSION_ADAPT_STRUCT_MSVC_REDEFINE_TEMPLATE_PARAMS(SEQ)
108 #define BOOST_FUSION_ADAPT_STRUCT_C_BASE( \
109 TEMPLATE_PARAMS_SEQ,NAME_SEQ,I,PREFIX,ATTRIBUTE,ATTRIBUTE_TUPEL_SIZE) \
112 BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS(TEMPLATE_PARAMS_SEQ) \
114 struct access::struct_member< \
115 BOOST_FUSION_ADAPT_STRUCT_UNPACK_NAME(NAME_SEQ) \
120 BOOST_PP_TUPLE_ELEM(ATTRIBUTE_TUPEL_SIZE, 0, ATTRIBUTE) \
122 BOOST_FUSION_ADAPT_STRUCT_MSVC_REDEFINE_TEMPLATE_PARAMS( \
123 TEMPLATE_PARAMS_SEQ) \
125 typedef attribute_type type; \
127 template<typename Seq> \
132 typename mpl::eval_if< \
134 , add_const<attribute_type> \
135 , mpl::identity<attribute_type> \
143 return seq.PREFIX() \
144 BOOST_PP_TUPLE_ELEM(ATTRIBUTE_TUPEL_SIZE, 1, ATTRIBUTE); \
150 BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS(TEMPLATE_PARAMS_SEQ) \
152 struct struct_member_name< \
153 BOOST_FUSION_ADAPT_STRUCT_UNPACK_NAME(NAME_SEQ) \
157 typedef char const* type; \
162 return BOOST_PP_STRINGIZE( \
163 BOOST_PP_TUPLE_ELEM(ATTRIBUTE_TUPEL_SIZE,1,ATTRIBUTE)); \
167 #define BOOST_FUSION_ADAPT_STRUCT_BASE( \
168 TEMPLATE_PARAMS_SEQ, \
173 ATTRIBUTES_CALLBACK) \
181 BOOST_FUSION_ADAPT_STRUCT_TAG_OF_SPECIALIZATION( \
182 BOOST_PP_EMPTY(), TEMPLATE_PARAMS_SEQ, NAME_SEQ, TAG) \
183 BOOST_FUSION_ADAPT_STRUCT_TAG_OF_SPECIALIZATION( \
184 const, TEMPLATE_PARAMS_SEQ, NAME_SEQ, TAG) \
187 namespace extension \
190 BOOST_PP_DEC(BOOST_PP_SEQ_SIZE(ATTRIBUTES_SEQ)), \
191 BOOST_PP_SEQ_FOR_EACH_I_R, \
192 BOOST_PP_TUPLE_EAT(4))( \
194 BOOST_FUSION_ADAPT_STRUCT_BASE_UNPACK_AND_CALL, \
195 (ATTRIBUTES_CALLBACK,TEMPLATE_PARAMS_SEQ,NAME_SEQ), \
196 BOOST_PP_SEQ_TAIL(ATTRIBUTES_SEQ)) \
199 BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS( \
200 TEMPLATE_PARAMS_SEQ) \
202 struct struct_size<BOOST_FUSION_ADAPT_STRUCT_UNPACK_NAME(NAME_SEQ)> \
203 : mpl::int_<BOOST_PP_DEC(BOOST_PP_SEQ_SIZE(ATTRIBUTES_SEQ))> \
207 BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS( \
208 TEMPLATE_PARAMS_SEQ) \
210 struct struct_is_view< \
211 BOOST_FUSION_ADAPT_STRUCT_UNPACK_NAME(NAME_SEQ) \
213 : mpl::BOOST_PP_IF(IS_VIEW,true_,false_) \
221 struct sequence_tag; \
224 BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS( \
225 TEMPLATE_PARAMS_SEQ) \
227 struct sequence_tag<BOOST_FUSION_ADAPT_STRUCT_UNPACK_NAME(NAME_SEQ)> \
229 typedef fusion::fusion_sequence_tag type; \
233 BOOST_FUSION_ADAPT_STRUCT_UNPACK_TEMPLATE_PARAMS( \
234 TEMPLATE_PARAMS_SEQ) \
236 struct sequence_tag< \
237 BOOST_FUSION_ADAPT_STRUCT_UNPACK_NAME(NAME_SEQ) const \
240 typedef fusion::fusion_sequence_tag type; \