ivl 679
ivl/details/core/keyword_operators/clone.hpp
00001 /* This file is part of the ivl C++ library <http://image.ntua.gr/ivl>.
00002    A C++ template library extending syntax towards mathematical notation.
00003 
00004    Copyright (C) 2012 Yannis Avrithis <iavr@image.ntua.gr>
00005    Copyright (C) 2012 Kimon Kontosis <kimonas@image.ntua.gr>
00006 
00007    ivl is free software; you can redistribute it and/or modify
00008    it under the terms of the GNU Lesser General Public License 
00009    version 3 as published by the Free Software Foundation.
00010 
00011    Alternatively, you can redistribute it and/or modify it under the terms 
00012    of the GNU General Public License version 2 as published by the Free 
00013    Software Foundation.
00014 
00015    ivl is distributed in the hope that it will be useful,
00016    but WITHOUT ANY WARRANTY; without even the implied warranty of
00017    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  
00018    See the GNU General Public License for more details.
00019 
00020    You should have received a copy of the GNU General Public License 
00021    and a copy of the GNU Lesser General Public License along 
00022    with ivl. If not, see <http://www.gnu.org/licenses/>. */
00023 
00024 #ifndef IVL_CORE_DETAILS_RVALUE_HPP
00025 #define IVL_CORE_DETAILS_RVALUE_HPP
00026 
00027 namespace ivl {
00028 
00029 template<class DERIVED>
00030 class rvalue_base
00031 : public types::rvalue_identifier
00032 {
00033 private:
00034         typedef DERIVED derived_type;
00035 
00036 public:
00037         const derived_type& derived() const
00038                 { return static_cast<const derived_type&>(*this); }
00039 
00040         derived_type& derived()
00041                 { return static_cast<derived_type&>(*this); }
00042 };
00043 
00044 namespace core_details {
00045 
00046 template <class A, class IS_LINEAR>
00047 struct rval_tool {};
00048 
00049 template <class A>
00050 struct rval_tool<A, types::t_false>
00051 {
00052         typedef typename A::create_similar type;
00053         static inline type operate(const A& a) { return type(a); }
00054 };
00055 
00056 template <class A>
00057 struct rval_tool<A, types::t_true>
00058 {
00059         typedef const A& type;
00060         static inline const A& operate(const A& a) { a; }
00061 };
00062 
00063 template <class A>
00064 struct resolve_rval
00065 : public rval_tool<A, typename A::is_linear>
00066 {
00067         typedef typename resolve_rval::type type;
00068 };
00069 
00070 /*
00071 template <class A>
00072 struct resolve_is_subreference
00073 : public rval_tool<A, typename types::t_and<typename A::is_writeable,
00074         types::is_base<types::subreference_identifier, A> >::type>
00075 {
00076         typedef typename resolve_rval::type type;
00077 };
00078 */
00079 
00080 } /* namespace core_details */
00081 
00107 template<class A>
00108 typename core_details::resolve_rval<A>::type rval(const A& a)
00109 {
00110         return core_details::resolve_rval<A>::operate(a);
00111 }
00112 
00120 template <class A>
00121 inline A& by_ref(const A& a) { return const_cast<A&>(a); }
00122 
00123 
00124 
00125 namespace core_details {
00126 
00127 template <class R, class A>
00128 class lval_wrapper : public A
00129 {
00130         const R& r;
00131 public:
00132         template <class X>
00133         lval_wrapper(const R& r, const X& x) : r(r), A(r) { }
00134         lval_wrapper(const lval_wrapper& x) : r(x.r), A(x) { }
00135 
00136         lval_wrapper& operator=(const lval_wrapper& x)
00137         { A::operator=(x); return *this; }
00138         using A::operator=;
00139 
00140         ~lval_wrapper()
00141         {
00142                 const_cast<typename types::remove_const<R>::type&>(r).assign(*this);
00143         }
00144 };
00145 
00146 template <class A, class IS_LINEAR>
00147 struct lval_tool {};
00148 
00149 template <class A>
00150 struct lval_tool<A, types::t_false>
00151 {
00152         typedef lval_wrapper<A, typename A::create_similar> type;
00153         static inline type operate(const A& a) { return type(a); }
00154 };
00155 
00156 template <class A>
00157 struct lval_tool<A, types::t_true>
00158 {
00159         typedef const A& type;
00160         static inline const A& operate(const A& a) { a; }
00161 };
00162 
00163 template <class A>
00164 struct resolve_lval
00165 : public lval_tool<A, typename A::is_linear>
00166 {
00167         typedef typename resolve_lval::type type;
00168 };
00169 
00170 } /* namespace core_details */
00171 
00176 template <class A>
00177 inline
00178 typename core_details::resolve_lval<A>::type lval(const A& a)
00179 {
00180         return core_details::resolve_lval<A>::operate(a);
00181 }
00182 
00183 }; /*namespace ivl*/
00184 
00185 #endif // IVL_CORE_DETAILS_RVALUE_HPP
 All Classes Namespaces Files Functions Variables Typedefs Enumerations