22 #include "libmesh/mesh_tools.h" 23 #include "libmesh/explicit_system.h" 24 #include "libmesh/numeric_vector.h" 25 #include "libmesh/elem.h" 26 #include "libmesh/node.h" 27 #include "libmesh/dof_map.h" 28 #include "libmesh/edge_edge2.h" 29 #include "libmesh/edge_edge3.h" 30 #include "libmesh/face_tri3.h" 31 #include "libmesh/face_tri6.h" 32 #include "libmesh/face_tri7.h" 33 #include "libmesh/face_quad4.h" 34 #include "libmesh/face_quad8.h" 35 #include "libmesh/face_quad9.h" 36 #include "libmesh/exodusII_io.h" 37 #include "libmesh/quadrature_gauss.h" 38 #include "libmesh/quadrature_nodal.h" 39 #include "libmesh/distributed_mesh.h" 40 #include "libmesh/replicated_mesh.h" 41 #include "libmesh/enum_to_string.h" 42 #include "libmesh/statistics.h" 43 #include "libmesh/equation_systems.h" 45 #include "metaphysicl/dualnumber.h" 58 #if NANOFLANN_VERSION < 0x150 85 if (_amg._secondary_node_to_nodal_normal.empty() ||
86 _amg._secondary_node_to_hh_nodal_tangents.empty())
87 mooseError(
"No entries found in the secondary node -> nodal geometry map.");
89 auto & problem = _app.feProblem();
90 auto & subproblem = _amg._on_displaced
91 ?
static_cast<SubProblem &
>(*problem.getDisplacedProblem())
92 : static_cast<SubProblem &>(problem);
93 auto & nodal_normals_es = subproblem.
es();
95 const std::string nodal_normals_sys_name =
"nodal_normals";
97 if (!_nodal_normals_system)
99 for (
const auto s :
make_range(nodal_normals_es.n_systems()))
100 if (!nodal_normals_es.get_system(s).is_initialized())
105 _nodal_normals_system =
106 &nodal_normals_es.template add_system<ExplicitSystem>(nodal_normals_sys_name);
107 _nnx_var_num = _nodal_normals_system->add_variable(
"nodal_normal_x",
FEType(
FIRST,
LAGRANGE)),
108 _nny_var_num = _nodal_normals_system->add_variable(
"nodal_normal_y",
FEType(
FIRST,
LAGRANGE));
109 _nnz_var_num = _nodal_normals_system->add_variable(
"nodal_normal_z",
FEType(
FIRST,
LAGRANGE));
124 nodal_normals_es.
reinit();
127 const DofMap & dof_map = _nodal_normals_system->get_dof_map();
128 std::vector<dof_id_type> dof_indices_nnx, dof_indices_nny, dof_indices_nnz;
129 std::vector<dof_id_type> dof_indices_t1x, dof_indices_t1y, dof_indices_t1z;
130 std::vector<dof_id_type> dof_indices_t2x, dof_indices_t2y, dof_indices_t2z;
132 for (MeshBase::const_element_iterator el = _amg._mesh.elements_begin(),
133 end_el = _amg._mesh.elements_end();
137 const Elem * elem = *el;
140 dof_map.
dof_indices(elem, dof_indices_nnx, _nnx_var_num);
141 dof_map.
dof_indices(elem, dof_indices_nny, _nny_var_num);
142 dof_map.
dof_indices(elem, dof_indices_nnz, _nnz_var_num);
144 dof_map.
dof_indices(elem, dof_indices_t1x, _t1x_var_num);
145 dof_map.
dof_indices(elem, dof_indices_t1y, _t1y_var_num);
146 dof_map.
dof_indices(elem, dof_indices_t1z, _t1z_var_num);
148 dof_map.
dof_indices(elem, dof_indices_t2x, _t2x_var_num);
149 dof_map.
dof_indices(elem, dof_indices_t2y, _t2y_var_num);
150 dof_map.
dof_indices(elem, dof_indices_t2z, _t2z_var_num);
158 auto it = _amg._secondary_node_to_nodal_normal.find(elem->
node_ptr(n));
159 if (it != _amg._secondary_node_to_nodal_normal.end())
161 _nodal_normals_system->solution->set(dof_indices_nnx[n], it->second(0));
162 _nodal_normals_system->solution->set(dof_indices_nny[n], it->second(1));
163 _nodal_normals_system->solution->set(dof_indices_nnz[n], it->second(2));
166 auto it_tangent = _amg._secondary_node_to_hh_nodal_tangents.find(elem->
node_ptr(n));
167 if (it_tangent != _amg._secondary_node_to_hh_nodal_tangents.end())
169 _nodal_normals_system->solution->set(dof_indices_t1x[n], it_tangent->second[0](0));
170 _nodal_normals_system->solution->set(dof_indices_t1y[n], it_tangent->second[0](1));
171 _nodal_normals_system->solution->set(dof_indices_t1z[n], it_tangent->second[0](2));
173 _nodal_normals_system->solution->set(dof_indices_t2x[n], it_tangent->second[1](0));
174 _nodal_normals_system->solution->set(dof_indices_t2y[n], it_tangent->second[1](1));
175 _nodal_normals_system->solution->set(dof_indices_t2z[n], it_tangent->second[1](2));
182 _nodal_normals_system->solution->close();
184 std::set<std::string> sys_names = {nodal_normals_sys_name};
193 "nodal_geometry_only.e", nodal_normals_es, &sys_names);
220 const std::pair<BoundaryID, BoundaryID> & boundary_key,
221 const std::pair<SubdomainID, SubdomainID> & subdomain_key,
225 const bool correct_edge_dropping,
226 const Real minimum_projection_angle)
231 _on_displaced(on_displaced),
236 _distributed(_mesh.mesh_dimension() == 3 ? true : (!_on_displaced && !_mesh.is_replicated())),
237 _correct_edge_dropping(correct_edge_dropping),
238 _minimum_projection_angle(minimum_projection_angle)
262 string_vec[2 * i] = std::to_string(primary_bnd_id);
263 string_vec[2 * i + 1] = std::to_string(secondary_bnd_id);
265 string_vec.back() =
_on_displaced ?
"displaced" :
"undisplaced";
266 return MooseUtils::join(string_vec,
"_");
311 "Must specify secondary and primary boundary ids before building node-to-elem maps.");
314 for (
const auto & secondary_elem :
321 for (
const auto & nd : secondary_elem->node_ref_range())
324 vec.push_back(secondary_elem);
329 for (
const auto & primary_elem :
336 for (
const auto & nd : primary_elem->node_ref_range())
339 vec.push_back(primary_elem);
347 std::vector<Point> nodal_normals(secondary_elem.
n_nodes());
351 return nodal_normals;
359 "Map should locate secondary element");
364 std::map<unsigned int, unsigned int>
367 std::map<unsigned int, unsigned int> secondary_ip_i_to_lower_secondary_i;
369 mooseAssert(secondary_ip,
"This should be non-null");
373 const auto & nd = lower_secondary_elem.
node_ref(i);
374 secondary_ip_i_to_lower_secondary_i[secondary_ip->
get_node_index(&nd)] = i;
377 return secondary_ip_i_to_lower_secondary_i;
380 std::map<unsigned int, unsigned int>
382 const Elem & lower_primary_elem,
383 const Elem & primary_elem,
386 std::map<unsigned int, unsigned int> primary_ip_i_to_lower_primary_i;
390 const auto & nd = lower_primary_elem.
node_ref(i);
391 primary_ip_i_to_lower_primary_i[primary_elem.
get_node_index(&nd)] = i;
394 return primary_ip_i_to_lower_primary_i;
397 std::array<MooseUtils::SemidynamicVector<Point, 9>, 2>
401 MooseUtils::SemidynamicVector<Point, 9> nodal_tangents_one(0);
402 MooseUtils::SemidynamicVector<Point, 9> nodal_tangents_two(0);
406 const auto & tangent_vectors =
408 nodal_tangents_one.push_back(tangent_vectors[0]);
409 nodal_tangents_two.push_back(tangent_vectors[1]);
412 return {{nodal_tangents_one, nodal_tangents_two}};
417 const std::vector<Real> & oned_xi1_pts)
const 419 std::vector<Point> xi1_pts(oned_xi1_pts.size());
421 xi1_pts[qp] = oned_xi1_pts[qp];
428 const std::vector<Point> & xi1_pts)
const 431 const auto num_qps = xi1_pts.size();
433 std::vector<Point> normals(num_qps);
445 normals[qp] += phi * nodal_normals[n];
449 for (
auto & normal : normals)
461 std::size_t node_unique_id_offset = 0;
469 const auto primary_bnd_id = pr.first;
470 const auto secondary_bnd_id = pr.second;
471 const auto num_primary_nodes =
472 std::distance(
_mesh.bid_nodes_begin(primary_bnd_id),
_mesh.bid_nodes_end(primary_bnd_id));
473 const auto num_secondary_nodes = std::distance(
_mesh.bid_nodes_begin(secondary_bnd_id),
474 _mesh.bid_nodes_end(secondary_bnd_id));
475 mooseAssert(num_primary_nodes,
476 "There are no primary nodes on boundary ID " 477 << primary_bnd_id <<
". Does that bondary ID even exist on the mesh?");
478 mooseAssert(num_secondary_nodes,
479 "There are no secondary nodes on boundary ID " 480 << secondary_bnd_id <<
". Does that bondary ID even exist on the mesh?");
482 node_unique_id_offset += num_primary_nodes + 2 * num_secondary_nodes;
486 for (MeshBase::const_element_iterator el =
_mesh.active_elements_begin(),
487 end_el =
_mesh.active_elements_end();
491 const Elem * secondary_elem = *el;
497 std::vector<Node *> new_nodes;
498 for (MooseIndex(secondary_elem->
n_nodes()) n = 0; n < secondary_elem->
n_nodes(); ++n)
502 Node *
const new_node = new_nodes.back();
506 std::unique_ptr<Elem> new_elem;
508 new_elem = std::make_unique<Edge3>();
510 new_elem = std::make_unique<Edge2>();
512 new_elem->processor_id() = secondary_elem->
processor_id();
513 new_elem->subdomain_id() = secondary_elem->
subdomain_id();
514 new_elem->set_id(local_id_index++);
515 new_elem->set_unique_id(new_elem->id());
517 for (MooseIndex(new_elem->n_nodes()) n = 0; n < new_elem->n_nodes(); ++n)
518 new_elem->set_node(n, new_nodes[n]);
529 std::make_pair(secondary_elem->
node_ptr(0), secondary_elem)),
531 std::make_pair(secondary_elem->
node_ptr(1), secondary_elem));
533 bool new_container_node0_found =
535 new_container_node1_found =
538 const Elem * node0_primary_candidate =
nullptr;
539 const Elem * node1_primary_candidate =
nullptr;
541 if (new_container_node0_found)
543 const auto & xi2_primary_elem_pair = new_container_it0->second;
544 msinfo.
xi2_a = xi2_primary_elem_pair.first;
545 node0_primary_candidate = xi2_primary_elem_pair.second;
548 if (new_container_node1_found)
550 const auto & xi2_primary_elem_pair = new_container_it1->second;
551 msinfo.
xi2_b = xi2_primary_elem_pair.first;
552 node1_primary_candidate = xi2_primary_elem_pair.second;
559 if (node0_primary_candidate == node1_primary_candidate)
574 auto val = pr.second;
576 const Node * primary_node = std::get<1>(key);
577 Real xi1 = val.first;
578 const Elem * secondary_elem = val.second;
588 for (MooseIndex(secondary_elem->
n_nodes()) n = 0; n < secondary_elem->
n_nodes(); ++n)
593 Elem * current_mortar_segment =
nullptr;
596 for (
const auto & mortar_segment_candidate : mortar_segment_set)
602 catch (std::out_of_range &)
604 mooseError(
"MortarSegmentInfo not found for the mortar segment candidate");
606 if (
info->xi1_a <= xi1 && xi1 <= info->xi1_b)
608 current_mortar_segment = mortar_segment_candidate;
614 if (current_mortar_segment ==
nullptr)
615 mooseError(
"Unable to find appropriate mortar segment during linear search!");
622 if (
info->xi1_a == xi1 || xi1 ==
info->xi1_b)
627 "new_id must be the same on all processes");
628 Node *
const new_node =
630 new_node->set_unique_id(new_id + node_unique_id_offset);
635 const Point normal =
getNormals(*secondary_elem, std::vector<Real>({xi1}))[0];
640 mooseError(
"We should already have built this primary node to elem pair!");
641 const std::vector<const Elem *> & primary_node_neighbors =
645 if (primary_node_neighbors.size() == 0 || primary_node_neighbors.size() > 2)
646 mooseError(
"We must have either 1 or 2 primary side nodal neighbors, but we had ",
647 primary_node_neighbors.size());
654 const Elem * left_primary_elem = primary_node_neighbors[0];
655 const Elem * right_primary_elem =
656 (primary_node_neighbors.size() == 2) ? primary_node_neighbors[1] :
nullptr;
662 std::array<Real, 2> secondary_node_cps;
663 std::vector<Real> primary_node_cps(primary_node_neighbors.size());
666 for (
unsigned int nid = 0; nid < 2; ++nid)
667 secondary_node_cps[nid] = normal.
cross(secondary_elem->
point(nid) - new_pt)(2);
669 for (MooseIndex(primary_node_neighbors) mnn = 0; mnn < primary_node_neighbors.size(); ++mnn)
671 const Elem * primary_neigh = primary_node_neighbors[mnn];
672 Point opposite = (primary_neigh->
node_ptr(0) == primary_node) ? primary_neigh->
point(1)
673 : primary_neigh->
point(0);
675 primary_node_cps[mnn] = cp(2);
679 bool orientation1_valid =
false, orientation2_valid =
false;
681 if (primary_node_neighbors.size() == 2)
684 orientation1_valid = (secondary_node_cps[0] * primary_node_cps[0] > 0.) &&
685 (secondary_node_cps[1] * primary_node_cps[1] > 0.);
687 orientation2_valid = (secondary_node_cps[0] * primary_node_cps[1] > 0.) &&
688 (secondary_node_cps[1] * primary_node_cps[0] > 0.);
690 else if (primary_node_neighbors.size() == 1)
693 orientation1_valid = (secondary_node_cps[0] * primary_node_cps[0] > 0.);
694 orientation2_valid = (secondary_node_cps[1] * primary_node_cps[0] > 0.);
697 mooseError(
"Invalid primary node neighbors size ", primary_node_neighbors.size());
703 if (orientation1_valid && orientation2_valid)
705 "AutomaticMortarGeneration: Both orientations cannot simultaneously be valid.");
711 if (!orientation1_valid && !orientation2_valid)
714 "AutomaticMortarGeneration: Unable to determine valid secondary-primary orientation. " 715 "Consequently we will consider projection of the primary node invalid and not split the " 717 "This situation can indicate there are very oblique projections between primary (mortar) " 718 "and secondary (non-mortar) surfaces for a good problem set up. It can also mean your " 719 "time step is too large. This message is only printed once."));
724 std::unique_ptr<Elem> new_elem_left;
726 new_elem_left = std::make_unique<Edge3>();
728 new_elem_left = std::make_unique<Edge2>();
730 new_elem_left->processor_id() = current_mortar_segment->
processor_id();
731 new_elem_left->subdomain_id() = current_mortar_segment->
subdomain_id();
732 new_elem_left->set_id(local_id_index++);
733 new_elem_left->set_unique_id(new_elem_left->id());
734 new_elem_left->set_node(0, current_mortar_segment->
node_ptr(0));
735 new_elem_left->set_node(1, new_node);
738 std::unique_ptr<Elem> new_elem_right;
740 new_elem_right = std::make_unique<Edge3>();
742 new_elem_right = std::make_unique<Edge2>();
744 new_elem_right->processor_id() = current_mortar_segment->
processor_id();
745 new_elem_right->subdomain_id() = current_mortar_segment->
subdomain_id();
746 new_elem_right->set_id(local_id_index++);
747 new_elem_right->set_unique_id(new_elem_right->id());
748 new_elem_right->set_node(0, new_node);
749 new_elem_right->set_node(1, current_mortar_segment->
node_ptr(1));
754 Point left_interior_point(0);
755 Real left_interior_xi = (xi1 +
info->xi1_a) / 2;
758 Real current_left_interior_eta =
759 (2. * left_interior_xi -
info->xi1_a -
info->xi1_b) / (
info->xi1_b -
info->xi1_a);
761 for (MooseIndex(current_mortar_segment->
n_nodes()) n = 0;
762 n < current_mortar_segment->
n_nodes();
765 current_mortar_segment->
point(n);
769 "new_id must be the same on all processes");
771 left_interior_point, new_interior_left_id, new_elem_left->processor_id());
772 new_elem_left->set_node(2, new_interior_node_left);
773 new_interior_node_left->set_unique_id(new_interior_left_id + node_unique_id_offset);
776 Point right_interior_point(0);
777 Real right_interior_xi = (xi1 +
info->xi1_b) / 2;
779 Real current_right_interior_eta =
780 (2. * right_interior_xi -
info->xi1_a -
info->xi1_b) / (
info->xi1_b -
info->xi1_a);
782 for (MooseIndex(current_mortar_segment->
n_nodes()) n = 0;
783 n < current_mortar_segment->
n_nodes();
786 current_mortar_segment->
point(n);
790 "new_id must be the same on all processes");
792 right_interior_point, new_interior_id_right, new_elem_right->processor_id());
793 new_elem_right->set_node(2, new_interior_node_right);
794 new_interior_node_right->set_unique_id(new_interior_id_right + node_unique_id_offset);
798 if (orientation2_valid)
799 std::swap(left_primary_elem, right_primary_elem);
803 if (left_primary_elem)
804 left_xi2 = (primary_node == left_primary_elem->
node_ptr(0)) ? -1 : +1;
805 if (right_primary_elem)
806 right_xi2 = (primary_node == right_primary_elem->
node_ptr(0)) ? -1 : +1;
815 mooseError(
"MortarSegmentInfo not found for current_mortar_segment.");
831 new_msinfo_left.
xi1_b = xi1;
832 new_msinfo_left.
xi2_b = left_xi2;
840 mortar_segment_set.insert(msm_new_elem);
850 new_msinfo_right.
xi1_b = current_msinfo.
xi1_b;
851 new_msinfo_right.
xi2_b = current_msinfo.
xi2_b;
853 new_msinfo_right.
xi1_a = xi1;
854 new_msinfo_right.
xi2_a = right_xi2;
859 mortar_segment_set.insert(msm_new_elem);
867 mortar_segment_set.erase(current_mortar_segment);
884 if (primary_elem ==
nullptr ||
abs(msinfo.
xi2_a) > 1.0 + TOLERANCE ||
885 abs(msinfo.
xi2_b) > 1.0 + TOLERANCE)
890 "We should have found the element");
891 auto & msm_set = it->second;
892 msm_set.erase(msm_elem);
915 std::unordered_set<Node *> msm_connected_nodes;
920 for (
auto & n : element->node_ref_range())
921 msm_connected_nodes.insert(&n);
924 if (!msm_connected_nodes.count(node))
933 "All mortar segment elements should have valid " 955 std::array<std::string, 3> file_pieces = {
958 "mortar_segment_mesh.e"};
959 mortar_segment_mesh_writer.
write(MooseUtils::join(file_pieces,
"_"));
977 dof_id_type local_secondary_sub_elems = 0, visible_primary_sub_elems = 0;
980 for (
const auto *
const el :
982 local_secondary_sub_elems += el->n_sub_elem();
984 visible_primary_sub_elems += el->n_sub_elem();
986 const dof_id_type per_rank_bound = local_secondary_sub_elems * visible_primary_sub_elems * 9;
987 std::vector<dof_id_type> per_rank_bounds;
991 start += per_rank_bounds[r];
1003 const auto primary_subd_id = pr.first;
1004 const auto secondary_subd_id = pr.second;
1009 3, mesh_adaptor, nanoflann::KDTreeSingleIndexAdaptorParams(10));
1012 kd_tree.buildIndex();
1016 auto get_sub_elem_nodes = [](
const ElemType type,
1017 const unsigned int sub_elem) -> std::vector<unsigned int>
1024 return {{0, 1, 2, 3}};
1038 mooseError(
"get_sub_elem_nodes: Invalid sub_elem: ", sub_elem);
1052 return {{4, 5, 6, 7}};
1054 mooseError(
"get_sub_elem_nodes: Invalid sub_elem: ", sub_elem);
1060 return {{0, 4, 8, 7}};
1062 return {{4, 1, 5, 8}};
1064 return {{8, 5, 2, 6}};
1066 return {{7, 8, 6, 3}};
1068 mooseError(
"get_sub_elem_nodes: Invalid sub_elem: ", sub_elem);
1071 mooseError(
"get_sub_elem_inds: Face element type: ",
1072 libMesh::Utility::enum_to_string<ElemType>(type),
1073 " invalid for 3D mortar");
1080 for (MeshBase::const_element_iterator el =
_mesh.active_local_elements_begin(),
1081 end_el =
_mesh.active_local_elements_end();
1085 const Elem * secondary_side_elem = *el;
1087 const Real secondary_volume = secondary_side_elem->
volume();
1090 if (secondary_side_elem->
subdomain_id() != secondary_subd_id)
1093 auto [secondary_elem_to_msm_map_it, insertion_happened] =
1095 std::set<Elem *, CompareDofObjectsByID>{});
1097 auto & secondary_to_msm_element_set = secondary_elem_to_msm_map_it->second;
1099 std::vector<std::unique_ptr<MortarSegmentHelper>> mortar_segment_helper(
1116 auto sub_elem_nodes = get_sub_elem_nodes(secondary_side_elem->
type(), sel);
1121 std::vector<Point> nodes(sub_elem_nodes.size());
1124 for (
auto iv :
make_range(sub_elem_nodes.size()))
1126 const auto n = sub_elem_nodes[iv];
1127 nodes[iv] = secondary_side_elem->
point(n);
1128 center += secondary_side_elem->
point(n);
1129 normal += nodal_normals[n];
1131 center /= sub_elem_nodes.size();
1132 normal = normal.
unit();
1135 mortar_segment_helper[sel] = std::make_unique<MortarSegmentHelper>(nodes, center, normal);
1146 std::array<Real, 3> query_pt;
1148 switch (secondary_side_elem->
type())
1152 center_point = mortar_segment_helper[0]->center();
1153 query_pt = {{center_point(0), center_point(1), center_point(2)}};
1157 center_point = mortar_segment_helper[1]->center();
1158 query_pt = {{center_point(0), center_point(1), center_point(2)}};
1161 center_point = mortar_segment_helper[4]->center();
1162 query_pt = {{center_point(0), center_point(1), center_point(2)}};
1165 center_point = secondary_side_elem->
point(8);
1166 query_pt = {{center_point(0), center_point(1), center_point(2)}};
1170 "Face element type: ", secondary_side_elem->
type(),
"not supported for 3D mortar");
1176 const std::size_t num_results = 3;
1179 std::vector<size_t> ret_index(num_results);
1180 std::vector<Real> out_dist_sqr(num_results);
1181 nanoflann::KNNResultSet<Real> result_set(num_results);
1182 result_set.init(&ret_index[0], &out_dist_sqr[0]);
1186 std::set<const Elem *, CompareDofObjectsByID> processed_primary_elems;
1190 bool primary_elem_found =
false;
1191 std::set<const Elem *, CompareDofObjectsByID> primary_elem_candidates;
1199 "Lower-dimensional element squared distance verification failed.");
1202 std::vector<const Elem *> & node_elems =
1206 for (
auto elem : node_elems)
1207 primary_elem_candidates.insert(elem);
1217 while (!primary_elem_candidates.empty())
1219 const Elem * primary_elem_candidate = *primary_elem_candidates.begin();
1222 if (processed_primary_elems.count(primary_elem_candidate))
1226 std::vector<Point> nodal_points;
1229 std::vector<std::vector<unsigned int>> elem_to_node_map;
1232 std::vector<std::pair<unsigned int, unsigned int>> sub_elem_map;
1241 auto sub_elem_nodes = get_sub_elem_nodes(primary_elem_candidate->
type(), p_el);
1244 std::vector<Point> primary_sub_elem(sub_elem_nodes.size());
1245 for (
auto iv :
make_range(sub_elem_nodes.size()))
1247 const auto n = sub_elem_nodes[iv];
1248 primary_sub_elem[iv] = primary_elem_candidate->
point(n);
1261 mortar_segment_helper[s_el]->getMortarSegments(
1262 primary_sub_elem, nodal_points, elem_to_node_map);
1265 for (
auto i = sub_elem_map.size(); i < elem_to_node_map.size(); ++i)
1266 sub_elem_map.push_back(std::make_pair(s_el, p_el));
1271 processed_primary_elems.insert(primary_elem_candidate);
1272 primary_elem_candidates.erase(primary_elem_candidate);
1275 if (!elem_to_node_map.empty())
1280 if (!primary_elem_found)
1282 primary_elem_found =
true;
1283 primary_elem_candidates.clear();
1290 if (neighbor ==
nullptr || neighbor->subdomain_id() != primary_subd_id)
1293 if (processed_primary_elems.count(neighbor))
1296 primary_elem_candidates.insert(neighbor);
1302 std::vector<Node *> new_nodes;
1303 for (
auto pt : nodal_points)
1305 pt, next_node_id++, secondary_side_elem->
processor_id()));
1311 std::unique_ptr<Elem> new_elem;
1312 if (elem_to_node_map[el].size() == 3)
1313 new_elem = std::make_unique<Tri3>();
1315 mooseError(
"Active mortar segments only supports TRI elements, 3 nodes expected " 1317 elem_to_node_map[el].size(),
1320 new_elem->processor_id() = secondary_side_elem->
processor_id();
1321 new_elem->subdomain_id() = secondary_side_elem->
subdomain_id();
1322 new_elem->set_id(next_elem_id++);
1326 new_elem->set_node(i, new_nodes[elem_to_node_map[el][i]]);
1329 if (new_elem->volume() / secondary_volume < TOLERANCE)
1347 secondary_to_msm_element_set.insert(msm_new_elem);
1361 if (mortar_segment_helper[sel]->remainder() == 1.0)
1363 mooseWarning(
"Some secondary elements on mortar interface were unable to identify" 1364 " a corresponding primary element; this may be expected depending on" 1365 " problem geometry but may indicate a failure of the element search" 1369 if (secondary_to_msm_element_set.empty())
1386 if (msm_el->type() !=
TRI3)
1387 msm_el->subdomain_id()++;
1393 if (msm_el->type() !=
TRI3)
1394 msm_el->subdomain_id()--;
1409 std::unordered_map<processor_id_type, std::vector<std::pair<dof_id_type, dof_id_type>>>
1420 const Elem * secondary_elem = pr.second.secondary_elem;
1421 const Elem * primary_elem = pr.second.primary_elem;
1424 coupling_info[secondary_elem->
processor_id()].emplace_back(
1433 coupling_info[secondary_elem->
processor_id()].emplace_back(
1442 coupling_info[secondary_elem->
processor_id()].emplace_back(secondary_elem->
id(),
1443 primary_elem->
id());
1467 auto action_functor =
1469 const std::vector<std::pair<dof_id_type, dof_id_type>> & coupling_info)
1471 for (
auto [i, j] : coupling_info)
1477 std::vector<AutomaticMortarGeneration::MsmSubdomainStats>
1480 std::vector<MsmSubdomainStats> result;
1484 std::unordered_map<dof_id_type, Real> primary_elems_to_volume;
1488 for (
const auto *
const secondary_el :
1489 _mesh.active_local_subdomain_element_ptr_range(secondary_subd_id))
1491 secondary.push_back(secondary_el->volume());
1496 for (
const auto *
const msm_elem : it->second)
1498 msm.push_back(msm_elem->volume());
1502 if (msm_info.primary_elem)
1504 if (msm_info.primary_elem->subdomain_id() != primary_subd_id)
1505 mooseError(
"Unhandled primary-secondary pairing when computing mortar segment " 1506 "statistics. This could happen if you have the same secondary " 1507 "lower-dimensional subdomain ID paired with multiple lower-dimensional " 1508 "primary subdomain IDs. Contact a MOOSE developer for help.");
1509 if (
const auto [it, inserted] =
1510 primary_elems_to_volume.emplace(msm_info.primary_elem->id(),
Real{});
1512 it->second = msm_info.primary_elem->volume();
1515 MooseUtils::absoluteFuzzyEqual(it->second, msm_info.primary_elem->volume()),
1516 "Volumes should be consistent");
1524 primary.reserve(primary_elems_to_volume.size());
1525 for (
const auto [_,
volume] : primary_elems_to_volume)
1526 primary.push_back(
volume);
1543 result.push_back(stats);
1548 primary_elems_to_volume.clear();
1562 Moose::out <<
"Mortar Interface Statistics:" << std::endl;
1563 for (
const auto & stats : all_stats)
1565 std::vector<std::string> col_names = {
"mesh",
"n_elems",
"max",
"min",
"median"};
1566 std::vector<std::string> subds = {
"secondary_lower",
"primary_lower",
"mortar_segment"};
1567 std::vector<size_t> n_elems = {
1568 stats.secondary_lower_n_elems, stats.primary_lower_n_elems, stats.msm_n_elems};
1569 std::vector<Real> maxs = {
1570 stats.secondary_lower_max_volume, stats.primary_lower_max_volume, stats.msm_max_volume};
1571 std::vector<Real> mins = {
1572 stats.secondary_lower_min_volume, stats.primary_lower_min_volume, stats.msm_min_volume};
1573 std::vector<Real> medians = {stats.secondary_lower_median_volume,
1574 stats.primary_lower_median_volume,
1575 stats.msm_median_volume};
1582 table.
addData<std::string>(col_names[0], subds[i]);
1583 table.
addData<
size_t>(col_names[1], n_elems[i]);
1589 Moose::out <<
"secondary subdomain: " << stats.secondary_subd_id
1590 <<
" \tprimary subdomain: " << stats.primary_subd_id << std::endl;
1607 const Real tol = (
dim() == 3) ? 0.1 : TOLERANCE;
1609 std::unordered_map<processor_id_type, std::set<dof_id_type>> proc_to_inactive_nodes_set;
1613 std::unordered_set<dof_id_type> inactive_node_ids;
1615 std::unordered_map<const Elem *, Real> active_volume{};
1619 active_volume[el] = 0.;
1627 active_volume[secondary_elem] += msm_elem->
volume();
1635 if (
abs(active_volume[el] / el->volume() - 1.0) > tol)
1639 inactive_node_ids.insert(el->node_id(n));
1645 const auto secondary_subd_id = pr.second;
1651 const auto pid = el->processor_id();
1658 for (
const auto n :
make_range(el->n_nodes()))
1660 const auto node_id = el->node_id(n);
1661 if (inactive_node_ids.find(node_id) != inactive_node_ids.end())
1662 proc_to_inactive_nodes_set[pid].insert(node_id);
1670 std::unordered_map<processor_id_type, std::vector<dof_id_type>> proc_to_inactive_nodes_vector;
1671 for (
const auto & proc_set : proc_to_inactive_nodes_set)
1672 proc_to_inactive_nodes_vector[proc_set.first].insert(
1673 proc_to_inactive_nodes_vector[proc_set.first].end(),
1674 proc_set.second.begin(),
1675 proc_set.second.end());
1679 const std::vector<dof_id_type> & sent_data)
1682 mooseError(
"Should not be communicating with self.");
1683 for (
const auto pr : sent_data)
1684 inactive_node_ids.insert(pr);
1689 for (
const auto node_id : inactive_node_ids)
1702 std::unordered_map<processor_id_type, std::set<dof_id_type>> proc_to_active_nodes_set;
1706 std::unordered_set<dof_id_type> active_local_nodes;
1715 active_local_nodes.insert(secondary_elem->
node_id(n));
1721 const auto secondary_subd_id = pr.second;
1734 for (
const auto n :
make_range(el->n_nodes()))
1736 const auto node_id = el->node_id(n);
1737 if (active_local_nodes.find(node_id) != active_local_nodes.end())
1738 proc_to_active_nodes_set[pid].insert(node_id);
1746 std::unordered_map<processor_id_type, std::vector<dof_id_type>> proc_to_active_nodes_vector;
1747 for (
const auto & proc_set : proc_to_active_nodes_set)
1749 proc_to_active_nodes_vector[proc_set.first].reserve(proc_to_active_nodes_set.size());
1750 for (
const auto node_id : proc_set.second)
1751 proc_to_active_nodes_vector[proc_set.first].push_back(node_id);
1756 const std::vector<dof_id_type> & sent_data)
1759 mooseError(
"Should not be communicating with self.");
1760 active_local_nodes.insert(sent_data.begin(), sent_data.end());
1771 for (
const auto n :
make_range(el->n_nodes()))
1772 if (active_local_nodes.find(el->node_id(n)) == active_local_nodes.end())
1781 std::unordered_set<const Elem *> active_local_elems;
1787 const Real tol = (
dim() == 3) ? 0.1 : TOLERANCE;
1789 std::unordered_map<const Elem *, Real> active_volume;
1798 active_volume[secondary_elem] += msm_elem->
volume();
1809 if (
abs(active_volume[secondary_elem] / secondary_elem->
volume() - 1.0) > tol)
1813 active_local_elems.insert(secondary_elem);
1821 if (active_local_elems.find(el) == active_local_elems.end())
1831 mooseAssert(
dim == 2 ||
dim == 3,
1832 "AutomaticMortarGeneration::computeNodalGeometry() is only valid for " 1833 "mortar constraints on 2D or 3D meshes.");
1842 std::map<dof_id_type, std::vector<std::pair<Point, Real>>> node_to_normals_map;
1850 for (MeshBase::const_element_iterator el =
_mesh.active_elements_begin(),
1851 end_el =
_mesh.active_elements_end();
1855 const Elem * secondary_elem = *el;
1865 nnx_fe_face->attach_quadrature_rule(&qface);
1866 const std::vector<Point> & face_normals = nnx_fe_face->get_normals();
1868 const auto & JxW = nnx_fe_face->get_JxW();
1873 mooseAssert(interior_parent,
1874 "No interior parent exists for element " 1875 << secondary_elem->
id()
1876 <<
". There may be a problem with your sideset set-up.");
1887 nnx_fe_face->reinit(interior_parent, s);
1889 for (MooseIndex(secondary_elem->
n_nodes()) n = 0; n < secondary_elem->
n_nodes(); ++n)
1891 auto & normals_and_weights_vec = node_to_normals_map[secondary_elem->
node_id(n)];
1892 normals_and_weights_vec.push_back(std::make_pair(
sign * face_normals[n], JxW[n]));
1899 for (
const auto & pr : node_to_normals_map)
1902 const auto & node_id = pr.first;
1903 const auto & normals_and_weights_vec = pr.second;
1906 for (
const auto & norm_and_weight : normals_and_weights_vec)
1907 nodal_normal += norm_and_weight.first * norm_and_weight.second;
1908 nodal_normal = nodal_normal.
unit();
1912 Point nodal_tangent_one;
1913 Point nodal_tangent_two;
1923 Point & nodal_tangent_one,
1924 Point & nodal_tangent_two)
const 1928 mooseAssert(MooseUtils::absoluteFuzzyEqual(nodal_normal.
norm(), 1),
1929 "The input nodal normal should have unity norm");
1931 const Real nx = nodal_normal(0);
1932 const Real ny = nodal_normal(1);
1933 const Real nz = nodal_normal(2);
1938 const Point h_vector(nx + 1.0, ny, nz);
1943 if (
abs(h_vector(0)) < TOLERANCE)
1945 nodal_tangent_one(0) = 0;
1946 nodal_tangent_one(1) = 1;
1947 nodal_tangent_one(2) = 0;
1949 nodal_tangent_two(0) = 0;
1950 nodal_tangent_two(1) = 0;
1951 nodal_tangent_two(2) = -1;
1958 nodal_tangent_one(0) = -2.0 * h_vector(0) * h_vector(1) / (h * h);
1959 nodal_tangent_one(1) = 1.0 - 2.0 * h_vector(1) * h_vector(1) / (h * h);
1960 nodal_tangent_one(2) = -2.0 * h_vector(1) * h_vector(2) / (h * h);
1962 nodal_tangent_two(0) = -2.0 * h_vector(0) * h_vector(2) / (h * h);
1963 nodal_tangent_two(1) = -2.0 * h_vector(1) * h_vector(2) / (h * h);
1964 nodal_tangent_two(2) = 1.0 - 2.0 * h_vector(2) * h_vector(2) / (h * h);
1980 const Node & secondary_node,
1981 const Node & primary_node,
1982 const std::vector<const Elem *> * secondary_node_neighbors,
1983 const std::vector<const Elem *> * primary_node_neighbors,
1985 const Elem & candidate_element,
1986 std::set<const Elem *> & rejected_elem_candidates)
1988 if (!secondary_node_neighbors)
1990 if (!primary_node_neighbors)
1993 std::vector<bool> primary_elems_mapped(primary_node_neighbors->size(),
false);
2021 std::array<Real, 2> secondary_node_neighbor_cps, primary_node_neighbor_cps;
2023 for (
const auto nn :
index_range(*secondary_node_neighbors))
2025 const Elem *
const secondary_neigh = (*secondary_node_neighbors)[nn];
2026 const Point opposite = (secondary_neigh->
node_ptr(0) == &secondary_node)
2027 ? secondary_neigh->
point(1)
2028 : secondary_neigh->
point(0);
2029 const Point cp = nodal_normal.
cross(opposite - secondary_node);
2030 secondary_node_neighbor_cps[nn] = cp(2);
2033 for (
const auto nn :
index_range(*primary_node_neighbors))
2035 const Elem *
const primary_neigh = (*primary_node_neighbors)[nn];
2036 const Point opposite = (primary_neigh->
node_ptr(0) == &primary_node) ? primary_neigh->
point(1)
2037 : primary_neigh->
point(0);
2038 const Point cp = nodal_normal.
cross(opposite - primary_node);
2039 primary_node_neighbor_cps[nn] = cp(2);
2043 bool found_match =
false;
2044 for (
const auto snn :
index_range(*secondary_node_neighbors))
2045 for (
const auto mnn :
index_range(*primary_node_neighbors))
2046 if (secondary_node_neighbor_cps[snn] * primary_node_neighbor_cps[mnn] > 0)
2049 if (primary_elems_mapped[mnn])
2051 primary_elems_mapped[mnn] =
true;
2055 const Real xi2 = (&primary_node == (*primary_node_neighbors)[mnn]->node_ptr(0)) ? -1 : +1;
2056 const auto secondary_key =
2057 std::make_pair(&secondary_node, (*secondary_node_neighbors)[snn]);
2058 const auto primary_val = std::make_pair(xi2, (*primary_node_neighbors)[mnn]);
2063 (&secondary_node == (*secondary_node_neighbors)[snn]->node_ptr(0)) ? -1 : +1;
2065 const auto primary_key =
2066 std::make_tuple(primary_node.
id(), &primary_node, (*primary_node_neighbors)[mnn]);
2067 const auto secondary_val = std::make_pair(xi1, (*secondary_node_neighbors)[snn]);
2075 rejected_elem_candidates.insert(&candidate_element);
2082 if (secondary_node_neighbors->size() == 1 && primary_node_neighbors->size() == 2)
2083 for (
const auto i :
index_range(primary_elems_mapped))
2084 if (!primary_elems_mapped[i])
2087 std::make_tuple(primary_node.
id(), &primary_node, (*primary_node_neighbors)[i]),
2088 std::make_pair(1,
nullptr));
2096 SubdomainID lower_dimensional_primary_subdomain_id,
2097 SubdomainID lower_dimensional_secondary_subdomain_id)
2104 3, mesh_adaptor, nanoflann::KDTreeSingleIndexAdaptorParams(10));
2107 kd_tree.buildIndex();
2109 for (MeshBase::const_element_iterator el =
_mesh.active_elements_begin(),
2110 end_el =
_mesh.active_elements_end();
2114 const Elem * secondary_side_elem = *el;
2117 if (secondary_side_elem->
subdomain_id() != lower_dimensional_secondary_subdomain_id)
2124 for (MooseIndex(secondary_side_elem->
n_vertices()) n = 0; n < secondary_side_elem->
n_vertices();
2127 const Node * secondary_node = secondary_side_elem->
node_ptr(n);
2131 const std::vector<const Elem *> & secondary_node_neighbors =
2136 bool is_mapped =
true;
2137 for (MooseIndex(secondary_node_neighbors) snn = 0; snn < secondary_node_neighbors.size();
2140 auto secondary_key = std::make_pair(secondary_node, secondary_node_neighbors[snn]);
2156 std::array<Real, 3> query_pt = {
2157 {(*secondary_node)(0), (*secondary_node)(1), (*secondary_node)(2)}};
2163 const std::size_t num_results = 3;
2166 std::vector<size_t> ret_index(num_results);
2167 std::vector<Real> out_dist_sqr(num_results);
2168 nanoflann::KNNResultSet<Real> result_set(num_results);
2169 result_set.init(&ret_index[0], &out_dist_sqr[0]);
2173 bool projection_succeeded =
false;
2177 std::set<const Elem *> rejected_primary_elem_candidates;
2183 for (MooseIndex(result_set) r = 0; r < result_set.size(); ++r)
2187 out_dist_sqr[r]) <= TOLERANCE,
2188 "Lower-dimensional element squared distance verification failed.");
2192 std::vector<const Elem *> & primary_elem_candidates =
2196 for (MooseIndex(primary_elem_candidates) e = 0; e < primary_elem_candidates.size(); ++e)
2198 const Elem * primary_elem_candidate = primary_elem_candidates[e];
2201 if (rejected_primary_elem_candidates.count(primary_elem_candidate))
2207 unsigned int current_iterate = 0, max_iterates = 10;
2213 for (MooseIndex(primary_elem_candidate->
n_nodes()) n = 0;
2214 n < primary_elem_candidate->
n_nodes();
2218 const auto u = x2 - (*secondary_node);
2219 const auto F = u(0) * nodal_normal(1) - u(1) * nodal_normal(0);
2224 if (F.derivatives())
2226 Real dxi2 = -F.value() / F.derivatives();
2234 current_iterate = max_iterates;
2235 }
while (++current_iterate < max_iterates);
2237 Real xi2 = xi2_dn.value();
2251 (
abs((primary_elem_candidate->
point(0) - primary_elem_candidate->
point(1)).unit() *
2261 const Node * primary_node = (xi2 < 0) ? primary_elem_candidate->
node_ptr(0)
2262 : primary_elem_candidate->
node_ptr(1);
2263 const bool created_mortar_segment =
2266 &secondary_node_neighbors,
2269 *primary_elem_candidate,
2270 rejected_primary_elem_candidates);
2272 if (!created_mortar_segment)
2278 for (MooseIndex(secondary_node_neighbors) nn = 0;
2279 nn < secondary_node_neighbors.size();
2282 const Elem * neigh = secondary_node_neighbors[nn];
2286 if (secondary_node == neigh_node)
2288 auto key = std::make_pair(neigh_node, neigh);
2289 auto val = std::make_pair(xi2, primary_elem_candidate);
2296 projection_succeeded =
true;
2301 rejected_primary_elem_candidates.insert(primary_elem_candidate);
2304 if (projection_succeeded)
2308 if (!projection_succeeded)
2312 _console <<
"Failed to find primary Elem into which secondary node " 2313 <<
static_cast<const Point &
>(*secondary_node) <<
", id '" 2314 << secondary_node->
id() <<
"', projects onto\n" 2333 <<
" secondary nodes were successfully projected\n" 2350 SubdomainID lower_dimensional_primary_subdomain_id,
2351 SubdomainID lower_dimensional_secondary_subdomain_id)
2358 3, mesh_adaptor, nanoflann::KDTreeSingleIndexAdaptorParams(10));
2361 kd_tree.buildIndex();
2363 std::unordered_set<dof_id_type> primary_nodes_visited;
2365 for (
const auto & primary_side_elem :
_mesh.active_element_ptr_range())
2368 if (primary_side_elem->subdomain_id() != lower_dimensional_primary_subdomain_id)
2373 for (MooseIndex(primary_side_elem->n_vertices()) n = 0; n < primary_side_elem->n_vertices();
2377 const Node * primary_node = primary_side_elem->node_ptr(n);
2380 const std::vector<const Elem *> & primary_node_neighbors =
2388 std::make_tuple(primary_node->
id(), primary_node, primary_node_neighbors[0]);
2389 if (!primary_nodes_visited.insert(primary_node->
id()).second ||
2394 Real query_pt[3] = {(*primary_node)(0), (*primary_node)(1), (*primary_node)(2)};
2400 const size_t num_results = 3;
2403 std::vector<size_t> ret_index(num_results);
2404 std::vector<Real> out_dist_sqr(num_results);
2405 nanoflann::KNNResultSet<Real> result_set(num_results);
2406 result_set.init(&ret_index[0], &out_dist_sqr[0]);
2410 bool projection_succeeded =
false;
2415 std::set<const Elem *> rejected_secondary_elem_candidates;
2419 for (MooseIndex(result_set) r = 0; r < result_set.size(); ++r)
2424 "Lower-dimensional element squared distance verification failed.");
2428 const std::vector<const Elem *> & secondary_elem_candidates =
2432 for (MooseIndex(secondary_elem_candidates) e = 0; e < secondary_elem_candidates.size(); ++e)
2434 const Elem * secondary_elem_candidate = secondary_elem_candidates[e];
2437 if (rejected_secondary_elem_candidates.count(secondary_elem_candidate))
2440 std::vector<Point> nodal_normals(secondary_elem_candidate->
n_nodes());
2450 unsigned int current_iterate = 0, max_iterates = 10;
2461 for (MooseIndex(secondary_elem_candidate->
n_nodes()) n = 0;
2462 n < secondary_elem_candidate->
n_nodes();
2466 x1 += phi * secondary_elem_candidate->
point(n);
2467 normals += phi * nodal_normals[n];
2470 const auto u = x1 - (*primary_node);
2472 const auto F = u(0) * normals(1) - u(1) * normals(0);
2480 Real dxi1 = -F.value() / F.derivatives();
2485 }
while (++current_iterate < max_iterates);
2487 Real xi1 = xi1_dn.value();
2491 if ((current_iterate < max_iterates) && (
abs(xi1) <= 1. +
_xi_tolerance) &&
2492 (
abs((primary_side_elem->point(0) - primary_side_elem->point(1)).unit() *
2506 const Node & secondary_node = (xi1 < 0) ? secondary_elem_candidate->
node_ref(0)
2507 : secondary_elem_candidate->
node_ref(1);
2508 bool created_mortar_segment =
false;
2515 &primary_node_neighbors,
2517 *secondary_elem_candidate,
2518 rejected_secondary_elem_candidates);
2520 rejected_secondary_elem_candidates.insert(secondary_elem_candidate);
2522 if (!created_mortar_segment)
2538 const Elem * neigh = primary_node_neighbors[0];
2542 if (primary_node == neigh_node)
2544 auto key = std::make_tuple(neigh_node->
id(), neigh_node, neigh);
2545 auto val = std::make_pair(xi1, secondary_elem_candidate);
2551 projection_succeeded =
true;
2557 rejected_secondary_elem_candidates.insert(secondary_elem_candidate);
2561 if (projection_succeeded)
2565 if (!projection_succeeded &&
_debug)
2567 _console <<
"\nFailed to find point from which primary node " 2568 <<
static_cast<const Point &
>(*primary_node) <<
" was projected." << std::endl
2575 std::vector<AutomaticMortarGeneration::MortarFilterIter>
2582 const auto & secondary_elems = secondary_it->second;
2583 std::vector<MortarFilterIter> ret;
2584 ret.reserve(secondary_elems.size());
2588 auto *
const secondary_elem = secondary_elems[i];
2594 mooseAssert(secondary_elem->active(),
2595 "We loop over active elements when building the mortar segment mesh, so we golly " 2596 "well hope this is active.");
2597 mooseAssert(!msm_it->second.empty(),
2598 "We should have removed all secondaries from this map if they do not have any " 2599 "mortar segments associated with them.");
2600 ret.push_back(msm_it);
virtual T maximum() const
std::set< SubdomainID > _primary_boundary_subdomain_ids
MetaPhysicL::DualNumber< V, D, asd > abs(const MetaPhysicL::DualNumber< V, D, asd > &a)
const Elem * getSecondaryLowerdElemFromSecondaryElem(dof_id_type secondary_elem_id) const
Return lower dimensional secondary element given its interior parent.
T fe_lagrange_2D_shape(const libMesh::ElemType type, const Order order, const unsigned int i, const VectorType< T > &p)
std::unique_ptr< FEGenericBase< Real > > build(const unsigned int dim, const FEType &fet)
void allgather(const T &send_data, std::vector< T, A > &recv_data) const
std::unordered_set< const Elem * > _inactive_local_lm_elems
List of inactive lagrange multiplier nodes (for elemental variables)
static const std::string name_param
The name of the parameter that contains the object name.
static const std::string app_param
The name of the parameter that contains the MooseApp.
SubdomainID secondary_subd_id
SubdomainID primary_subd_id
unsigned int get_node_index(const Node *node_ptr) const
void computeInactiveLMNodes()
Get list of secondary nodes that don't contribute to interaction with any primary element...
std::vector< std::pair< BoundaryID, BoundaryID > > _primary_secondary_boundary_id_pairs
A list of primary/secondary boundary id pairs corresponding to each side of the mortar interface...
const Elem * interior_parent() const
static const std::string type_param
The name of the parameter that contains the object type.
Real secondary_lower_min_volume
unsigned int _t2x_var_num
void clear()
Clears the mortar segment mesh and accompanying data structures.
void dof_indices(const Elem *const elem, std::vector< dof_id_type > &di) const
std::set< SubdomainID > _secondary_boundary_subdomain_ids
The secondary/primary lower-dimensional boundary subdomain ids are the secondary/primary boundary ids...
void mooseError(Args &&... args)
Emit an error message with the given stringified, concatenated args and terminate the application...
std::unordered_map< dof_id_type, std::set< Elem *, CompareDofObjectsByID > > _secondary_elems_to_mortar_segments
We maintain a mapping from lower-dimensional secondary elements in the original mesh to (sets of) ele...
std::map< unsigned int, unsigned int > getPrimaryIpToLowerElementMap(const Elem &primary_elem, const Elem &primary_elem_ip, const Elem &lower_secondary_elem) const
Compute on-the-fly mapping from primary interior parent nodes to its corresponding lower dimensional ...
unsigned int _t2z_var_num
const bool _periodic
Whether this object will be generating a mortar segment mesh for periodic constraints.
void swap(std::vector< T > &data, const std::size_t idx0, const std::size_t idx1, const libMesh::Parallel::Communicator &comm)
Swap function for serial or distributed vector of data.
unsigned int which_side_am_i(const Elem *e) const
void outputMortarMesh()
Write the mortar segment mesh to exodus.
Real _newton_tolerance
Newton solve tolerance for node projections.
Real secondary_lower_max_volume
void mooseWarning(Args &&... args)
Emit a warning message with the given stringified, concatenated args.
void output() override
Overload this function with the desired output activities.
std::string getOutputFileBase(bool for_non_moose_build_output=false) const
Get the output file base name.
MortarNodalGeometryOutput(const InputParameters ¶ms)
std::unordered_map< const Elem *, MortarSegmentInfo > _msm_elem_to_info
Map between Elems in the mortar segment mesh and their info structs.
const bool _distributed
Whether the mortar segment mesh is distributed.
Base class for MOOSE-based applications.
void buildMortarSegmentMesh()
Builds the mortar segment mesh once the secondary and primary node projections have been completed...
const Parallel::Communicator & comm() const
void msmStatistics()
Prints mortar segment mesh statistics to console (calls computeMsmStatistics internally) ...
unsigned int _nnx_var_num
void buildCouplingInformation()
build the _mortar_interface_coupling data
std::vector< Point > getNodalNormals(const Elem &secondary_elem) const
Special adaptor that works with subdomains of the Mesh.
The following methods are specializations for using the libMesh::Parallel::packed_range_* routines fo...
static const Real invalid_xi
unsigned int _nny_var_num
std::string mortarInterfaceName() const
Specialization of SubProblem for solving nonlinear equations plus auxiliary equations.
FEProblemBase & feProblem() const
unsigned int _nnz_var_num
void projectSecondaryNodes()
Project secondary nodes (find xi^(2) values) to the closest points on the primary surface...
const std::unordered_map< dof_id_type, std::set< Elem *, CompareDofObjectsByID > > & secondariesToMortarSegments() const
std::unordered_set< dof_id_type > _projected_secondary_nodes
Debugging container for printing information about fraction of successful projections for secondary n...
This class is a container/interface for the objects involved in automatic generation of mortar spaces...
const bool _debug
Whether to print debug output.
Statistics for one primary-secondary subdomain pair.
Based class for output objects.
const Elem * primary_elem
void push_parallel_vector_data(const Communicator &comm, MapToVectors &&data, const ActionFunctor &act_on_data)
uint8_t processor_id_type
std::set< SubdomainID > _secondary_ip_sub_ids
All the secondary interior parent subdomain IDs associated with the mortar mesh.
virtual libMesh::EquationSystems & es()=0
std::optional< dof_id_type > _msm_node_id_start
Cached per-rank starting ID for 3D MSM nodes/elements.
TypeVector< Real > unit() const
void libmesh_ignore(const Args &...)
Real secondary_lower_median_volume
void projectPrimaryNodes()
(Inverse) project primary nodes to the points on the secondary surface where they would have come fro...
unsigned int _t1z_var_num
std::size_t primary_lower_n_elems
CTSub CT_OPERATOR_BINARY CTMul CTCompareLess CTCompareGreater CTCompareEqual _arg template cos(_arg) *_arg.template D< dtag >()) CT_SIMPLE_UNARY_FUNCTION(cos
const Node & node_ref(const unsigned int i) const
AutomaticMortarGeneration(MooseApp &app, MeshBase &mesh_in, const std::pair< BoundaryID, BoundaryID > &boundary_key, const std::pair< SubdomainID, SubdomainID > &subdomain_key, bool on_displaced, bool periodic, const bool debug, const bool correct_edge_dropping, const Real minimum_projection_angle)
Must be constructed with a reference to the Mesh we are generating mortar spaces for.
virtual unsigned int n_nodes() const=0
unsigned int _t2y_var_num
void buildNodeToElemMaps()
Once the secondary_requested_boundary_ids and primary_requested_boundary_ids containers have been fil...
std::unordered_map< dof_id_type, const Elem * > _secondary_element_to_secondary_lowerd_element
Map from full dimensional secondary element id to lower dimensional secondary element.
virtual void write_equation_systems(const std::string &fname, const EquationSystems &es, const std::set< std::string > *system_names=nullptr) override
std::map< std::tuple< dof_id_type, const Node *, const Elem * >, std::pair< Real, const Elem * > > _primary_node_and_elem_to_xi1_secondary_elem
Same type of container, but for mapping (Primary Node ID, Primary Node, Primary Elem) -> (xi^(1)...
void projectPrimaryNodesSinglePair(SubdomainID lower_dimensional_primary_subdomain_id, SubdomainID lower_dimensional_secondary_subdomain_id)
Helper function used internally by AutomaticMortarGeneration::project_primary_nodes().
std::unordered_map< dof_id_type, std::vector< const Elem * > > _nodes_to_primary_elem_map
SimpleRange< IndexType > as_range(const std::pair< IndexType, IndexType > &p)
const bool _correct_edge_dropping
Flag to enable regressed treatment of edge dropping where all LM DoFs on edge dropping element are st...
bool processAlignedNodes(const Node &secondary_node, const Node &primary_node, const std::vector< const Elem *> *secondary_node_neighbors, const std::vector< const Elem *> *primary_node_neighbors, const VectorValue< Real > &nodal_normal, const Elem &candidate_element, std::set< const Elem *> &rejected_element_candidates)
Process aligned nodes.
An inteface for the _console for outputting to the Console object.
unsigned int _t1y_var_num
std::array< MooseUtils::SemidynamicVector< Point, 9 >, 2 > getNodalTangents(const Elem &secondary_elem) const
Compute the two nodal tangents, which are built on-the-fly.
std::unique_ptr< InputParameters > _output_params
Storage for the input parameters used by the mortar nodal geometry output.
std::set< BoundaryID > _primary_requested_boundary_ids
The boundary ids corresponding to all the primary surfaces.
std::map< unsigned int, unsigned int > getSecondaryIpToLowerElementMap(const Elem &lower_secondary_elem) const
Compute on-the-fly mapping from secondary interior parent nodes to lower dimensional nodes...
TypeVector< typename CompareTypes< Real, T2 >::supertype > cross(const TypeVector< T2 > &v) const
nanoflann::KDTreeSingleIndexAdaptor< subdomain_adatper_t, NanoflannMeshSubdomainAdaptor< 3 >, 3 > subdomain_kd_tree_t
void computeNodalGeometry()
Computes and stores the nodal normal/tangent vectors in a local data structure instead of using the E...
Real _xi_tolerance
Tolerance for checking projection xi values.
static InputParameters validParams()
void computeIncorrectEdgeDroppingInactiveLMNodes()
Computes inactive secondary nodes when incorrect edge dropping behavior is enabled (any node touching...
std::unordered_set< dof_id_type > _failed_secondary_node_projections
Secondary nodes that failed to project.
void buildMortarSegmentMesh3d()
Builds the mortar segment mesh once the secondary and primary node projections have been completed...
const Elem * secondary_elem
Real primary_lower_max_volume
void projectSecondaryNodesSinglePair(SubdomainID lower_dimensional_primary_subdomain_id, SubdomainID lower_dimensional_secondary_subdomain_id)
Helper function responsible for projecting secondary nodes onto primary elements for a single primary...
Provides a way for users to bail out of the current solve.
void set_unique_id(unique_id_type new_id)
virtual unsigned int n_vertices() const=0
DIE A HORRIBLE DEATH HERE typedef LIBMESH_DEFAULT_SCALAR_TYPE Real
std::unordered_map< std::pair< const Node *, const Elem * >, std::pair< Real, const Elem * > > _secondary_node_and_elem_to_xi2_primary_elem
Similar to the map above, but associates a (Secondary Node, Secondary Elem) pair to a (xi^(2)...
Holds xi^(1), xi^(2), and other data for a given mortar segment.
Real primary_lower_min_volume
Generic class for solving transient nonlinear problems.
virtual SimpleRange< element_iterator > active_local_subdomain_elements_ptr_range(subdomain_id_type sid)=0
subdomain_id_type subdomain_id() const
const bool _on_displaced
Whether this object is on the displaced mesh.
const Node * node_ptr(const unsigned int i) const
unsigned int spatial_dimension() const
virtual void write(const std::string &fname) override
std::vector< MsmSubdomainStats > computeMsmStatistics()
Computes mortar segment mesh statistics and returns one entry per subdomain pair. ...
void initOutput()
initialize mortar-mesh based output
void addOutput(std::shared_ptr< Output > output)
Adds an existing output object to the warehouse.
virtual Real volume() const
std::unordered_map< const Node *, std::array< Point, 2 > > _secondary_node_to_hh_nodal_tangents
Container for storing the nodal tangent/binormal vectors associated with each secondary node (Househo...
IntRange< T > make_range(T beg, T end)
KOKKOS_INLINE_FUNCTION T sign(T x)
Returns the sign of a value.
unsigned int _t1x_var_num
Real primary_lower_median_volume
unsigned int mesh_dimension() const
unsigned int level ElemType type std::set< subdomain_id_type > ss processor_id_type pid unsigned int level std::set< subdomain_id_type > virtual ss SimpleRange< element_iterator > active_subdomain_elements_ptr_range(subdomain_id_type sid)=0
virtual T minimum() const
std::size_t secondary_lower_n_elems
T fe_lagrange_1D_shape(const Order order, const unsigned int i, const T &xi)
const Real _minimum_projection_angle
Parameter to control which angle (in degrees) is admissible for the creation of mortar segments...
MeshBase & _mesh
Reference to the mesh stored in equation_systems.
virtual const Point & point(const dof_id_type i) const=0
std::vector< std::pair< SubdomainID, SubdomainID > > _primary_secondary_subdomain_id_pairs
A list of primary/secondary subdomain id pairs corresponding to each side of the mortar interface...
const ConsoleStream _console
An instance of helper class to write streams to the Console objects.
SimpleRange< NeighborPtrIter > neighbor_ptr_range()
virtual unsigned int n_sub_elem() const=0
std::set< BoundaryID > _secondary_requested_boundary_ids
The boundary ids corresponding to all the secondary surfaces.
std::unordered_map< const Node *, Point > _secondary_node_to_nodal_normal
Container for storing the nodal normal vector associated with each secondary node.
virtual const Node * node_ptr(const dof_id_type i) const=0
std::unordered_set< const Node * > _inactive_local_lm_nodes
processor_id_type processor_id() const
virtual Order default_order() const=0
std::unordered_map< dof_id_type, std::unordered_set< dof_id_type > > _mortar_interface_coupling
Used by the AugmentSparsityOnInterface functor to determine whether a given Elem is coupled to any ot...
std::set< SubdomainID > _primary_ip_sub_ids
All the primary interior parent subdomain IDs associated with the mortar mesh.
SearchParams SearchParameters
AutomaticMortarGeneration & _amg
The mortar generation object that we will query for nodal normal and tangent information.
void set_hdf5_writing(bool write_hdf5)
processor_id_type processor_id() const
void householderOrthogolization(const Point &normal, Point &tangent_one, Point &tangent_two) const
Householder orthogonalization procedure to obtain proper basis for tangent and binormal vectors...
virtual ElemType type() const=0
dof_id_type node_id(const unsigned int i) const
std::vector< Point > getNormals(const Elem &secondary_elem, const std::vector< Point > &xi1_pts) const
Compute the normals at given reference points on a secondary element.
static InputParameters validParams()
MooseApp & _app
The Moose app.
const Point & point(const unsigned int i) const
std::unique_ptr< MeshBase > _mortar_segment_mesh
1D Mesh of mortar segment elements which gets built by the call to build_mortar_segment_mesh().
auto index_range(const T &sizable)
void set_extra_integer(const unsigned int index, const dof_id_type value)
OutputWarehouse & getOutputWarehouse()
Get the OutputWarehouse objects.
std::unordered_map< dof_id_type, std::vector< const Elem * > > _nodes_to_secondary_elem_map
Map from nodes to connected lower-dimensional elements on the secondary/primary subdomains.
std::unordered_map< const Elem *, unsigned int > _lower_elem_to_side_id
Keeps track of the mapping between lower-dimensional elements and the side_id of the interior_parent ...
void computeInactiveLMElems()
Get list of secondary elems without any corresponding primary elements.
void set_union(T &data, const unsigned int root_id) const