6#include "eigen3/Eigen/Core"
7#include "eigen3/Eigen/Dense"
8#include "eigen3/unsupported/Eigen/CXX11/Tensor"
26 using Matrix = Eigen::Matrix<double, nsd, nsd>;
29 using Tensor = Eigen::TensorFixedSize<double, Eigen::Sizes<nsd, nsd, nsd, nsd>>;
32 template <
typename MatrixType>
33 MatrixType convert_to_eigen_matrix(
const Array<double>& src) {
35 for (
int i = 0; i < mat.rows(); ++i)
36 for (
int j = 0; j < mat.cols(); ++j)
37 mat(i, j) = src(i, j);
42 template <
typename MatrixType>
43 void convert_to_array(
const MatrixType& mat, Array<double>& dest) {
44 for (
int i = 0; i < mat.rows(); ++i)
45 for (
int j = 0; j < mat.cols(); ++j)
46 dest(i, j) = mat(i, j);
50 template <
typename MatrixType>
51 void copy_Dm(
const MatrixType& mat, Array<double>& dest) {
52 if ((mat.rows() != dest.nrows()) || (mat.cols() != dest.ncols())) {
53 auto mat_dims = (std::stringstream() <<
"(" << mat.rows() <<
"x" << mat.cols() <<
")").str();
54 auto dest_dims = (std::stringstream() <<
"(" << dest.nrows() <<
"x" << dest.ncols() <<
")").str();
55 svmp::raise<svmp::FE::InvalidArgumentException>( SVMP_HERE,
"The 'mat" + mat_dims +
"' and 'dest" + dest_dims +
56 "' arrays have incompatible sizes.");
59 for (
int i = 0; i < mat.rows(); ++i) {
60 for (
int j = 0; j < mat.cols(); ++j) {
61 dest(i, j) = mat(i, j);
67 Eigen::Matrix<double, nsd, 1> cross_product(
const Eigen::Matrix<double, nsd, 1>& u,
const Eigen::Matrix<double, nsd, 1>& v) {
68 if constexpr (nsd == 2) {
69 return Eigen::Matrix<double, 2, 1>(v(1), - v(0));
71 else if constexpr (nsd == 3) {
75 throw std::runtime_error(
"[cross_product] Invalid number of spatial dimensions '" + std::to_string(nsd) +
"'. Valid dimensions are 2 or 3.");
79 double mat_ddot(
const Array<double>& A,
const Array<double>& B,
const int nd);
82 double double_dot_product(
const Matrix<nsd>& A,
const Matrix<nsd>& B) {
83 return A.cwiseProduct(B).sum();
86 double mat_det(
const Array<double>& A,
const int nd);
87 Array<double> mat_dev(
const Array<double>& A,
const int nd);
91 Array<double> mat_id(
const int nsd);
92 Array<double>
mat_inv(
const Array<double>& A,
const int nd,
bool debug =
false);
93 Array<double>
mat_inv_ge(
const Array<double>& A,
const int nd,
bool debug =
false);
94 Array<double>
mat_inv_lp(
const Array<double>& A,
const int nd);
97 Array<double>
mat_mul(
const Array<double>& A,
const Array<double>& B);
98 void mat_mul(
const Array<double>& A,
const Array<double>& B, Array<double>& result);
99 void mat_mul6x3(
const Array<double>& A,
const Array<double>& B, Array<double>& C);
101 Array<double>
mat_symm(
const Array<double>& A,
const int nd);
104 double mat_trace(
const Array<double>& A,
const int nd);
117 double_dot_product(
const Tensor<nsd>& A,
const std::array<int, 2>& dimsA,
118 const Tensor<nsd>& B,
const std::array<int, 2>& dimsB) {
121 Eigen::array<Eigen::IndexPair<int>, 2> contractionDims = {
122 Eigen::IndexPair<int>(dimsA[0], dimsB[0]),
123 Eigen::IndexPair<int>(dimsA[1], dimsB[1])
127 return A.contract(B, contractionDims);
150 for (
int i = 0; i < nsd; ++i) {
151 for (
int j = 0; j < nsd; ++j) {
152 for (
int k = 0; k < nsd; ++k) {
153 for (
int l = 0; l < nsd; ++l) {
154 C(i,j,k,l) = A(i,j) * B(k,l);
182 for (
int i = 0; i < nsd; ++i) {
183 for (
int j = 0; j < nsd; ++j) {
208 for (
int i = 0; i < nsd; ++i) {
209 for (
int j = 0; j < nsd; ++j) {
210 for (
int k = 0; k < nsd; ++k) {
211 for (
int l = 0; l < nsd; ++l) {
212 C(i,j,k,l) = 0.5 * (A(i,k) * B(j,l) + A(i,l) * B(j,k));
241 for (
int i = 0; i < nsd; ++i) {
242 for (
int j = 0; j < nsd; ++j) {
243 for (
int k = 0; k < nsd; ++k) {
244 for (
int l = 0; l < nsd; ++l) {
245 B(i,j,k,l) = A(k,l,i,j);
254 Array<double>
transpose(
const Array<double>& A);
Exception hierarchy for error handling in the FE library.
The Tensor4 template class implements a simple interface to 4th order tensors.
Definition Tensor4.h:20
The Vector template class is used for storing int and double data.
Definition Vector.h:24
The classes defined here duplicate the data structures in the Fortran MATFUN module defined in MATFUN...
Definition mat_fun.cpp:14
Tensor< nsd > symmetric_dyadic_product(const Matrix< nsd > &A, const Matrix< nsd > &B)
Create a 4th order tensor from symmetric outer product of two matrices: C_ijkl = 0....
Definition mat_fun.h:202
void ten_init(const int nd)
Initialize tensor index pointer.
Definition mat_fun.cpp:760
Tensor4< double > ten_symm_prod(const Array< double > &A, const Array< double > &B, const int nd)
Create a 4th order tensor from symmetric outer product of two matrices.
Definition mat_fun.cpp:852
Array< double > ten_mddot(const Tensor4< double > &A, const Array< double > &B, const int nd)
Double dot product of a 4th order tensor and a 2nd order tensor.
Definition mat_fun.cpp:822
Array< double > mat_symm(const Array< double > &A, const int nd)
Symmetric part of a matrix, S = (A + A.T)/2.
Definition mat_fun.cpp:555
Tensor4< double > ten_ids(const int nd)
Create a 4th order order symmetric identity tensor.
Definition mat_fun.cpp:803
Array< double > transpose(const Array< double > &A)
Reproduces Fortran TRANSPOSE.
Definition mat_fun.cpp:888
Vector< double > mat_mul(const Array< double > &A, const Vector< double > &v)
Multiply a matrix by a vector.
Definition mat_fun.cpp:466
double mat_trace(const Array< double > &A, const int nd)
Trace of second order matrix of rank nd.
Definition mat_fun.cpp:587
Tensor4< double > ten_ddot(const Tensor4< double > &A, const Tensor4< double > &B, const int nd)
Double dot product of 2 4th order tensors T_ijkl = A_ijmn * B_klmn.
Definition mat_fun.cpp:626
Tensor4< double > ten_dyad_prod(const Array< double > &A, const Array< double > &B, const int nd)
Create a 4th order tensor from outer product of two matrices.
Definition mat_fun.cpp:784
Array< double > mat_inv_lp(const Array< double > &A, const int nd)
This function computes inverse of a square matrix using Lapack functions (DGETRF + DGETRI)
Definition mat_fun.cpp:396
Array< double > mat_inv(const Array< double > &A, const int nd, bool debug)
This function computes inverse of a square matrix.
Definition mat_fun.cpp:109
Array< double > mat_inv_ge(const Array< double > &Ain, const int n, bool debug)
This function computes inverse of a square matrix using Gauss Elimination method.
Definition mat_fun.cpp:166
Array< double > mat_symm_prod(const Vector< double > &u, const Vector< double > &v, const int nd)
Create a matrix from symmetric product of two vectors.
Definition mat_fun.cpp:572
Array< double > mat_dyad_prod(const Vector< double > &u, const Vector< double > &v, const int nd)
Create a matrix from outer product of two vectors.
Definition mat_fun.cpp:81
Tensor< nsd > fourth_order_identity()
Create a 4th order identity tensor: I_ijkl = 0.5 * (δ_ik * δ_jl + δ_il * δ_jk)
Definition mat_fun.h:176
double mat_ddot(const Array< double > &A, const Array< double > &B, const int nd)
Double dot product of 2 square matrices.
Definition mat_fun.cpp:20
Tensor4< double > ten_ddot_2412(const Tensor4< double > &A, const Tensor4< double > &B, const int nd)
T_ijkl = A_imjn * B_mnkl.
Definition mat_fun.cpp:671
Tensor4< double > ten_asym_prod12(const Array< double > &A, const Array< double > &B, const int nd)
Create a 4th order tensor from antisymmetric outer product of two matrices.
Definition mat_fun.cpp:604
Tensor< nsd > dyadic_product(const Matrix< nsd > &A, const Matrix< nsd > &B)
Compute the dyadic product of two 2nd order tensors A and B, C_ijkl = A_ij * B_kl.
Definition mat_fun.h:145