248 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			248 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| /* Copyright 2003-2013 Joaquin M Lopez Munoz.
 | |
|  * Distributed under the Boost Software License, Version 1.0.
 | |
|  * (See accompanying file LICENSE_1_0.txt or copy at
 | |
|  * http://www.boost.org/LICENSE_1_0.txt)
 | |
|  *
 | |
|  * See http://www.boost.org/libs/multi_index for library home page.
 | |
|  */
 | |
| 
 | |
| #ifndef BOOST_MULTI_INDEX_DETAIL_VARTEMPL_SUPPORT_HPP
 | |
| #define BOOST_MULTI_INDEX_DETAIL_VARTEMPL_SUPPORT_HPP
 | |
| 
 | |
| #if defined(_MSC_VER)
 | |
| #pragma once
 | |
| #endif
 | |
| 
 | |
| /* Utilities for emulation of variadic template functions. Variadic packs are
 | |
|  * replaced by lists of BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS parameters:
 | |
|  *
 | |
|  *   - typename... Args            --> BOOST_MULTI_INDEX_TEMPLATE_PARAM_PACK
 | |
|  *   - Args&&... args              --> BOOST_MULTI_INDEX_FUNCTION_PARAM_PACK
 | |
|  *   - std::forward<Args>(args)... --> BOOST_MULTI_INDEX_FORWARD_PARAM_PACK
 | |
|  *
 | |
|  * Forwarding emulated with Boost.Move. A template functions foo_imp
 | |
|  * defined in such way accepts *exactly* BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS
 | |
|  * arguments: variable number of arguments is emulated by providing a set of
 | |
|  * overloads foo forwarding to foo_impl with
 | |
|  * 
 | |
|  *   BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL
 | |
|  *   BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL_EXTRA_ARG (initial extra arg)
 | |
|  *
 | |
|  * which fill the extra args with boost::multi_index::detail::noarg's.
 | |
|  * boost::multi_index::detail::vartempl_placement_new works the opposite
 | |
|  * way: it acceps a full a pointer x to Value and a
 | |
|  * BOOST_MULTI_INDEX_FUNCTION_PARAM_PACK and forwards to
 | |
|  * new(x) Value(args) where args is the argument pack after discarding
 | |
|  * noarg's.
 | |
|  *
 | |
|  * Emulation decays to the real thing when the compiler supports variadic
 | |
|  * templates and move semantics natively.
 | |
|  */
 | |
| 
 | |
| #include <boost/config.hpp>
 | |
| 
 | |
| #if defined(BOOST_NO_CXX11_RVALUE_REFERENCES)||\
 | |
|     defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
 | |
| 
 | |
| #include <boost/move/core.hpp>
 | |
| #include <boost/move/utility.hpp>
 | |
| #include <boost/preprocessor/arithmetic/add.hpp>
 | |
| #include <boost/preprocessor/arithmetic/sub.hpp>
 | |
| #include <boost/preprocessor/cat.hpp>
 | |
| #include <boost/preprocessor/control/if.hpp>
 | |
| #include <boost/preprocessor/facilities/empty.hpp>
 | |
| #include <boost/preprocessor/facilities/intercept.hpp>
 | |
| #include <boost/preprocessor/logical/and.hpp>
 | |
| #include <boost/preprocessor/punctuation/comma.hpp>
 | |
| #include <boost/preprocessor/punctuation/comma_if.hpp>
 | |
| #include <boost/preprocessor/repetition/enum.hpp>
 | |
| #include <boost/preprocessor/repetition/enum_params.hpp>
 | |
| #include <boost/preprocessor/repetition/repeat_from_to.hpp>
 | |
| #include <boost/preprocessor/seq/elem.hpp>
 | |
| 
 | |
| #if !defined(BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS)
 | |
| #define BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS 5
 | |
| #endif
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_TEMPLATE_PARAM_PACK                        \
 | |
| BOOST_PP_ENUM_PARAMS(                                                \
 | |
|   BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,typename T)
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_VARTEMPL_ARG(z,n,_)                        \
 | |
| BOOST_FWD_REF(BOOST_PP_CAT(T,n)) BOOST_PP_CAT(t,n)
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_FUNCTION_PARAM_PACK                        \
 | |
| BOOST_PP_ENUM(                                                       \
 | |
|   BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,                             \
 | |
|   BOOST_MULTI_INDEX_VARTEMPL_ARG,~)
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_VARTEMPL_FORWARD_ARG(z,n,_)                \
 | |
| boost::forward<BOOST_PP_CAT(T,n)>(BOOST_PP_CAT(t,n))
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_FORWARD_PARAM_PACK                         \
 | |
