349 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable File
		
	
	
			
		
		
	
	
			349 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable File
		
	
	
| 
 | |
| //  (C) Copyright Edward Diener 2011,2012
 | |
| //  Use, modification and distribution are subject to 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).
 | |
| 
 | |
| /*
 | |
| 
 | |
|   The succeeding comments in this file are in doxygen format.
 | |
| 
 | |
| */
 | |
| 
 | |
| /** \file
 | |
| */
 | |
| 
 | |
| #if !defined(BOOST_TTI_HAS_TEMPLATE_HPP)
 | |
| #define BOOST_TTI_HAS_TEMPLATE_HPP
 | |
| 
 | |
| #include <boost/config.hpp>
 | |
| #include <boost/tti/gen/has_template_gen.hpp>
 | |
| #include <boost/preprocessor/config/config.hpp>
 | |
| #include <boost/preprocessor/control/iif.hpp>
 | |
| 
 | |
| #if BOOST_PP_VARIADICS
 | |
| 
 | |
| #include <boost/preprocessor/comparison/equal.hpp>
 | |
| #include <boost/preprocessor/variadic/elem.hpp>
 | |
| #include <boost/preprocessor/variadic/size.hpp>
 | |
| #include <boost/tti/detail/dvm_template_params.hpp>
 | |
| 
 | |
| /// Expands to a metafunction which tests whether an inner class template with a particular name exists.
 | |
| /**
 | |
| 
 | |
|     trait = the name of the metafunction.
 | |
|     ...   = variadic parameters.
 | |
|     
 | |
|             The first variadic parameter is the inner class template name.
 | |
|             
 | |
|             Following variadic parameters are optional.
 | |
|             
 | |
|             If no following variadic parameters exist, then the inner class template 
 | |
|             being introspected must be all template type parameters ( template parameters 
 | |
|             starting with `class` or `typename` ) and any number of template type parameters
 | |
|             can occur.
 | |
|             
 | |
|             If the second variadic parameter is BOOST_PP_NIL and no other variadic 
 | |
|             parameter is given, then just as in the previous case the inner class template 
 | |
|             being introspected must be all template type parameters ( template parameters 
 | |
|             starting with `class` or `typename` ) and any number of template type parameters
 | |
|             can occur. This form is allowed in order to be consistent with using the 
 | |
|             non-variadic form of this macro.
 | |
|             
 | |
|             If the second variadic parameter is a Boost preprocessor library array and no other 
 | |
|             variadic parameter is given, then the inner class template must have its template 
 | |
|             parameters matching the sequence in the tuple portion of the Boost PP array. This 
 | |
|             form is allowed in order to be consistent with using the non-variadic form of this 
 | |
|             macro.
 | |
|             
 | |
|             Otherwise the inner class template must have its template parameters matching the 
 | |
|             sequence of the optional variadic parameters.
 | |
|     
 | |
|     generates a metafunction called "trait" where 'trait' is the first macro parameter.
 | |
|     
 | |
|               template<class BOOST_TTI_TP_T>
 | |
|               struct trait
 | |
|                 {
 | |
|                 static const value = unspecified;
 | |
|                 typedef mpl::bool_<true-or-false> type;
 | |
|                 };
 | |
| 
 | |
|               The metafunction types and return:
 | |
|     
 | |
|                 BOOST_TTI_TP_T = the enclosing type in which to look for our 'name'.
 | |
|                 
 | |
|                 returns = 'value' is true if the 'name' template exists within the enclosing type,
 | |
|                           otherwise 'value' is false.
 | |
|                           
 | |
|     Examples:
 | |
|     
 | |
|     1) Search for an inner class template called 'MyTemplate', with all template type parameters,
 | |
|        nested within the class 'MyClass' using a metafunction name of 'MyMeta'.
 | |
|     
 | |
|        BOOST_TTI_TRAIT_HAS_TEMPLATE(MyMeta,MyTemplate)
 | |
|     
 | |
|        or
 | |
|     
 | |
|        BOOST_TTI_TRAIT_HAS_TEMPLATE(MyMeta,MyTemplate,BOOST_PP_NIL) // Non-variadic macro form
 | |
|     
 | |
|        MyMeta<MyClass>::value
 | |
|     
 | |
|        is a compile time boolean constant which is either 'true' or 'false'
 | |
|        if the nested template exists.
 | |
|     
 | |
|     2) Search for an inner class template called 'MyTemplate', with template parameters 
 | |
|        of 'class T,int x,template<class> class U', nested within the class 'MyClass' 
 | |
|        using a metafunction name of 'MyMeta'.
 | |
|     
 | |
|        BOOST_TTI_TRAIT_HAS_TEMPLATE(MyMeta,MyTemplate,class,int,template<class> class)
 | |
|     
 | |
|        or
 | |
|     
 | |
|        BOOST_TTI_TRAIT_HAS_TEMPLATE(MyMeta,MyTemplate,(3,(class,int,template<class> class))) // Non-variadic macro form
 | |
|     
 | |
|        MyMeta<MyClass>::value
 | |
|     
 | |
|        is a compile time boolean constant which is either 'true' or 'false'
 | |
|        if the nested template exists.
 | |
|     
 | |
| */
 | |
