/////////////////////////////////////////////////////////////////////////////// /// \file ref.hpp /// Utility for storing a sub-expr by reference // // Copyright 2007 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_ { 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 tag::proto_ref fusion_tag; 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_, 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. template struct ref_ > {}; } #undef BOOST_PROTO_ARG namespace result_of { template struct unref { typedef T type; typedef T &reference; typedef T const &const_reference; }; template struct unref > { typedef T type; typedef T &reference; typedef T &const_reference; }; template struct unref > { typedef T type; typedef T const &reference; typedef T const &const_reference; }; template struct unref { typedef T type; typedef T &reference; typedef T &const_reference; }; template struct unref { typedef T type; typedef T const &reference; typedef T const &const_reference; }; template struct unref { typedef T (&type)[N]; typedef T (&reference)[N]; typedef T (&const_reference)[N]; }; template struct unref { typedef T const (&type)[N]; typedef T const (&reference)[N]; typedef T const (&const_reference)[N]; }; } namespace functional { struct unref { template struct result {}; template struct result : result_of::unref::type> {}; template T &operator()(T &t) const { return t; } template T const &operator()(T const &t) const { return t; } template T &operator()(ref_ &t) const { return t.expr; } template T &operator()(ref_ const &t) const { return t.expr; } }; } functional::unref const unref = {}; }} #if defined(_MSC_VER) && (_MSC_VER >= 1020) # pragma warning(pop) #endif #endif