operators_arithmetic.hpp 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  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. #ifndef BOOST_UBLAS_TENSOR_OPERATORS_ARITHMETIC_HPP
  12. #define BOOST_UBLAS_TENSOR_OPERATORS_ARITHMETIC_HPP
  13. #include "expression.hpp"
  14. #include "expression_evaluation.hpp"
  15. #include "multi_index_utility.hpp"
  16. #include "functions.hpp"
  17. #include <type_traits>
  18. #include <functional>
  19. #include <algorithm>
  20. namespace boost{
  21. namespace numeric{
  22. namespace ublas {
  23. template<class element_type, class storage_format, class storage_type>
  24. class tensor;
  25. template<class E>
  26. class matrix_expression;
  27. template<class E>
  28. class vector_expression;
  29. }
  30. }
  31. }
  32. #define FIRST_ORDER_OPERATOR_RIGHT(OP, EXPR_TYPE_L, EXPR_TYPE_R) \
  33. template<class T, class L, class R> \
  34. auto operator OP ( boost::numeric::ublas:: EXPR_TYPE_L <T,L> const& lhs, boost::numeric::ublas:: EXPR_TYPE_R <R> const& rhs) { \
  35. return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), \
  36. [](auto const& l, auto const& r){ return l OP r; }); \
  37. } \
  38. FIRST_ORDER_OPERATOR_RIGHT (*, detail:: tensor_expression , vector_expression)
  39. FIRST_ORDER_OPERATOR_RIGHT (+, detail:: tensor_expression , vector_expression)
  40. FIRST_ORDER_OPERATOR_RIGHT (-, detail:: tensor_expression , vector_expression)
  41. FIRST_ORDER_OPERATOR_RIGHT (/, detail:: tensor_expression , vector_expression)
  42. FIRST_ORDER_OPERATOR_RIGHT (*, detail:: tensor_expression , matrix_expression)
  43. FIRST_ORDER_OPERATOR_RIGHT (+, detail:: tensor_expression , matrix_expression)
  44. FIRST_ORDER_OPERATOR_RIGHT (-, detail:: tensor_expression , matrix_expression)
  45. FIRST_ORDER_OPERATOR_RIGHT (/, detail:: tensor_expression , matrix_expression)
  46. #define FIRST_ORDER_OPERATOR_LEFT(OP, EXPR_TYPE_L, EXPR_TYPE_R) \
  47. template<class T, class L, class R> \
  48. auto operator OP ( boost::numeric::ublas:: EXPR_TYPE_L <L> const& lhs, boost::numeric::ublas:: EXPR_TYPE_R <T,R> const& rhs) { \
  49. return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), \
  50. [](auto const& l, auto const& r){ return l OP r; }); \
  51. } \
  52. FIRST_ORDER_OPERATOR_LEFT (*, vector_expression, detail:: tensor_expression)
  53. FIRST_ORDER_OPERATOR_LEFT (+, vector_expression, detail:: tensor_expression)
  54. FIRST_ORDER_OPERATOR_LEFT (-, vector_expression, detail:: tensor_expression)
  55. FIRST_ORDER_OPERATOR_LEFT (/, vector_expression, detail:: tensor_expression)
  56. FIRST_ORDER_OPERATOR_LEFT (*, matrix_expression, detail:: tensor_expression)
  57. FIRST_ORDER_OPERATOR_LEFT (+, matrix_expression, detail:: tensor_expression)
  58. FIRST_ORDER_OPERATOR_LEFT (-, matrix_expression, detail:: tensor_expression)
  59. FIRST_ORDER_OPERATOR_LEFT (/, matrix_expression, detail:: tensor_expression)
  60. template<class T, class L, class R>
  61. auto operator+( boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
  62. return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), [](auto const& l, auto const& r){ return l + r; });
  63. }
  64. template<class T, class L, class R>
  65. auto operator-( boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
  66. return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), [](auto const& l, auto const& r){ return l - r; });
  67. // return boost::numeric::ublas::detail::make_lambda<T>([&lhs,&rhs](std::size_t i){ return lhs(i) - rhs(i);});
  68. }
  69. template<class T, class L, class R>
  70. auto operator*( boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
  71. return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), [](auto const& l, auto const& r){ return l * r; });
  72. }
  73. template<class T, class L, class R>
  74. auto operator/( boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
  75. return boost::numeric::ublas::detail::make_binary_tensor_expression<T> (lhs(), rhs(), [](auto const& l, auto const& r){ return l / r; });
  76. }
  77. // Overloaded Arithmetic Operators with Scalars
  78. template<class T, class R>
  79. auto operator+(typename T::const_reference lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
  80. return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (rhs(), [lhs](auto const& r){ return lhs + r; });
  81. //return boost::numeric::ublas::detail::make_lambda<T>( [&lhs,&rhs](std::size_t i) {return lhs + rhs(i); } );
  82. }
  83. template<class T, class R>
  84. auto operator-(typename T::const_reference lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
  85. return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (rhs(), [lhs](auto const& r){ return lhs - r; });
  86. }
  87. template<class T, class R>
  88. auto operator*(typename T::const_reference lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
  89. return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (rhs(), [lhs](auto const& r){ return lhs * r; });
  90. }
  91. template<class T, class R>
  92. auto operator/(typename T::const_reference lhs, boost::numeric::ublas::detail::tensor_expression<T,R> const& rhs) {
  93. return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (rhs(), [lhs](auto const& r){ return lhs / r; });
  94. }
  95. template<class T, class L>
  96. auto operator+(boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, typename T::const_reference rhs) {
  97. return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (lhs(), [rhs] (auto const& l) { return l + rhs; } );
  98. }
  99. template<class T, class L>
  100. auto operator-(boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, typename T::const_reference rhs) {
  101. return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (lhs(), [rhs] (auto const& l) { return l - rhs; } );
  102. }
  103. template<class T, class L>
  104. auto operator*(boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, typename T::const_reference rhs) {
  105. return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (lhs(), [rhs] (auto const& l) { return l * rhs; } );
  106. }
  107. template<class T, class L>
  108. auto operator/(boost::numeric::ublas::detail::tensor_expression<T,L> const& lhs, typename T::const_reference rhs) {
  109. return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (lhs(), [rhs] (auto const& l) { return l / rhs; } );
  110. }
  111. template<class T, class D>
  112. auto& operator += (T& lhs, const boost::numeric::ublas::detail::tensor_expression<T,D> &expr) {
  113. boost::numeric::ublas::detail::eval(lhs, expr(), [](auto& l, auto const& r) { l+=r; } );
  114. return lhs;
  115. }
  116. template<class T, class D>
  117. auto& operator -= (T& lhs, const boost::numeric::ublas::detail::tensor_expression<T,D> &expr) {
  118. boost::numeric::ublas::detail::eval(lhs, expr(), [](auto& l, auto const& r) { l-=r; } );
  119. return lhs;
  120. }
  121. template<class T, class D>
  122. auto& operator *= (T& lhs, const boost::numeric::ublas::detail::tensor_expression<T,D> &expr) {
  123. boost::numeric::ublas::detail::eval(lhs, expr(), [](auto& l, auto const& r) { l*=r; } );
  124. return lhs;
  125. }
  126. template<class T, class D>
  127. auto& operator /= (T& lhs, const boost::numeric::ublas::detail::tensor_expression<T,D> &expr) {
  128. boost::numeric::ublas::detail::eval(lhs, expr(), [](auto& l, auto const& r) { l/=r; } );
  129. return lhs;
  130. }
  131. template<class E, class F, class A>
  132. auto& operator += (boost::numeric::ublas::tensor<E,F,A>& lhs, typename boost::numeric::ublas::tensor<E,F,A>::const_reference r) {
  133. boost::numeric::ublas::detail::eval(lhs, [r](auto& l) { l+=r; } );
  134. return lhs;
  135. }
  136. template<class E, class F, class A>
  137. auto& operator -= (boost::numeric::ublas::tensor<E,F,A>& lhs, typename boost::numeric::ublas::tensor<E,F,A>::const_reference r) {
  138. boost::numeric::ublas::detail::eval(lhs, [r](auto& l) { l-=r; } );
  139. return lhs;
  140. }
  141. template<class E, class F, class A>
  142. auto& operator *= (boost::numeric::ublas::tensor<E,F,A>& lhs, typename boost::numeric::ublas::tensor<E,F,A>::const_reference r) {
  143. boost::numeric::ublas::detail::eval(lhs, [r](auto& l) { l*=r; } );
  144. return lhs;
  145. }
  146. template<class E, class F, class A>
  147. auto& operator /= (boost::numeric::ublas::tensor<E,F,A>& lhs, typename boost::numeric::ublas::tensor<E,F,A>::const_reference r) {
  148. boost::numeric::ublas::detail::eval(lhs, [r](auto& l) { l/=r; } );
  149. return lhs;
  150. }
  151. template<class T, class D>
  152. auto const& operator +(const boost::numeric::ublas::detail::tensor_expression<T,D>& lhs) {
  153. return lhs;
  154. }
  155. template<class T, class D>
  156. auto operator -(boost::numeric::ublas::detail::tensor_expression<T,D> const& lhs) {
  157. return boost::numeric::ublas::detail::make_unary_tensor_expression<T> (lhs(), [] (auto const& l) { return -l; } );
  158. }
  159. /** @brief Performs a tensor contraction, not an elementwise multiplication
  160. *
  161. */
  162. template<class tensor_type_left, class tuple_type_left, class tensor_type_right, class tuple_type_right>
  163. auto operator*(
  164. std::pair< tensor_type_left const&, tuple_type_left > lhs,
  165. std::pair< tensor_type_right const&, tuple_type_right > rhs)
  166. {
  167. using namespace boost::numeric::ublas;
  168. auto const& tensor_left = lhs.first;
  169. auto const& tensor_right = rhs.first;
  170. auto multi_index_left = lhs.second;
  171. auto multi_index_right = rhs.second;
  172. static constexpr auto num_equal_ind = number_equal_indexes<tuple_type_left, tuple_type_right>::value;
  173. if constexpr ( num_equal_ind == 0 ){
  174. return tensor_left * tensor_right;
  175. }
  176. else if constexpr ( num_equal_ind==std::tuple_size<tuple_type_left>::value && std::is_same<tuple_type_left, tuple_type_right>::value ){
  177. return boost::numeric::ublas::inner_prod( tensor_left, tensor_right );
  178. }
  179. else {
  180. auto array_index_pairs = index_position_pairs(multi_index_left,multi_index_right);
  181. auto index_pairs = array_to_vector( array_index_pairs );
  182. return boost::numeric::ublas::prod( tensor_left, tensor_right, index_pairs.first, index_pairs.second );
  183. }
  184. }
  185. #endif