| BOOST_PP_ENUM(                                                       \
 | |
|   BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,                             \
 | |
|   BOOST_MULTI_INDEX_VARTEMPL_FORWARD_ARG,~)
 | |
| 
 | |
| namespace boost{namespace multi_index{namespace detail{
 | |
| struct noarg{};
 | |
| }}}
 | |
| 
 | |
| /* call vartempl function without args */
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_NULL_PARAM_PACK                            \
 | |
| BOOST_PP_ENUM_PARAMS(                                                \
 | |
|   BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,                             \
 | |
|   boost::multi_index::detail::noarg() BOOST_PP_INTERCEPT)
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_TEMPLATE_N(n)                              \
 | |
| template<BOOST_PP_ENUM_PARAMS(n,typename T)>
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_TEMPLATE_0(n)
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL_AUX(z,n,data)        \
 | |
| BOOST_PP_IF(n,                                                       \
 | |
|   BOOST_MULTI_INDEX_TEMPLATE_N,                                      \
 | |
|   BOOST_MULTI_INDEX_TEMPLATE_0)(n)                                   \
 | |
| BOOST_PP_SEQ_ELEM(0,data) /* ret */                                  \
 | |
| BOOST_PP_SEQ_ELEM(1,data) /* name_from */ (                          \
 | |
|   BOOST_PP_ENUM(n,BOOST_MULTI_INDEX_VARTEMPL_ARG,~))                 \
 | |
| {                                                                    \
 | |
|   return BOOST_PP_SEQ_ELEM(2,data) /* name_to */ (                   \
 | |
|     BOOST_PP_ENUM(n,BOOST_MULTI_INDEX_VARTEMPL_FORWARD_ARG,~)        \
 | |
|     BOOST_PP_COMMA_IF(                                               \
 | |
|       BOOST_PP_AND(                                                  \
 | |
|         n,BOOST_PP_SUB(BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,n)))    \
 | |
|     BOOST_PP_ENUM_PARAMS(                                            \
 | |
|       BOOST_PP_SUB(BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,n),         \
 | |
|       boost::multi_index::detail::noarg() BOOST_PP_INTERCEPT)        \
 | |
|   );                                                                 \
 | |
| }
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL(                     \
 | |
|   ret,name_from,name_to)                                             \
 | |
| BOOST_PP_REPEAT_FROM_TO(                                             \
 | |
|   0,BOOST_PP_ADD(BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,1),           \
 | |
|   BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL_AUX,                       \
 | |
|   (ret)(name_from)(name_to))
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL_EXTRA_ARG_AUX(       \
 | |
|   z,n,data)                                                          \
 | |
| BOOST_PP_IF(n,                                                       \
 | |
|   BOOST_MULTI_INDEX_TEMPLATE_N,                                      \
 | |
|   BOOST_MULTI_INDEX_TEMPLATE_0)(n)                                   \
 | |
| BOOST_PP_SEQ_ELEM(0,data) /* ret */                                  \
 | |
| BOOST_PP_SEQ_ELEM(1,data) /* name_from */ (                          \
 | |
|   BOOST_PP_SEQ_ELEM(3,data) BOOST_PP_SEQ_ELEM(4,data) /* extra arg */\
 | |
|   BOOST_PP_COMMA_IF(n)                                               \
 | |
|   BOOST_PP_ENUM(n,BOOST_MULTI_INDEX_VARTEMPL_ARG,~))                 \
 | |
| {                                                                    \
 | |
|   return BOOST_PP_SEQ_ELEM(2,data) /* name_to */ (                   \
 | |
|     BOOST_PP_SEQ_ELEM(4,data) /* extra_arg_name */                   \
 | |
|     BOOST_PP_COMMA_IF(n)                                             \
 | |
|     BOOST_PP_ENUM(n,BOOST_MULTI_INDEX_VARTEMPL_FORWARD_ARG,~)        \
 | |
|     BOOST_PP_COMMA_IF(                                               \
 | |
|       BOOST_PP_SUB(BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,n))         \
 | |
|     BOOST_PP_ENUM_PARAMS(                                            \
 | |
|       BOOST_PP_SUB(BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,n),         \
 | |
|       boost::multi_index::detail::noarg() BOOST_PP_INTERCEPT)        \
 | |
|   );                                                                 \
 | |
| }
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL_EXTRA_ARG(           \
 | |
|   ret,name_from,name_to,extra_arg_type,extra_arg_name)               \
 | |
