13 #include <MGIS/Behaviour/Behaviour.hxx>
14 #include <MGIS/Behaviour/BehaviourData.hxx>
15 #include "MGIS/LibrariesManager.hxx"
16 #include "MGIS/Behaviour/Integrate.hxx"
19using namespace mgis::behaviour;
23 constexpr double inv_sqr2 = boost::math::constants::half_root_two<double>();
25 #define VOIGT_VEC_SYMM_3D(VEC) \
26 VEC[0], inv_sqr2 *VEC[3], inv_sqr2 *VEC[4], VEC[1], inv_sqr2 *VEC[5], VEC[2]
28 #define VOIGT_VEC_SYMM_2D(VEC) VEC[0], inv_sqr2 *VEC[3], VEC[1]
30 #define VOIGT_VEC_SYMM_2D_FULL(VEC) \
31 VEC[0], inv_sqr2 *VEC[3], 0., VEC[1], 0., VEC[2]
33 #define VOIGT_VEC_3D(VEC) \
34 VEC[0], VEC[3], VEC[5], VEC[4], VEC[1], VEC[7], VEC[6], VEC[8], VEC[2]
36 #define VOIGT_VEC_2D(VEC) VEC[0], VEC[3], VEC[4], VEC[1]
38 #define VOIGT_VEC_2D_FULL(VEC) \
39 VEC[0], VEC[3], 0., VEC[4], VEC[1], 0., 0., 0., VEC[2]
41template <ModelHypothesis MH>
struct MFrontEleType;
45 MFrontEleType() =
delete;
46 ~MFrontEleType() =
delete;
50 using PostProcDomainEle = PostProcBrokenMeshInMoabBase<DomainEle>;
57 MFrontEleType() =
delete;
58 ~MFrontEleType() =
delete;
62 using PostProcDomainEle = PostProcBrokenMeshInMoabBase<DomainEle>;
69 MFrontEleType() =
delete;
70 ~MFrontEleType() =
delete;
74 using PostProcDomainEle = PostProcBrokenMeshInMoabBase<DomainEle>;
80template <ModelHypothesis MH, AssemblyType AT = AssemblyType::PETSC>
81struct MFrontInterfaceImpl :
public MFrontInterface {
85 using DomainEle =
typename MFrontEleType<MH>::DomainEle;
86 using DomainEleOp =
typename MFrontEleType<MH>::DomainEleOp;
87 using PostProcDomainEle =
typename MFrontEleType<MH>::PostProcDomainEle;
89 static constexpr int DIM = MFrontEleType<MH>::SPACE_DIM;
91 using OpInternalForce =
92 typename FormsIntegrators<DomainEleOp>::template Assembly<AT>::
94 using OpAssembleLhsFiniteStrains =
95 typename FormsIntegrators<DomainEleOp>::template Assembly<
98 using OpAssembleLhsSmallStrains =
99 typename FormsIntegrators<DomainEleOp>::template Assembly<AT>::
106 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
110 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
114 setUpdateInternalVariablesOperators(ForcesAndSourcesCore::RuleHookFun rule,
117 MoFEMErrorCode setPostProcessOperators(ForcesAndSourcesCore::RuleHookFun rule,
123 std::string fe_name)
override;
126 string fe_name)
override;
129 setMonitor(boost::shared_ptr<MoFEM::FEMethod> monitor_ptr)
override {
131 monitorPtr = monitor_ptr;
138 string optionsPrefix;
140 SmartPetscObj<DM> dM;
143 PetscBool saveDomain;
145 PetscBool testJacobian;
146 PetscReal randomFieldScale;
148 bool isFiniteKinematics;
152 boost::shared_ptr<PostProcDomainEle> postProcFe;
153 boost::shared_ptr<DomainEle> updateIntVariablesElePtr;
155 boost::shared_ptr<moab::Interface> moabGaussIntPtr;
157 boost::shared_ptr<MoFEM::FEMethod> monitorPtr;
159 boost::shared_ptr<CommonData> commonDataPtr;
162template <ModelHypothesis MH>
163boost::shared_ptr<MFrontInterface>
168 return boost::make_shared<MFrontInterfaceImpl<MH, PETSC>>(m_field);
170 return boost::make_shared<MFrontInterfaceImpl<MH, SCHUR>>(m_field);
172 return boost::make_shared<MFrontInterfaceImpl<MH, BLOCK_MAT>>(m_field);
174 return boost::make_shared<MFrontInterfaceImpl<MH, BLOCK_SCHUR>>(m_field);
179 return boost::shared_ptr<MFrontInterface>();
182boost::shared_ptr<MFrontInterface>
188 return createMFrontInterfaceImpl<TRIDIMENSIONAL>(m_field, at);
190 return createMFrontInterfaceImpl<PLANESTRAIN>(m_field, at);
192 return createMFrontInterfaceImpl<AXISYMMETRICAL>(m_field, at);
196 return boost::shared_ptr<MFrontInterface>();
199struct MFrontInterface::CommonData {
204 const int nb_gauss_pts,
const int var_size,
205 const int grad_size,
const int stress_size,
213 boost::shared_ptr<MatrixDouble> mGradPtr;
214 boost::shared_ptr<MatrixDouble> mStressPtr;
215 boost::shared_ptr<MatrixDouble> mFullStrainPtr;
216 boost::shared_ptr<MatrixDouble> mFullStressPtr;
218 boost::shared_ptr<MatrixDouble> mPrevGradPtr;
219 boost::shared_ptr<MatrixDouble> mPrevStressPtr;
221 boost::shared_ptr<MatrixDouble> mDispPtr;
222 boost::shared_ptr<MatrixDouble> materialTangentPtr;
223 boost::shared_ptr<MatrixDouble> mFullTangentPtr;
224 boost::shared_ptr<MatrixDouble> internalVariablePtr;
228 std::map<int, BlockData> setOfBlocksData;
229 std::map<EntityHandle, int> blocksIDmap;
235 Tag internalVariableTag;
240enum DataTags { RHS = 0, LHS };
242struct MFrontInterface::CommonData::BlockData {
245 : isFiniteStrain(false), behaviourPath(
"src/libBehaviour.so"),
246 behaviourName(
"LinearElasticity") {
249 externalVariable = 0;
255 behDataPtr->K[0] = 0;
257 behDataPtr->K[0] = 5;
262 MoFEMErrorCode setBlockBehaviourData(
bool set_params_from_blocks);
267 string behaviourPath;
268 string behaviourName;
270 boost::shared_ptr<mgis::behaviour::Behaviour> mGisBehaviour;
271 mgis::behaviour::BehaviourDataView bView;
272 boost::shared_ptr<mgis::behaviour::BehaviourData> behDataPtr;
279 vector<double> params;
283 double externalVariable;
288MoFEMErrorCode MFrontInterface::CommonData::BlockData::setBlockBehaviourData(
289 bool set_params_from_blocks) {
293 auto &mgis_bv = *mGisBehaviour;
295 sizeIntVar = getArraySize(mgis_bv.isvs, mgis_bv.hypothesis);
296 sizeExtVar = getArraySize(mgis_bv.esvs, mgis_bv.hypothesis);
297 sizeGradVar = getArraySize(mgis_bv.gradients, mgis_bv.hypothesis);
299 getArraySize(mgis_bv.thermodynamic_forces, mgis_bv.hypothesis);
301 behDataPtr = boost::make_shared<BehaviourData>(BehaviourData{mgis_bv});
302 bView = make_view(*behDataPtr);
303 const int total_number_of_params = mgis_bv.mps.size();
305 if (set_params_from_blocks) {
307 if (params.size() < total_number_of_params)
309 "Not enough parameters supplied for this block. We have %ld "
310 "provided where %d are necessary for this block",
311 params.size(), total_number_of_params);
313 for (
int dd = 0;
dd < total_number_of_params; ++
dd) {
314 setMaterialProperty(behDataPtr->s0, dd, params[dd]);
315 setMaterialProperty(behDataPtr->s1, dd, params[dd]);
319 if (isFiniteStrain) {
320 behDataPtr->K[0] = 0;
321 behDataPtr->K[1] = 2;
322 behDataPtr->K[2] = 2;
324 behDataPtr->K[0] = 0;
325 behDataPtr->K[1] = 0;
328 for (
auto &mb : {&behDataPtr->s0, &behDataPtr->s1}) {
329 mb->dissipated_energy = dIssipation;
330 mb->stored_energy = storedEnergy;
331 setExternalStateVariable(*mb, 0, externalVariable);
340 string block_name =
"MAT_MFRONT";
342 if (it->getName().compare(0, block_name.size(), block_name) == 0) {
343 std::vector<double> block_data;
345 CHKERR it->getAttributes(block_data);
346 const int id = it->getMeshsetId();
348 CHKERR mField.get_moab().get_entities_by_dimension(
349 meshset, dim, setOfBlocksData[
id].eNts,
true);
350 for (
auto ent : setOfBlocksData[id].eNts)
351 blocksIDmap[ent] = id;
353 setOfBlocksData[id].iD = id;
354 setOfBlocksData[id].params.resize(block_data.size());
356 for (
int n = 0;
n != block_data.size();
n++)
357 setOfBlocksData[
id].params[
n] = block_data[
n];
365 const EntityHandle fe_ent,
const int nb_gauss_pts,
const int var_size,
369 auto mget_tag_data = [&](
Tag &m_tag, boost::shared_ptr<MatrixDouble> &m_mat,
370 const int &m_size,
bool is_def_grad =
false) {
375 rval = mField.get_moab().tag_get_by_ptr(
376 m_tag, &fe_ent, 1, (
const void **)&tag_data, &tag_size);
378 if (
rval != MB_SUCCESS || tag_size != m_size * nb_gauss_pts) {
379 m_mat->resize(nb_gauss_pts, m_size,
false);
383 for (
int gg = 0; gg != nb_gauss_pts; ++gg) {
388 void const *tag_data2[] = {&*m_mat->data().begin()};
389 const int tag_size2 = m_mat->data().size();
390 CHKERR mField.get_moab().tag_set_by_ptr(m_tag, &fe_ent, 1, tag_data2,
394 nb_gauss_pts, m_size,
395 ublas::shallow_array_adaptor<double>(tag_size, tag_data));
403 CHKERR mget_tag_data(internalVariableTag, internalVariablePtr, var_size);
404 CHKERR mget_tag_data(stressTag, mPrevStressPtr, grad_size);
405 CHKERR mget_tag_data(gradientTag, mPrevGradPtr, stress_size,
true);
411MFrontInterface::CommonData::setInternalVar(
const EntityHandle fe_ent) {
414 auto mset_tag_data = [&](
Tag &m_tag, boost::shared_ptr<MatrixDouble> &m_mat) {
416 void const *tag_data[] = {&*m_mat->data().begin()};
417 const int tag_size = m_mat->data().size();
418 CHKERR mField.get_moab().tag_set_by_ptr(m_tag, &fe_ent, 1, tag_data,
423 CHKERR mset_tag_data(internalVariableTag, internalVariablePtr);
424 CHKERR mset_tag_data(stressTag, mPrevStressPtr);
425 CHKERR mset_tag_data(gradientTag, mPrevGradPtr);
432 const int default_length = 0;
433 CHKERR mField.get_moab().tag_get_handle(
434 "_INTERNAL_VAR", default_length, MB_TYPE_DOUBLE, internalVariableTag,
435 MB_TAG_CREAT | MB_TAG_VARLEN | MB_TAG_SPARSE, PETSC_NULLPTR);
436 CHKERR mField.get_moab().tag_get_handle(
437 "_STRESS_TAG", default_length, MB_TYPE_DOUBLE, stressTag,
438 MB_TAG_CREAT | MB_TAG_VARLEN | MB_TAG_SPARSE, PETSC_NULLPTR);
439 CHKERR mField.get_moab().tag_get_handle(
440 "_GRAD_TAG", default_length, MB_TYPE_DOUBLE, gradientTag,
441 MB_TAG_CREAT | MB_TAG_VARLEN | MB_TAG_SPARSE, PETSC_NULLPTR);
449 for (
auto &[
id, data] : setOfBlocksData) {
450 CHKERR mField.get_moab().tag_clear_data(internalVariableTag, data.eNts,
452 CHKERR mField.get_moab().tag_clear_data(stressTag, data.eNts, &zero);
453 CHKERR mField.get_moab().tag_clear_data(gradientTag, data.eNts, &zero);
458template <ModelHypothesis MH, AssemblyType AT>
462 isFiniteKinematics =
false;
463 saveGauss = PETSC_FALSE;
464 saveDomain = PETSC_TRUE;
465 testJacobian = PETSC_FALSE;
466 randomFieldScale = 1.0;
467 optionsPrefix =
"mf_";
468 monitorPtr =
nullptr;
471 if (!LogManager::checkIfChannelExist(
"MFrontInterfaceWorld")) {
472 auto core_log = logging::core::get();
474 core_log->add_sink(LogManager::createSink(LogManager::getStrmWorld(),
475 "MFrontInterfaceWorld"));
476 core_log->add_sink(LogManager::createSink(LogManager::getStrmSync(),
477 "MFrontInterfaceSync"));
478 core_log->add_sink(LogManager::createSink(LogManager::getStrmSelf(),
479 "MFrontInterfaceSelf"));
481 LogManager::setLog(
"MFrontInterfaceWorld");
482 LogManager::setLog(
"MFrontInterfaceSync");
483 LogManager::setLog(
"MFrontInterfaceSelf");
490 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::noisy) <<
"MFront Interface created";
493template <ModelHypothesis MH, AssemblyType AT>
494MoFEMErrorCode MFrontInterfaceImpl<MH, AT>::getCommandLineParameters() {
497 PetscOptionsBegin(PETSC_COMM_WORLD, optionsPrefix.c_str(),
"",
"none");
499 CHKERR PetscOptionsBool(
"-save_gauss",
"save gauss pts (internal variables)",
500 "", saveGauss, &saveGauss, PETSC_NULLPTR);
501 CHKERR PetscOptionsBool(
"-save_domain",
"save results on a domain mesh",
"",
502 saveDomain, &saveDomain, PETSC_NULLPTR);
504 CHKERR PetscOptionsBool(
"-test_jacobian",
"test Jacobian (LHS matrix)",
"",
505 testJacobian, &testJacobian, PETSC_NULLPTR);
506 CHKERR PetscOptionsReal(
"-random_field_scale",
507 "scale for the finite difference jacobian",
"",
508 randomFieldScale, &randomFieldScale, PETSC_NULLPTR);
511 moabGaussIntPtr = boost::shared_ptr<moab::Interface>(
new moab::Core());
513 commonDataPtr = boost::make_shared<MFrontInterface::CommonData>(mField);
515 commonDataPtr->setBlocks(DIM);
516 commonDataPtr->createTags();
518 commonDataPtr->mGradPtr = boost::make_shared<MatrixDouble>();
519 commonDataPtr->mStressPtr = boost::make_shared<MatrixDouble>();
520 commonDataPtr->mFullStrainPtr = boost::make_shared<MatrixDouble>();
521 commonDataPtr->mFullStressPtr = boost::make_shared<MatrixDouble>();
522 commonDataPtr->mDispPtr = boost::make_shared<MatrixDouble>();
523 commonDataPtr->mPrevGradPtr = boost::make_shared<MatrixDouble>();
524 commonDataPtr->mPrevStressPtr = boost::make_shared<MatrixDouble>();
525 commonDataPtr->materialTangentPtr = boost::make_shared<MatrixDouble>();
526 commonDataPtr->mFullTangentPtr = boost::make_shared<MatrixDouble>();
527 commonDataPtr->internalVariablePtr = boost::make_shared<MatrixDouble>();
529 if (commonDataPtr->setOfBlocksData.empty() ||
530 commonDataPtr->blocksIDmap.empty()) {
532 "No blocksets on the mesh have been provided for MFront (e.g. "
536 auto check_lib_finite_strain = [&](
const std::string &lib,
537 const std::string &beh_name,
bool &flag) {
540 ifstream
f(lib.c_str());
542 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::error)
543 <<
"Problem with the behaviour path: " << lib;
545 auto &lm = mgis::LibrariesManager::get();
546 flag =
bool(lm.getBehaviourType(lib, beh_name) == 2) &&
547 (lm.getBehaviourKinematic(lib, beh_name) == 3);
551 auto op = FiniteStrainBehaviourOptions{};
552 op.stress_measure = FiniteStrainBehaviourOptions::PK1;
553 op.tangent_operator = FiniteStrainBehaviourOptions::DPK1_DF;
555 for (
auto &block : commonDataPtr->setOfBlocksData) {
556 const int &
id = block.first;
557 auto &lib_path = block.second.behaviourPath;
558 auto &name = block.second.behaviourName;
559 const string param_name =
"-block_" + to_string(
id);
560 const string param_path =
"-lib_path_" + to_string(
id);
561 const string param_from_blocks =
"-params_" + to_string(
id);
562 PetscBool set_from_blocks = PETSC_FALSE;
566 CHKERR PetscOptionsBool(param_from_blocks.c_str(),
567 "set parameters from blocks",
"", set_from_blocks,
568 &set_from_blocks, PETSC_NULLPTR);
570 CHKERR PetscOptionsString(param_name.c_str(),
"name of the behaviour",
"",
571 "LinearElasticity", char_name, 255, &is_param);
573 name = string(char_name);
574 string default_lib_path =
575 "src/libBehaviour." + string(DEFAULT_LIB_EXTENSION);
576 CHKERR PetscOptionsString(
577 param_path.c_str(),
"path to the behaviour library",
"",
578 default_lib_path.c_str(), char_name, 255, &is_param);
580 lib_path = string(char_name);
581 auto &mgis_bv_ptr = block.second.mGisBehaviour;
582 bool is_finite_strain =
false;
584 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform)
585 <<
"Loading behaviour from " << lib_path;
587 CHKERR check_lib_finite_strain(lib_path, name, is_finite_strain);
589 mgis::behaviour::Hypothesis
h;
592 h = mgis::behaviour::Hypothesis::TRIDIMENSIONAL;
593 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform)
594 <<
"Model hypothesis: TRIDIMENSIONAL";
597 h = mgis::behaviour::Hypothesis::PLANESTRAIN;
598 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform)
599 <<
"Model hypothesis: PLANESTRAIN";
602 h = mgis::behaviour::Hypothesis::AXISYMMETRICAL;
603 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform)
604 <<
"Model hypothesis: AXISYMMETRICAL";
610 if (is_finite_strain) {
611 mgis_bv_ptr = boost::make_shared<Behaviour>(load(op, lib_path, name,
h));
612 block.second.isFiniteStrain =
true;
614 mgis_bv_ptr = boost::make_shared<Behaviour>(load(lib_path, name,
h));
616 CHKERR block.second.setBlockBehaviourData(set_from_blocks);
617 for (
size_t dd = 0;
dd < mgis_bv_ptr->mps.size(); ++
dd) {
619 PetscBool is_set = PETSC_FALSE;
620 string param_cmd =
"-param_" + to_string(
id) +
"_" + to_string(dd);
621 CHKERR PetscOptionsScalar(param_cmd.c_str(),
"parameter from cmd",
"",
622 my_param, &my_param, &is_set);
625 setMaterialProperty(block.second.behDataPtr->s0, dd, my_param);
626 setMaterialProperty(block.second.behDataPtr->s1, dd, my_param);
629 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform)
630 << mgis_bv_ptr->behaviour <<
" behaviour loaded on block "
633 if (is_finite_strain)
634 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform)
635 <<
"Finite Strain Kinematics";
637 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform)
638 <<
"Small Strain Kinematics";
640 if (mgis_bv_ptr->isvs.size()) {
641 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform) <<
"Internal variables:";
642 for (
const auto &is : mgis_bv_ptr->isvs) {
643 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform) <<
": " << is.name;
647 if (mgis_bv_ptr->esvs.size()) {
648 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform) <<
"External variables:";
649 for (
const auto &es : mgis_bv_ptr->esvs) {
650 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform) <<
": " << es.name;
654 auto it = block.second.behDataPtr->s0.material_properties.begin();
657 if (mgis_bv_ptr->mps.size()) {
658 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform) <<
"Material properties:";
659 for (
const auto &mp : mgis_bv_ptr->mps) {
660 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform)
661 << nb++ <<
" : " << mp.name <<
" = " << *it++;
665 if (mgis_bv_ptr->params.size()) {
666 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform) <<
"Real parameters:";
667 for (
const auto &p : mgis_bv_ptr->params) {
668 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform) << nb++ <<
" : " << p;
672 if (mgis_bv_ptr->iparams.size()) {
673 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform) <<
"Integer parameters:";
674 for (
const auto &p : mgis_bv_ptr->iparams) {
675 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform) << nb++ <<
" : " << p;
679 if (mgis_bv_ptr->usparams.size()) {
680 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform)
681 <<
"Unsigned short parameters:";
682 for (
const auto &p : mgis_bv_ptr->usparams) {
683 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::inform) << nb++ <<
" : " << p;
690 auto check_behaviours_kinematics = [&](
bool &is_finite_kin) {
693 commonDataPtr->setOfBlocksData.begin()->second.isFiniteStrain;
694 for (
auto &block : commonDataPtr->setOfBlocksData) {
695 if (block.second.isFiniteStrain != is_finite_kin)
697 "All used MFront behaviours have to be of same kinematics "
704 CHKERR check_behaviours_kinematics(isFiniteKinematics);
706 auto get_base = [&]() {
708 CHKERR mField.get_moab().get_entities_by_dimension(0, DIM, domain_ents,
710 if (domain_ents.empty())
728 fieldBase = get_base();
729 MOFEM_LOG(
"MFrontInterfaceWorld", Sev::verbose)
730 <<
"MFront post process projection base: "
751template <
typename T>
inline size_t get_paraview_size(T &vsize) {
752 return vsize > 1 ? (vsize > 3 ? 9 : 3) : 1;
756template <
typename T1,
typename T2>
765 if (std::is_same<T2, Tensor4Pack<3>>::value) {
766 D(N0, N0, N0, N0) =
K[0];
767 D(N0, N0, N1, N1) =
K[1];
768 D(N0, N0, N2, N2) =
K[2];
769 D(N0, N0, N0, N1) =
K[3];
770 D(N0, N0, N1, N0) =
K[4];
771 D(N0, N0, N0, N2) =
K[5];
772 D(N0, N0, N2, N0) =
K[6];
773 D(N0, N0, N1, N2) =
K[7];
774 D(N0, N0, N2, N1) =
K[8];
775 D(N1, N1, N0, N0) =
K[9];
776 D(N1, N1, N1, N1) =
K[10];
777 D(N1, N1, N2, N2) =
K[11];
778 D(N1, N1, N0, N1) =
K[12];
779 D(N1, N1, N1, N0) =
K[13];
780 D(N1, N1, N0, N2) =
K[14];
781 D(N1, N1, N2, N0) =
K[15];
782 D(N1, N1, N1, N2) =
K[16];
783 D(N1, N1, N2, N1) =
K[17];
784 D(N2, N2, N0, N0) =
K[18];
785 D(N2, N2, N1, N1) =
K[19];
786 D(N2, N2, N2, N2) =
K[20];
787 D(N2, N2, N0, N1) =
K[21];
788 D(N2, N2, N1, N0) =
K[22];
789 D(N2, N2, N0, N2) =
K[23];
790 D(N2, N2, N2, N0) =
K[24];
791 D(N2, N2, N1, N2) =
K[25];
792 D(N2, N2, N2, N1) =
K[26];
793 D(N0, N1, N0, N0) =
K[27];
794 D(N0, N1, N1, N1) =
K[28];
795 D(N0, N1, N2, N2) =
K[29];
796 D(N0, N1, N0, N1) =
K[30];
797 D(N0, N1, N1, N0) =
K[31];
798 D(N0, N1, N0, N2) =
K[32];
799 D(N0, N1, N2, N0) =
K[33];
800 D(N0, N1, N1, N2) =
K[34];
801 D(N0, N1, N2, N1) =
K[35];
802 D(N1, N0, N0, N0) =
K[36];
803 D(N1, N0, N1, N1) =
K[37];
804 D(N1, N0, N2, N2) =
K[38];
805 D(N1, N0, N0, N1) =
K[39];
806 D(N1, N0, N1, N0) =
K[40];
807 D(N1, N0, N0, N2) =
K[41];
808 D(N1, N0, N2, N0) =
K[42];
809 D(N1, N0, N1, N2) =
K[43];
810 D(N1, N0, N2, N1) =
K[44];
811 D(N0, N2, N0, N0) =
K[45];
812 D(N0, N2, N1, N1) =
K[46];
813 D(N0, N2, N2, N2) =
K[47];
814 D(N0, N2, N0, N1) =
K[48];
815 D(N0, N2, N1, N0) =
K[49];
816 D(N0, N2, N0, N2) =
K[50];
817 D(N0, N2, N2, N0) =
K[51];
818 D(N0, N2, N1, N2) =
K[52];
819 D(N0, N2, N2, N1) =
K[53];
820 D(N2, N0, N0, N0) =
K[54];
821 D(N2, N0, N1, N1) =
K[55];
822 D(N2, N0, N2, N2) =
K[56];
823 D(N2, N0, N0, N1) =
K[57];
824 D(N2, N0, N1, N0) =
K[58];
825 D(N2, N0, N0, N2) =
K[59];
826 D(N2, N0, N2, N0) =
K[60];
827 D(N2, N0, N1, N2) =
K[61];
828 D(N2, N0, N2, N1) =
K[62];
829 D(N1, N2, N0, N0) =
K[63];
830 D(N1, N2, N1, N1) =
K[64];
831 D(N1, N2, N2, N2) =
K[65];
832 D(N1, N2, N0, N1) =
K[66];
833 D(N1, N2, N1, N0) =
K[67];
834 D(N1, N2, N0, N2) =
K[68];
835 D(N1, N2, N2, N0) =
K[69];
836 D(N1, N2, N1, N2) =
K[70];
837 D(N1, N2, N2, N1) =
K[71];
838 D(N2, N1, N0, N0) =
K[72];
839 D(N2, N1, N1, N1) =
K[73];
840 D(N2, N1, N2, N2) =
K[74];
841 D(N2, N1, N0, N1) =
K[75];
842 D(N2, N1, N1, N0) =
K[76];
843 D(N2, N1, N0, N2) =
K[77];
844 D(N2, N1, N2, N0) =
K[78];
845 D(N2, N1, N1, N2) =
K[79];
846 D(N2, N1, N2, N1) =
K[80];
850 if (std::is_same<T2, Tensor4Pack<2>>::value) {
851 D(N0, N0, N0, N0) =
K[0];
852 D(N0, N0, N1, N1) =
K[1];
854 D(N0, N0, N0, N1) =
K[3];
855 D(N0, N0, N1, N0) =
K[4];
856 D(N1, N1, N0, N0) =
K[5];
857 D(N1, N1, N1, N1) =
K[6];
859 D(N1, N1, N0, N1) =
K[8];
860 D(N1, N1, N1, N0) =
K[9];
862 D(N0, N1, N0, N0) =
K[15];
863 D(N0, N1, N1, N1) =
K[16];
865 D(N0, N1, N0, N1) =
K[18];
866 D(N0, N1, N1, N0) =
K[19];
867 D(N1, N0, N0, N0) =
K[20];
868 D(N1, N0, N1, N1) =
K[21];
870 D(N1, N0, N0, N1) =
K[23];
871 D(N1, N0, N1, N0) =
K[24];
875 if (std::is_same<T2, DdgPack<3>>::value) {
876 D(N0, N0, N0, N0) =
K[0];
877 D(N0, N0, N1, N1) =
K[1];
878 D(N0, N0, N2, N2) =
K[2];
880 D(N0, N0, N0, N1) = inv_sqr2 *
K[3];
881 D(N0, N0, N0, N2) = inv_sqr2 *
K[4];
882 D(N0, N0, N1, N2) = inv_sqr2 *
K[5];
884 D(N1, N1, N0, N0) =
K[6];
885 D(N1, N1, N1, N1) =
K[7];
886 D(N1, N1, N2, N2) =
K[8];
888 D(N1, N1, N0, N1) = inv_sqr2 *
K[9];
889 D(N1, N1, N0, N2) = inv_sqr2 *
K[10];
890 D(N1, N1, N1, N2) = inv_sqr2 *
K[11];
892 D(N2, N2, N0, N0) =
K[12];
893 D(N2, N2, N1, N1) =
K[13];
894 D(N2, N2, N2, N2) =
K[14];
896 D(N2, N2, N0, N1) = inv_sqr2 *
K[15];
897 D(N2, N2, N0, N2) = inv_sqr2 *
K[16];
898 D(N2, N2, N1, N2) = inv_sqr2 *
K[17];
900 D(N0, N1, N0, N0) = inv_sqr2 *
K[18];
901 D(N0, N1, N1, N1) = inv_sqr2 *
K[19];
902 D(N0, N1, N2, N2) = inv_sqr2 *
K[20];
904 D(N0, N1, N0, N1) = 0.5 *
K[21];
905 D(N0, N1, N0, N2) = 0.5 *
K[22];
906 D(N0, N1, N1, N2) = 0.5 *
K[23];
908 D(N0, N2, N0, N0) = inv_sqr2 *
K[24];
909 D(N0, N2, N1, N1) = inv_sqr2 *
K[25];
910 D(N0, N2, N2, N2) = inv_sqr2 *
K[26];
912 D(N0, N2, N0, N1) = 0.5 *
K[27];
913 D(N0, N2, N0, N2) = 0.5 *
K[28];
914 D(N0, N2, N1, N2) = 0.5 *
K[29];
916 D(N1, N2, N0, N0) = inv_sqr2 *
K[30];
917 D(N1, N2, N1, N1) = inv_sqr2 *
K[31];
918 D(N1, N2, N2, N2) = inv_sqr2 *
K[32];
920 D(N1, N2, N0, N1) = 0.5 *
K[33];
921 D(N1, N2, N0, N2) = 0.5 *
K[34];
922 D(N1, N2, N1, N2) = 0.5 *
K[35];
926 if (std::is_same<T2, DdgPack<2>>::value) {
928 D(N0, N0, N0, N0) =
K[0];
929 D(N0, N0, N1, N1) =
K[1];
932 D(N0, N0, N0, N1) = inv_sqr2 *
K[3];
934 D(N1, N1, N0, N0) =
K[4];
935 D(N1, N1, N1, N1) =
K[5];
938 D(N1, N1, N0, N1) = inv_sqr2 *
K[7];
942 D(N0, N1, N0, N0) = inv_sqr2 *
K[12];
943 D(N0, N1, N1, N1) = inv_sqr2 *
K[13];
947 D(N0, N1, N0, N1) = 0.5 *
K[15];
953template <
bool IS_LARGE_STRAIN,
typename T1,
typename T2>
962 if constexpr (IS_LARGE_STRAIN) {
963 D(N0, N0, N0, N0) =
K[0];
964 D(N0, N0, N1, N1) =
K[1];
965 D(N0, N0, N2, N2) =
K[2];
966 D(N0, N0, N0, N1) =
K[3];
967 D(N0, N0, N1, N0) =
K[4];
968 D(N1, N1, N0, N0) =
K[5];
969 D(N1, N1, N1, N1) =
K[6];
970 D(N1, N1, N2, N2) =
K[7];
971 D(N1, N1, N0, N1) =
K[8];
972 D(N1, N1, N1, N0) =
K[9];
973 D(N2, N2, N0, N0) =
K[10];
974 D(N2, N2, N1, N1) =
K[11];
975 D(N2, N2, N2, N2) =
K[12];
976 D(N2, N2, N0, N1) =
K[13];
977 D(N2, N2, N1, N0) =
K[14];
978 D(N0, N1, N0, N0) =
K[15];
979 D(N0, N1, N1, N1) =
K[16];
980 D(N0, N1, N2, N2) =
K[17];
981 D(N0, N1, N0, N1) =
K[18];
982 D(N0, N1, N1, N0) =
K[19];
983 D(N1, N0, N0, N0) =
K[20];
984 D(N1, N0, N1, N1) =
K[21];
985 D(N1, N0, N2, N2) =
K[22];
986 D(N1, N0, N0, N1) =
K[23];
987 D(N1, N0, N1, N0) =
K[24];
990 D(N0, N0, N0, N0) =
K[0];
991 D(N0, N0, N1, N1) =
K[1];
992 D(N0, N0, N2, N2) =
K[2];
994 D(N0, N0, N0, N1) = inv_sqr2 *
K[3];
996 D(N1, N1, N0, N0) =
K[4];
997 D(N1, N1, N1, N1) =
K[5];
998 D(N1, N1, N2, N2) =
K[6];
1000 D(N1, N1, N0, N1) = inv_sqr2 *
K[7];
1002 D(N2, N2, N0, N0) =
K[8];
1003 D(N2, N2, N1, N1) =
K[9];
1004 D(N2, N2, N2, N2) =
K[10];
1006 D(N2, N2, N0, N1) = inv_sqr2 *
K[11];
1007 D(N2, N2, N1, N0) =
D(N2, N2, N0, N1);
1009 D(N0, N1, N0, N0) = inv_sqr2 *
K[12];
1010 D(N0, N1, N1, N1) = inv_sqr2 *
K[13];
1012 D(N0, N1, N2, N2) = inv_sqr2 *
K[14];
1013 D(N1, N0, N2, N2) =
D(N0, N1, N2, N2);
1015 D(N0, N1, N0, N1) = 0.5 *
K[15];
1021template <
typename T> T get_tangent_tensor(MatrixDouble &mat);
1024Tensor4Pack<3> get_tangent_tensor<Tensor4Pack<3>>(
MatrixDouble &mat) {
1025 return getFTensor4FromMat<3, 3, 3, 3>(mat);
1029Tensor4Pack<2> get_tangent_tensor<Tensor4Pack<2>>(
MatrixDouble &mat) {
1030 return getFTensor4FromMat<2, 2, 2, 2>(mat);
1033template <> DdgPack<3> get_tangent_tensor<DdgPack<3>>(
MatrixDouble &mat) {
1034 return getFTensor4DdgFromMat<3, 3>(mat);
1037template <> DdgPack<2> get_tangent_tensor<DdgPack<2>>(
MatrixDouble &mat) {
1038 return getFTensor4DdgFromMat<2, 2>(mat);
1041template <
bool IS_LARGE_STRAIN, ModelHypothesis MH>
1043mgis_integration(
size_t gg, Tensor2Pack<MFrontEleType<MH>::SPACE_DIM> &t_grad,
1044 Tensor1Pack<MFrontEleType<MH>::SPACE_DIM> &t_disp,
1045 Tensor1PackCoords &t_coords,
1046 MFrontInterface::CommonData &common_data,
1047 MFrontInterface::CommonData::BlockData &block_data) {
1050 static constexpr int DIM = MFrontEleType<MH>::SPACE_DIM;
1052 int check_integration;
1053 MatrixDouble &mat_int = *common_data.internalVariablePtr;
1055 MatrixDouble &mat_stress0 = *common_data.mPrevStressPtr;
1057 int &size_of_vars = block_data.sizeIntVar;
1058 int &size_of_grad = block_data.sizeGradVar;
1059 int &size_of_stress = block_data.sizeStressVar;
1061 auto &mgis_bv = *block_data.mGisBehaviour;
1066 if constexpr (IS_LARGE_STRAIN) {
1070 setGradient(block_data.behDataPtr->s1, 0, size_of_grad,
1073 setGradient(block_data.behDataPtr->s1, 0, size_of_grad,
1074 &*getVoigtVec<DIM>(t_strain).data());
1078 t_strain(
i,
j) = (t_grad(
i,
j) || t_grad(
j,
i)) / 2;
1081 block_data.behDataPtr->s1, 0, size_of_grad,
1084 setGradient(block_data.behDataPtr->s1, 0, size_of_grad,
1085 &*getVoigtVecSymm<DIM>(t_strain).data());
1091 setGradient(block_data.behDataPtr->s0, 0, size_of_grad, &*grad0_vec.begin());
1093 auto stress0_vec =
getVectorAdaptor(&mat_stress0.data()[gg * size_of_stress],
1095 setThermodynamicForce(block_data.behDataPtr->s0, 0, size_of_stress,
1096 &*stress0_vec.begin());
1101 setInternalStateVariable(block_data.behDataPtr->s0, 0, size_of_vars,
1102 &*internal_var.begin());
1105 check_integration = mgis::behaviour::integrate(block_data.bView, mgis_bv);
1106 switch (check_integration) {
1108 MOFEM_LOG(
"WORLD", Sev::error) <<
"Mfront integration failed";
1112 <<
"Mfront integration succeeded but results are unreliable";
1122template <
bool UPDATE,
bool IS_LARGE_STRAIN, ModelHypothesis MH>
1124 static constexpr int DIM = MFrontEleType<MH>::SPACE_DIM;
1125 using DomainEleOp =
typename MFrontEleType<MH>::DomainEleOp;
1128 boost::shared_ptr<MFrontInterface::CommonData> common_data_ptr,
1129 boost::shared_ptr<FEMethod> monitor_ptr)
1131 commonDataPtr(common_data_ptr), monitorPtr(monitor_ptr) {
1132 std::fill(&DomainEleOp::doEntities[MBEDGE],
1133 &DomainEleOp::doEntities[MBMAXTYPE],
false);
1138 boost::shared_ptr<MFrontInterface::CommonData> commonDataPtr;
1139 boost::shared_ptr<FEMethod> monitorPtr;
1142template <ModelHypothesis MH>
1143using OpUpdateVariablesFiniteStrains = OpStressTmp<true, true, MH>;
1145template <ModelHypothesis MH>
1146using OpUpdateVariablesSmallStrains = OpStressTmp<true, false, MH>;
1148template <ModelHypothesis MH>
1149using OpStressFiniteStrains = OpStressTmp<false, true, MH>;
1151template <ModelHypothesis MH>
1152using OpStressSmallStrains = OpStressTmp<false, false, MH>;
1154template <
bool UPDATE,
bool IS_LARGE_STRAIN, ModelHypothesis MH>
1155MoFEMErrorCode OpStressTmp<UPDATE, IS_LARGE_STRAIN, MH>::doWork(
int side,
1166 const size_t nb_gauss_pts = commonDataPtr->mGradPtr->size2();
1167 auto fe_ent = DomainEleOp::getNumeredEntFiniteElementPtr()->getEnt();
1168 auto id = commonDataPtr->blocksIDmap.at(fe_ent);
1169 auto &dAta = commonDataPtr->setOfBlocksData.at(
id);
1171 if (monitorPtr ==
nullptr)
1173 "Time Monitor (FEMethod) has not been set for MFrontInterfaceImpl. "
1174 "Make sure to call setMonitor before calling "
1175 "opFactoryDomainRhs and opFactoryDomainLhs");
1177 dAta.setTag(DataTags::RHS);
1178 dAta.behDataPtr->dt = monitorPtr->ts_dt;
1179 dAta.bView.dt = monitorPtr->ts_dt;
1181 CHKERR commonDataPtr->getInternalVar(fe_ent, nb_gauss_pts, dAta.sizeIntVar,
1182 dAta.sizeGradVar, dAta.sizeStressVar,
1185 MatrixDouble &mat_int = *commonDataPtr->internalVariablePtr;
1186 MatrixDouble &mat_grad0 = *commonDataPtr->mPrevGradPtr;
1187 MatrixDouble &mat_stress0 = *commonDataPtr->mPrevStressPtr;
1189 auto t_grad = getFTensor2FromMat<DIM, DIM>(*(commonDataPtr->mGradPtr));
1190 auto t_disp = getFTensor1FromMat<DIM>(*(commonDataPtr->mDispPtr));
1191 auto t_coords = DomainEleOp::getFTensor1CoordsAtGaussPts();
1193 commonDataPtr->mStressPtr->resize(DIM * DIM, nb_gauss_pts);
1194 commonDataPtr->mStressPtr->clear();
1195 auto t_stress = getFTensor2FromMat<DIM, DIM>(*(commonDataPtr->mStressPtr));
1197 commonDataPtr->mFullStressPtr->resize(3 * 3, nb_gauss_pts);
1198 commonDataPtr->mFullStressPtr->clear();
1199 auto t_full_stress =
1200 getFTensor2FromMat<3, 3>(*(commonDataPtr->mFullStressPtr));
1202 for (
size_t gg = 0; gg != nb_gauss_pts; ++gg) {
1204 CHKERR mgis_integration<IS_LARGE_STRAIN, MH>(gg, t_grad, t_disp, t_coords,
1205 *commonDataPtr, dAta);
1207 if constexpr (DIM == 3) {
1209 if constexpr (IS_LARGE_STRAIN) {
1211 VOIGT_VEC_3D(getThermodynamicForce(dAta.behDataPtr->s1, 0)));
1214 VOIGT_VEC_SYMM_3D(getThermodynamicForce(dAta.behDataPtr->s1, 0))));
1216 t_stress(
i,
j) = t_force(
i,
j);
1217 }
else if constexpr (DIM == 2) {
1220 if constexpr (IS_LARGE_STRAIN) {
1222 VOIGT_VEC_2D(getThermodynamicForce(dAta.behDataPtr->s1, 0)));
1224 VOIGT_VEC_2D_FULL(getThermodynamicForce(dAta.behDataPtr->s1, 0)));
1227 VOIGT_VEC_SYMM_2D(getThermodynamicForce(dAta.behDataPtr->s1, 0))));
1230 getThermodynamicForce(dAta.behDataPtr->s1, 0))));
1232 t_stress(
I,
J) = t_force(
I,
J);
1233 t_full_stress(
i,
j) = t_full_force(
i,
j);
1236 if constexpr (UPDATE) {
1237 for (
int dd = 0;
dd != dAta.sizeIntVar; ++
dd) {
1238 mat_int(gg, dd) = *getInternalStateVariable(dAta.behDataPtr->s1, dd);
1240 for (
int dd = 0;
dd != dAta.sizeGradVar; ++
dd) {
1241 mat_grad0(gg, dd) = *getGradient(dAta.behDataPtr->s1, dd);
1243 for (
int dd = 0;
dd != dAta.sizeStressVar; ++
dd) {
1244 mat_stress0(gg, dd) = *getThermodynamicForce(dAta.behDataPtr->s1, dd);
1255 if constexpr (UPDATE) {
1256 CHKERR commonDataPtr->setInternalVar(fe_ent);
1262template <
typename T, ModelHypothesis MH>
1264 static constexpr int DIM = MFrontEleType<MH>::SPACE_DIM;
1265 using DomainEleOp =
typename MFrontEleType<MH>::DomainEleOp;
1268 boost::shared_ptr<MFrontInterface::CommonData> common_data_ptr,
1269 boost::shared_ptr<FEMethod> monitor_ptr)
1271 commonDataPtr(common_data_ptr), monitorPtr(monitor_ptr) {
1272 std::fill(&DomainEleOp::doEntities[MBEDGE],
1273 &DomainEleOp::doEntities[MBMAXTYPE],
false);
1278 boost::shared_ptr<MFrontInterface::CommonData> commonDataPtr;
1279 boost::shared_ptr<FEMethod> monitorPtr;
1282template <ModelHypothesis MH>
1283using OpTangentFiniteStrains =
1284 struct OpTangent<Tensor4Pack<MFrontEleType<MH>::
SPACE_DIM>, MH>;
1286template <ModelHypothesis MH>
1287using OpTangentSmallStrains =
1288 struct OpTangent<DdgPack<MFrontEleType<MH>::SPACE_DIM>, MH>;
1290template <typename T, ModelHypothesis MH>
1291MoFEMErrorCode OpTangent<T, MH>::doWork(int side, EntityType type,
1295 const size_t nb_gauss_pts = commonDataPtr->mGradPtr->size2();
1296 auto fe_ent = DomainEleOp::getNumeredEntFiniteElementPtr()->getEnt();
1297 auto id = commonDataPtr->blocksIDmap.at(fe_ent);
1298 auto &dAta = commonDataPtr->setOfBlocksData.at(
id);
1300 if (monitorPtr ==
nullptr)
1302 "Time Monitor (FEMethod) has not been set for MFrontInterfaceImpl. "
1303 "Make sure to call setMonitor(monitor_ptr) before calling "
1306 dAta.setTag(DataTags::LHS);
1307 dAta.behDataPtr->dt = monitorPtr->ts_dt;
1308 dAta.bView.dt = monitorPtr->ts_dt;
1310 constexpr bool IS_LARGE_STRAIN = std::is_same<T, Tensor4Pack<3>>::value ||
1311 std::is_same<T, Tensor4Pack<2>>::value;
1313 CHKERR commonDataPtr->getInternalVar(fe_ent, nb_gauss_pts, dAta.sizeIntVar,
1314 dAta.sizeGradVar, dAta.sizeStressVar,
1317 MatrixDouble &S_E = *(commonDataPtr->materialTangentPtr);
1320 size_t tens_size = 36;
1322 if constexpr (DIM == 2) {
1324 if constexpr (IS_LARGE_STRAIN)
1329 if constexpr (IS_LARGE_STRAIN)
1333 S_E.resize(tens_size, nb_gauss_pts,
false);
1334 auto D1 = get_tangent_tensor<T>(S_E);
1336 size_t full_tens_size = 81;
1337 F_E.resize(full_tens_size, nb_gauss_pts,
false);
1340 auto D2 = get_tangent_tensor<Tensor4Pack<3>>(F_E);
1342 auto t_grad = getFTensor2FromMat<DIM, DIM>(*(commonDataPtr->mGradPtr));
1343 auto t_disp = getFTensor1FromMat<DIM>(*(commonDataPtr->mDispPtr));
1344 auto t_coords = DomainEleOp::getFTensor1CoordsAtGaussPts();
1346 for (
size_t gg = 0; gg != nb_gauss_pts; ++gg) {
1348 CHKERR mgis_integration<IS_LARGE_STRAIN, MH>(gg, t_grad, t_disp, t_coords,
1349 *commonDataPtr, dAta);
1351 CHKERR get_tensor4_from_voigt(&*dAta.behDataPtr->K.begin(), D1);
1352 CHKERR get_full_tensor4_from_voigt<IS_LARGE_STRAIN>(
1353 &*dAta.behDataPtr->K.begin(), D2);
1365template <AssemblyType AT>
1366struct OpAxisymmetricRhs
1367 :
public FormsIntegrators<
1368 MFrontEleType<AXISYMMETRICAL>::DomainEleOp>::Assembly<AT>
::OpBase {
1369 using DomainEleOp =
typename MFrontEleType<AXISYMMETRICAL>::DomainEleOp;
1370 using OpBase =
typename FormsIntegrators<DomainEleOp>::Assembly<
AT>
::OpBase;
1374 boost::shared_ptr<MFrontInterface::CommonData> common_data_ptr)
1376 commonDataPtr(common_data_ptr) {};
1379 boost::shared_ptr<MFrontInterface::CommonData> commonDataPtr;
1384template <AssemblyType AT>
1389 const double vol = OpBase::getMeasure();
1391 auto t_w = OpBase::getFTensor0IntegrationWeight();
1394 auto t_full_stress =
1395 getFTensor2FromMat<3, 3>(*(commonDataPtr->mFullStressPtr));
1400 auto t_nf = OpBase::template getNf<2>();
1405 const double alpha = t_w * vol * 2. * M_PI;
1409 t_nf(0) += alpha * t_full_stress(2, 2) * t_base;
1422template <AssemblyType AT>
1423struct OpAxisymmetricLhs
1424 :
public FormsIntegrators<
1425 MFrontEleType<AXISYMMETRICAL>::DomainEleOp>::Assembly<AT>
::OpBase {
1426 using DomainEleOp =
typename MFrontEleType<AXISYMMETRICAL>::DomainEleOp;
1427 using OpBase =
typename FormsIntegrators<DomainEleOp>::Assembly<
AT>
::OpBase;
1431 boost::shared_ptr<MFrontInterface::CommonData> common_data_ptr)
1433 commonDataPtr(common_data_ptr) {};
1436 boost::shared_ptr<MFrontInterface::CommonData> commonDataPtr;
1441template <AssemblyType AT>
1447 const double vol = OpBase::getMeasure();
1449 auto t_w = OpBase::getFTensor0IntegrationWeight();
1455 auto t_coords = OpBase::getFTensor1CoordsAtGaussPts();
1462 getFTensor4FromMat<3, 3, 3, 3, 1>(*(commonDataPtr->mFullTangentPtr));
1468 const double r_cylinder = t_coords(0);
1471 const double alpha = t_w * vol * 2. * M_PI;
1478 auto t_m = OpBase::template getLocMat<2>(2 * rr);
1489 alpha * t_D(N0, N0, N2, N2) * t_col_base * t_row_diff_base(0);
1492 alpha * t_D(N1, N1, N2, N2) * t_col_base * t_row_diff_base(1);
1495 alpha * t_D(N2, N2, N0, N0) * t_col_diff_base(0) * t_row_base;
1498 alpha * t_D(N2, N2, N1, N1) * t_col_diff_base(1) * t_row_base;
1500 t_m(0, 0) += alpha * (t_D(N2, N2, N2, N2) / r_cylinder) * t_col_base *
1504 alpha * t_D(N2, N2, N0, N1) * t_col_diff_base(1) * t_row_base;
1507 alpha * t_D(N2, N2, N1, N0) * t_col_diff_base(0) * t_row_base;
1510 alpha * t_D(N0, N1, N2, N2) * t_col_base * t_row_diff_base(1);
1513 alpha * t_D(N1, N0, N2, N2) * t_col_base * t_row_diff_base(0);
1537template <ModelHypothesis MH, AssemblyType AT>
1539 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
1543 auto jacobian = [&](
const double r,
const double,
const double) {
1544 if (MH == AXISYMMETRICAL)
1545 return 2. * M_PI *
r;
1550 auto add_domain_ops_rhs = [&](
auto &pipeline) {
1551 if (isFiniteKinematics)
1553 new OpStressFiniteStrains<MH>(
field_name, commonDataPtr, monitorPtr));
1556 new OpStressSmallStrains<MH>(
field_name, commonDataPtr, monitorPtr));
1559 new OpInternalForce(
field_name, commonDataPtr->mStressPtr, jacobian));
1561 if (MH == AXISYMMETRICAL)
1562 pipeline.push_back(
new OpAxisymmetricRhs<AT>(
field_name, commonDataPtr));
1565 auto add_domain_base_ops = [&](
auto &pipeline) {
1566 pipeline.push_back(
new OpCalculateVectorFieldValues<DIM>(
1568 pipeline.push_back(
new OpCalculateVectorFieldGradient<DIM, DIM>(
1572 add_domain_base_ops(pip);
1573 add_domain_ops_rhs(pip);
1578template <ModelHypothesis MH, AssemblyType AT>
1580 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
1584 auto jacobian = [&](
const double r,
const double,
const double) {
1585 if (MH == AXISYMMETRICAL)
1586 return 2. * M_PI *
r;
1591 auto add_domain_ops_lhs = [&](
auto &pipeline) {
1592 if (isFiniteKinematics) {
1593 pipeline.push_back(
new OpTangentFiniteStrains<MH>(
1595 pipeline.push_back(
new OpAssembleLhsFiniteStrains(
1599 new OpTangentSmallStrains<MH>(
field_name, commonDataPtr, monitorPtr));
1600 pipeline.push_back(
new OpAssembleLhsSmallStrains(
1604 if (MH == AXISYMMETRICAL)
1605 pipeline.push_back(
new OpAxisymmetricLhs<AT>(
field_name, commonDataPtr));
1608 auto add_domain_base_ops = [&](
auto &pipeline) {
1609 pipeline.push_back(
new OpCalculateVectorFieldValues<DIM>(
1611 pipeline.push_back(
new OpCalculateVectorFieldGradient<DIM, DIM>(
1615 add_domain_base_ops(pip);
1616 add_domain_ops_lhs(pip);
1621template <ModelHypothesis MH>
1622struct OpSaveGaussPts :
public MFrontEleType<MH>
::DomainEleOp {
1623 static constexpr int DIM = MFrontEleType<MH>::SPACE_DIM;
1624 using DomainEleOp =
typename MFrontEleType<MH>::DomainEleOp;
1626 OpSaveGaussPts(
const std::string
field_name, moab::Interface &moab_mesh,
1627 boost::shared_ptr<MFrontInterface::CommonData> common_data_ptr)
1629 commonDataPtr(common_data_ptr), fieldName(
field_name) {
1630 std::fill(&DomainEleOp::doEntities[MBEDGE],
1631 &DomainEleOp::doEntities[MBMAXTYPE],
false);
1637 auto fe_ent = DomainEleOp::getNumeredEntFiniteElementPtr()->getEnt();
1638 auto id = commonDataPtr->blocksIDmap.at(fe_ent);
1639 auto &dAta = commonDataPtr->setOfBlocksData.at(
id);
1640 auto &mgis_bv = *dAta.mGisBehaviour;
1642 int &size_of_vars = dAta.sizeIntVar;
1643 int &size_of_grad = dAta.sizeGradVar;
1644 int &size_of_stress = dAta.sizeStressVar;
1646 auto get_tag = [&](std::string name,
size_t size) {
1647 std::array<double, 9> def;
1648 std::fill(def.begin(), def.end(), 0);
1650 CHKERR internalVarMesh.tag_get_handle(name.c_str(), size, MB_TYPE_DOUBLE,
1651 th, MB_TAG_CREAT | MB_TAG_SPARSE,
1656 auto t_stress = getFTensor2FromMat<3, 3>(*(commonDataPtr->mStressPtr));
1660 auto th_disp = get_tag(fieldName, 3);
1661 auto th_stress = get_tag(mgis_bv.thermodynamic_forces[0].name, 9);
1662 auto th_grad = get_tag(mgis_bv.gradients[0].name, 9);
1664 size_t nb_gauss_pts = commonDataPtr->mGradPtr->size2();
1665 auto t_grad = getFTensor2FromMat<3, 3>(*(commonDataPtr->mGradPtr));
1666 auto t_disp = getFTensor1FromMat<3>(*(commonDataPtr->mDispPtr));
1667 CHKERR commonDataPtr->getInternalVar(fe_ent, nb_gauss_pts, size_of_vars,
1668 size_of_grad, size_of_stress);
1670 MatrixDouble &mat_int = *commonDataPtr->internalVariablePtr;
1671 vector<Tag> tags_vec;
1674 for (
auto c : mgis_bv.isvs) {
1675 auto vsize = getVariableSize(
c, mgis_bv.hypothesis);
1676 const size_t parav_siz = get_paraview_size(vsize);
1677 tags_vec.emplace_back(get_tag(
c.name, parav_siz));
1680 if (!(side == 0 && type == MBVERTEX))
1683 for (
size_t gg = 0; gg != nb_gauss_pts; ++gg) {
1685 double coords[] = {0, 0, 0};
1687 for (
int dd = 0;
dd != 3;
dd++)
1688 coords[dd] = DomainEleOp::getCoordsAtGaussPts()(gg,
dd);
1690 CHKERR internalVarMesh.create_vertex(coords, vertex);
1693 auto it = tags_vec.begin();
1694 for (
auto c : mgis_bv.isvs) {
1695 auto vsize = getVariableSize(
c, mgis_bv.hypothesis);
1696 const size_t parav_siz = get_paraview_size(vsize);
1698 getVariableOffset(mgis_bv.isvs,
c.name, mgis_bv.hypothesis);
1702 tag_vec.resize(parav_siz);
1704 CHKERR internalVarMesh.tag_set_data(*it, &vertex, 1, &*tag_vec.begin());
1710 array<double, 9> my_stress_vec{
1711 t_stress(0, 0), t_stress(1, 1), t_stress(2, 2),
1712 t_stress(0, 1), t_stress(1, 0), t_stress(0, 2),
1713 t_stress(2, 0), t_stress(1, 2), t_stress(2, 1)};
1718 array<double, 9> grad1_vec;
1722 grad1_vec = getVoigtVec<3>(t_strain);
1725 t_strain(
i,
j) = (t_grad(
i,
j) || t_grad(
j,
i)) / 2;
1726 grad1_vec = getVoigtVecSymm<3>(t_strain);
1729 CHKERR internalVarMesh.tag_set_data(th_stress, &vertex, 1,
1730 my_stress_vec.data());
1731 CHKERR internalVarMesh.tag_set_data(th_grad, &vertex, 1,
1733 CHKERR internalVarMesh.tag_set_data(th_disp, &vertex, 1, &*disps.begin());
1744 boost::shared_ptr<MFrontInterface::CommonData> commonDataPtr;
1745 moab::Interface &internalVarMesh;
1746 std::string fieldName;
1749template <ModelHypothesis MH, AssemblyType AT>
1750MoFEMErrorCode MFrontInterfaceImpl<MH, AT>::setUpdateInternalVariablesOperators(
1751 ForcesAndSourcesCore::RuleHookFun rule, std::string
field_name) {
1754 updateIntVariablesElePtr = boost::make_shared<DomainEle>(mField);
1756 updateIntVariablesElePtr->getRuleHook = rule;
1759 updateIntVariablesElePtr->getOpPtrVector(), {H1});
1761 updateIntVariablesElePtr->getOpPtrVector().push_back(
1762 new OpCalculateVectorFieldGradient<DIM, DIM>(
field_name,
1763 commonDataPtr->mGradPtr));
1764 updateIntVariablesElePtr->getOpPtrVector().push_back(
1765 new OpCalculateVectorFieldValues<DIM>(
field_name,
1766 commonDataPtr->mDispPtr));
1767 if (isFiniteKinematics)
1768 updateIntVariablesElePtr->getOpPtrVector().push_back(
1769 new OpUpdateVariablesFiniteStrains<MH>(
field_name, commonDataPtr,
1772 updateIntVariablesElePtr->getOpPtrVector().push_back(
1773 new OpUpdateVariablesSmallStrains<MH>(
field_name, commonDataPtr,
1775 if (saveGauss && (MH == TRIDIMENSIONAL)) {
1776 auto &moab_gauss = *moabGaussIntPtr;
1777 updateIntVariablesElePtr->getOpPtrVector().push_back(
1778 new OpSaveGaussPts<MH>(
field_name, moab_gauss, commonDataPtr));
1784template <
bool IS_LARGE_STRAIN, ModelHypothesis MH>
1785struct OpSaveStress :
public MFrontEleType<MH>
::DomainEleOp {
1786 static constexpr int DIM = MFrontEleType<MH>::SPACE_DIM;
1787 using DomainEleOp =
typename MFrontEleType<MH>::DomainEleOp;
1790 boost::shared_ptr<MFrontInterface::CommonData> common_data_ptr)
1792 commonDataPtr(common_data_ptr) {
1793 std::fill(&DomainEleOp::doEntities[MBEDGE],
1794 &DomainEleOp::doEntities[MBMAXTYPE],
false);
1799 const size_t nb_gauss_pts = commonDataPtr->mGradPtr->size2();
1800 auto fe_ent = DomainEleOp::getNumeredEntFiniteElementPtr()->getEnt();
1801 auto id = commonDataPtr->blocksIDmap.at(fe_ent);
1802 auto &dAta = commonDataPtr->setOfBlocksData.at(
id);
1804 int &size_of_stress = dAta.sizeStressVar;
1805 int &size_of_grad = dAta.sizeGradVar;
1807 CHKERR commonDataPtr->getInternalVar(fe_ent, nb_gauss_pts, dAta.sizeIntVar,
1808 dAta.sizeGradVar, dAta.sizeStressVar,
1811 MatrixDouble &mat_stress = *commonDataPtr->mPrevStressPtr;
1814 commonDataPtr->mFullStressPtr->resize(3 * 3, nb_gauss_pts);
1815 commonDataPtr->mFullStressPtr->clear();
1816 auto t_full_stress =
1817 getFTensor2FromMat<3, 3>(*(commonDataPtr->mFullStressPtr));
1819 commonDataPtr->mFullStrainPtr->resize(3 * 3, nb_gauss_pts);
1820 commonDataPtr->mFullStrainPtr->clear();
1821 auto t_full_strain =
1822 getFTensor2FromMat<3, 3>(*(commonDataPtr->mFullStrainPtr));
1827 for (
size_t gg = 0; gg != nb_gauss_pts; ++gg) {
1830 &mat_stress.data()[gg * size_of_stress], size_of_stress);
1837 if constexpr (IS_LARGE_STRAIN) {
1838 if constexpr (DIM == 3) {
1841 }
else if constexpr (DIM == 2) {
1846 if constexpr (DIM == 3) {
1851 }
else if constexpr (DIM == 2) {
1859 t_full_stress(
i,
j) = t_stress(
i,
j);
1860 t_full_strain(
i,
j) = t_grad(
i,
j);
1870 boost::shared_ptr<MFrontInterface::CommonData> commonDataPtr;
1873template <ModelHypothesis MH, AssemblyType AT>
1874MoFEMErrorCode MFrontInterfaceImpl<MH, AT>::setPostProcessOperators(
1875 ForcesAndSourcesCore::RuleHookFun rule, std::string fe_name,
1879 postProcFe = boost::make_shared<PostProcDomainEle>(mField);
1881 auto &pip = postProcFe->getOpPtrVector();
1883 pip.push_back(
new OpCalculateVectorFieldValues<DIM>(
field_name,
1884 commonDataPtr->mDispPtr));
1886 auto entity_data_l2 = boost::make_shared<EntitiesFieldData>(MBENTITYSET);
1887 auto mass_ptr = boost::make_shared<MatrixDouble>();
1888 auto strain_coeffs_ptr = boost::make_shared<MatrixDouble>();
1889 auto stress_coeffs_ptr = boost::make_shared<MatrixDouble>();
1891 auto op_this =
new OpLoopThis<DomainEle>(mField, fe_name, Sev::noisy);
1892 pip.push_back(op_this);
1893 pip.push_back(
new OpDGProjectionEvaluation(commonDataPtr->mFullStrainPtr,
1894 strain_coeffs_ptr, entity_data_l2,
1896 pip.push_back(
new OpDGProjectionEvaluation(commonDataPtr->mFullStressPtr,
1897 stress_coeffs_ptr, entity_data_l2,
1900 auto fe_physics_ptr = op_this->getThisFEPtr();
1901 fe_physics_ptr->getRuleHook = rule;
1903 fe_physics_ptr->getOpPtrVector().push_back(
new OpDGProjectionMassMatrix(
1904 order, mass_ptr, entity_data_l2, fieldBase,
L2));
1905 if (isFiniteKinematics) {
1906 fe_physics_ptr->getOpPtrVector().push_back(
1907 new OpSaveStress<true, MH>(
field_name, commonDataPtr));
1909 fe_physics_ptr->getOpPtrVector().push_back(
1910 new OpSaveStress<false, MH>(
field_name, commonDataPtr));
1912 fe_physics_ptr->getOpPtrVector().push_back(
new OpDGProjectionCoefficients(
1913 commonDataPtr->mFullStrainPtr, strain_coeffs_ptr, mass_ptr,
1914 entity_data_l2, fieldBase,
L2));
1915 fe_physics_ptr->getOpPtrVector().push_back(
new OpDGProjectionCoefficients(
1916 commonDataPtr->mFullStressPtr, stress_coeffs_ptr, mass_ptr,
1917 entity_data_l2, fieldBase,
L2));
1919 using OpPPMapVec = OpPostProcMapInMoab<DIM, DIM>;
1920 using OpPPMapTen = OpPostProcMapInMoab<3, 3>;
1922 pip.push_back(
new OpPPMapVec(
1924 postProcFe->getPostProcMesh(), postProcFe->getMapGaussPts(),
1928 {{field_name, commonDataPtr->mDispPtr}},
1936 pip.push_back(
new OpPPMapTen(
1938 postProcFe->getPostProcMesh(), postProcFe->getMapGaussPts(),
1944 {{
"STRAIN", commonDataPtr->mFullStrainPtr},
1945 {
"STRESS", commonDataPtr->mFullStressPtr}},
1954template <ModelHypothesis MH, AssemblyType AT>
1956 SmartPetscObj<DM> dm,
1960 auto make_vtks = [&]() {
1965 CHKERR postProcFe->writeFile(
"out_" + optionsPrefix +
1966 boost::lexical_cast<std::string>(step) +
1971 string file_name =
"out_" + optionsPrefix +
"gauss_" +
1972 boost::lexical_cast<std::string>(step) +
".h5m";
1974 CHKERR moabGaussIntPtr->write_file(file_name.c_str(),
"MOAB",
1975 "PARALLEL=WRITE_PART");
1976 CHKERR moabGaussIntPtr->delete_mesh();
1987template <ModelHypothesis MH, AssemblyType AT>
1989MFrontInterfaceImpl<MH, AT>::updateInternalVariables(SmartPetscObj<DM> dm,
1990 std::string fe_name) {
2006template <ModelHypothesis MH, AssemblyType AT>
2008 : mField(m_field) {}
2010template <ModelHypothesis MH, AssemblyType AT>
2011MoFEMErrorCode MFrontInterfaceImpl<MH, AT>::getCommandLineParameters() {
2014 "MFront has not been enabled in this build of MoFEM");
2018template <ModelHypothesis MH, AssemblyType AT>
2020 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
2024 "MFront has not been enabled in this build of MoFEM");
2028template <ModelHypothesis MH, AssemblyType AT>
2030 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
2034 "MFront has not been enabled in this build of MoFEM");
2038template <ModelHypothesis MH, AssemblyType AT>
2039MoFEMErrorCode MFrontInterfaceImpl<MH, AT>::setUpdateInternalVariablesOperators(
2043 "MFront has not been enabled in this build of MoFEM");
2047template <ModelHypothesis MH, AssemblyType AT>
2048MoFEMErrorCode MFrontInterfaceImpl<MH, AT>::setPostProcessOperators(
2053 "MFront has not been enabled in this build of MoFEM");
2057template <ModelHypothesis MH, AssemblyType AT>
2059 SmartPetscObj<DM> dm,
2063 "MFront has not been enabled in this build of MoFEM");
2067template <ModelHypothesis MH, AssemblyType AT>
2069MFrontInterfaceImpl<MH, AT>::updateInternalVariables(SmartPetscObj<DM> dm,
2070 std::string fe_name) {
2073 "MFront has not been enabled in this build of MoFEM");
2083template struct MFrontInterfaceImpl<TRIDIMENSIONAL, AssemblyType::PETSC>;
2084template struct MFrontInterfaceImpl<AXISYMMETRICAL, AssemblyType::PETSC>;
2085template struct MFrontInterfaceImpl<PLANESTRAIN, AssemblyType::PETSC>;
2087template struct MFrontInterfaceImpl<TRIDIMENSIONAL, AssemblyType::BLOCK_SCHUR>;
2088template struct MFrontInterfaceImpl<AXISYMMETRICAL, AssemblyType::BLOCK_SCHUR>;
2089template struct MFrontInterfaceImpl<PLANESTRAIN, AssemblyType::BLOCK_SCHUR>;
static MoFEMErrorCode setBlocks(MoFEM::Interface &m_field, boost::shared_ptr< map< int, BlockData > > &block_sets_ptr)
DomainEle::UserDataOperator DomainEleOp
Finire element operator type.
ElementsAndOps< SPACE_DIM >::DomainEle DomainEle
FieldApproximationBase
approximation base
@ AINSWORTH_LEGENDRE_BASE
Ainsworth Cole (Legendre) approx. base .
#define CHK_THROW_MESSAGE(err, msg)
Check and throw MoFEM exception.
#define MoFEMFunctionReturnHot(a)
Last executable line of each PETSc function used for error handling. Replaces return()
@ L2
field with C-1 continuity
#define MoFEMFunctionBegin
First executable line of each MoFEM function, used for error handling. Final line of MoFEM functions ...
@ MOFEM_DATA_INCONSISTENCY
static const char *const ApproximationBaseNames[]
#define MoFEMFunctionReturn(a)
Last executable line of each PETSc function used for error handling. Replaces return()
#define CHKERR
Inline error check.
#define MoFEMFunctionBeginHot
First executable line of each MoFEM function, used for error handling. Final line of MoFEM functions ...
PetscErrorCode DMoFEMLoopFiniteElements(DM dm, const char fe_name[], MoFEM::FEMethod *method, CacheTupleWeakPtr cache_ptr=CacheTupleSharedPtr())
Executes FEMethod for finite elements in DM.
#define MOFEM_LOG(channel, severity)
Log.
#define MOFEM_LOG_TAG(channel, tag)
Tag channel.
#define _IT_CUBITMESHSETS_BY_SET_TYPE_FOR_LOOP_(MESHSET_MANAGER, CUBITBCTYPE, IT)
Iterator that loops over a specific Cubit MeshSet having a particular BC meshset in a moFEM field.
FTensor::Index< 'i', SPACE_DIM > i
const double c
speed of light (cm/ns)
const double n
refractive index of diffusive medium
FTensor::Index< 'J', DIM1 > J
FTensor::Index< 'j', 3 > j
Tensors class implemented by Walter Landry.
Tensor2_Expr< Kronecker_Delta< T >, T, Dim0, Dim1, i, j > kronecker_delta(const Index< i, Dim0 > &, const Index< j, Dim1 > &)
Rank 2.
const Tensor2_symmetric_Expr< const ddTensor0< T, Dim, i, j >, typename promote< T, double >::V, Dim, i, j > dd(const Tensor0< T * > &a, const Index< i, Dim > index1, const Index< j, Dim > index2, const Tensor1< int, Dim > &d_ijk, const Tensor1< double, Dim > &d_xyz)
static MoFEMErrorCodeGeneric< moab::ErrorCode > rval
PetscErrorCode MoFEMErrorCode
MoFEM/PETSc error code.
UBlasMatrix< double > MatrixDouble
UBlasVector< double > VectorDouble
MatrixShallowArrayAdaptor< double > MatrixAdaptor
Matrix adaptor.
MatrixBoundedArray< double, 9 > MatrixDouble3by3
implementation of Data Operators for Forces and Sources
auto getVectorAdaptor(T1 ptr, const size_t n)
Get Vector adaptor.
auto type_from_handle(const EntityHandle h)
get type from entity handle
auto getVoigtVecAxisymm(T &t_mat, const double hoop_term)
auto to_non_symm(const FTensor::Tensor2_symmetric< T, DIM > &symm)
boost::shared_ptr< MFrontInterface > createMFrontInterface(MoFEM::Interface &m_field, ModelHypothesis mh, AssemblyType at=AssemblyType::PETSC)
create mfront interface
ModelHypothesis
Enumeration of model hypotheses supported by MFront interface.
@ AXISYMMETRICAL
Axisymmetrical model hypothesis.
@ PLANESTRAIN
Plane strain model hypothesis.
@ TRIDIMENSIONAL
3D model hypothesis.
auto getVoigtVecSymmAxisymm(T &t_mat, const double hoop_term)
MoFEMErrorCode opFactoryDomainRhs(MoFEM::Interface &m_field, std::string block_name, Pip &pip, std::string u, std::string ep, std::string tau)
MoFEMErrorCode opFactoryDomainLhs(MoFEM::Interface &m_field, std::string block_name, Pip &pip, std::string u, std::string ep, std::string tau)
FormsIntegrators< DomainEleOp >::Assembly< A >::LinearForm< I >::OpGradTimesTensor< 1, FIELD_DIM, SPACE_DIM > OpGradTimesTensor
constexpr IntegrationType I
constexpr auto field_name
OpBaseImpl< PETSC, EdgeEleOp > OpBase
Deprecated interface functions.
Data on single entity (This is passed as argument to DataOperator::doWork)
FTensor::Tensor1< FTensor::PackPtr< double *, Tensor_Dim >, Tensor_Dim > getFTensor1DiffN(const FieldApproximationBase base)
Get derivatives of base functions.
FTensor::Tensor0< FTensor::PackPtr< double *, 1 > > getFTensor0N(const FieldApproximationBase base)
Get base function as Tensor0.
MatrixDouble & getN(const FieldApproximationBase base)
get base functions this return matrix (nb. of rows is equal to nb. of Gauss pts, nb....
boost::function< int(int order_row, int order_col, int order_data)> RuleHookFun
CommonData(MoFEM::Interface &m_field)
int nbRows
number of dofs on rows
int nbIntegrationPts
number of integration points
virtual MoFEMErrorCode iNtegrate(EntData &row_data, EntData &col_data)
Integrate grad-grad operator.
int nbCols
number if dof on column
int nbRowBaseFunctions
number or row base functions
Volume finite element base.
constexpr AssemblyType AT