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 |