| #define BOOST_TTI_TRAIT_HAS_TEMPLATE(trait,...) \
 | |
|   BOOST_PP_IIF \
 | |
|     ( \
 | |
|     BOOST_PP_EQUAL \
 | |
|       ( \
 | |
|       BOOST_PP_VARIADIC_SIZE(__VA_ARGS__), \
 | |
|       1 \
 | |
|       ), \
 | |
|     BOOST_TTI_DETAIL_VM_TRAIT_HAS_TEMPLATE, \
 | |
|     BOOST_TTI_DETAIL_VM_CHECK_MORE_THAN_TWO \
 | |
|     ) \
 | |
|     (trait,__VA_ARGS__) \
 | |
| /**/
 | |
| 
 | |
| /// Expands to a metafunction which tests whether an inner class template with a particular name exists.
 | |
| /**
 | |
| 
 | |
|     ...   = variadic parameters.
 | |
|     
 | |
|             The first variadic parameter is the inner class template name.
 | |
|             
 | |
|             Following variadic parameters are optional.
 | |
|             
 | |
|             If no following variadic parameters exist, then the inner class template 
 | |
|             being introspected must be all template type parameters ( template parameters 
 | |
|             starting with `class` or `typename` ) and any number of template type parameters
 | |
|             can occur.
 | |
|             
 | |
|             If the second variadic parameter is BOOST_PP_NIL and no other variadic 
 | |
|             parameter is given, then just as in the previous case the inner class template 
 | |
|             being introspected must be all template type parameters ( template parameters 
 | |
|             starting with `class` or `typename` ) and any number of template type parameters
 | |
|             can occur. This form is allowed in order to be consistent with using the 
 | |
|             non-variadic form of this macro.
 | |
|             
 | |
|             If the second variadic parameter is a Boost preprocessor library array and no other 
 | |
|             variadic parameter is given, then the inner class template must have its template 
 | |
|             parameters matching the sequence in the tuple portion of the Boost PP array. This 
 | |
|             form is allowed in order to be consistent with using the non-variadic form of this 
 | |
|             macro.
 | |
|             
 | |
|             Otherwise the inner class template must have its template parameters matching the 
 | |
|             sequence of the optional variadic parameters.
 | |
|     
 | |
|     generates a metafunction called "has_template_'name'" where 'name' is the first variadic parameter.
 | |
|     
 | |
|               template<class BOOST_TTI_TP_T>
 | |
|               struct has_template_'name'
 | |
|                 {
 | |
|                 static const value = unspecified;
 | |
|                 typedef mpl::bool_<true-or-false> type;
 | |
|                 };
 | |
| 
 | |
|               The metafunction types and return:
 | |
|     
 | |
|                 BOOST_TTI_TP_T = the enclosing type in which to look for our 'name'.
 | |
|                 
 | |
|                 returns = 'value' is true if the 'name' template exists within the enclosing type,
 | |
|                           otherwise 'value' is false.
 | |
|                           
 | |
|     Examples:
 | |
|     
 | |
|     1) Search for an inner class template called 'MyTemplate', with all template type parameters,
 | |
|        nested within the class 'MyClass'.
 | |
|     
 | |
|        BOOST_TTI_HAS_TEMPLATE(MyTemplate)
 | |
|     
 | |
|        or
 | |
|     
 | |
|        BOOST_TTI_HAS_TEMPLATE(MyTemplate,BOOST_PP_NIL) // Non-variadic macro form
 | |
|     
 | |
|        has_template_MyTemplate<MyClass>::value
 | |
|     
 | |
|        is a compile time boolean constant which is either 'true' or 'false'
 | |
|        if the nested template exists.
 | |
|     
 | |
|     2) Search for an inner class template called 'MyTemplate' with template parameters 
 | |
|        of 'class T,int x,template<class> class U' nested within the class 'MyClass'.
 | |
|     
 | |
|        BOOST_TTI_HAS_TEMPLATE(MyTemplate,class,int,template<class> class)
 | |
|     
 | |
|        or
 | |
|     
 | |
|        BOOST_TTI_HAS_TEMPLATE(MyTemplate,(3,(class,int,template<class> class))) // Non-variadic macro form
 | |
|     
 | |
|        has_template_MyTemplate<MyClass>::value
 | |
|     
 | |
|        is a compile time boolean constant which is either 'true' or 'false'
 | |
|        if the nested template exists.
 | |
|     
 | |
| */
 | |
