Eigen-unsupported  5.0.1-dev+284dcc12
 
Loading...
Searching...
No Matches
qrsolv.h
1// IWYU pragma: private
2#include "./InternalHeaderCheck.h"
3
4namespace Eigen {
5
6namespace internal {
7
8// TODO : once qrsolv2 is removed, use ColPivHouseholderQR or PermutationMatrix instead of ipvt
9template <typename Scalar>
10void qrsolv(Matrix<Scalar, Dynamic, Dynamic> &s,
11 // TODO : use a PermutationMatrix once lmpar is no more:
12 const VectorXi &ipvt, const Matrix<Scalar, Dynamic, 1> &diag, const Matrix<Scalar, Dynamic, 1> &qtb,
13 Matrix<Scalar, Dynamic, 1> &x, Matrix<Scalar, Dynamic, 1> &sdiag)
14
15{
16 typedef DenseIndex Index;
17
18 /* Local variables */
19 Index i, j, k, l;
20 Scalar temp;
21 Index n = s.cols();
22 Matrix<Scalar, Dynamic, 1> wa(n);
23 JacobiRotation<Scalar> givens;
24
25 /* Function Body */
26 // the following will only change the lower triangular part of s, including
27 // the diagonal, though the diagonal is restored afterward
28
29 /* copy r and (q transpose)*b to preserve input and initialize s. */
30 /* in particular, save the diagonal elements of r in x. */
31 x = s.diagonal();
32 wa = qtb;
33
34 s.topLeftCorner(n, n).template triangularView<StrictlyLower>() = s.topLeftCorner(n, n).transpose();
35
36 /* eliminate the diagonal matrix d using a givens rotation. */
37 for (j = 0; j < n; ++j) {
38 /* prepare the row of d to be eliminated, locating the */
39 /* diagonal element using p from the qr factorization. */
40 l = ipvt[j];
41 if (diag[l] == 0.) break;
42 sdiag.tail(n - j).setZero();
43 sdiag[j] = diag[l];
44
45 /* the transformations to eliminate the row of d */
46 /* modify only a single element of (q transpose)*b */
47 /* beyond the first n, which is initially zero. */
48 Scalar qtbpj = 0.;
49 for (k = j; k < n; ++k) {
50 /* determine a givens rotation which eliminates the */
51 /* appropriate element in the current row of d. */
52 givens.makeGivens(-s(k, k), sdiag[k]);
53
54 /* compute the modified diagonal element of r and */
55 /* the modified element of ((q transpose)*b,0). */
56 s(k, k) = givens.c() * s(k, k) + givens.s() * sdiag[k];
57 temp = givens.c() * wa[k] + givens.s() * qtbpj;
58 qtbpj = -givens.s() * wa[k] + givens.c() * qtbpj;
59 wa[k] = temp;
60
61 /* accumulate the transformation in the row of s. */
62 for (i = k + 1; i < n; ++i) {
63 temp = givens.c() * s(i, k) + givens.s() * sdiag[i];
64 sdiag[i] = -givens.s() * s(i, k) + givens.c() * sdiag[i];
65 s(i, k) = temp;
66 }
67 }
68 }
69
70 /* solve the triangular system for z. if the system is */
71 /* singular, then obtain a least squares solution. */
72 Index nsing;
73 for (nsing = 0; nsing < n && sdiag[nsing] != 0; nsing++) {
74 }
75
76 wa.tail(n - nsing).setZero();
77 s.topLeftCorner(nsing, nsing).transpose().template triangularView<Upper>().solveInPlace(wa.head(nsing));
78
79 // restore
80 sdiag = s.diagonal();
81 s.diagonal() = x;
82
83 /* permute the components of z back to components of x. */
84 for (j = 0; j < n; ++j) x[ipvt[j]] = wa[j];
85}
86
87} // end namespace internal
88
89} // end namespace Eigen
Matrix< int, Dynamic, 1 > VectorXi
Namespace containing all symbols from the Eigen library.
EIGEN_DEFAULT_DENSE_INDEX_TYPE Index