7 #ifndef LIN_CORE_OPERATIONS_TENSOR_OPERATIONS_HPP_ 8 #define LIN_CORE_OPERATIONS_TENSOR_OPERATIONS_HPP_ 10 #include "../config.hpp" 11 #include "../traits.hpp" 12 #include "../types.hpp" 18 #include <type_traits> 68 template <
typename T,
typename U>
70 :
conjunction<matches_tensor<T>, matches_tensor<U>> { };
89 template <
typename T,
typename U>
91 :
conjunction<matches_tensor<T>, matches_scalar<U>> { };
110 template <
typename T,
typename U>
112 :
conjunction<matches_scalar<T>, matches_tensor<U>> { };
130 template <
typename T,
typename U>
132 :
conjunction<matches_scalar<T>, matches_scalar<U>> { };
143 template <
class C,
class D, std::enable_if_t<
152 template <
typename T,
class C, std::enable_if_t<
158 template <
class C,
typename T, std::enable_if_t<
164 template <
typename T,
typename U, std::enable_if_t<
166 constexpr
auto add(T
const &t, U
const &u) {
170 template <
typename T,
class C, std::enable_if_t<
176 template <
typename T,
typename U, std::enable_if_t<
178 inline constexpr
auto cast(U
const &u) {
182 template <
class C,
class D, std::enable_if_t<
183 internal::matches_tensor_tensor<C, D>::value,
size_t> = 0>
191 template <
typename T,
class C, std::enable_if_t<
192 internal::matches_scalar_tensor<T, C>::value,
size_t> = 0>
197 template <
class C,
typename T, std::enable_if_t<
198 internal::matches_tensor_scalar<C, T>::value,
size_t> = 0>
203 template <
typename T,
typename U, std::enable_if_t<
204 internal::matches_scalar_scalar<T, U>::value,
size_t> = 0>
205 inline constexpr
auto divide(T
const &t, U
const &u) {
209 template <
class C, std::enable_if_t<
210 internal::matches_tensor<C>::value,
size_t> = 0>
212 typename C::Traits::elem_t f = c(0) * c(0);
213 for (
size_t i = 1; i < c.
size(); i++) f += c(i) * c(i);
217 template <
typename T, std::enable_if_t<
219 constexpr
auto fro(T
const &t) {
223 template <
class C,
class D, std::enable_if_t<
224 internal::matches_tensor_tensor<C, D>::value,
size_t> = 0>
232 template <
typename T,
class C, std::enable_if_t<
233 internal::matches_scalar_tensor<T, C>::value,
size_t> = 0>
238 template <
class C,
typename T, std::enable_if_t<
239 internal::matches_tensor_scalar<C, T>::value,
size_t> = 0>
244 template <
typename T,
typename U, std::enable_if_t<
245 internal::matches_scalar_scalar<T, U>::value,
size_t> = 0>
246 inline constexpr
auto multiply(T
const &t, U
const &u) {
250 template <
class C, std::enable_if_t<
251 internal::matches_tensor<C>::value,
size_t> = 0>
256 template <
typename T, std::enable_if_t<
257 internal::matches_scalar<T>::value,
size_t> = 0>
258 inline constexpr
auto negate(T
const &t) {
262 template <
class C, std::enable_if_t<
263 internal::matches_tensor<C>::value,
size_t> = 0>
268 template <
typename T, std::enable_if_t<
269 internal::matches_scalar<T>::value,
size_t> = 0>
270 inline constexpr
auto sign(T
const &t) {
274 template <
class C, std::enable_if_t<
275 internal::matches_tensor<C>::value,
size_t> = 0>
280 template <
typename T, std::enable_if_t<
281 internal::matches_scalar<T>::value,
size_t> = 0>
282 inline constexpr
auto square(T
const &t) {
286 template <
class C,
class D, std::enable_if_t<
287 internal::matches_tensor_tensor<C, D>::value,
size_t> = 0>
295 template <
typename T,
class C, std::enable_if_t<
296 internal::matches_scalar_tensor<T, C>::value,
size_t> = 0>
301 template <
class C,
typename T, std::enable_if_t<
302 internal::matches_tensor_scalar<C, T>::value,
size_t> = 0>
307 template <
typename T,
typename U, std::enable_if_t<
308 internal::matches_scalar_scalar<T, U>::value,
size_t> = 0>
309 inline constexpr
auto subtract(T
const &t, U
const &u) {
315 typename C::Traits::elem_t x = c(0);
320 template <
class C, std::enable_if_t<
321 internal::matches_tensor<C>::value,
size_t> = 0>
326 template <
class C, std::enable_if_t<
327 internal::matches_tensor<C>::value,
size_t> = 0>
Tests if a type is a scalar.
Definition: tensor_operations.hpp:50
Tests if a pair of types is a "scalar tensor" pair.
Definition: tensor_operations.hpp:111
constexpr size_t size() const
Definition: stream.hpp:90
Definition: functors.hpp:16
Definition: functors.hpp:140
Definition: functors.hpp:41
Proxy to a lazily evalutated transpose operation.
Definition: mapping_transpose.hpp:22
Logical NOT operation for template metaprogramming.
Definition: utilities.hpp:91
Tests if a pair of types is a "tensor scalar" pair.
Definition: tensor_operations.hpp:90
#define LIN_ASSERT(x)
Asserts the condition x when assertions are enabled within lin.
Definition: config.hpp:22
Definition: functors.hpp:83
Tests if a pair of types is a "scalar scalar" pair.
Definition: tensor_operations.hpp:131
std::size_t size_t
Type tracking tensor dimensions and sizing.
Definition: config.hpp:33
Definition: functors.hpp:107
constexpr size_t rows() const
Definition: stream.hpp:76
Tests if a pair of types is a "tensor tensor" pair.
Definition: tensor_operations.hpp:69
Definition: functors.hpp:131
Logical AND operation for template metaprogramming.
Definition: utilities.hpp:54
Proxy to a lazily evalutated element wise operation.
Definition: stream_element_wise_operator.hpp:30
constexpr size_t cols() const
Definition: stream.hpp:82
Tests if a type is a tensor.
Definition: tensor_operations.hpp:36
Definition: config.hpp:27
Proxy to a lazily evalutated transpose operation.
Definition: stream_transpose.hpp:22
Definition: functors.hpp:116
Definition: functors.hpp:50
Tests if a type has traits.
Definition: tensor.hpp:158