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_OPERATION_LVAL_HPP 00025 #define IVL_CORE_DETAILS_OPERATION_LVAL_HPP 00026 00027 namespace ivl { 00028 00029 00030 namespace core_details { 00031 00032 template <class R, class A> 00033 class lval_wrapper : public A 00034 { 00035 const R& r; 00036 public: 00037 template <class X> 00038 lval_wrapper(const R& r, const X& x) : r(r), A(r) { } 00039 lval_wrapper(const lval_wrapper& x) : r(x.r), A(x) { } 00040 00041 lval_wrapper& operator=(const lval_wrapper& x) 00042 { A::operator=(x); return *this; } 00043 using A::operator=; 00044 00045 ~lval_wrapper() 00046 { 00047 const_cast<typename types::remove_const<R>::type&>(r).assign(*this); 00048 } 00049 }; 00050 00051 template <class A, class IS_LINEAR> 00052 struct lval_tool {}; 00053 00054 template <class A> 00055 struct lval_tool<A, types::t_false> 00056 { 00057 typedef lval_wrapper<A, typename A::create_similar> type; 00058 static inline type operate(const A& a) { return type(a); } 00059 }; 00060 00061 template <class A> 00062 struct lval_tool<A, types::t_true> 00063 { 00064 typedef const A& type; 00065 static inline const A& operate(const A& a) { a; } 00066 }; 00067 00068 template <class A> 00069 struct resolve_lval 00070 : public lval_tool<A, typename A::is_linear> 00071 { 00072 typedef typename resolve_lval::type type; 00073 }; 00074 00075 } /* namespace core_details */ 00076 00081 template <class A> 00082 inline 00083 typename core_details::resolve_lval<A>::type lval(const A& a) 00084 { 00085 return core_details::resolve_lval<A>::operate(a); 00086 } 00087 00088 }; /*namespace ivl*/ 00089 00090 #endif // IVL_CORE_DETAILS_OPERATION_LVAL_HPP