| #define BOOST_TTI_HAS_TEMPLATE(...) \
 | |
|   BOOST_TTI_TRAIT_HAS_TEMPLATE \
 | |
|     ( \
 | |
|     BOOST_TTI_HAS_TEMPLATE_GEN \
 | |
|       ( \
 | |
|       BOOST_PP_VARIADIC_ELEM(0,__VA_ARGS__) \
 | |
|       ), \
 | |
|     __VA_ARGS__ \
 | |
|     ) \
 | |
| /**/
 | |
| 
 | |
| #else // !BOOST_PP_VARIADICS
 | |
| 
 | |
| #include <boost/preprocessor/detail/is_binary.hpp>
 | |
| #include <boost/tti/detail/dtemplate.hpp>
 | |
| #include <boost/tti/detail/dtemplate_params.hpp>
 | |
| 
 | |
| /// Expands to a metafunction which tests whether an inner class template with a particular name exists.
 | |
| /**
 | |
| 
 | |
|     trait  = the name of the metafunction.
 | |
|     name   = the inner class template name.
 | |
|     params = If the  parameter is BOOST_PP_NIL the inner class template 
 | |
|              being introspected must be all template type parameters ( template parameters 
 | |
|              starting with `class` or `typename` ) and any number of template type parameters
 | |
|              can occur.
 | |
|             
 | |
|              If the parameter is a Boost preprocessor library array, then the inner class 
 | |
|              template must have its template parameters matching the sequence in the tuple portion 
 | |
|              of the Boost PP array.
 | |
|             
 | |
|              Otherwise a compiler error occurs.
 | |
|     
 | |
|     generates a metafunction called "trait" where 'trait' is the first macro parameter.
 | |
|     
 | |
|               template<class BOOST_TTI_TP_T>
 | |
|               struct trait
 | |
|                 {
 | |
|                 static const value = unspecified;
 | |
|                 typedef mpl::bool_<true-or-false> type;
 | |
|                 };
 | |
| 
 | |
|               The metafunction types and return:
 | |
|     
 | |
|                 BOOST_TTI_TP_T = the enclosing type in which to look for our 'name'.
 | |
|                 
 | |
|                 returns = 'value' is true if the 'name' template exists within the enclosing type,
 | |
|                           otherwise 'value' is false.
 | |
|                           
 | |
|     Examples:
 | |
|     
 | |
|     1) Search for an inner class template called 'MyTemplate', with all template type parameters,
 | |
|        nested within the class 'MyClass' using a metafunction name of 'MyMeta'.
 | |
|     
 | |
|        BOOST_TTI_TRAIT_HAS_TEMPLATE(MyMeta,MyTemplate,BOOST_PP_NIL)
 | |
|     
 | |
|        MyMeta<MyClass>::value
 | |
|     
 | |
|        is a compile time boolean constant which is either 'true' or 'false'
 | |
|        if the nested template exists.
 | |
|     
 | |
|     2) Search for an inner class template called 'MyTemplate', with template parameters 
 | |
|        of 'class T,int x,template<class> class U', nested within the class 'MyClass' 
 | |
|        using a metafunction name of 'MyMeta'.
 | |
|     
 | |
|        BOOST_TTI_TRAIT_HAS_TEMPLATE(MyMeta,MyTemplate,(3,(class,int,template<class> class)))
 | |
|     
 | |
|        MyMeta<MyClass>::value
 | |
|     
 | |
|        is a compile time boolean constant which is either 'true' or 'false'
 | |
|        if the nested template exists.
 | |
|     
 | |
| */
 | |
