/////////////////////////////////////////////////////////////////////////////// /// \file ref.hpp /// Utility for storing a sub-expr by reference // // 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_REF_HPP_EAN_04_01_2005 #define BOOST_PROTO_REF_HPP_EAN_04_01_2005 #include #include #include #include #include #include #include #if defined(_MSC_VER) && (_MSC_VER >= 1020) # pragma warning(push) # pragma warning(disable : 4510) // default constructor could not be generated # pragma warning(disable : 4512) // assignment operator could not be generated # pragma warning(disable : 4610) // user defined constructor required #endif namespace boost { namespace proto { #define BOOST_PROTO_ARG(z, n, data) \ typedef \ typename Expr::BOOST_PP_CAT(proto_arg, n) \ BOOST_PP_CAT(proto_arg, n); \ /**/ namespace refns_ { /// \brief A simple reference wrapper for a Proto expression type, /// used by expr\<\> to hold children expressions by reference. /// /// ref_\<\> is used by expr\<\> to hold children /// expression types by reference. It forwards enough of the child /// expression's interface so that expr\<\> can handle children /// uniformly regardless of whether it is stored by reference or by /// value. /// /// This type is largely an implementation detail. template struct ref_ { typedef typename Expr::proto_base_expr proto_base_expr; typedef typename Expr::proto_tag proto_tag; typedef typename Expr::proto_args proto_args; typedef typename Expr::proto_arity proto_arity; typedef typename Expr::proto_domain proto_domain; typedef void proto_is_ref_; typedef void proto_is_expr_; typedef Expr proto_derived_expr; BOOST_PP_REPEAT(BOOST_PROTO_MAX_ARITY, BOOST_PROTO_ARG, _) typename mpl::if_c< is_const::value , proto_base_expr const & , proto_base_expr & >::type proto_base() const { return this->expr.proto_base(); } static ref_ make(Expr &expr) { ref_ that = {expr}; return that; } Expr &expr; }; // ref_-to-ref_ is not allowed. this will cause a compile error. /// INTERNAL ONLY template struct ref_ > {}; } #undef BOOST_PROTO_ARG namespace result_of { /// \brief Trait for stripping top-level references /// and reference wrappers. template struct unref { typedef T type; ///< Suitable for return by value typedef T &reference; ///< Suitable for return by reference typedef T const &const_reference; ///< Suitable for return by const reference }; /// \brief Trait for stripping top-level references /// and reference wrappers. template struct unref > { typedef T type; ///< Suitable for return by value typedef T &reference; ///< Suitable for return by reference typedef T &const_reference; ///< Suitable for return by const reference }; /// \brief Trait for stripping top-level references /// and reference wrappers. template struct unref > { typedef T type; ///< Suitable for return by value typedef T const &reference; ///< Suitable for return by reference typedef T const &const_reference; ///< Suitable for return by const reference }; /// \brief Trait for stripping top-level references /// and reference wrappers. template struct unref { typedef T type; ///< Suitable for return by value typedef T &reference; ///< Suitable for return by reference typedef T &const_reference; ///< Suitable for return by const reference }; /// \brief Trait for stripping top-level references /// and reference wrappers. template struct unref { typedef T type; ///< Suitable for return by value typedef T const &reference; ///< Suitable for return by reference typedef T const &const_reference; ///< Suitable for return by const reference }; /// \brief Trait for stripping top-level references /// and reference wrappers. template struct unref { typedef T (&type)[N]; ///< Suitable for return by value typedef T (&reference)[N]; ///< Suitable for return by reference typedef T (&const_reference)[N]; ///< Suitable for return by const reference }; /// \brief Trait for stripping top-level references /// and reference wrappers. template struct unref { typedef T const (&type)[N]; ///< Suitable for return by value typedef T const (&reference)[N]; ///< Suitable for return by reference typedef T const (&const_reference)[N]; ///< Suitable for return by const reference }; } namespace functional { /// \brief A callable PolymorphicFunctionObject equivalent /// to the proto::unref() function that removes /// top-level reference wrappers. struct unref { BOOST_PROTO_CALLABLE() template struct result; template struct result { typedef BOOST_PROTO_UNCVREF(T) uncvref_type; typedef typename result_of::unref::type type; }; /// \brief Remove a top-level ref_\<\> reference wrapper, /// if it exists. /// \param t The object to unwrap /// \return If \c T t is a ref_\<\>, return t.expr. /// Otherwise, return \c t. template T &operator()(T &t) const { return t; } /// \overload /// template T const &operator()(T const &t) const { return t; } /// \overload /// template T &operator()(ref_ &t) const { return t.expr; } /// \overload /// template T &operator()(ref_ const &t) const { return t.expr; } }; } /// \brief Remove a top-level ref_\<\> reference wrapper, if /// it exists. /// \param t The object to unwrap /// \throw nothrow /// \return If \c T t is a ref_\<\>, return t.expr. /// Otherwise, return \c t. template T &unref(T &t BOOST_PROTO_DISABLE_IF_IS_CONST(T)) { return t; } /// \overload /// template T const &unref(T const &t) { return t; } /// \overload /// template T &unref(ref_ &t) { return t.expr; } /// \overload /// template T &unref(ref_ const &t) { return t.expr; } }} #if defined(_MSC_VER) && (_MSC_VER >= 1020) # pragma warning(pop) #endif #endif