ivl 679
|
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