| #define BOOST_TTI_TRAIT_HAS_TEMPLATE(trait,name,params) \
 | |
|   BOOST_PP_IIF \
 | |
|     ( \
 | |
|     BOOST_PP_IS_BINARY(params), \
 | |
|     BOOST_TTI_DETAIL_TRAIT_HAS_TEMPLATE_CHECK_PARAMS, \
 | |
|     BOOST_TTI_DETAIL_TRAIT_CHECK_IS_NIL \
 | |
|     ) \
 | |
|     (trait,name,params) \
 | |
| /**/
 | |
|   
 | |
| /// Expands to a metafunction which tests whether an inner class template with a particular name exists.
 | |
| /**
 | |
| 
 | |
|     name   = the inner class template name.
 | |
|     params = If the  parameter is BOOST_PP_NIL the inner class template 
 | |
|              being introspected must be all template type parameters ( template parameters 
 | |
|              starting with `class` or `typename` ) and any number of template type parameters
 | |
|              can occur.
 | |
|             
 | |
|              If the parameter is a Boost preprocessor library array, then the inner class 
 | |
|              template must have its template parameters matching the sequence in the tuple portion 
 | |
|              of the Boost PP array.
 | |
|             
 | |
|              Otherwise a compiler error occurs.
 | |
|     
 | |
|     generates a metafunction called "has_template_'name'" where 'name' is the first macro parameter.
 | |
|     
 | |
|               template<class BOOST_TTI_TP_T>
 | |
|               struct trait
 | |
|                 {
 | |
|                 static const value = unspecified;
 | |
|                 typedef mpl::bool_<true-or-false> type;
 | |
|                 };
 | |
| 
 | |
|               The metafunction types and return:
 | |
|     
 | |
|                 BOOST_TTI_TP_T = the enclosing type in which to look for our 'name'.
 | |
|                 
 | |
|                 returns = 'value' is true if the 'name' template exists within the enclosing type,
 | |
|                           otherwise 'value' is false.
 | |
|                           
 | |
|     Examples:
 | |
|     
 | |
|     1) Search for an inner class template called 'MyTemplate', with all template type parameters,
 | |
|        nested within the class 'MyClass'.
 | |
|     
 | |
|        BOOST_TTI_HAS_TEMPLATE(MyTemplate,BOOST_PP_NIL)
 | |
|     
 | |
|        has_template_MyTemplate<MyClass>::value
 | |
|     
 | |
|        is a compile time boolean constant which is either 'true' or 'false'
 | |
|        if the nested template exists.
 | |
|     
 | |
|     2) Search for an inner class template called 'MyTemplate' with template parameters 
 | |
|        of 'class T,int x,template<class> class U' nested within the class 'MyClass'.
 | |
|     
 | |
|        BOOST_TTI_HAS_TEMPLATE(MyTemplate,(3,(class,int,template<class> class)))
 | |
|     
 | |
|        has_template_MyTemplate<MyClass>::value
 | |
|     
 | |
|        is a compile time boolean constant which is either 'true' or 'false'
 | |
|        if the nested template exists.
 | |
|        
 | |
| */
 | |
| #define BOOST_TTI_HAS_TEMPLATE(name,params) \
 | |
|   BOOST_TTI_TRAIT_HAS_TEMPLATE \
 | |
|   ( \
 | |
|   BOOST_TTI_HAS_TEMPLATE_GEN(name), \
 | |
|   name, \
 | |
|   params \
 | |
|   ) \
 | |
| /**/
 | |
| 
 | |
| #endif // BOOST_PP_VARIADICS
 | |
| #endif // BOOST_TTI_HAS_TEMPLATE_HPP
 |