6#include <spdlog/fmt/fmt.h>
28template <
typename ShapeFunctionDisplacement,
typename ShapeFunction,
29 int DisplacementDim,
typename ConstitutiveTraits>
31 DisplacementDim, ConstitutiveTraits>::
32 ThermoRichardsMechanicsLocalAssembler(
36 bool const is_axially_symmetric,
40 e, integration_method, is_axially_symmetric, process_data)
42 unsigned const n_integration_points =
45 ip_data_.resize(n_integration_points);
47 auto const shape_matrices_u =
50 DisplacementDim>(e, is_axially_symmetric,
53 auto const shape_matrices =
55 DisplacementDim>(e, is_axially_symmetric,
58 for (
unsigned ip = 0; ip < n_integration_points; ip++)
61 auto const& sm_u = shape_matrices_u[ip];
64 sm_u.integralMeasure * sm_u.detJ;
67 ip_data.dNdx_u = sm_u.dNdx;
70 ip_data.N_p = shape_matrices[ip].N;
71 ip_data.dNdx_p = shape_matrices[ip].dNdx;
75template <
typename ShapeFunctionDisplacement,
typename ShapeFunction,
76 int DisplacementDim,
typename ConstitutiveTraits>
84 assert(local_x.size() ==
87 auto const p_L = local_x.template segment<pressure_size>(
pressure_index);
91 constexpr double dt = std::numeric_limits<double>::quiet_NaN();
97 typename ConstitutiveTraits::ConstitutiveSetting
const constitutive_setting;
98 auto models = ConstitutiveTraits::createConstitutiveModels(
101 unsigned const n_integration_points =
103 for (
unsigned ip = 0; ip < n_integration_points; ip++)
109 std::nullopt, this->
element_.getID(),
134 .template value<double>(variables, x_position, t, dt);
135 std::get<PrevState<SaturationData>>(this->
prev_states_[ip])->S_L =
141 constitutive_setting.init(models, t, dt, x_position, media_data,
150 "The medium has no saturation property required to compute "
159template <
typename ShapeFunctionDisplacement,
typename ShapeFunction,
160 int DisplacementDim,
typename ConstitutiveTraits>
163 ConstitutiveTraits>::
164 convertInitialStressType(
unsigned const ip,
169 double const p_at_ip)
171 bool constexpr is_strain_temperature_constitutive =
174 ConstitutiveTraits>::value;
175 if (is_strain_temperature_constitutive &&
182 if (!is_strain_temperature_constitutive &&
188 double const alpha_b =
190 .template value<double>(variables, x_position, t, 0.0 );
192 double const bishop =
194 .template value<double>(variables, x_position, t, 0.0 );
196 ConstitutiveTraits::ConstitutiveSetting::convertInitialStressType(
201template <
typename ShapeFunctionDisplacement,
typename ShapeFunction,
202 int DisplacementDim,
typename ConstitutiveTraits>
205 ConstitutiveTraits>::
206 assembleWithJacobian(
double const t,
double const dt,
207 std::vector<double>
const& local_x,
208 std::vector<double>
const& local_x_prev,
209 std::vector<double>& local_rhs_data,
210 std::vector<double>& local_Jac_data)
220 typename ConstitutiveTraits::ConstitutiveSetting constitutive_setting;
226 std::nullopt, this->
element_.getID(),
234 local_x, local_x_prev,
240 loc_mat += loc_mat_current_ip;
249template <
typename ShapeFunctionDisplacement,
typename ShapeFunction,
250 int DisplacementDim,
typename ConstitutiveTraits>
253 ConstitutiveTraits>::
260 loc_mat.storage_p_a_p =
261 loc_mat.storage_p_a_p.colwise().sum().eval().asDiagonal();
262 loc_mat.storage_p_a_S =
263 loc_mat.storage_p_a_S.colwise().sum().eval().asDiagonal();
264 loc_mat.storage_p_a_S_Jpp =
265 loc_mat.storage_p_a_S_Jpp.colwise().sum().eval().asDiagonal();
269template <
typename ShapeFunctionDisplacement,
typename ShapeFunction,
270 int DisplacementDim,
typename ConstitutiveTraits>
273 ConstitutiveTraits>::
274 addToLocalMatrixData(
276 std::vector<double>
const& local_x,
277 std::vector<double>
const& local_x_prev,
281 std::vector<double>& local_rhs_data,
282 std::vector<double>& local_Jac_data)
const
284 constexpr auto local_matrix_dim =
286 assert(local_x.size() == local_matrix_dim);
289 typename ShapeMatricesTypeDisplacement::template MatrixType<
290 local_matrix_dim, local_matrix_dim>>(
291 local_Jac_data, local_matrix_dim, local_matrix_dim);
295 template VectorType<local_matrix_dim>>(
296 local_rhs_data, local_matrix_dim);
298 local_Jac.noalias() = loc_mat.Jac;
299 local_rhs.noalias() = -loc_mat.res;
304 block_TT(local_Jac).noalias() += loc_mat.M_TT / dt + loc_mat.K_TT;
306 loc_mat.K_Tp + loc_mat.dK_TT_dp + loc_mat.M_Tp / dt;
308 block_pT(local_Jac).noalias() += loc_mat.M_pT / dt + loc_mat.K_pT;
310 loc_mat.K_pp + loc_mat.storage_p_a_p / dt + loc_mat.storage_p_a_S_Jpp;
311 block_pu(local_Jac).noalias() = loc_mat.M_pu / dt;
316 auto const [T, p_L, u] =
localDOF(local_x);
317 auto const [T_prev, p_L_prev, u_prev] =
localDOF(local_x_prev);
319 block_T(local_rhs).noalias() -= loc_mat.M_TT * (T - T_prev) / dt +
320 loc_mat.K_TT * T + loc_mat.K_Tp * p_L +
321 loc_mat.M_Tp * (p_L - p_L_prev) / dt;
322 block_p(local_rhs).noalias() -=
323 loc_mat.K_pp * p_L + loc_mat.K_pT * T +
324 (loc_mat.storage_p_a_p + loc_mat.storage_p_a_S) * (p_L - p_L_prev) /
326 loc_mat.M_pu * (u - u_prev) / dt + loc_mat.M_pT * (T - T_prev) / dt;
329template <
typename ShapeFunctionDisplacement,
typename ShapeFunction,
330 int DisplacementDim,
typename ConstitutiveTraits>
333 ConstitutiveTraits>::
334 assembleWithJacobianSingleIP(
335 double const t,
double const dt,
337 std::vector<double>
const& local_x,
338 std::vector<double>
const& local_x_prev,
341 ConstitutiveTraits>
::IpData const& ip_data,
342 typename ConstitutiveTraits::ConstitutiveSetting& CS,
347 typename ConstitutiveTraits::StatefulData& current_state,
348 typename ConstitutiveTraits::StatefulDataPrev
const& prev_state,
350 typename ConstitutiveTraits::OutputData& output_data)
const
352 auto const& N_u = ip_data.N_u;
353 auto const& dNdx_u = ip_data.dNdx_u;
356 auto const& N = ip_data.N_p;
357 auto const& dNdx = ip_data.dNdx_p;
361 ShapeFunctionDisplacement::NPOINTS,
366 typename ConstitutiveTraits::ConstitutiveData CD;
368 auto const [T, p_L, u] =
localDOF(local_x);
369 auto const [T_prev, p_L_prev, u_prev] =
localDOF(local_x_prev);
372 auto models = ConstitutiveTraits::createConstitutiveModels(
374 typename ConstitutiveTraits::ConstitutiveTempData tmp;
376 double const T_ip = N * T;
377 double const T_prev_ip = N * T_prev;
380 double const p_cap_ip = -N * p_L;
381 double const p_cap_prev_ip = -N * p_L_prev;
386 CS.eval(models, t, dt, x_position,
388 {T_ip, T_prev_ip, grad_T_ip},
389 {p_cap_ip, p_cap_prev_ip, grad_p_cap_ip},
390 eps, current_state, prev_state, mat_state, tmp, output_data,
395 NodalMatrix
const NTN = N.transpose() * N;
396 NodalMatrix
const dNTdN = dNdx.transpose() * dNdx;
401 DisplacementDim)>::identity2;
402 typename ShapeMatricesTypeDisplacement::template MatrixType<
404 BTI2N = B.transpose() * identity2 * N;
427 dNdx.transpose() * std::get<EqPData<DisplacementDim>>(CD).rhs_p_dNT_V;
433 static_cast<int>(this->
process_data_.apply_body_force_for_deformation) *
435 std::get<GravityData<DisplacementDim>>(CD).volumetric_body_force;
438 out.storage_p_a_p.noalias() =
439 std::get<EqPData<DisplacementDim>>(CD).storage_p_a_p_X_NTN * NTN;
440 out.storage_p_a_S.noalias() =
441 std::get<TRMStorageData>(CD).storage_p_a_S_X_NTN * NTN;
442 out.storage_p_a_S_Jpp.noalias() =
443 std::get<TRMStorageData>(CD).storage_p_a_S_Jpp_X_NTN * NTN;
447 std::get<EqTData<DisplacementDim>>(CD).M_TT_X_NTN * NTN;
449 std::get<TRMVaporDiffusionData<DisplacementDim>>(CD).M_Tp_X_NTN * NTN;
452 std::get<EqPData<DisplacementDim>>(CD).M_pT_X_NTN * NTN;
454 std::get<EqPData<DisplacementDim>>(CD).M_pu_X_BTI2N * BTI2N.transpose();
459 std::get<TRMHeatStorageAndFluxData<DisplacementDim>>(CD)
463 (std::get<EqTData<DisplacementDim>>(CD).K_TT_NT_V_dN.transpose() *
465 std::get<TRMVaporDiffusionData<DisplacementDim>>(CD).K_TT_X_dNTdN *
468 out.dK_TT_dp.noalias() =
470 (std::get<TRMHeatStorageAndFluxData<DisplacementDim>>(CD)
471 .K_Tp_NT_V_dN.transpose() *
473 std::get<TRMHeatStorageAndFluxData<DisplacementDim>>(CD).K_Tp_X_NTN *
477 std::get<ThermoOsmosisData<DisplacementDim>>(CD).K_Tp_Laplace *
483 dNdx.transpose() * std::get<EqPData<DisplacementDim>>(CD).K_pp_Laplace *
489 std::get<ThermoOsmosisData<DisplacementDim>>(CD).K_pT_Laplace * dNdx;
493 std::get<TRMVaporDiffusionData<DisplacementDim>>(CD).J_pT_X_dNTdN *
496 std::get<TRMStorageData>(CD).J_pp_X_NTN * NTN +
497 std::get<EqPData<DisplacementDim>>(CD).J_pp_X_BTI2NT_u_dot_N *
498 BTI2N.transpose() * (u - u_prev) / dt *
501 std::get<EqPData<DisplacementDim>>(CD).J_pp_dNT_V_N * N;
505 std::get<SolidMechanicsDataStateless<DisplacementDim>>(CD).J_uT_BT_K_N *
509 std::get<SolidMechanicsDataStateless<DisplacementDim>>(CD)
513 std::get<GravityData<DisplacementDim>>(CD).J_up_HT_V_N * N;
516 std::get<SolidMechanicsDataStateless<DisplacementDim>>(CD)
520 out *= ip_data.integration_weight;
523template <
typename ShapeFunctionDisplacement,
typename ShapeFunction,
524 int DisplacementDim,
typename ConstitutiveTraits>
527 ConstitutiveTraits>::
528 computeSecondaryVariableConcrete(
double const t,
double const dt,
529 Eigen::VectorXd
const& local_x,
530 Eigen::VectorXd
const& local_x_prev)
532 auto const T =
block_T(local_x);
533 auto const p_L =
block_p(local_x);
534 auto const u =
block_u(local_x);
536 auto const T_prev =
block_T(local_x_prev);
537 auto const p_L_prev =
block_p(local_x_prev);
539 auto const e_id = this->
element_.getID();
541 auto& medium = *process_data.media_map.getMedium(e_id);
543 unsigned const n_integration_points =
546 typename ConstitutiveTraits::ConstitutiveSetting constitutive_setting;
548 auto models = ConstitutiveTraits::createConstitutiveModels(
550 typename ConstitutiveTraits::ConstitutiveTempData tmp;
551 typename ConstitutiveTraits::ConstitutiveData CD;
553 for (
unsigned ip = 0; ip < n_integration_points; ip++)
561 auto const& N = ip_data.N_p;
562 auto const& N_u = ip_data.N_u;
563 auto const& dNdx_u = ip_data.dNdx_u;
564 auto const& dNdx = ip_data.dNdx_p;
567 std::nullopt, this->
element_.getID(),
577 ShapeFunctionDisplacement::NPOINTS,
581 double const T_ip = N * T;
582 double const T_prev_ip = N * T_prev;
585 double const p_cap_ip = -N * p_L;
586 double const p_cap_prev_ip = -N * p_L_prev;
591 constitutive_setting.eval(models,
594 {T_ip, T_prev_ip, grad_T_ip},
595 {p_cap_ip, p_cap_prev_ip, grad_p_cap_ip},
602 ShapeFunction,
typename ShapeFunctionDisplacement::MeshElement,
604 *process_data.pressure_interpolated);
606 ShapeFunction,
typename ShapeFunctionDisplacement::MeshElement,
607 DisplacementDim>(this->element_, this->is_axially_symmetric_, T,
608 *process_data.temperature_interpolated);
Property const & property(PropertyType const &p) const
double gas_phase_pressure
double capillary_pressure
double liquid_phase_pressure
constexpr double getWeight() const
MathLib::WeightedPoint const & getWeightedPoint(unsigned const igp) const
std::optional< MathLib::Point3d > const getCoordinates() const
MatrixType< _kelvin_vector_size, _number_of_dof > BMatrixType
static auto block_TT(auto &mat)
static constexpr auto & N_u_op
void addToLocalMatrixData(double const dt, std::vector< double > const &local_x, std::vector< double > const &local_x_prev, LocalMatrices const &loc_mat, std::vector< double > &local_rhs_data, std::vector< double > &local_Jac_data) const
static constexpr int temperature_size
static constexpr int temperature_index
static constexpr int pressure_size
void assembleWithJacobianSingleIP(double const t, double const dt, ParameterLib::SpatialPosition const &x_position, std::vector< double > const &local_x, std::vector< double > const &local_x_prev, IpData const &ip_data, typename ConstitutiveTraits::ConstitutiveSetting &CS, MaterialPropertyLib::Medium &medium, LocalMatrices &out, typename ConstitutiveTraits::StatefulData ¤t_state, typename ConstitutiveTraits::StatefulDataPrev const &prev_state, MaterialStateData< DisplacementDim > &mat_state, typename ConstitutiveTraits::OutputData &output_data) const
ShapeMatrixPolicyType< ShapeFunctionDisplacement, DisplacementDim > ShapeMatricesTypeDisplacement
static auto block_p(auto &vec)
typename ShapeMatricesType::GlobalDimVectorType GlobalDimVectorType
static auto block_pu(auto &mat)
std::vector< IpData > ip_data_
void convertInitialStressType(unsigned const ip, double const t, ParameterLib::SpatialPosition const x_position, MaterialPropertyLib::Medium const &medium, MPL::VariableArray const &variables, double const p_at_ip)
typename BMatricesType::KelvinVectorType KelvinVectorType
static auto block_up(auto &mat)
static auto block_Tp(auto &mat)
ThermoRichardsMechanicsLocalAssembler(ThermoRichardsMechanicsLocalAssembler const &)=delete
void setInitialConditionsConcrete(Eigen::VectorXd const local_x, double const t, int const process_id) override
ShapeMatrixPolicyType< ShapeFunction, DisplacementDim > ShapeMatricesType
static auto block_pp(auto &mat)
IntegrationPointData< ShapeMatricesTypeDisplacement, ShapeMatricesType, DisplacementDim, ShapeFunctionDisplacement::NPOINTS > IpData
static auto block_pT(auto &mat)
static auto block_uT(auto &mat)
static auto block_uu(auto &mat)
static constexpr auto localDOF(std::vector< double > const &x)
static constexpr int displacement_size
static constexpr int pressure_index
static auto block_u(auto &vec)
static auto block_T(auto &vec)
void massLumping(LocalMatrices &loc_mat) const
@ bishops_effective_stress
constexpr int kelvin_vector_dimensions(int const displacement_dim)
Kelvin vector dimensions for given displacement dimension.
Eigen::Map< Vector > createZeroedVector(std::vector< double > &data, Eigen::VectorXd::Index size)
Eigen::Map< Matrix > createZeroedMatrix(std::vector< double > &data, Eigen::MatrixXd::Index rows, Eigen::MatrixXd::Index cols)
void shapeFunctionInterpolate(const NodalValues &, const ShapeMatrix &)
void interpolateToHigherOrderNodes(MeshLib::Element const &element, bool const is_axially_symmetric, Eigen::MatrixBase< EigenMatrixType > const &node_values, MeshLib::PropertyVector< double > &interpolated_values_global_vector)
std::vector< typename ShapeMatricesType::ShapeMatrices, Eigen::aligned_allocator< typename ShapeMatricesType::ShapeMatrices > > initShapeMatrices(MeshLib::Element const &e, bool const is_axially_symmetric, IntegrationMethod const &integration_method)
std::array< double, 3 > interpolateCoordinates(MeshLib::Element const &e, typename ShapeMatricesType::ShapeMatrices::ShapeType const &N)
auto & get(Tuples &... ts)
BMatrixType computeBMatrix(DNDX_Type const &dNdx, N_Type const &N, const double radius, const bool is_axially_symmetric)
Fills a B-matrix based on given shape function dN/dx values.
MatrixType< ShapeFunction::NPOINTS, ShapeFunction::NPOINTS > NodalMatrixType
static Eigen::Matrix< double, KelvinVectorSize, 1 > const identity2
Kelvin mapping of 2nd order identity tensor.
bool const is_axially_symmetric_
NumLib::GenericIntegrationMethod const & integration_method_
std::vector< typename ConstitutiveTraits::OutputData > output_data_
ThermoRichardsMechanicsProcessData< DisplacementDim, ConstitutiveTraits > & process_data_
LocalAssemblerInterface(MeshLib::Element const &e, NumLib::GenericIntegrationMethod const &integration_method, bool const is_axially_symmetric, ThermoRichardsMechanicsProcessData< DisplacementDim, ConstitutiveTraits > &process_data)
std::vector< typename ConstitutiveTraits::StatefulDataPrev > prev_states_
std::vector< MaterialStateData< DisplacementDim > > material_states_
ConstitutiveTraits::SolidConstitutiveRelation const & solid_material_
std::vector< typename ConstitutiveTraits::StatefulData > current_states_
MeshLib::Element const & element_