lin
tensor_operations.hpp
Go to the documentation of this file.
1 // vim: set tabstop=2:softtabstop=2:shiftwidth=2:expandtab
2 
7 #ifndef LIN_CORE_OPERATIONS_TENSOR_OPERATIONS_HPP_
8 #define LIN_CORE_OPERATIONS_TENSOR_OPERATIONS_HPP_
9 
10 #include "../config.hpp"
11 #include "../traits.hpp"
12 #include "../types.hpp"
13 #include "functors.hpp"
14 #include "mapping_transpose.hpp"
16 #include "stream_transpose.hpp"
17 
18 #include <type_traits>
19 
20 namespace lin {
21 namespace internal {
22 
35 template <typename T>
36 struct matches_tensor : has_traits<T> { };
37 
49 template <typename T>
50 struct matches_scalar : negation<matches_tensor<T>> { };
51 
68 template <typename T, typename U>
70  : conjunction<matches_tensor<T>, matches_tensor<U>> { };
71 
89 template <typename T, typename U>
91  : conjunction<matches_tensor<T>, matches_scalar<U>> { };
92 
110 template <typename T, typename U>
112  : conjunction<matches_scalar<T>, matches_tensor<U>> { };
113 
130 template <typename T, typename U>
132  : conjunction<matches_scalar<T>, matches_scalar<U>> { };
133 
134 template <class C>
135 class StreamTranspose;
136 
137 } // namespace internal
138 
143 template <class C, class D, std::enable_if_t<
145 inline constexpr auto add(internal::Stream<C> const &c, internal::Stream<D> const &d) {
146  LIN_ASSERT(c.rows() == d.rows());
147  LIN_ASSERT(c.cols() == d.cols());
148 
150 }
151 
152 template <typename T, class C, std::enable_if_t<
154 inline constexpr auto add(T const &t, internal::Stream<C> const &c) {
156 }
157 
158 template <class C, typename T, std::enable_if_t<
160 inline constexpr auto add(internal::Stream<C> const &c, T const &t) {
162 }
163 
164 template <typename T, typename U, std::enable_if_t<
166 constexpr auto add(T const &t, U const &u) {
167  return internal::add()(t, u);
168 }
169 
170 template <typename T, class C, std::enable_if_t<
172 inline constexpr auto cast(internal::Stream<C> const &c) {
174 }
175 
176 template <typename T, typename U, std::enable_if_t<
178 inline constexpr auto cast(U const &u) {
179  return internal::cast<T>()(u);
180 }
181 
182 template <class C, class D, std::enable_if_t<
183  internal::matches_tensor_tensor<C, D>::value, size_t> = 0>
184 inline constexpr auto divide(internal::Stream<C> const &c, internal::Stream<D> const &d) {
185  LIN_ASSERT(c.rows() == d.rows());
186  LIN_ASSERT(c.cols() == d.cols());
187 
189 }
190 
191 template <typename T, class C, std::enable_if_t<
192  internal::matches_scalar_tensor<T, C>::value, size_t> = 0>
193 inline constexpr auto divide(T const &t, internal::Stream<C> const &c) {
195 }
196 
197 template <class C, typename T, std::enable_if_t<
198  internal::matches_tensor_scalar<C, T>::value, size_t> = 0>
199 inline constexpr auto divide(internal::Stream<C> const &c, T const &t) {
201 }
202 
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) {
206  return internal::divide()(t, u);
207 }
208 
209 template <class C, std::enable_if_t<
210  internal::matches_tensor<C>::value, size_t> = 0>
211 constexpr auto fro(internal::Stream<C> const &c) {
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);
214  return f;
215 }
216 
217 template <typename T, std::enable_if_t<
219 constexpr auto fro(T const &t) {
220  return t * t;
221 }
222 
223 template <class C, class D, std::enable_if_t<
224  internal::matches_tensor_tensor<C, D>::value, size_t> = 0>
225 inline constexpr auto multiply(internal::Stream<C> const &c, internal::Stream<D> const &d) {
226  LIN_ASSERT(c.rows() == d.rows());
227  LIN_ASSERT(c.cols() == d.cols());
228 
230 }
231 
232 template <typename T, class C, std::enable_if_t<
233  internal::matches_scalar_tensor<T, C>::value, size_t> = 0>
234 inline constexpr auto multiply(T const &t, internal::Stream<C> const &c) {
236 }
237 
238 template <class C, typename T, std::enable_if_t<
239  internal::matches_tensor_scalar<C, T>::value, size_t> = 0>
240 inline constexpr auto multiply(internal::Stream<C> const &c, T const &t) {
242 }
243 
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) {
247  return internal::multiply()(t, u);
248 }
249 
250 template <class C, std::enable_if_t<
251  internal::matches_tensor<C>::value, size_t> = 0>
252 inline constexpr auto negate(internal::Stream<C> const &c) {
254 }
255 
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) {
259  return internal::negate()(t);
260 }
261 
262 template <class C, std::enable_if_t<
263  internal::matches_tensor<C>::value, size_t> = 0>
264 inline constexpr auto sign(internal::Stream<C> const &c) {
266 }
267 
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) {
271  return internal::sign()(t);
272 }
273 
274 template <class C, std::enable_if_t<
275  internal::matches_tensor<C>::value, size_t> = 0>
276 inline constexpr auto square(internal::Stream<C> const &c) {
278 }
279 
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) {
283  return internal::square()(t);
284 }
285 
286 template <class C, class D, std::enable_if_t<
287  internal::matches_tensor_tensor<C, D>::value, size_t> = 0>
288 inline constexpr auto subtract(internal::Stream<C> const &c, internal::Stream<D> const &d) {
289  LIN_ASSERT(c.rows() == d.rows());
290  LIN_ASSERT(c.cols() == d.cols());
291 
293 }
294 
295 template <typename T, class C, std::enable_if_t<
296  internal::matches_scalar_tensor<T, C>::value, size_t> = 0>
297 inline constexpr auto subtract(T const &t, internal::Stream<C> const &c) {
299 }
300 
301 template <class C, typename T, std::enable_if_t<
302  internal::matches_tensor_scalar<C, T>::value, size_t> = 0>
303 inline constexpr auto subtract(internal::Stream<C> const &c, T const &t) {
305 }
306 
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) {
310  return internal::subtract()(t, u);
311 }
312 
313 template <class C>
314 constexpr auto sum(internal::Stream<C> const &c) {
315  typename C::Traits::elem_t x = c(0);
316  for (lin::size_t i = 1; i < c.size(); i++) x += c(i);
317  return x;
318 }
319 
320 template <class C, std::enable_if_t<
321  internal::matches_tensor<C>::value, size_t> = 0>
322 constexpr auto transpose(internal::Mapping<C> &c) {
324 }
325 
326 template <class C, std::enable_if_t<
327  internal::matches_tensor<C>::value, size_t> = 0>
328 constexpr auto transpose(internal::Stream<C> const &c) {
330 }
331 
335 } // namespace lin
336 
337 #endif
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