| BOOST_PP_REPEAT_FROM_TO(                                             \
 | |
|   0,BOOST_PP_ADD(BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,1),           \
 | |
|   BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL_EXTRA_ARG_AUX,             \
 | |
|   (ret)(name_from)(name_to)(extra_arg_type)(extra_arg_name))
 | |
| 
 | |
| namespace boost{
 | |
|   
 | |
| namespace multi_index{
 | |
|   
 | |
| namespace detail{
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_VARTEMPL_TO_PLACEMENT_NEW_AUX(z,n,name)    \
 | |
| template<                                                            \
 | |
|   typename Value                                                     \
 | |
|   BOOST_PP_COMMA_IF(n)                                               \
 | |
|   BOOST_PP_ENUM_PARAMS(n,typename T)                                 \
 | |
| >                                                                    \
 | |
| Value* name(                                                         \
 | |
|   Value* x                                                           \
 | |
|   BOOST_PP_COMMA_IF(n)                                               \
 | |
|   BOOST_PP_ENUM(n,BOOST_MULTI_INDEX_VARTEMPL_ARG,~)                  \
 | |
|   BOOST_PP_COMMA_IF(                                                 \
 | |
|     BOOST_PP_SUB(BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,n))           \
 | |
|   BOOST_PP_ENUM_PARAMS(                                              \
 | |
|     BOOST_PP_SUB(BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,n),           \
 | |
|     BOOST_FWD_REF(noarg) BOOST_PP_INTERCEPT))                        \
 | |
| {                                                                    \
 | |
|   return new(x) Value(                                               \
 | |
|     BOOST_PP_ENUM(n,BOOST_MULTI_INDEX_VARTEMPL_FORWARD_ARG,~));      \
 | |
| }
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_VARTEMPL_TO_PLACEMENT_NEW(name)            \
 | |
| BOOST_PP_REPEAT_FROM_TO(                                             \
 | |
|   0,BOOST_PP_ADD(BOOST_MULTI_INDEX_LIMIT_VARTEMPL_ARGS,1),           \
 | |
|   BOOST_MULTI_INDEX_VARTEMPL_TO_PLACEMENT_NEW_AUX,                   \
 | |
|   name)
 | |
| 
 | |
| BOOST_MULTI_INDEX_VARTEMPL_TO_PLACEMENT_NEW(vartempl_placement_new)
 | |
| 
 | |
| #undef BOOST_MULTI_INDEX_VARTEMPL_TO_PLACEMENT_NEW_AUX
 | |
| #undef BOOST_MULTI_INDEX_VARTEMPL_TO_PLACEMENT_NEW
 | |
| 
 | |
| } /* namespace multi_index::detail */
 | |
| 
 | |
| } /* namespace multi_index */
 | |
| 
 | |
| } /* namespace boost */
 | |
| 
 | |
| #else
 | |
| 
 | |
| /* native variadic templates support */
 | |
| 
 | |
| #include <utility>
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_TEMPLATE_PARAM_PACK typename... Args
 | |
| #define BOOST_MULTI_INDEX_FUNCTION_PARAM_PACK Args&&... args
 | |
| #define BOOST_MULTI_INDEX_FORWARD_PARAM_PACK  std::forward<Args>(args)...
 | |
| #define BOOST_MULTI_INDEX_NULL_PARAM_PACK
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL(                     \
 | |
|   ret,name_from,name_to)                                             \
 | |
| template<typename... Args> ret name_from(Args&&... args)             \
 | |
| {                                                                    \
 | |
|   return name_to(std::forward<Args>(args)...);                       \
 | |
| }
 | |
| 
 | |
| #define BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL_EXTRA_ARG(           \
 | |
|   ret,name_from,name_to,extra_arg_type,extra_arg_name)               \
 | |
| template<typename... Args> ret name_from(                            \
 | |
|   extra_arg_type extra_arg_name,Args&&... args)                      \
 | |
| {                                                                    \
 | |
|   return name_to(extra_arg_name,std::forward<Args>(args)...);        \
 | |
| }
 | |
| 
 | |
| namespace boost{
 | |
|   
 | |
| namespace multi_index{
 | |
|   
 | |
| namespace detail{
 | |
| 
 | |
| template<typename Value,typename... Args>
 | |
| Value* vartempl_placement_new(Value*x,Args&&... args)
 | |
| {
 | |
|   return new(x) Value(std::forward<Args>(args)...);
 | |
| }
 | |
| 
 | |
| } /* namespace multi_index::detail */
 | |
| 
 | |
| } /* namespace multi_index */
 | |
| 
 | |
| } /* namespace boost */
 | |
| 
 | |
| #endif
 | |
| #endif
 | 
