• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //
2 //  Copyright (c) 2018-2019, Cem Bassoy, cem.bassoy@gmail.com
3 //
4 //  Distributed under the Boost Software License, Version 1.0. (See
5 //  accompanying file LICENSE_1_0.txt or copy at
6 //  http://www.boost.org/LICENSE_1_0.txt)
7 //
8 //  The authors gratefully acknowledge the support of
9 //  Fraunhofer IOSB, Ettlingen, Germany
10 //
11 
12 #ifndef BOOST_UBLAS_TENSOR_OPERATORS_ARITHMETIC_HPP
13 #define BOOST_UBLAS_TENSOR_OPERATORS_ARITHMETIC_HPP
14 
15 #include "expression.hpp"
16 #include "expression_evaluation.hpp"
17 #include "multi_index_utility.hpp"
18 #include "functions.hpp"
19 
20 #include <type_traits>
21 #include <functional>
22 #include <algorithm>
23 
24 namespace boost{
25 namespace numeric{
26 namespace ublas {
27 
28 
29 template<class element_type, class storage_format, class storage_type>
30 class tensor;
31 
32 template<class E>
33 class matrix_expression;
34 
35 
36 template<class E>
37 class vector_expression;
38 
39 }
40 }
41 }
42 
43 #define FIRST_ORDER_OPERATOR_RIGHT(OP, EXPR_TYPE_L, EXPR_TYPE_R) \
44 template<class T, class L, class R> \
45 auto operator OP ( boost::numeric::ublas:: EXPR_TYPE_L <T,L> const& lhs, boost::numeric::ublas:: EXPR_TYPE_R <R> const& rhs) { \
46 	return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), \
47 	  [](auto const& l, auto const& r){ return l OP r; }); \
48 } \
49 
50 FIRST_ORDER_OPERATOR_RIGHT (*, detail:: tensor_expression , vector_expression)
51 FIRST_ORDER_OPERATOR_RIGHT (+, detail:: tensor_expression , vector_expression)
52 FIRST_ORDER_OPERATOR_RIGHT (-, detail:: tensor_expression , vector_expression)
53 FIRST_ORDER_OPERATOR_RIGHT (/, detail:: tensor_expression , vector_expression)
54 
55 FIRST_ORDER_OPERATOR_RIGHT (*, detail:: tensor_expression , matrix_expression)
56 FIRST_ORDER_OPERATOR_RIGHT (+, detail:: tensor_expression , matrix_expression)
57 FIRST_ORDER_OPERATOR_RIGHT (-, detail:: tensor_expression , matrix_expression)
58 FIRST_ORDER_OPERATOR_RIGHT (/, detail:: tensor_expression , matrix_expression)
59 
60 
61 #define FIRST_ORDER_OPERATOR_LEFT(OP, EXPR_TYPE_L, EXPR_TYPE_R) \
62 template<class T, class L, class R> \
63 auto operator OP ( boost::numeric::ublas:: EXPR_TYPE_L <L> const& lhs, boost::numeric::ublas:: EXPR_TYPE_R <T,R> const& rhs) { \
64 	return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), \
65 	  [](auto const& l, auto const& r){ return l OP r; }); \
66 } \
67 
68 FIRST_ORDER_OPERATOR_LEFT (*, vector_expression, detail:: tensor_expression)
69 FIRST_ORDER_OPERATOR_LEFT (+, vector_expression, detail:: tensor_expression)
70 FIRST_ORDER_OPERATOR_LEFT (-, vector_expression, detail:: tensor_expression)
71 FIRST_ORDER_OPERATOR_LEFT (/, vector_expression, detail:: tensor_expression)
72 
73 FIRST_ORDER_OPERATOR_LEFT (*, matrix_expression, detail:: tensor_expression)
74 FIRST_ORDER_OPERATOR_LEFT (+, matrix_expression, detail:: tensor_expression)
75 FIRST_ORDER_OPERATOR_LEFT (-, matrix_expression, detail:: tensor_expression)
76 FIRST_ORDER_OPERATOR_LEFT (/, matrix_expression, detail:: tensor_expression)
77 
78 
79 
80 
81 template<class T, class L, class R>
operator +(boost::numeric::ublas::detail::tensor_expression<T,L> const & lhs,boost::numeric::ublas::detail::tensor_expression<T,R> const & rhs)82 auto operator+( boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
83 	return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), [](auto const& l, auto const& r){ return l + r; });
84 }
85 template<class T, class L, class R>
operator -(boost::numeric::ublas::detail::tensor_expression<T,L> const & lhs,boost::numeric::ublas::detail::tensor_expression<T,R> const & rhs)86 auto operator-( boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
87 	return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), [](auto const& l, auto const& r){ return l - r; });
88 //	return boost::numeric::ublas::detail::make_lambda<T>([&lhs,&rhs](std::size_t i){ return lhs(i) - rhs(i);});
89 }
90 template<class T, class L, class R>
operator *(boost::numeric::ublas::detail::tensor_expression<T,L> const & lhs,boost::numeric::ublas::detail::tensor_expression<T,R> const & rhs)91 auto operator*( boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
92 	return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), [](auto const& l, auto const& r){ return l * r; });
93 }
94 template<class T, class L, class R>
operator /(boost::numeric::ublas::detail::tensor_expression<T,L> const & lhs,boost::numeric::ublas::detail::tensor_expression<T,R> const & rhs)95 auto operator/( boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
96 	return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), [](auto const& l, auto const& r){ return l / r; });
97 }
98 
99 
100 // Overloaded Arithmetic Operators with Scalars
101 template<class T, class R>
operator +(typename T::const_reference lhs,boost::numeric::ublas::detail::tensor_expression<T,R> const & rhs)102 auto operator+(typename T::const_reference lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
103 	return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (rhs(), [lhs](auto const& r){ return lhs + r; });
104 	//return boost::numeric::ublas::detail::make_lambda<T>( [&lhs,&rhs](std::size_t i) {return lhs + rhs(i); } );
105 }
106 template<class T, class R>
operator -(typename T::const_reference lhs,boost::numeric::ublas::detail::tensor_expression<T,R> const & rhs)107 auto operator-(typename T::const_reference lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
108 	return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (rhs(), [lhs](auto const& r){ return lhs - r; });
109 }
110 template<class T, class R>
operator *(typename T::const_reference lhs,boost::numeric::ublas::detail::tensor_expression<T,R> const & rhs)111 auto operator*(typename T::const_reference lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
112 	return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (rhs(), [lhs](auto const& r){ return lhs * r; });
113 }
114 template<class T, class R>
operator /(typename T::const_reference lhs,boost::numeric::ublas::detail::tensor_expression<T,R> const & rhs)115 auto operator/(typename T::const_reference lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
116 	return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (rhs(), [lhs](auto const& r){ return lhs / r; });
117 }
118 
119 
120 template<class T, class L>
operator +(boost::numeric::ublas::detail::tensor_expression<T,L> const & lhs,typename T::const_reference rhs)121 auto operator+(boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, typename T::const_reference rhs) {
122 	return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (lhs(), [rhs] (auto const& l) { return l + rhs; } );
123 }
124 template<class T, class L>
operator -(boost::numeric::ublas::detail::tensor_expression<T,L> const & lhs,typename T::const_reference rhs)125 auto operator-(boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, typename T::const_reference rhs) {
126 	return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (lhs(), [rhs] (auto const& l) { return l - rhs; } );
127 }
128 template<class T, class L>
operator *(boost::numeric::ublas::detail::tensor_expression<T,L> const & lhs,typename T::const_reference rhs)129 auto operator*(boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, typename T::const_reference rhs) {
130 	return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (lhs(), [rhs] (auto const& l) { return l * rhs; } );
131 }
132 template<class T, class L>
operator /(boost::numeric::ublas::detail::tensor_expression<T,L> const & lhs,typename T::const_reference rhs)133 auto operator/(boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, typename T::const_reference rhs) {
134 	return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (lhs(), [rhs] (auto const& l) { return l / rhs; } );
135 }
136 
137 
138 
139 template<class T, class D>
operator +=(T & lhs,const boost::numeric::ublas::detail::tensor_expression<T,D> & expr)140 auto& operator += (T& lhs, const boost::numeric::ublas::detail::tensor_expression<T,D> &expr) {
141 	boost::numeric::ublas::detail::eval(lhs, expr(), [](auto& l, auto const& r) { l+=r; } );
142 	return lhs;
143 }
144 
145 template<class T, class D>
operator -=(T & lhs,const boost::numeric::ublas::detail::tensor_expression<T,D> & expr)146 auto& operator -= (T& lhs, const boost::numeric::ublas::detail::tensor_expression<T,D> &expr) {
147 	boost::numeric::ublas::detail::eval(lhs, expr(), [](auto& l, auto const& r) { l-=r; } );
148 	return lhs;
149 }
150 
151 template<class T, class D>
operator *=(T & lhs,const boost::numeric::ublas::detail::tensor_expression<T,D> & expr)152 auto& operator *= (T& lhs, const boost::numeric::ublas::detail::tensor_expression<T,D> &expr) {
153 	boost::numeric::ublas::detail::eval(lhs, expr(), [](auto& l, auto const& r) { l*=r; } );
154 	return lhs;
155 }
156 
157 template<class T, class D>
operator /=(T & lhs,const boost::numeric::ublas::detail::tensor_expression<T,D> & expr)158 auto& operator /= (T& lhs, const boost::numeric::ublas::detail::tensor_expression<T,D> &expr) {
159 	boost::numeric::ublas::detail::eval(lhs, expr(), [](auto& l, auto const& r) { l/=r; } );
160 	return lhs;
161 }
162 
163 
164 
165 
166 template<class E, class F, class A>
operator +=(boost::numeric::ublas::tensor<E,F,A> & lhs,typename boost::numeric::ublas::tensor<E,F,A>::const_reference r)167 auto& operator += (boost::numeric::ublas::tensor<E,F,A>& lhs, typename boost::numeric::ublas::tensor<E,F,A>::const_reference r) {
168 	boost::numeric::ublas::detail::eval(lhs, [r](auto& l) { l+=r; } );
169 	return lhs;
170 }
171 
172 template<class E, class F, class A>
operator -=(boost::numeric::ublas::tensor<E,F,A> & lhs,typename boost::numeric::ublas::tensor<E,F,A>::const_reference r)173 auto& operator -= (boost::numeric::ublas::tensor<E,F,A>& lhs, typename boost::numeric::ublas::tensor<E,F,A>::const_reference r) {
174 	boost::numeric::ublas::detail::eval(lhs, [r](auto& l) { l-=r; } );
175 	return lhs;
176 }
177 
178 template<class E, class F, class A>
operator *=(boost::numeric::ublas::tensor<E,F,A> & lhs,typename boost::numeric::ublas::tensor<E,F,A>::const_reference r)179 auto& operator *= (boost::numeric::ublas::tensor<E,F,A>& lhs, typename boost::numeric::ublas::tensor<E,F,A>::const_reference r) {
180 	boost::numeric::ublas::detail::eval(lhs, [r](auto& l) { l*=r; } );
181 	return lhs;
182 }
183 
184 template<class E, class F, class A>
operator /=(boost::numeric::ublas::tensor<E,F,A> & lhs,typename boost::numeric::ublas::tensor<E,F,A>::const_reference r)185 auto& operator /= (boost::numeric::ublas::tensor<E,F,A>& lhs, typename boost::numeric::ublas::tensor<E,F,A>::const_reference r) {
186 	boost::numeric::ublas::detail::eval(lhs, [r](auto& l) { l/=r; } );
187 	return lhs;
188 }
189 
190 
191 
192 
193 
194 
195 template<class T, class D>
operator +(const boost::numeric::ublas::detail::tensor_expression<T,D> & lhs)196 auto const& operator +(const boost::numeric::ublas::detail::tensor_expression<T,D>& lhs) {
197 	return lhs;
198 }
199 
200 template<class T, class D>
operator -(boost::numeric::ublas::detail::tensor_expression<T,D> const & lhs)201 auto operator -(boost::numeric::ublas::detail::tensor_expression<T,D> const& lhs) {
202 	return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (lhs(), [] (auto const& l) { return -l; } );
203 }
204 
205 
206 
207 
208 
209 /** @brief Performs a tensor contraction, not an elementwise multiplication
210 	*
211 */
212 
213 template<class tensor_type_left, class tuple_type_left, class tensor_type_right, class tuple_type_right>
operator *(std::pair<tensor_type_left const &,tuple_type_left> lhs,std::pair<tensor_type_right const &,tuple_type_right> rhs)214 auto operator*(
215 		std::pair< tensor_type_left  const&, tuple_type_left  > lhs,
216 		std::pair< tensor_type_right const&, tuple_type_right > rhs)
217 {
218 
219 	using namespace boost::numeric::ublas;
220 
221 	auto const& tensor_left  = lhs.first;
222 	auto const& tensor_right = rhs.first;
223 
224 	auto multi_index_left = lhs.second;
225 	auto multi_index_right = rhs.second;
226 
227 	static constexpr auto num_equal_ind = number_equal_indexes<tuple_type_left, tuple_type_right>::value;
228 
229 	if constexpr ( num_equal_ind == 0  ){
230 		return tensor_left * tensor_right;
231 	}
232 	else if constexpr ( num_equal_ind==std::tuple_size<tuple_type_left>::value && std::is_same<tuple_type_left, tuple_type_right>::value ){
233 
234 		return boost::numeric::ublas::inner_prod( tensor_left, tensor_right );
235 	}
236 	else {
237 		auto array_index_pairs = index_position_pairs(multi_index_left,multi_index_right);
238 		auto index_pairs = array_to_vector(  array_index_pairs  );
239 		return boost::numeric::ublas::prod( tensor_left, tensor_right, index_pairs.first, index_pairs.second );
240 	}
241 
242 }
243 
244 #endif
245