You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and dots ('.'), can be up to 35 characters long. Letters must be lowercase.
310 lines
11 KiB
310 lines
11 KiB
/////////////////////////////////////////////////////////////////////////////// |
|
/// \file repeat.hpp |
|
/// Contains macros to ease the generation of repetitious code constructs |
|
// |
|
// Copyright 2008 Eric Niebler. 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) |
|
|
|
#ifndef BOOST_PROTO_REPEAT_HPP_EAN_11_24_2008 |
|
#define BOOST_PROTO_REPEAT_HPP_EAN_11_24_2008 |
|
|
|
#include <boost/preprocessor/cat.hpp> |
|
#include <boost/preprocessor/facilities/intercept.hpp> |
|
#include <boost/preprocessor/repetition/enum.hpp> |
|
#include <boost/preprocessor/repetition/enum_params.hpp> |
|
#include <boost/preprocessor/repetition/enum_binary_params.hpp> |
|
#include <boost/preprocessor/iteration/local.hpp> |
|
#include <boost/preprocessor/tuple/elem.hpp> |
|
#include <boost/proto/proto_fwd.hpp> // for BOOST_PROTO_MAX_ARITY |
|
|
|
//////////////////////////////////////////// |
|
/// INTERNAL ONLY |
|
#define BOOST_PROTO_ref_a_aux(Z, N, DATA)\ |
|
boost::ref(BOOST_PP_CAT(proto_a, N)) |
|
|
|
/// \brief Generates a sequence like <tt>typename A0, typename A1, ...</tt> |
|
/// |
|
#define BOOST_PROTO_typename_A(N)\ |
|
BOOST_PP_ENUM_PARAMS(N, typename proto_A) |
|
|
|
/// \brief Generates a sequence like <tt>A0 const &, A1 const &, ...</tt> |
|
/// |
|
#define BOOST_PROTO_A_const_ref(N)\ |
|
BOOST_PP_ENUM_BINARY_PARAMS(N, proto_A, const & BOOST_PP_INTERCEPT) |
|
|
|
/// \brief Generates a sequence like <tt>A0 &, A1 &, ...</tt> |
|
/// |
|
#define BOOST_PROTO_A_ref(N)\ |
|
BOOST_PP_ENUM_BINARY_PARAMS(N, proto_A, & BOOST_PP_INTERCEPT) |
|
|
|
/// \brief Generates a sequence like <tt>A0, A1, ...</tt> |
|
/// |
|
#define BOOST_PROTO_A(N)\ |
|
BOOST_PP_ENUM_PARAMS(N, proto_A) |
|
|
|
/// \brief Generates a sequence like <tt>A0 const, A1 const, ...</tt> |
|
/// |
|
#define BOOST_PROTO_A_const(N)\ |
|
BOOST_PP_ENUM_PARAMS(N, const proto_A) |
|
|
|
/// \brief Generates a sequence like <tt>A0 const &a0, A1 const &a0, ...</tt> |
|
/// |
|
#define BOOST_PROTO_A_const_ref_a(N)\ |
|
BOOST_PP_ENUM_BINARY_PARAMS(N, proto_A, const &proto_a) |
|
|
|
/// \brief Generates a sequence like <tt>A0 &a0, A1 &a0, ...</tt> |
|
/// |
|
#define BOOST_PROTO_A_ref_a(N)\ |
|
BOOST_PP_ENUM_BINARY_PARAMS(N, proto_A, &proto_a) |
|
|
|
/// \brief Generates a sequence like <tt>boost::ref(a0), boost::ref(a1), ...</tt> |
|
/// |
|
#define BOOST_PROTO_ref_a(N)\ |
|
BOOST_PP_ENUM(N, BOOST_PROTO_ref_a_aux, ~) |
|
|
|
/// \brief Generates a sequence like <tt>a0, a1, ...</tt> |
|
/// |
|
#define BOOST_PROTO_a(N)\ |
|
BOOST_PP_ENUM_PARAMS(N, proto_a) |
|
|
|
//////////////////////////////////////////// |
|
/// INTERNAL ONLY |
|
#define BOOST_PROTO_invoke(Z, N, DATA)\ |
|
BOOST_PP_TUPLE_ELEM(5,0,DATA)(N, BOOST_PP_TUPLE_ELEM(5,1,DATA), BOOST_PP_TUPLE_ELEM(5,2,DATA), BOOST_PP_TUPLE_ELEM(5,3,DATA), BOOST_PP_TUPLE_ELEM(5,4,DATA)) |
|
|
|
/// \brief Repeatedly invoke the specified macro. |
|
/// |
|
/// BOOST_PROTO_REPEAT_FROM_TO_EX() is used generate the kind of repetitive code that is typical |
|
/// of DSELs built with Proto. BOOST_PROTO_REPEAT_FROM_TO_EX(FROM, TO, MACRO, typename_A, A, A_a, a) is equivalent to: |
|
/// |
|
/// \code |
|
/// MACRO(FROM, typename_A, A, A_a, a) |
|
/// MACRO(FROM+1, typename_A, A, A_a, a) |
|
/// ... |
|
/// MACRO(TO-1, typename_A, A, A_a, a) |
|
/// \endcode |
|
#define BOOST_PROTO_REPEAT_FROM_TO_EX(FROM, TO, MACRO, typename_A, A, A_a, a)\ |
|
BOOST_PP_REPEAT_FROM_TO(FROM, TO, BOOST_PROTO_invoke, (MACRO, typename_A, A, A_a, a)) |
|
|
|
/// \brief Repeatedly invoke the specified macro. |
|
/// |
|
/// BOOST_PROTO_REPEAT_FROM_TO() is used generate the kind of repetitive code that is typical |
|
/// of DSELs built with Proto. BOOST_PROTO_REPEAT_FROM_TO(FROM, TO, MACRO) is equivalent to: |
|
/// |
|
/// \code |
|
/// MACRO(FROM, BOOST_PROTO_typename_A, BOOST_PROTO_A_const_ref, BOOST_PROTO_A_const_ref_a, BOOST_PROTO_ref_a) |
|
/// MACRO(FROM+1, BOOST_PROTO_typename_A, BOOST_PROTO_A_const_ref, BOOST_PROTO_A_const_ref_a, BOOST_PROTO_ref_a) |
|
/// ... |
|
/// MACRO(TO-1, BOOST_PROTO_typename_A, BOOST_PROTO_A_const_ref, BOOST_PROTO_A_const_ref_a, BOOST_PROTO_ref_a) |
|
/// \endcode |
|
/// |
|
/// Example: |
|
/// |
|
/** \code |
|
|
|
// Generate BOOST_PROTO_MAX_ARITY-1 overloads of the |
|
// following construct() function template. |
|
#define M0(N, typename_A, A_const_ref, A_const_ref_a, ref_a) \ |
|
template<typename T, typename_A(N)> \ |
|
typename proto::result_of::make_expr< \ |
|
proto::tag::function \ |
|
, construct_helper<T> \ |
|
, A_const_ref(N) \ |
|
>::type const \ |
|
construct(A_const_ref_a(N)) \ |
|
{ \ |
|
return proto::make_expr< \ |
|
proto::tag::function \ |
|
>( \ |
|
construct_helper<T>() \ |
|
, ref_a(N) \ |
|
); \ |
|
} |
|
BOOST_PROTO_REPEAT_FROM_TO(1, BOOST_PROTO_MAX_ARITY, M0) |
|
#undef M0 |
|
|
|
\endcode |
|
**/ |
|
/// The above invocation of BOOST_PROTO_REPEAT_FROM_TO() will generate |
|
/// the following code: |
|
/// |
|
/// \code |
|
/// template<typename T, typename A0> |
|
/// typename proto::result_of::make_expr< |
|
/// proto::tag::function |
|
/// , construct_helper<T> |
|
/// , A0 const & |
|
/// >::type const |
|
/// construct(A0 const & a0) |
|
/// { |
|
/// return proto::make_expr< |
|
/// proto::tag::function |
|
/// >( |
|
/// construct_helper<T>() |
|
/// , boost::ref(a0) |
|
/// ); |
|
/// } |
|
/// |
|
/// template<typename T, typename A0, typename A1> |
|
/// typename proto::result_of::make_expr< |
|
/// proto::tag::function |
|
/// , construct_helper<T> |
|
/// , A0 const & |
|
/// , A1 const & |
|
/// >::type const |
|
/// construct(A0 const & a0, A1 const & a1) |
|
/// { |
|
/// return proto::make_expr< |
|
/// proto::tag::function |
|
/// >( |
|
/// construct_helper<T>() |
|
/// , boost::ref(a0) |
|
/// , boost::ref(a1) |
|
/// ); |
|
/// } |
|
/// |
|
/// // ... and so on, up to BOOST_PROTO_MAX_ARITY-1 arguments ... |
|
/// \endcode |
|
#define BOOST_PROTO_REPEAT_FROM_TO(FROM, TO, MACRO)\ |
|
BOOST_PROTO_REPEAT_FROM_TO_EX(FROM, TO, MACRO, BOOST_PROTO_typename_A, BOOST_PROTO_A_const_ref, BOOST_PROTO_A_const_ref_a, BOOST_PROTO_ref_a) |
|
|
|
/// \brief Repeatedly invoke the specified macro. |
|
/// |
|
/// BOOST_PROTO_REPEAT_EX() is used generate the kind of repetitive code that is typical |
|
/// of DSELs built with Proto. BOOST_PROTO_REPEAT_EX(MACRO, typename_A, A, A_a, a) is equivalent to: |
|
/// |
|
/// \code |
|
/// MACRO(1, typename_A, A, A_a, a) |
|
/// MACRO(2, typename_A, A, A_a, a) |
|
/// ... |
|
/// MACRO(BOOST_PROTO_MAX_ARITY, typename_A, A, A_a, a) |
|
/// \endcode |
|
#define BOOST_PROTO_REPEAT_EX(MACRO, typename_A, A, A_a, a)\ |
|
BOOST_PROTO_REPEAT_FROM_TO_EX(1, BOOST_PP_INC(BOOST_PROTO_MAX_ARITY), MACRO, BOOST_PROTO_typename_A, BOOST_PROTO_A_const_ref, BOOST_PROTO_A_const_ref_a, BOOST_PROTO_ref_a) |
|
|
|
/// \brief Repeatedly invoke the specified macro. |
|
/// |
|
/// BOOST_PROTO_REPEAT() is used generate the kind of repetitive code that is typical |
|
/// of DSELs built with Proto. BOOST_PROTO_REPEAT(MACRO) is equivalent to: |
|
/// |
|
/// \code |
|
/// MACRO(1, BOOST_PROTO_typename_A, BOOST_PROTO_A_const_ref, BOOST_PROTO_A_const_ref_a, BOOST_PROTO_ref_a) |
|
/// MACRO(2, BOOST_PROTO_typename_A, BOOST_PROTO_A_const_ref, BOOST_PROTO_A_const_ref_a, BOOST_PROTO_ref_a) |
|
/// ... |
|
/// MACRO(BOOST_PROTO_MAX_ARITY, BOOST_PROTO_typename_A, BOOST_PROTO_A_const_ref, BOOST_PROTO_A_const_ref_a, BOOST_PROTO_ref_a) |
|
/// \endcode |
|
#define BOOST_PROTO_REPEAT(MACRO)\ |
|
BOOST_PROTO_REPEAT_FROM_TO(1, BOOST_PP_INC(BOOST_PROTO_MAX_ARITY), MACRO) |
|
|
|
/// \brief Repeatedly invoke the specified macro. |
|
/// |
|
/// BOOST_PROTO_LOCAL_ITERATE() is used generate the kind of repetitive code that is typical |
|
/// of DSELs built with Proto. This macro causes the user-defined macro BOOST_PROTO_LOCAL_MACRO to |
|
/// be expanded with values in the range specified by BOOST_PROTO_LOCAL_LIMITS. |
|
/// |
|
/// Usage: |
|
/// |
|
/// \code |
|
/// #include BOOST_PROTO_LOCAL_ITERATE() |
|
/// \endcode |
|
/// |
|
/// Example: |
|
/// |
|
/** \code |
|
|
|
// Generate BOOST_PROTO_MAX_ARITY-1 overloads of the |
|
// following construct() function template. |
|
#define BOOST_PROTO_LOCAL_MACRO(N, typename_A, A_const_ref, \ |
|
A_const_ref_a, ref_a) \ |
|
template<typename T, typename_A(N)> \ |
|
typename proto::result_of::make_expr< \ |
|
proto::tag::function \ |
|
, construct_helper<T> \ |
|
, A_const_ref(N) \ |
|
>::type const \ |
|
construct(A_const_ref_a(N)) \ |
|
{ \ |
|
return proto::make_expr< \ |
|
proto::tag::function \ |
|
>( \ |
|
construct_helper<T>() \ |
|
, ref_a(N) \ |
|
); \ |
|
} |
|
#define BOOST_PROTO_LOCAL_LIMITS (1, BOOST_PP_DEC(BOOST_PROTO_MAX_ARITY)) |
|
#include BOOST_PROTO_LOCAL_ITERATE() |
|
|
|
\endcode |
|
**/ |
|
/// The above inclusion of BOOST_PROTO_LOCAL_ITERATE() will generate |
|
/// the following code: |
|
/// |
|
/// \code |
|
/// template<typename T, typename A0> |
|
/// typename proto::result_of::make_expr< |
|
/// proto::tag::function |
|
/// , construct_helper<T> |
|
/// , A0 const & |
|
/// >::type const |
|
/// construct(A0 const & a0) |
|
/// { |
|
/// return proto::make_expr< |
|
/// proto::tag::function |
|
/// >( |
|
/// construct_helper<T>() |
|
/// , boost::ref(a0) |
|
/// ); |
|
/// } |
|
/// |
|
/// template<typename T, typename A0, typename A1> |
|
/// typename proto::result_of::make_expr< |
|
/// proto::tag::function |
|
/// , construct_helper<T> |
|
/// , A0 const & |
|
/// , A1 const & |
|
/// >::type const |
|
/// construct(A0 const & a0, A1 const & a1) |
|
/// { |
|
/// return proto::make_expr< |
|
/// proto::tag::function |
|
/// >( |
|
/// construct_helper<T>() |
|
/// , boost::ref(a0) |
|
/// , boost::ref(a1) |
|
/// ); |
|
/// } |
|
/// |
|
/// // ... and so on, up to BOOST_PROTO_MAX_ARITY-1 arguments ... |
|
/// \endcode |
|
/// |
|
/// If BOOST_PROTO_LOCAL_LIMITS is not defined by the user, it defaults |
|
/// to (1, BOOST_PROTO_MAX_ARITY) |
|
/// |
|
/// At each iteration, BOOST_PROTO_LOCAL_MACRO is invoked with the current |
|
/// iteration number and the following 4 macro parameters: |
|
/// |
|
/// \li BOOST_PROTO_LOCAL_typename_A |
|
/// \li BOOST_PROTO_LOCAL_A |
|
/// \li BOOST_PROTO_LOCAL_A_a |
|
/// \li BOOST_PROTO_LOCAL_a |
|
/// |
|
/// If these macros are not defined by the user, they default respectively to: |
|
/// |
|
/// \li BOOST_PROTO_typename_A |
|
/// \li BOOST_PROTO_A_const_ref |
|
/// \li BOOST_PROTO_A_const_ref_a |
|
/// \li BOOST_PROTO_ref_a |
|
/// |
|
/// After including BOOST_PROTO_LOCAL_ITERATE(), the following macros are |
|
/// automatically undefined: |
|
/// |
|
/// \li BOOST_PROTO_LOCAL_MACRO |
|
/// \li BOOST_PROTO_LOCAL_LIMITS |
|
/// \li BOOST_PROTO_LOCAL_typename_A |
|
/// \li BOOST_PROTO_LOCAL_A |
|
/// \li BOOST_PROTO_LOCAL_A_a |
|
/// \li BOOST_PROTO_LOCAL_a |
|
#define BOOST_PROTO_LOCAL_ITERATE() <boost/proto/detail/local.hpp> |
|
|
|
#endif
|
|
|