PolyFEM
Loading...
Searching...
No Matches
MatrixUtils.hpp
Go to the documentation of this file.
1#pragma once
2
4
5#include <Eigen/Dense>
6#include <Eigen/Sparse>
7
8namespace polyfem
9{
10 namespace utils
11 {
12 // Show some stats about the matrix M: det, singular values, condition number, etc
13 void show_matrix_stats(const Eigen::MatrixXd &M);
14
15 template <typename T>
16 T matrix_inner_product(const Eigen::Matrix<T, Eigen::Dynamic, Eigen::Dynamic> &A, const Eigen::Matrix<T, Eigen::Dynamic, Eigen::Dynamic> &B)
17 {
18 return (A.array() * B.array()).sum();
19 }
20
21 template <typename T, int rows, int cols, int option, int maxRow, int maxCol>
22 T determinant(const Eigen::Matrix<T, rows, cols, option, maxRow, maxCol> &mat)
23 {
24 assert(mat.rows() == mat.cols());
25
26 if (mat.rows() == 1)
27 return mat(0);
28 else if (mat.rows() == 2)
29 return mat(0, 0) * mat(1, 1) - mat(0, 1) * mat(1, 0);
30 else if (mat.rows() == 3)
31 return mat(0, 0) * (mat(1, 1) * mat(2, 2) - mat(1, 2) * mat(2, 1)) - mat(0, 1) * (mat(1, 0) * mat(2, 2) - mat(1, 2) * mat(2, 0)) + mat(0, 2) * (mat(1, 0) * mat(2, 1) - mat(1, 1) * mat(2, 0));
32
33 assert(false);
34 return T(0);
35 }
36
37 template <typename T>
38 Eigen::Matrix<T, Eigen::Dynamic, Eigen::Dynamic, 0, 3, 3> inverse(const Eigen::Matrix<T, Eigen::Dynamic, Eigen::Dynamic, 0, 3, 3> &mat)
39 {
40 assert(mat.rows() == mat.cols());
41
42 if (mat.rows() == 1)
43 {
44 Eigen::Matrix<T, 1, 1> inv;
45 inv(0, 0) = T(1.) / mat(0);
46
47 return inv;
48 }
49 else if (mat.rows() == 2)
50 {
51 Eigen::Matrix<T, 2, 2> inv;
52 T det = determinant(mat);
53 inv(0, 0) = mat(1, 1) / det;
54 inv(0, 1) = -mat(0, 1) / det;
55 inv(1, 0) = -mat(1, 0) / det;
56 inv(1, 1) = mat(0, 0) / det;
57
58 return inv;
59 }
60 else if (mat.rows() == 3)
61 {
62 Eigen::Matrix<T, 3, 3> inv;
63 T det = determinant(mat);
64 inv(0, 0) = (-mat(1, 2) * mat(2, 1) + mat(1, 1) * mat(2, 2)) / det;
65 inv(0, 1) = (mat(0, 2) * mat(2, 1) - mat(0, 1) * mat(2, 2)) / det;
66 inv(0, 2) = (-mat(0, 2) * mat(1, 1) + mat(0, 1) * mat(1, 2)) / det;
67 inv(1, 0) = (mat(1, 2) * mat(2, 0) - mat(1, 0) * mat(2, 2)) / det;
68 inv(1, 1) = (-mat(0, 2) * mat(2, 0) + mat(0, 0) * mat(2, 2)) / det;
69 inv(1, 2) = (mat(0, 2) * mat(1, 0) - mat(0, 0) * mat(1, 2)) / det;
70 inv(2, 0) = (-mat(1, 1) * mat(2, 0) + mat(1, 0) * mat(2, 1)) / det;
71 inv(2, 1) = (mat(0, 1) * mat(2, 0) - mat(0, 0) * mat(2, 1)) / det;
72 inv(2, 2) = (-mat(0, 1) * mat(1, 0) + mat(0, 0) * mat(1, 1)) / det;
73
74 return inv;
75 }
76
77 assert(false);
78 Eigen::Matrix<T, 1, 1> inv;
79 inv(0, 0) = T(0);
80 return inv;
81 }
82
83 inline Eigen::SparseMatrix<double> sparse_identity(int rows, int cols)
84 {
85 Eigen::SparseMatrix<double> I(rows, cols);
86 I.setIdentity();
87 return I;
88 }
89
91 Eigen::VectorXd flatten(const Eigen::MatrixXd &X);
92
94 Eigen::MatrixXd unflatten(const Eigen::VectorXd &x, int dim);
95
96 void vector2matrix(const Eigen::VectorXd &vec, Eigen::MatrixXd &mat);
97
101 Eigen::SparseMatrix<double> lump_matrix(const Eigen::SparseMatrix<double> &M);
102
110 const int full_size,
111 const int reduced_size,
112 const std::vector<int> &removed_vars,
113 const StiffnessMatrix &full,
114 StiffnessMatrix &reduced);
115
122 Eigen::MatrixXd reorder_matrix(
123 const Eigen::MatrixXd &in,
124 const Eigen::VectorXi &in_to_out,
125 int out_blocks = -1,
126 const int block_size = 1);
127
134 Eigen::MatrixXd unreorder_matrix(
135 const Eigen::MatrixXd &out,
136 const Eigen::VectorXi &in_to_out,
137 int in_blocks = -1,
138 const int block_size = 1);
139
144 Eigen::MatrixXi map_index_matrix(
145 const Eigen::MatrixXi &in,
146 const Eigen::VectorXi &index_mapping);
147
148 template <typename DstMat, typename SrcMat>
149 void append_rows(DstMat &dst, const SrcMat &src)
150 {
151 if (src.rows() == 0)
152 return;
153 if (dst.cols() == 0)
154 dst.resize(dst.rows(), src.cols());
155 assert(dst.cols() == src.cols());
156 dst.conservativeResize(dst.rows() + src.rows(), dst.cols());
157 dst.bottomRows(src.rows()) = src;
158 }
159
160 template <typename DstMat>
161 void append_rows_of_zeros(DstMat &dst, const size_t n_zero_rows)
162 {
163 assert(dst.cols() > 0);
164 if (n_zero_rows == 0)
165 return;
166 dst.conservativeResize(dst.rows() + n_zero_rows, dst.cols());
167 dst.bottomRows(n_zero_rows).setZero();
168 }
169 } // namespace utils
170} // namespace polyfem
Eigen::MatrixXd vec
Definition Assembler.cpp:72
int x
T matrix_inner_product(const Eigen::Matrix< T, Eigen::Dynamic, Eigen::Dynamic > &A, const Eigen::Matrix< T, Eigen::Dynamic, Eigen::Dynamic > &B)
Eigen::SparseMatrix< double > lump_matrix(const Eigen::SparseMatrix< double > &M)
Lump each row of a matrix into the diagonal.
void show_matrix_stats(const Eigen::MatrixXd &M)
Eigen::MatrixXd reorder_matrix(const Eigen::MatrixXd &in, const Eigen::VectorXi &in_to_out, int out_blocks=-1, const int block_size=1)
Reorder row blocks in a matrix.
Eigen::SparseMatrix< double > sparse_identity(int rows, int cols)
void vector2matrix(const Eigen::VectorXd &vec, Eigen::MatrixXd &mat)
Eigen::Matrix< T, Eigen::Dynamic, Eigen::Dynamic, 0, 3, 3 > inverse(const Eigen::Matrix< T, Eigen::Dynamic, Eigen::Dynamic, 0, 3, 3 > &mat)
Eigen::MatrixXd unreorder_matrix(const Eigen::MatrixXd &out, const Eigen::VectorXi &in_to_out, int in_blocks=-1, const int block_size=1)
Undo the reordering of row blocks in a matrix.
Eigen::MatrixXd unflatten(const Eigen::VectorXd &x, int dim)
Unflatten rowwises, so every dim elements in x become a row.
void append_rows_of_zeros(DstMat &dst, const size_t n_zero_rows)
void append_rows(DstMat &dst, const SrcMat &src)
Eigen::VectorXd flatten(const Eigen::MatrixXd &X)
Flatten rowwises.
Eigen::MatrixXi map_index_matrix(const Eigen::MatrixXi &in, const Eigen::VectorXi &index_mapping)
Map the entrys of an index matrix to new indices.
void full_to_reduced_matrix(const int full_size, const int reduced_size, const std::vector< int > &removed_vars, const StiffnessMatrix &full, StiffnessMatrix &reduced)
Map a full size matrix to a reduced one by dropping rows and columns.
T determinant(const Eigen::Matrix< T, rows, cols, option, maxRow, maxCol > &mat)
Eigen::SparseMatrix< double, Eigen::ColMajor > StiffnessMatrix
Definition Types.hpp:22