lin
substitutions.hpp
Go to the documentation of this file.
1 // vim: set tabstop=2:softtabstop=2:shiftwidth=2:expandtab
2 
3 //
4 // MIT License
5 //
6 // Copyright (c) 2020 kylekrol
7 // Copyright (c) 2020 Pathfinder for Autonomous Navigation (PAN)
8 //
9 // Permission is hereby granted, free of charge, to any person obtaining a copy
10 // of this software and associated documentation files (the "Software"), to deal
11 // in the Software without restriction, including without limitation the rights
12 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
13 // copies of the Software, and to permit persons to whom the Software is
14 // furnished to do so, subject to the following conditions:
15 //
16 // The above copyright notice and this permission notice shall be included in
17 // all copies or substantial portions of the Software.
18 //
19 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
20 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
21 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
22 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
23 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
24 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
25 // SOFTWARE.
26 //
27 
38 #ifndef LIN_SUBSTITUTIONS_HPP_
39 #define LIN_SUBSTITUTIONS_HPP_
40 
41 #include "core.hpp"
42 #include "references.hpp"
43 
44 #include <type_traits>
45 
46 namespace lin {
47 namespace internal {
48 
60 template <class C, class D, class E>
62  can_multiply<C, D>, have_same_elements<C, D, E>, have_same_dimensions<D, E>,
63  is_square<C>
64  > {};
65 
77 template <class C, class D, class E>
78 struct can_forward_sub : can_backward_sub<C, D, E> {};
79 
80 } // namespace internal
81 
115 template <class C, class D, class E, typename =
116  std::enable_if_t<internal::can_backward_sub<C, D, E>::value>>
118  LIN_ASSERT(U.rows() == U.cols());
119  LIN_ASSERT(U.cols() == Y.rows());
120  LIN_ASSERT(Y.rows() == X.rows());
121  LIN_ASSERT(Y.cols() == X.cols());
122 
123  // Solve for the last row
124  size_t m = U.rows() - 1;
125  row(X, m) = row(Y, m) / U(m, m);
126 
127  // Solve for the other rows in descending order
128  size_t n = m - 1;
129  do {
130  typedef typename C::Traits::elem_t Elem;
132  typedef std::conditional_t<internal::is_col_vector<E>::value, Vector<Elem, 0, E::Traits::max_rows>, Matrix<Elem, 0, E::Traits::cols, E::Traits::max_rows, E::Traits::max_cols>> V;
133 
134  row(X, n) = (row(Y, n) - ref<T>(U, n, n + 1, m - n) * ref<V>(X, n + 1, 0, m - n, X.cols())) / U(n, n);
135  } while (n--);
136 
137  // TODO : Change this to a meaningfull error code
138  return 0;
139 }
140 
171 template <class C, class D, class E, typename =
172  std::enable_if_t<internal::can_backward_sub<C, D, E>::value>>
174  X.resize(Y.rows(), Y.cols());
175  return backward_sub(U, static_cast<internal::Mapping<D> &>(X), Y);
176 }
177 
211 template <class C, class D, class E, typename =
212  std::enable_if_t<internal::can_forward_sub<C, D, E>::value>>
214  LIN_ASSERT(L.rows() == L.cols());
215  LIN_ASSERT(L.cols() == Y.rows());
216  LIN_ASSERT(Y.rows() == X.rows());
217  LIN_ASSERT(Y.cols() == X.cols());
218 
219  // Solve for the first row
220  row(X, 0) = row(Y, 0) / L(0, 0);
221 
222  // Solve for other rows in ascending order
223  for (size_t n = 1; n < X.rows(); n++) {
224  typedef typename C::Traits::elem_t Elem;
226  typedef std::conditional_t<internal::is_col_vector<E>::value, Vector<Elem, 0, E::Traits::max_rows>, Matrix<Elem, 0, E::Traits::cols, E::Traits::max_rows, E::Traits::max_cols>> V;
227 
228  row(X, n) = (row(Y, n) - ref<T>(L, n, 0, n) * ref<V>(X, 0, 0, n, X.cols())) / L(n, n);
229  }
230 
231  // TODO : Change this to a meaningfull error code
232  return 0;
233 }
234 
265 template <class C, class D, class E, typename =
266  std::enable_if_t<internal::can_forward_sub<C, D, E>::value>>
267 constexpr size_t forward_sub(internal::Mapping<C> const &L, internal::Base<D> &X, internal::Mapping<E> const &Y) {
268  X.resize(Y.rows(), Y.cols());
269  return forward_sub(L, static_cast<internal::Mapping<D> &>(X), Y);
270 }
271 } // namespace lin
272 
273 #endif
Value backed tensor interface with resizing support.
Definition: base.hpp:40
constexpr void resize(size_t r, size_t c)
Resizes a tensor&#39;s dimensions.
Definition: dimensions.hpp:75
Generic vector.
Definition: vector.hpp:33
Definition: substitutions.hpp:78
constexpr size_t backward_sub(internal::Mapping< C > const &U, internal::Base< D > &X, internal::Mapping< E > const &Y)
Solves a linear system using backward substitution.
Definition: substitutions.hpp:173
Generic matrix.
Definition: matrix.hpp:35
#define LIN_ASSERT(x)
Asserts the condition x when assertions are enabled within lin.
Definition: config.hpp:22
Generic row vector.
Definition: vector.hpp:131
constexpr size_t rows() const
Definition: stream.hpp:76
Logical AND operation for template metaprogramming.
Definition: utilities.hpp:54
constexpr size_t cols() const
Definition: stream.hpp:82
Definition: config.hpp:27
constexpr size_t forward_sub(internal::Mapping< C > const &L, internal::Base< D > &X, internal::Mapping< E > const &Y)
Solves a linear system using forward substitution.
Definition: substitutions.hpp:267
Definition: substitutions.hpp:61
constexpr auto row(internal::Mapping< D > &mapping, size_t i)
Creates a mapping reference of a particular row of a given tensor.
Definition: references.hpp:330