Line data Source code
1 : // The libMesh Finite Element Library.
2 : // Copyright (C) 2002-2025 Benjamin S. Kirk, John W. Peterson, Roy H. Stogner
3 :
4 : // This library is free software; you can redistribute it and/or
5 : // modify it under the terms of the GNU Lesser General Public
6 : // License as published by the Free Software Foundation; either
7 : // version 2.1 of the License, or (at your option) any later version.
8 :
9 : // This library is distributed in the hope that it will be useful,
10 : // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 : // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 : // Lesser General Public License for more details.
13 :
14 : // You should have received a copy of the GNU Lesser General Public
15 : // License along with this library; if not, write to the Free Software
16 : // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17 :
18 :
19 :
20 : // Local includes
21 : #include "libmesh/elem.h"
22 : #include "libmesh/elem_range.h"
23 : #include "libmesh/libmesh_logging.h"
24 : #include "libmesh/mesh_base.h"
25 : #include "libmesh/mesh_communication.h"
26 : #include "libmesh/mesh_serializer.h"
27 : #include "libmesh/mesh_tools.h"
28 : #include "libmesh/node_range.h"
29 : #include "libmesh/parallel.h"
30 : #include "libmesh/parallel_algebra.h"
31 : #include "libmesh/parallel_ghost_sync.h"
32 : #include "libmesh/sphere.h"
33 : #include "libmesh/threads.h"
34 : #include "libmesh/enum_to_string.h"
35 : #include "libmesh/enum_elem_type.h"
36 : #include "libmesh/int_range.h"
37 : #include "libmesh/utility.h"
38 : #include "libmesh/boundary_info.h"
39 :
40 : #ifndef NDEBUG
41 : # include "libmesh/remote_elem.h"
42 : #endif
43 :
44 : // C++ includes
45 : #include <limits>
46 : #include <numeric> // for std::accumulate
47 : #include <set>
48 : #include <unordered_map>
49 : #include <unordered_set>
50 :
51 :
52 :
53 : // ------------------------------------------------------------
54 : // anonymous namespace for helper classes
55 : namespace {
56 :
57 : using namespace libMesh;
58 :
59 : /**
60 : * SumElemWeight(Range) sums the number of nodes per element
61 : * for each element in the provided range. The join() method
62 : * defines how to combine the reduction operation from two
63 : * distinct instances of this class which may be executed on
64 : * separate threads.
65 : */
66 : class SumElemWeight
67 : {
68 : public:
69 0 : SumElemWeight () :
70 0 : _weight(0)
71 0 : {}
72 :
73 0 : SumElemWeight (SumElemWeight &, Threads::split) :
74 0 : _weight(0)
75 0 : {}
76 :
77 0 : void operator()(const ConstElemRange & range)
78 : {
79 0 : for (const auto & elem : range)
80 0 : _weight += elem->n_nodes();
81 0 : }
82 :
83 0 : dof_id_type weight() const
84 0 : { return _weight; }
85 :
86 : // If we don't have threads we never need a join, and icpc yells a
87 : // warning if it sees an anonymous function that's never used
88 : #if LIBMESH_USING_THREADS
89 0 : void join (const SumElemWeight & other)
90 0 : { _weight += other.weight(); }
91 : #endif
92 :
93 : private:
94 : dof_id_type _weight;
95 : };
96 :
97 :
98 : /**
99 : * FindBBox(Range) computes the bounding box for the objects
100 : * in the specified range. This class may be split and subranges
101 : * can be executed on separate threads. The join() method
102 : * defines how the results from two separate threads are combined.
103 : */
104 : class FindBBox
105 : {
106 : public:
107 2297438 : FindBBox () : _bbox()
108 32814 : {}
109 :
110 40652 : FindBBox (FindBBox & other, Threads::split) :
111 40652 : _bbox(other._bbox)
112 13773 : {}
113 :
114 40636 : void operator()(const ConstNodeRange & range)
115 : {
116 5073100 : for (const auto & node : range)
117 : {
118 457357 : libmesh_assert(node);
119 5032464 : _bbox.union_with(*node);
120 : }
121 40636 : }
122 :
123 2349758 : void operator()(const ConstElemRange & range)
124 : {
125 27081965 : for (const auto & elem : range)
126 : {
127 1274884 : libmesh_assert(elem);
128 24732207 : _bbox.union_with(elem->loose_bounding_box());
129 : }
130 2349758 : }
131 :
132 16688 : Point & min() { return _bbox.min(); }
133 :
134 16688 : Point & max() { return _bbox.max(); }
135 :
136 : // If we don't have threads we never need a join, and icpc yells a
137 : // warning if it sees an anonymous function that's never used
138 : #if LIBMESH_USING_THREADS
139 13773 : void join (const FindBBox & other)
140 : {
141 40652 : _bbox.union_with(other._bbox);
142 26879 : }
143 : #endif
144 :
145 48940 : libMesh::BoundingBox & bbox ()
146 : {
147 48940 : return _bbox;
148 : }
149 :
150 : private:
151 : BoundingBox _bbox;
152 : };
153 :
154 : #ifdef DEBUG
155 2786430 : void assert_semiverify_dofobj(const Parallel::Communicator & communicator,
156 : const DofObject * d,
157 : unsigned int sysnum = libMesh::invalid_uint)
158 : {
159 2786430 : if (d)
160 : {
161 2710542 : const unsigned int n_sys = d->n_systems();
162 :
163 5421084 : std::vector<unsigned int> n_vars (n_sys, 0);
164 5793166 : for (unsigned int s = 0; s != n_sys; ++s)
165 3082624 : if (sysnum == s ||
166 : sysnum == libMesh::invalid_uint)
167 2710542 : n_vars[s] = d->n_vars(s);
168 :
169 : const unsigned int tot_n_vars =
170 2710542 : std::accumulate(n_vars.begin(), n_vars.end(), 0);
171 :
172 5421084 : std::vector<unsigned int> n_comp (tot_n_vars, 0);
173 5421084 : std::vector<dof_id_type> first_dof (tot_n_vars, 0);
174 :
175 5793166 : for (unsigned int s = 0, i=0; s != n_sys; ++s)
176 : {
177 3082624 : if (sysnum != s &&
178 : sysnum != libMesh::invalid_uint)
179 372082 : continue;
180 :
181 9533250 : for (unsigned int v = 0; v != n_vars[s]; ++v, ++i)
182 : {
183 6822708 : n_comp[i] = d->n_comp(s,v);
184 6822708 : first_dof[i] = n_comp[i] ? d->dof_number(s,v,0) : DofObject::invalid_id;
185 : }
186 : }
187 :
188 2710542 : libmesh_assert(communicator.semiverify(&n_sys));
189 2710542 : libmesh_assert(communicator.semiverify(&n_vars));
190 2710542 : libmesh_assert(communicator.semiverify(&n_comp));
191 2710542 : libmesh_assert(communicator.semiverify(&first_dof));
192 : }
193 : else
194 : {
195 75888 : const unsigned int * p_ui = nullptr;
196 75888 : const std::vector<unsigned int> * p_vui = nullptr;
197 75888 : const std::vector<dof_id_type> * p_vdid = nullptr;
198 :
199 75888 : libmesh_assert(communicator.semiverify(p_ui));
200 75888 : libmesh_assert(communicator.semiverify(p_vui));
201 75888 : libmesh_assert(communicator.semiverify(p_vui));
202 75888 : libmesh_assert(communicator.semiverify(p_vdid));
203 : }
204 2786430 : }
205 :
206 :
207 :
208 : #ifdef LIBMESH_ENABLE_UNIQUE_ID
209 9197454 : void assert_dofobj_unique_id(const Parallel::Communicator & comm,
210 : const DofObject * d,
211 : const std::unordered_set<unique_id_type> & unique_ids)
212 : {
213 : // Duplicating some semiverify code here so we can reuse
214 : // tempmin,tempmax afterward
215 :
216 : unique_id_type tempmin, tempmax;
217 9197454 : if (d)
218 : {
219 8815826 : tempmin = tempmax = d->unique_id();
220 : }
221 : else
222 : {
223 381628 : TIMPI::Attributes<unique_id_type>::set_highest(tempmin);
224 381628 : TIMPI::Attributes<unique_id_type>::set_lowest(tempmax);
225 : }
226 9197454 : comm.min(tempmin);
227 9197454 : comm.max(tempmax);
228 18013280 : bool invalid = d && ((d->unique_id() != tempmin) ||
229 8815826 : (d->unique_id() != tempmax));
230 9197454 : comm.max(invalid);
231 :
232 : // First verify that everything is in sync
233 9197454 : libmesh_assert(!invalid);
234 :
235 : // Then verify that any remote id doesn't duplicate a local one.
236 9197454 : if (!d && tempmin == tempmax)
237 50164 : libmesh_assert(!unique_ids.count(tempmin));
238 9197454 : }
239 : #endif // LIBMESH_ENABLE_UNIQUE_ID
240 : #endif // DEBUG
241 :
242 291668 : void find_nodal_neighbors_helper(const dof_id_type global_id,
243 : const std::vector<const Elem *> & node_to_elem_vec,
244 : std::vector<const Node *> & neighbors)
245 : {
246 : // We'll construct a std::set<const Node *> for more efficient
247 : // searching while finding the nodal neighbors, and return it to the
248 : // user in a std::vector.
249 16432 : std::set<const Node *> neighbor_set;
250 :
251 : // Look through the elements that contain this node
252 : // find the local node id... then find the side that
253 : // node lives on in the element
254 : // next, look for the _other_ node on that side
255 : // That other node is a "nodal_neighbor"... save it
256 1699882 : for (const auto & elem : node_to_elem_vec)
257 : {
258 : // We only care about active elements...
259 1408214 : if (elem->active())
260 : {
261 : // Which local node number is global_id?
262 1368546 : unsigned local_node_number = elem->local_node(global_id);
263 :
264 : // Make sure it was found
265 39668 : libmesh_assert_not_equal_to(local_node_number, libMesh::invalid_uint);
266 :
267 1408214 : const unsigned short n_edges = elem->n_edges();
268 :
269 : // If this element has no edges, the edge-based algorithm below doesn't make sense.
270 1408214 : if (!n_edges)
271 : {
272 1846 : switch (elem->type())
273 : {
274 710 : case EDGE2:
275 : {
276 20 : switch (local_node_number)
277 : {
278 355 : case 0:
279 : // The other node is a nodal neighbor
280 365 : neighbor_set.insert(elem->node_ptr(1));
281 355 : break;
282 :
283 355 : case 1:
284 : // The other node is a nodal neighbor
285 365 : neighbor_set.insert(elem->node_ptr(0));
286 355 : break;
287 :
288 0 : default:
289 0 : libmesh_error_msg("Invalid local node number: " << local_node_number << " found." << std::endl);
290 : }
291 20 : break;
292 : }
293 :
294 852 : case EDGE3:
295 : {
296 24 : switch (local_node_number)
297 : {
298 : // The outside nodes have node 2 as a neighbor
299 710 : case 0:
300 : case 1:
301 730 : neighbor_set.insert(elem->node_ptr(2));
302 710 : break;
303 :
304 : // The middle node has the outer nodes as neighbors
305 142 : case 2:
306 146 : neighbor_set.insert(elem->node_ptr(0));
307 146 : neighbor_set.insert(elem->node_ptr(1));
308 142 : break;
309 :
310 0 : default:
311 0 : libmesh_error_msg("Invalid local node number: " << local_node_number << " found." << std::endl);
312 : }
313 24 : break;
314 : }
315 :
316 284 : case EDGE4:
317 : {
318 8 : switch (local_node_number)
319 : {
320 71 : case 0:
321 : // The left-middle node is a nodal neighbor
322 73 : neighbor_set.insert(elem->node_ptr(2));
323 71 : break;
324 :
325 71 : case 1:
326 : // The right-middle node is a nodal neighbor
327 73 : neighbor_set.insert(elem->node_ptr(3));
328 71 : break;
329 :
330 : // The left-middle node
331 71 : case 2:
332 73 : neighbor_set.insert(elem->node_ptr(0));
333 73 : neighbor_set.insert(elem->node_ptr(3));
334 71 : break;
335 :
336 : // The right-middle node
337 71 : case 3:
338 73 : neighbor_set.insert(elem->node_ptr(1));
339 73 : neighbor_set.insert(elem->node_ptr(2));
340 71 : break;
341 :
342 0 : default:
343 0 : libmesh_error_msg("Invalid local node number: " << local_node_number << " found." << std::endl);
344 : }
345 8 : break;
346 : }
347 :
348 0 : default:
349 0 : libmesh_error_msg("Unrecognized ElemType: " << Utility::enum_to_string(elem->type()) << std::endl);
350 : }
351 : }
352 :
353 1408214 : const auto elem_order = Elem::type_to_default_order_map[elem->type()];
354 :
355 : // Index of the current edge
356 39668 : unsigned current_edge = 0;
357 :
358 1408214 : const unsigned short n_nodes = elem->n_nodes();
359 :
360 5830662 : while (current_edge < n_edges)
361 : {
362 : // Find the edge the node is on
363 124576 : bool found_edge = false;
364 11188890 : for (; current_edge<n_edges; ++current_edge)
365 10298266 : if (elem->is_node_on_edge(local_node_number, current_edge))
366 : {
367 99488 : found_edge = true;
368 99488 : break;
369 : }
370 :
371 : // Did we find one?
372 4422448 : if (found_edge)
373 : {
374 3531824 : const Node * node_to_save = nullptr;
375 :
376 : // Find another node in this element on this edge
377 28524392 : for (unsigned other_node_this_edge = 0; other_node_this_edge != n_nodes; other_node_this_edge++)
378 : {
379 46258488 : const bool both_vertices = elem->is_vertex(local_node_number) &&
380 21265920 : elem->is_vertex(other_node_this_edge);
381 32651016 : if ( elem->is_node_on_edge(other_node_this_edge, current_edge) && // On the current edge
382 25038584 : elem->node_id(other_node_this_edge) != global_id && // But not the original node
383 : // vertex nodes on the same edge of higher order elements are not nodal neighbors
384 4272128 : (elem_order == 1 || !both_vertices))
385 : {
386 : // We've found a nodal neighbor! Save a pointer to it..
387 3785578 : node_to_save = elem->node_ptr(other_node_this_edge);
388 :
389 : // Make sure we found something
390 106636 : libmesh_assert(node_to_save != nullptr);
391 :
392 3678942 : neighbor_set.insert(node_to_save);
393 : }
394 : }
395 : }
396 :
397 : // Keep looking for edges, node may be on more than one edge
398 4422448 : current_edge++;
399 : }
400 : } // if (elem->active())
401 : } // for
402 :
403 : // Assign the entries from the set to the vector. Note: this
404 : // replaces any existing contents in neighbors and modifies its size
405 : // accordingly.
406 283452 : neighbors.assign(neighbor_set.begin(), neighbor_set.end());
407 291668 : }
408 :
409 : }
410 :
411 :
412 : namespace libMesh
413 : {
414 :
415 : // ------------------------------------------------------------
416 : // MeshTools functions
417 :
418 : namespace MeshTools
419 : {
420 :
421 0 : dof_id_type total_weight(const MeshBase & mesh)
422 : {
423 0 : if (!mesh.is_serial())
424 : {
425 0 : libmesh_parallel_only(mesh.comm());
426 0 : dof_id_type weight = MeshTools::weight (mesh, mesh.processor_id());
427 0 : mesh.comm().sum(weight);
428 : dof_id_type unpartitioned_weight =
429 0 : MeshTools::weight (mesh, DofObject::invalid_processor_id);
430 0 : return weight + unpartitioned_weight;
431 : }
432 :
433 0 : SumElemWeight sew;
434 :
435 0 : Threads::parallel_reduce (ConstElemRange (mesh.elements_begin(),
436 0 : mesh.elements_end()),
437 : sew);
438 0 : return sew.weight();
439 :
440 : }
441 :
442 :
443 :
444 0 : dof_id_type weight(const MeshBase & mesh, const processor_id_type pid)
445 : {
446 0 : SumElemWeight sew;
447 :
448 0 : Threads::parallel_reduce (ConstElemRange (mesh.pid_elements_begin(pid),
449 0 : mesh.pid_elements_end(pid)),
450 : sew);
451 0 : return sew.weight();
452 : }
453 :
454 :
455 :
456 3991 : void build_nodes_to_elem_map (const MeshBase & mesh,
457 : std::vector<std::vector<dof_id_type>> & nodes_to_elem_map)
458 : {
459 : // A vector indexed over all nodes is too inefficient to use for a
460 : // distributed mesh. Use the unordered_map API instead.
461 3991 : if (!mesh.is_serial())
462 : libmesh_deprecated();
463 :
464 3991 : nodes_to_elem_map.resize (mesh.max_node_id());
465 :
466 100916 : for (const auto & elem : mesh.element_ptr_range())
467 192936 : for (auto & node : elem->node_ref_range())
468 : {
469 4104 : libmesh_assert_less (node.id(), nodes_to_elem_map.size());
470 4104 : libmesh_assert_less (elem->id(), mesh.n_elem());
471 :
472 147780 : nodes_to_elem_map[node.id()].push_back(elem->id());
473 3763 : }
474 3991 : }
475 :
476 :
477 :
478 0 : void build_nodes_to_elem_map (const MeshBase & mesh,
479 : std::vector<std::vector<const Elem *>> & nodes_to_elem_map)
480 : {
481 : // A vector indexed over all nodes is too inefficient to use for a
482 : // distributed mesh. Use the unordered_map API instead.
483 0 : if (!mesh.is_serial())
484 : libmesh_deprecated();
485 :
486 0 : nodes_to_elem_map.resize (mesh.max_node_id());
487 :
488 0 : for (const auto & elem : mesh.element_ptr_range())
489 0 : for (auto & node : elem->node_ref_range())
490 : {
491 0 : libmesh_assert_less (node.id(), nodes_to_elem_map.size());
492 :
493 0 : nodes_to_elem_map[node.id()].push_back(elem);
494 0 : }
495 0 : }
496 :
497 :
498 :
499 243232 : void build_nodes_to_elem_map (const MeshBase & mesh,
500 : std::unordered_map<dof_id_type, std::vector<dof_id_type>> & nodes_to_elem_map)
501 : {
502 7220 : nodes_to_elem_map.clear();
503 :
504 52440932 : for (const auto & elem : mesh.element_ptr_range())
505 145263145 : for (auto & node : elem->node_ref_range())
506 116532561 : nodes_to_elem_map[node.id()].push_back(elem->id());
507 243232 : }
508 :
509 :
510 :
511 3840 : void build_nodes_to_elem_map (const MeshBase & mesh,
512 : std::unordered_map<dof_id_type, std::vector<const Elem *>> & nodes_to_elem_map)
513 : {
514 110 : nodes_to_elem_map.clear();
515 :
516 565218 : for (const auto & elem : mesh.element_ptr_range())
517 1983679 : for (auto & node : elem->node_ref_range())
518 1691127 : nodes_to_elem_map[node.id()].push_back(elem);
519 3840 : }
520 :
521 :
522 :
523 : std::unordered_set<dof_id_type>
524 4628 : find_boundary_nodes(const MeshBase & mesh)
525 : {
526 136 : std::unordered_set<dof_id_type> boundary_nodes;
527 :
528 : // Loop over elements, find those on boundary, and
529 : // mark them as true in on_boundary.
530 959170 : for (const auto & elem : mesh.active_element_ptr_range())
531 2721159 : for (auto s : elem->side_index_range())
532 2294362 : if (elem->neighbor_ptr(s) == nullptr) // on the boundary
533 : {
534 139426 : auto nodes_on_side = elem->nodes_on_side(s);
535 :
536 699937 : for (auto & local_id : nodes_on_side)
537 585299 : boundary_nodes.insert(elem->node_ptr(local_id)->id());
538 4356 : }
539 :
540 4628 : return boundary_nodes;
541 : }
542 :
543 : std::unordered_set<dof_id_type>
544 1930 : find_block_boundary_nodes(const MeshBase & mesh)
545 : {
546 60 : std::unordered_set<dof_id_type> block_boundary_nodes;
547 :
548 : // Loop over elements, find those on boundary, and
549 : // mark them as true in on_boundary.
550 778728 : for (const auto & elem : mesh.active_element_ptr_range())
551 2186902 : for (auto s : elem->side_index_range())
552 1840302 : if (elem->neighbor_ptr(s) && (elem->neighbor_ptr(s)->subdomain_id() != elem->subdomain_id()))
553 : {
554 0 : auto nodes_on_side = elem->nodes_on_side(s);
555 :
556 0 : for (auto & local_id : nodes_on_side)
557 0 : block_boundary_nodes.insert(elem->node_ptr(local_id)->id());
558 1810 : }
559 :
560 1930 : return block_boundary_nodes;
561 : }
562 :
563 :
564 :
565 : libMesh::BoundingBox
566 1155381 : create_bounding_box (const MeshBase & mesh)
567 : {
568 : // This function must be run on all processors at once
569 16126 : libmesh_parallel_only(mesh.comm());
570 :
571 16126 : FindBBox find_bbox;
572 :
573 : // Start with any unpartitioned elements we know about locally
574 2294636 : Threads::parallel_reduce (ConstElemRange (mesh.pid_elements_begin(DofObject::invalid_processor_id),
575 1171507 : mesh.pid_elements_end(DofObject::invalid_processor_id)),
576 : find_bbox);
577 :
578 : // And combine with our local elements
579 1155381 : find_bbox.bbox().union_with(create_local_bounding_box(mesh));
580 :
581 : // Compare the bounding boxes across processors
582 1155381 : mesh.comm().min(find_bbox.min());
583 1155381 : mesh.comm().max(find_bbox.max());
584 :
585 1155381 : return find_bbox.bbox();
586 : }
587 :
588 :
589 :
590 : libMesh::BoundingBox
591 19490 : create_nodal_bounding_box (const MeshBase & mesh)
592 : {
593 : // This function must be run on all processors at once
594 562 : libmesh_parallel_only(mesh.comm());
595 :
596 562 : FindBBox find_bbox;
597 :
598 : // Start with any unpartitioned nodes we know about locally
599 38418 : Threads::parallel_reduce (ConstNodeRange (mesh.pid_nodes_begin(DofObject::invalid_processor_id),
600 20052 : mesh.pid_nodes_end(DofObject::invalid_processor_id)),
601 : find_bbox);
602 :
603 : // Add our local nodes
604 38418 : Threads::parallel_reduce (ConstNodeRange (mesh.local_nodes_begin(),
605 38418 : mesh.local_nodes_end()),
606 : find_bbox);
607 :
608 : // Compare the bounding boxes across processors
609 19490 : mesh.comm().min(find_bbox.min());
610 19490 : mesh.comm().max(find_bbox.max());
611 :
612 19490 : return find_bbox.bbox();
613 : }
614 :
615 :
616 :
617 : Sphere
618 0 : bounding_sphere(const MeshBase & mesh)
619 : {
620 0 : libMesh::BoundingBox bbox = create_bounding_box(mesh);
621 :
622 0 : const Real diag = (bbox.second - bbox.first).norm();
623 0 : const Point cent = (bbox.second + bbox.first)/2;
624 :
625 0 : return Sphere (cent, .5*diag);
626 : }
627 :
628 :
629 :
630 : libMesh::BoundingBox
631 1155381 : create_local_bounding_box (const MeshBase & mesh)
632 : {
633 16126 : FindBBox find_bbox;
634 :
635 2294636 : Threads::parallel_reduce (ConstElemRange (mesh.local_elements_begin(),
636 1171507 : mesh.local_elements_end()),
637 : find_bbox);
638 :
639 1155381 : return find_bbox.bbox();
640 : }
641 :
642 :
643 :
644 : libMesh::BoundingBox
645 0 : create_processor_bounding_box (const MeshBase & mesh,
646 : const processor_id_type pid)
647 : {
648 : // This can only be run in parallel, with consistent arguments.
649 0 : libmesh_parallel_only(mesh.comm());
650 0 : libmesh_assert(mesh.comm().verify(pid));
651 :
652 0 : libmesh_assert_less (pid, mesh.n_processors());
653 :
654 0 : FindBBox find_bbox;
655 :
656 0 : Threads::parallel_reduce (ConstElemRange (mesh.pid_elements_begin(pid),
657 0 : mesh.pid_elements_end(pid)),
658 : find_bbox);
659 :
660 : // Compare the bounding boxes across processors
661 0 : mesh.comm().min(find_bbox.min());
662 0 : mesh.comm().max(find_bbox.max());
663 :
664 0 : return find_bbox.bbox();
665 : }
666 :
667 :
668 :
669 : Sphere
670 0 : processor_bounding_sphere (const MeshBase & mesh,
671 : const processor_id_type pid)
672 : {
673 : libMesh::BoundingBox bbox =
674 0 : create_processor_bounding_box(mesh, pid);
675 :
676 0 : const Real diag = (bbox.second - bbox.first).norm();
677 0 : const Point cent = (bbox.second + bbox.first)/2;
678 :
679 0 : return Sphere (cent, .5*diag);
680 : }
681 :
682 :
683 :
684 : libMesh::BoundingBox
685 0 : create_subdomain_bounding_box (const MeshBase & mesh,
686 : const subdomain_id_type sid)
687 : {
688 : // This can only be run in parallel, with consistent arguments.
689 0 : libmesh_parallel_only(mesh.comm());
690 0 : libmesh_assert(mesh.comm().verify(sid));
691 :
692 0 : FindBBox find_bbox;
693 :
694 : Threads::parallel_reduce
695 0 : (ConstElemRange (mesh.active_local_subdomain_elements_begin(sid),
696 0 : mesh.active_local_subdomain_elements_end(sid)),
697 : find_bbox);
698 :
699 : // Compare the bounding boxes across processors
700 0 : mesh.comm().min(find_bbox.min());
701 0 : mesh.comm().max(find_bbox.max());
702 :
703 0 : return find_bbox.bbox();
704 : }
705 :
706 :
707 :
708 : Sphere
709 0 : subdomain_bounding_sphere (const MeshBase & mesh,
710 : const subdomain_id_type sid)
711 : {
712 : libMesh::BoundingBox bbox =
713 0 : create_subdomain_bounding_box(mesh, sid);
714 :
715 0 : const Real diag = (bbox.second - bbox.first).norm();
716 0 : const Point cent = (bbox.second + bbox.first)/2;
717 :
718 0 : return Sphere (cent, .5*diag);
719 : }
720 :
721 :
722 :
723 0 : void elem_types (const MeshBase & mesh,
724 : std::vector<ElemType> & et)
725 : {
726 : // Loop over the the elements. If the current element type isn't in
727 : // the vector, insert it.
728 0 : for (const auto & elem : mesh.element_ptr_range())
729 0 : if (!std::count(et.begin(), et.end(), elem->type()))
730 0 : et.push_back(elem->type());
731 0 : }
732 :
733 :
734 :
735 0 : dof_id_type n_elem_of_type (const MeshBase & mesh,
736 : const ElemType type)
737 : {
738 0 : return static_cast<dof_id_type>(std::distance(mesh.type_elements_begin(type),
739 0 : mesh.type_elements_end (type)));
740 : }
741 :
742 :
743 :
744 0 : dof_id_type n_active_elem_of_type (const MeshBase & mesh,
745 : const ElemType type)
746 : {
747 0 : return static_cast<dof_id_type>(std::distance(mesh.active_type_elements_begin(type),
748 0 : mesh.active_type_elements_end (type)));
749 : }
750 :
751 0 : dof_id_type n_non_subactive_elem_of_type_at_level(const MeshBase & mesh,
752 : const ElemType type,
753 : const unsigned int level)
754 : {
755 0 : dof_id_type cnt = 0;
756 :
757 : // iterate over the elements of the specified type
758 0 : for (const auto & elem : as_range(mesh.type_elements_begin(type),
759 0 : mesh.type_elements_end(type)))
760 0 : if ((elem->level() == level) && !elem->subactive())
761 0 : cnt++;
762 :
763 0 : return cnt;
764 : }
765 :
766 :
767 1425 : unsigned int n_active_local_levels(const MeshBase & mesh)
768 : {
769 1425 : unsigned int nl = 0;
770 :
771 40866 : for (auto & elem : mesh.active_local_element_ptr_range())
772 41845 : nl = std::max(elem->level() + 1, nl);
773 :
774 1425 : return nl;
775 : }
776 :
777 :
778 :
779 1425 : unsigned int n_active_levels(const MeshBase & mesh)
780 : {
781 42 : libmesh_parallel_only(mesh.comm());
782 :
783 1425 : unsigned int nl = n_active_local_levels(mesh);
784 :
785 2808 : for (const auto & elem : as_range(mesh.unpartitioned_elements_begin(),
786 5616 : mesh.unpartitioned_elements_end()))
787 0 : if (elem->active())
788 1341 : nl = std::max(elem->level() + 1, nl);
789 :
790 1425 : mesh.comm().max(nl);
791 1425 : return nl;
792 : }
793 :
794 :
795 :
796 1652430 : unsigned int n_local_levels(const MeshBase & mesh)
797 : {
798 1652430 : unsigned int nl = 0;
799 :
800 4874238 : for (const auto & elem : as_range(mesh.local_elements_begin(),
801 36514058 : mesh.local_elements_end()))
802 33438006 : nl = std::max(elem->level() + 1, nl);
803 :
804 1652430 : return nl;
805 : }
806 :
807 :
808 :
809 1652430 : unsigned int n_levels(const MeshBase & mesh)
810 : {
811 28338 : libmesh_parallel_only(mesh.comm());
812 :
813 1652430 : unsigned int nl = n_local_levels(mesh);
814 :
815 3669858 : for (const auto & elem : as_range(mesh.unpartitioned_elements_begin(),
816 20088704 : mesh.unpartitioned_elements_end()))
817 15131430 : nl = std::max(elem->level() + 1, nl);
818 :
819 1652430 : mesh.comm().max(nl);
820 :
821 : // n_levels() is only valid and should only be called in cases where
822 : // the mesh is validly distributed (or serialized). Let's run an
823 : // expensive test in debug mode to make sure this is such a case.
824 : #ifdef DEBUG
825 28338 : const unsigned int paranoid_nl = paranoid_n_levels(mesh);
826 28338 : libmesh_assert_equal_to(nl, paranoid_nl);
827 : #endif
828 1652430 : return nl;
829 : }
830 :
831 :
832 :
833 38446 : unsigned int paranoid_n_levels(const MeshBase & mesh)
834 : {
835 28636 : libmesh_parallel_only(mesh.comm());
836 :
837 38446 : unsigned int nl = 0;
838 4051009 : for (const auto & elem : mesh.element_ptr_range())
839 4012265 : nl = std::max(elem->level() + 1, nl);
840 :
841 38446 : mesh.comm().max(nl);
842 38446 : return nl;
843 : }
844 :
845 :
846 :
847 639 : dof_id_type n_connected_components(const MeshBase & mesh,
848 : Real constraint_tol)
849 : {
850 36 : LOG_SCOPE("n_connected_components()", "MeshTools");
851 :
852 : // Yes, I'm being lazy. This is for mesh analysis before a
853 : // simulation, not anything going in any loops.
854 639 : if (!mesh.is_serial_on_zero())
855 0 : libmesh_not_implemented();
856 :
857 639 : dof_id_type n_components = 0;
858 :
859 657 : if (mesh.processor_id())
860 : {
861 522 : mesh.comm().broadcast(n_components);
862 531 : return n_components;
863 : }
864 :
865 : // All nodes in a set here are connected (at least indirectly) to
866 : // all other nodes in the same set, but have not yet been discovered
867 : // to be connected to nodes in other sets.
868 9 : std::vector<std::unordered_set<const Node *>> components;
869 :
870 : // With a typical mesh with few components and somewhat-contiguous
871 : // ordering, vector performance should be fine. With a mesh with
872 : // many components or completely scrambled ordering, performance
873 : // can be a disaster.
874 3204 : auto find_component = [&components](const Node * n) {
875 267 : std::unordered_set<const Node *> * component = nullptr;
876 :
877 8132 : for (auto & c: components)
878 4928 : if (c.find(n) != c.end())
879 : {
880 105 : libmesh_assert(component == nullptr);
881 105 : component = &c;
882 : }
883 :
884 3204 : return component;
885 108 : };
886 :
887 : auto add_to_component =
888 2010 : [&find_component]
889 2412 : (std::unordered_set<const Node *> & component, const Node * n) {
890 :
891 2412 : auto current_component = find_component(n);
892 : // We may already know we're in the desired component
893 2412 : if (&component == current_component)
894 51 : return;
895 :
896 : // If we're unknown, we should be in the desired component
897 1950 : else if (!current_component)
898 147 : component.insert(n);
899 :
900 : // If we think we're in another component, it should actually be
901 : // part of the desired component
902 : else
903 : {
904 3 : component.merge(*current_component);
905 3 : libmesh_assert(current_component->empty());
906 : }
907 99 : };
908 :
909 9 : auto & constraint_rows = mesh.get_constraint_rows();
910 :
911 1659 : for (const auto & elem : mesh.element_ptr_range())
912 : {
913 792 : const Node * first_node = elem->node_ptr(0);
914 :
915 792 : auto component = find_component(first_node);
916 :
917 : // If we didn't find one, make a new one, reusing an existing
918 : // slot if possible or growing our vector if necessary
919 792 : if (!component)
920 684 : for (auto & c: components)
921 354 : if (c.empty())
922 0 : component = &c;
923 :
924 432 : if (!component)
925 219 : component = &components.emplace_back();
926 :
927 3234 : for (const Node & n : elem->node_ref_range())
928 : {
929 2376 : add_to_component(*component, &n);
930 :
931 2376 : auto it = constraint_rows.find(&n);
932 2376 : if (it == constraint_rows.end())
933 195 : continue;
934 :
935 72 : for (const auto & [pr, val] : it->second)
936 : {
937 : // Ignore too-trivial constraint coefficients if
938 : // we get a non-default-0 constraint_tol
939 39 : if (std::abs(val) < constraint_tol)
940 0 : continue;
941 :
942 36 : const Elem * spline_elem = pr.first;
943 3 : libmesh_assert(spline_elem == mesh.elem_ptr(spline_elem->id()));
944 :
945 : const Node * spline_node =
946 36 : spline_elem->node_ptr(pr.second);
947 :
948 36 : add_to_component(*component, spline_node);
949 : }
950 : }
951 90 : }
952 :
953 327 : for (auto & component : components)
954 219 : if (!component.empty())
955 144 : ++n_components;
956 :
957 : // We calculated this on proc 0; now let everyone else know too
958 108 : mesh.comm().broadcast(n_components);
959 :
960 108 : return n_components;
961 90 : }
962 :
963 :
964 :
965 0 : void get_not_subactive_node_ids(const MeshBase & mesh,
966 : std::set<dof_id_type> & not_subactive_node_ids)
967 : {
968 0 : for (const auto & elem : mesh.element_ptr_range())
969 0 : if (!elem->subactive())
970 0 : for (auto & n : elem->node_ref_range())
971 0 : not_subactive_node_ids.insert(n.id());
972 0 : }
973 :
974 :
975 :
976 485036 : dof_id_type n_elem (const MeshBase::const_element_iterator & begin,
977 : const MeshBase::const_element_iterator & end)
978 : {
979 950552 : return cast_int<dof_id_type>(std::distance(begin, end));
980 : }
981 :
982 :
983 :
984 0 : dof_id_type n_nodes (const MeshBase::const_node_iterator & begin,
985 : const MeshBase::const_node_iterator & end)
986 : {
987 0 : return cast_int<dof_id_type>(std::distance(begin, end));
988 : }
989 :
990 :
991 :
992 426 : Real volume (const MeshBase & mesh,
993 : unsigned int dim)
994 : {
995 12 : libmesh_parallel_only(mesh.comm());
996 :
997 426 : if (dim == libMesh::invalid_uint)
998 426 : dim = mesh.mesh_dimension();
999 :
1000 426 : Real vol = 0;
1001 :
1002 : // first my local elements
1003 889 : for (const auto & elem : as_range(mesh.local_elements_begin(),
1004 2807 : mesh.local_elements_end()))
1005 588 : if (elem->dim() == dim)
1006 990 : vol += elem->volume();
1007 :
1008 : // then count any unpartitioned objects, once
1009 438 : if (mesh.processor_id() == 0)
1010 138 : for (const auto & elem : as_range(mesh.unpartitioned_elements_begin(),
1011 276 : mesh.unpartitioned_elements_end()))
1012 0 : if (elem->dim() == dim)
1013 60 : vol += elem->volume();
1014 :
1015 426 : mesh.comm().sum(vol);
1016 426 : return vol;
1017 : }
1018 :
1019 :
1020 :
1021 1425 : unsigned int n_p_levels (const MeshBase & mesh)
1022 : {
1023 42 : libmesh_parallel_only(mesh.comm());
1024 :
1025 1425 : unsigned int max_p_level = 0;
1026 :
1027 : // first my local elements
1028 7068 : for (const auto & elem : as_range(mesh.local_elements_begin(),
1029 48788 : mesh.local_elements_end()))
1030 48773 : max_p_level = std::max(elem->p_level(), max_p_level);
1031 :
1032 : // then any unpartitioned objects
1033 2808 : for (const auto & elem : as_range(mesh.unpartitioned_elements_begin(),
1034 5616 : mesh.unpartitioned_elements_end()))
1035 1341 : max_p_level = std::max(elem->p_level(), max_p_level);
1036 :
1037 1425 : mesh.comm().max(max_p_level);
1038 1425 : return max_p_level + 1;
1039 : }
1040 :
1041 :
1042 :
1043 0 : void find_nodal_neighbors(const MeshBase &,
1044 : const Node & node,
1045 : const std::vector<std::vector<const Elem *>> & nodes_to_elem_map,
1046 : std::vector<const Node *> & neighbors)
1047 : {
1048 0 : find_nodal_neighbors_helper(node.id(), nodes_to_elem_map[node.id()],
1049 : neighbors);
1050 0 : }
1051 :
1052 :
1053 :
1054 291668 : void find_nodal_neighbors(const MeshBase &,
1055 : const Node & node,
1056 : const std::unordered_map<dof_id_type, std::vector<const Elem *>> & nodes_to_elem_map,
1057 : std::vector<const Node *> & neighbors)
1058 : {
1059 : const std::vector<const Elem *> node_to_elem_vec =
1060 299884 : libmesh_map_find(nodes_to_elem_map, node.id());
1061 291668 : find_nodal_neighbors_helper(node.id(), node_to_elem_vec, neighbors);
1062 291668 : }
1063 :
1064 :
1065 :
1066 0 : void find_hanging_nodes_and_parents(const MeshBase & mesh,
1067 : std::map<dof_id_type, std::vector<dof_id_type>> & hanging_nodes)
1068 : {
1069 : // Loop through all the elements
1070 0 : for (auto & elem : mesh.active_local_element_ptr_range())
1071 0 : if (elem->type() == QUAD4)
1072 0 : for (auto s : elem->side_index_range())
1073 : {
1074 : // Loop over the sides looking for sides that have hanging nodes
1075 : // This code is inspired by compute_proj_constraints()
1076 0 : const Elem * neigh = elem->neighbor_ptr(s);
1077 :
1078 : // If not a boundary side
1079 0 : if (neigh != nullptr)
1080 : {
1081 : // Is there a coarser element next to this one?
1082 0 : if (neigh->level() < elem->level())
1083 : {
1084 0 : const Elem * ancestor = elem;
1085 0 : while (neigh->level() < ancestor->level())
1086 0 : ancestor = ancestor->parent();
1087 0 : unsigned int s_neigh = neigh->which_neighbor_am_i(ancestor);
1088 0 : libmesh_assert_less (s_neigh, neigh->n_neighbors());
1089 :
1090 : // Couple of helper uints...
1091 0 : unsigned int local_node1=0;
1092 0 : unsigned int local_node2=0;
1093 :
1094 0 : bool found_in_neighbor = false;
1095 :
1096 : // Find the two vertices that make up this side
1097 0 : while (!elem->is_node_on_side(local_node1++,s)) { }
1098 0 : local_node1--;
1099 :
1100 : // Start looking for the second one with the next node
1101 0 : local_node2=local_node1+1;
1102 :
1103 : // Find the other one
1104 0 : while (!elem->is_node_on_side(local_node2++,s)) { }
1105 0 : local_node2--;
1106 :
1107 : //Pull out their global ids:
1108 0 : dof_id_type node1 = elem->node_id(local_node1);
1109 0 : dof_id_type node2 = elem->node_id(local_node2);
1110 :
1111 : // Now find which node is present in the neighbor
1112 : // FIXME This assumes a level one rule!
1113 : // The _other_ one is the hanging node
1114 :
1115 : // First look for the first one
1116 : // FIXME could be streamlined a bit
1117 0 : for (unsigned int n=0;n<neigh->n_sides();n++)
1118 0 : if (neigh->node_id(n) == node1)
1119 0 : found_in_neighbor=true;
1120 :
1121 0 : dof_id_type hanging_node=0;
1122 :
1123 0 : if (!found_in_neighbor)
1124 0 : hanging_node=node1;
1125 : else // If it wasn't node1 then it must be node2!
1126 0 : hanging_node=node2;
1127 :
1128 : // Reset for reuse
1129 0 : local_node1=0;
1130 :
1131 : // Find the first node that makes up the side in the neighbor (these should be the parent nodes)
1132 0 : while (!neigh->is_node_on_side(local_node1++,s_neigh)) { }
1133 0 : local_node1--;
1134 :
1135 0 : local_node2=local_node1+1;
1136 :
1137 : // Find the second node...
1138 0 : while (!neigh->is_node_on_side(local_node2++,s_neigh)) { }
1139 0 : local_node2--;
1140 :
1141 : // Save them if we haven't already found the parents for this one
1142 0 : if (hanging_nodes[hanging_node].size()<2)
1143 : {
1144 0 : hanging_nodes[hanging_node].push_back(neigh->node_id(local_node1));
1145 0 : hanging_nodes[hanging_node].push_back(neigh->node_id(local_node2));
1146 : }
1147 : }
1148 : }
1149 0 : }
1150 0 : }
1151 :
1152 :
1153 :
1154 36 : void clear_spline_nodes(MeshBase & mesh)
1155 : {
1156 6 : std::vector<Elem *> nodeelem_to_delete;
1157 :
1158 8737 : for (auto & elem : mesh.element_ptr_range())
1159 5073 : if (elem->type() == NODEELEM &&
1160 4140 : elem->mapping_type() == RATIONAL_BERNSTEIN_MAP)
1161 4170 : nodeelem_to_delete.push_back(elem);
1162 :
1163 3 : auto & constraint_rows = mesh.get_constraint_rows();
1164 :
1165 : // All our constraint_rows ought to be for spline constraints we're
1166 : // about to get rid of.
1167 : #ifndef NDEBUG
1168 762 : for (auto & node_row : constraint_rows)
1169 2064 : for (auto pr : node_row.second)
1170 : {
1171 1305 : const Elem * elem = pr.first.first;
1172 1305 : libmesh_assert(elem->type() == NODEELEM);
1173 1305 : libmesh_assert(elem->mapping_type() == RATIONAL_BERNSTEIN_MAP);
1174 : }
1175 : #endif
1176 :
1177 3 : constraint_rows.clear();
1178 :
1179 4176 : for (Elem * elem : nodeelem_to_delete)
1180 : {
1181 690 : Node * node = elem->node_ptr(0);
1182 4140 : mesh.delete_elem(elem);
1183 4140 : mesh.delete_node(node);
1184 : }
1185 36 : }
1186 :
1187 :
1188 :
1189 936 : bool valid_is_prepared (const MeshBase & mesh)
1190 : {
1191 84 : LOG_SCOPE("libmesh_assert_valid_is_prepared()", "MeshTools");
1192 :
1193 936 : if (!mesh.is_prepared())
1194 0 : return true;
1195 :
1196 978 : std::unique_ptr<MeshBase> mesh_clone = mesh.clone();
1197 :
1198 : // Try preparing (without allowing repartitioning or renumbering, to
1199 : // avoid false assertion failures)
1200 68 : bool old_allow_renumbering = mesh_clone->allow_renumbering();
1201 42 : mesh_clone->allow_renumbering(false);
1202 : bool old_allow_remote_element_removal =
1203 68 : mesh_clone->allow_remote_element_removal();
1204 68 : bool old_skip_partitioning = mesh_clone->skip_partitioning();
1205 42 : mesh_clone->skip_partitioning(true);
1206 42 : mesh_clone->allow_remote_element_removal(false);
1207 936 : mesh_clone->prepare_for_use();
1208 42 : mesh_clone->allow_renumbering(old_allow_renumbering);
1209 42 : mesh_clone->allow_remote_element_removal(old_allow_remote_element_removal);
1210 42 : mesh_clone->skip_partitioning(old_skip_partitioning);
1211 :
1212 936 : return (mesh == *mesh_clone);
1213 868 : }
1214 :
1215 :
1216 :
1217 : #ifndef NDEBUG
1218 :
1219 122 : void libmesh_assert_equal_n_systems (const MeshBase & mesh)
1220 : {
1221 244 : LOG_SCOPE("libmesh_assert_equal_n_systems()", "MeshTools");
1222 :
1223 122 : unsigned int n_sys = libMesh::invalid_uint;
1224 :
1225 10842 : for (const auto & elem : mesh.element_ptr_range())
1226 : {
1227 10720 : if (n_sys == libMesh::invalid_uint)
1228 122 : n_sys = elem->n_systems();
1229 : else
1230 10598 : libmesh_assert_equal_to (elem->n_systems(), n_sys);
1231 : }
1232 :
1233 29237 : for (const auto & node : mesh.node_ptr_range())
1234 : {
1235 29115 : if (n_sys == libMesh::invalid_uint)
1236 0 : n_sys = node->n_systems();
1237 : else
1238 29115 : libmesh_assert_equal_to (node->n_systems(), n_sys);
1239 : }
1240 122 : }
1241 :
1242 :
1243 :
1244 : #ifdef LIBMESH_ENABLE_AMR
1245 0 : void libmesh_assert_old_dof_objects (const MeshBase & mesh)
1246 : {
1247 0 : LOG_SCOPE("libmesh_assert_old_dof_objects()", "MeshTools");
1248 :
1249 0 : for (const auto & elem : mesh.element_ptr_range())
1250 : {
1251 0 : if (elem->refinement_flag() == Elem::JUST_REFINED ||
1252 0 : elem->refinement_flag() == Elem::INACTIVE)
1253 0 : continue;
1254 :
1255 0 : if (elem->has_dofs())
1256 0 : libmesh_assert(elem->get_old_dof_object());
1257 :
1258 0 : for (auto & node : elem->node_ref_range())
1259 0 : if (node.has_dofs())
1260 0 : libmesh_assert(node.get_old_dof_object());
1261 : }
1262 0 : }
1263 : #else
1264 : void libmesh_assert_old_dof_objects (const MeshBase &) {}
1265 : #endif // LIBMESH_ENABLE_AMR
1266 :
1267 :
1268 :
1269 0 : void libmesh_assert_valid_node_pointers(const MeshBase & mesh)
1270 : {
1271 0 : LOG_SCOPE("libmesh_assert_valid_node_pointers()", "MeshTools");
1272 :
1273 : // Here we specifically do not want "auto &" because we need to
1274 : // reseat the (temporary) pointer variable in the loop below,
1275 : // without modifying the original.
1276 0 : for (const Elem * elem : mesh.element_ptr_range())
1277 : {
1278 0 : libmesh_assert (elem);
1279 0 : while (elem)
1280 : {
1281 0 : elem->libmesh_assert_valid_node_pointers();
1282 0 : for (auto n : elem->neighbor_ptr_range())
1283 0 : if (n && n != remote_elem)
1284 0 : n->libmesh_assert_valid_node_pointers();
1285 :
1286 0 : libmesh_assert_not_equal_to (elem->parent(), remote_elem);
1287 0 : elem = elem->parent();
1288 : }
1289 : }
1290 0 : }
1291 :
1292 :
1293 :
1294 8698 : void libmesh_assert_valid_remote_elems(const MeshBase & mesh)
1295 : {
1296 17396 : LOG_SCOPE("libmesh_assert_valid_remote_elems()", "MeshTools");
1297 :
1298 645131 : for (const auto & elem : as_range(mesh.local_elements_begin(),
1299 1290262 : mesh.local_elements_end()))
1300 : {
1301 636433 : libmesh_assert (elem);
1302 :
1303 : // We currently don't allow active_local_elements to have
1304 : // remote_elem neighbors
1305 636433 : if (elem->active())
1306 2782245 : for (auto n : elem->neighbor_ptr_range())
1307 2239277 : libmesh_assert_not_equal_to (n, remote_elem);
1308 :
1309 : #ifdef LIBMESH_ENABLE_AMR
1310 636433 : const Elem * parent = elem->parent();
1311 636433 : if (parent)
1312 355748 : libmesh_assert_not_equal_to (parent, remote_elem);
1313 :
1314 : // We can only be strict about active elements' subactive
1315 : // children
1316 636433 : if (elem->active() && elem->has_children())
1317 26661 : for (auto & child : elem->child_ref_range())
1318 21348 : libmesh_assert_not_equal_to (&child, remote_elem);
1319 : #endif
1320 : }
1321 8698 : }
1322 :
1323 :
1324 :
1325 388 : void libmesh_assert_valid_elem_ids(const MeshBase & mesh)
1326 : {
1327 776 : LOG_SCOPE("libmesh_assert_valid_elem_ids()", "MeshTools");
1328 :
1329 388 : processor_id_type lastprocid = 0;
1330 388 : dof_id_type lastelemid = 0;
1331 :
1332 13378 : for (const auto & elem : mesh.active_element_ptr_range())
1333 : {
1334 12990 : libmesh_assert (elem);
1335 12990 : processor_id_type elemprocid = elem->processor_id();
1336 12990 : dof_id_type elemid = elem->id();
1337 :
1338 12990 : libmesh_assert_greater_equal (elemid, lastelemid);
1339 12990 : libmesh_assert_greater_equal (elemprocid, lastprocid);
1340 :
1341 12990 : lastelemid = elemid;
1342 12990 : lastprocid = elemprocid;
1343 : }
1344 388 : }
1345 :
1346 :
1347 :
1348 816 : void libmesh_assert_valid_amr_elem_ids(const MeshBase & mesh)
1349 : {
1350 1632 : LOG_SCOPE("libmesh_assert_valid_amr_elem_ids()", "MeshTools");
1351 :
1352 320464 : for (const auto & elem : mesh.element_ptr_range())
1353 : {
1354 319648 : libmesh_assert (elem);
1355 :
1356 319648 : const Elem * parent = elem->parent();
1357 :
1358 319648 : if (parent)
1359 : {
1360 24548 : libmesh_assert_greater_equal (elem->id(), parent->id());
1361 24548 : libmesh_assert_greater_equal (elem->processor_id(), parent->processor_id());
1362 : }
1363 : }
1364 816 : }
1365 :
1366 :
1367 :
1368 12256 : void libmesh_assert_valid_amr_interior_parents(const MeshBase & mesh)
1369 : {
1370 24512 : LOG_SCOPE("libmesh_assert_valid_amr_interior_parents()", "MeshTools");
1371 :
1372 1395355 : for (const auto & elem : mesh.element_ptr_range())
1373 : {
1374 1383099 : libmesh_assert (elem);
1375 :
1376 : // We can skip to the next element if we're full-dimension
1377 : // and therefore don't have any interior parents
1378 1383099 : if (elem->dim() >= LIBMESH_DIM)
1379 623182 : continue;
1380 :
1381 759917 : const Elem * ip = elem->interior_parent();
1382 :
1383 759917 : const Elem * parent = elem->parent();
1384 :
1385 759917 : if (ip && (ip != remote_elem) && parent)
1386 : {
1387 1000 : libmesh_assert_equal_to (ip->top_parent(),
1388 : elem->top_parent()->interior_parent());
1389 :
1390 1000 : if (ip->level() == elem->level())
1391 1000 : libmesh_assert_equal_to (ip->parent(),
1392 : parent->interior_parent());
1393 : else
1394 : {
1395 0 : libmesh_assert_less (ip->level(), elem->level());
1396 0 : libmesh_assert_equal_to (ip, parent->interior_parent());
1397 : }
1398 : }
1399 : }
1400 12256 : }
1401 :
1402 :
1403 :
1404 0 : void libmesh_assert_contiguous_dof_ids(const MeshBase & mesh, unsigned int sysnum)
1405 : {
1406 0 : LOG_SCOPE("libmesh_assert_contiguous_dof_ids()", "MeshTools");
1407 :
1408 0 : if (mesh.n_processors() == 1)
1409 0 : return;
1410 :
1411 0 : libmesh_parallel_only(mesh.comm());
1412 :
1413 0 : dof_id_type min_dof_id = std::numeric_limits<dof_id_type>::max(),
1414 0 : max_dof_id = std::numeric_limits<dof_id_type>::min();
1415 :
1416 : // Figure out what our local dof id range is
1417 0 : for (const auto * node : mesh.local_node_ptr_range())
1418 : {
1419 0 : for (auto v : make_range(node->n_vars(sysnum)))
1420 0 : for (auto c : make_range(node->n_comp(sysnum, v)))
1421 : {
1422 0 : dof_id_type id = node->dof_number(sysnum, v, c);
1423 0 : min_dof_id = std::min (min_dof_id, id);
1424 0 : max_dof_id = std::max (max_dof_id, id);
1425 : }
1426 : }
1427 :
1428 : // Make sure no other processors' ids are inside it
1429 0 : for (const auto * node : mesh.node_ptr_range())
1430 : {
1431 0 : if (node->processor_id() == mesh.processor_id())
1432 0 : continue;
1433 0 : for (auto v : make_range(node->n_vars(sysnum)))
1434 0 : for (auto c : make_range(node->n_comp(sysnum, v)))
1435 : {
1436 0 : dof_id_type id = node->dof_number(sysnum, v, c);
1437 0 : libmesh_assert (id < min_dof_id ||
1438 : id > max_dof_id);
1439 : }
1440 : }
1441 : }
1442 :
1443 :
1444 :
1445 : template <>
1446 17694 : void libmesh_assert_topology_consistent_procids<Elem>(const MeshBase & mesh)
1447 : {
1448 35388 : LOG_SCOPE("libmesh_assert_topology_consistent_procids()", "MeshTools");
1449 :
1450 : // This parameter is not used when !LIBMESH_ENABLE_AMR
1451 17694 : libmesh_ignore(mesh);
1452 :
1453 : // If we're adaptively refining, check processor ids for consistency
1454 : // between parents and children.
1455 : #ifdef LIBMESH_ENABLE_AMR
1456 :
1457 : // Ancestor elements we won't worry about, but subactive and active
1458 : // elements ought to have parents with consistent processor ids
1459 2642334 : for (const auto & elem : mesh.element_ptr_range())
1460 : {
1461 2624640 : libmesh_assert(elem);
1462 :
1463 2624640 : if (!elem->active() && !elem->subactive())
1464 288060 : continue;
1465 :
1466 2336580 : const Elem * parent = elem->parent();
1467 :
1468 2336580 : if (parent)
1469 : {
1470 1157608 : libmesh_assert(parent->has_children());
1471 1157608 : processor_id_type parent_procid = parent->processor_id();
1472 1157608 : bool matching_child_id = false;
1473 : // If we've got a remote_elem then we don't know whether
1474 : // it's responsible for the parent's processor id; all
1475 : // we can do is assume it is and let its processor fail
1476 : // an assert if there's something wrong.
1477 7222200 : for (auto & child : parent->child_ref_range())
1478 12129184 : if (&child == remote_elem ||
1479 6064592 : child.processor_id() == parent_procid)
1480 5909956 : matching_child_id = true;
1481 1157608 : libmesh_assert(matching_child_id);
1482 : }
1483 : }
1484 : #endif
1485 17694 : }
1486 :
1487 :
1488 :
1489 : template <>
1490 26966 : void libmesh_assert_topology_consistent_procids<Node>(const MeshBase & mesh)
1491 : {
1492 26966 : LOG_SCOPE("libmesh_assert_topology_consistent_procids()", "MeshTools");
1493 :
1494 26966 : if (mesh.n_processors() == 1)
1495 0 : return;
1496 :
1497 26966 : libmesh_parallel_only(mesh.comm());
1498 :
1499 : // We want this test to be valid even when called after nodes have
1500 : // been added asynchronously but before they're renumbered.
1501 : //
1502 : // Plus, some code (looking at you, stitch_meshes) modifies
1503 : // DofObject ids without keeping max_elem_id()/max_node_id()
1504 : // consistent, but that's done in a safe way for performance
1505 : // reasons, so we'll play along and just figure out new max ids
1506 : // ourselves.
1507 26966 : dof_id_type parallel_max_node_id = 0;
1508 7279169 : for (const auto & node : mesh.node_ptr_range())
1509 7252203 : parallel_max_node_id = std::max<dof_id_type>(parallel_max_node_id,
1510 7252203 : node->id()+1);
1511 26966 : mesh.comm().max(parallel_max_node_id);
1512 :
1513 :
1514 53932 : std::vector<bool> node_touched_by_me(parallel_max_node_id, false);
1515 :
1516 2223802 : for (const auto & elem : as_range(mesh.local_elements_begin(),
1517 4447604 : mesh.local_elements_end()))
1518 : {
1519 2196836 : libmesh_assert (elem);
1520 :
1521 16558620 : for (auto & node : elem->node_ref_range())
1522 : {
1523 14361784 : dof_id_type nodeid = node.id();
1524 14361784 : node_touched_by_me[nodeid] = true;
1525 : }
1526 : }
1527 53932 : std::vector<bool> node_touched_by_anyone(node_touched_by_me);
1528 26966 : mesh.comm().max(node_touched_by_anyone);
1529 :
1530 3640996 : for (const auto & node : mesh.local_node_ptr_range())
1531 : {
1532 3614030 : libmesh_assert(node);
1533 3614030 : dof_id_type nodeid = node->id();
1534 3614030 : libmesh_assert(!node_touched_by_anyone[nodeid] ||
1535 : node_touched_by_me[nodeid]);
1536 : }
1537 : }
1538 :
1539 :
1540 :
1541 0 : void libmesh_assert_canonical_node_procids (const MeshBase & mesh)
1542 : {
1543 0 : for (const auto & elem : mesh.active_element_ptr_range())
1544 0 : for (auto & node : elem->node_ref_range())
1545 0 : libmesh_assert_equal_to
1546 : (node.processor_id(),
1547 : node.choose_processor_id(node.processor_id(),
1548 : elem->processor_id()));
1549 0 : }
1550 :
1551 :
1552 :
1553 : #ifdef LIBMESH_ENABLE_AMR
1554 1226 : void libmesh_assert_valid_refinement_tree(const MeshBase & mesh)
1555 : {
1556 2452 : LOG_SCOPE("libmesh_assert_valid_refinement_tree()", "MeshTools");
1557 :
1558 59722 : for (const auto & elem : mesh.element_ptr_range())
1559 : {
1560 58496 : libmesh_assert(elem);
1561 58496 : if (elem->has_children())
1562 17748 : for (auto & child : elem->child_ref_range())
1563 14928 : if (&child != remote_elem)
1564 14000 : libmesh_assert_equal_to (child.parent(), elem);
1565 58496 : if (elem->active())
1566 : {
1567 55676 : libmesh_assert(!elem->ancestor());
1568 55676 : libmesh_assert(!elem->subactive());
1569 : }
1570 2820 : else if (elem->ancestor())
1571 : {
1572 2820 : libmesh_assert(!elem->subactive());
1573 : }
1574 : else
1575 0 : libmesh_assert(elem->subactive());
1576 :
1577 58496 : if (elem->p_refinement_flag() == Elem::JUST_REFINED)
1578 0 : libmesh_assert_greater(elem->p_level(), 0);
1579 : }
1580 1226 : }
1581 : #else
1582 : void libmesh_assert_valid_refinement_tree(const MeshBase &)
1583 : {
1584 : }
1585 : #endif // LIBMESH_ENABLE_AMR
1586 :
1587 : #endif // !NDEBUG
1588 :
1589 :
1590 :
1591 : #ifdef DEBUG
1592 :
1593 0 : void libmesh_assert_no_links_to_elem(const MeshBase & mesh,
1594 : const Elem * bad_elem)
1595 : {
1596 0 : for (const auto & elem : mesh.element_ptr_range())
1597 : {
1598 0 : libmesh_assert (elem);
1599 0 : libmesh_assert_not_equal_to (elem->parent(), bad_elem);
1600 0 : for (auto n : elem->neighbor_ptr_range())
1601 0 : libmesh_assert_not_equal_to (n, bad_elem);
1602 :
1603 : #ifdef LIBMESH_ENABLE_AMR
1604 0 : if (elem->has_children())
1605 0 : for (auto & child : elem->child_ref_range())
1606 0 : libmesh_assert_not_equal_to (&child, bad_elem);
1607 : #endif
1608 : }
1609 0 : }
1610 :
1611 :
1612 566 : void libmesh_assert_equal_points (const MeshBase & mesh)
1613 : {
1614 1132 : LOG_SCOPE("libmesh_assert_equal_points()", "MeshTools");
1615 :
1616 566 : dof_id_type pmax_node_id = mesh.max_node_id();
1617 566 : mesh.comm().max(pmax_node_id);
1618 :
1619 915766 : for (dof_id_type i=0; i != pmax_node_id; ++i)
1620 : {
1621 915200 : const Point * p = mesh.query_node_ptr(i);
1622 :
1623 915200 : libmesh_assert(mesh.comm().semiverify(p));
1624 : }
1625 566 : }
1626 :
1627 :
1628 566 : void libmesh_assert_equal_connectivity (const MeshBase & mesh)
1629 : {
1630 1132 : LOG_SCOPE("libmesh_assert_equal_connectivity()", "MeshTools");
1631 :
1632 566 : dof_id_type pmax_elem_id = mesh.max_elem_id();
1633 566 : mesh.comm().max(pmax_elem_id);
1634 :
1635 1142058 : for (dof_id_type i=0; i != pmax_elem_id; ++i)
1636 : {
1637 1141492 : const Elem * e = mesh.query_elem_ptr(i);
1638 :
1639 2282984 : std::vector<dof_id_type> nodes;
1640 1141492 : if (e)
1641 5771288 : for (auto n : e->node_index_range())
1642 4629912 : nodes.push_back(e->node_id(n));
1643 :
1644 1141492 : libmesh_assert(mesh.comm().semiverify(e ? &nodes : nullptr));
1645 : }
1646 566 : }
1647 :
1648 :
1649 0 : void libmesh_assert_connected_nodes (const MeshBase & mesh)
1650 : {
1651 0 : LOG_SCOPE("libmesh_assert_connected_nodes()", "MeshTools");
1652 :
1653 0 : std::set<const Node *> used_nodes;
1654 :
1655 0 : for (const auto & elem : mesh.element_ptr_range())
1656 : {
1657 0 : libmesh_assert (elem);
1658 :
1659 0 : for (auto & n : elem->node_ref_range())
1660 0 : used_nodes.insert(&n);
1661 : }
1662 :
1663 0 : for (const auto & node : mesh.node_ptr_range())
1664 : {
1665 0 : libmesh_assert(node);
1666 0 : libmesh_assert(used_nodes.count(node));
1667 : }
1668 0 : }
1669 :
1670 :
1671 :
1672 13244 : void libmesh_assert_valid_constraint_rows (const MeshBase & mesh)
1673 : {
1674 13244 : libmesh_parallel_only(mesh.comm());
1675 :
1676 13244 : const auto & constraint_rows = mesh.get_constraint_rows();
1677 :
1678 13244 : bool have_constraint_rows = !constraint_rows.empty();
1679 13244 : mesh.comm().max(have_constraint_rows);
1680 13244 : if (!have_constraint_rows)
1681 13204 : return;
1682 :
1683 12700 : for (auto & row : constraint_rows)
1684 : {
1685 12660 : const Node * node = row.first;
1686 12660 : libmesh_assert(node == mesh.node_ptr(node->id()));
1687 :
1688 44126 : for (auto & pr : row.second)
1689 : {
1690 31466 : const Elem * spline_elem = pr.first.first;
1691 31466 : libmesh_assert(spline_elem == mesh.elem_ptr(spline_elem->id()));
1692 : }
1693 : }
1694 :
1695 40 : dof_id_type pmax_node_id = mesh.max_node_id();
1696 40 : mesh.comm().max(pmax_node_id);
1697 :
1698 16990 : for (dof_id_type i=0; i != pmax_node_id; ++i)
1699 : {
1700 16950 : const Node * node = mesh.query_node_ptr(i);
1701 :
1702 16950 : bool have_constraint = constraint_rows.count(node);
1703 :
1704 16950 : const std::size_t my_n_constraints = have_constraint ?
1705 12660 : libmesh_map_find(constraint_rows, node).size() : std::size_t(-1);
1706 16950 : const std::size_t * n_constraints = node ?
1707 16950 : &my_n_constraints : nullptr;
1708 :
1709 16950 : libmesh_assert(mesh.comm().semiverify(n_constraints));
1710 : }
1711 : }
1712 :
1713 :
1714 :
1715 33212 : void libmesh_assert_valid_boundary_ids(const MeshBase & mesh)
1716 : {
1717 33212 : LOG_SCOPE("libmesh_assert_valid_boundary_ids()", "MeshTools");
1718 :
1719 33212 : if (mesh.n_processors() == 1)
1720 688 : return;
1721 :
1722 32524 : libmesh_parallel_only(mesh.comm());
1723 :
1724 32524 : const BoundaryInfo & boundary_info = mesh.get_boundary_info();
1725 :
1726 32524 : dof_id_type pmax_elem_id = mesh.max_elem_id();
1727 32524 : mesh.comm().max(pmax_elem_id);
1728 :
1729 4421554 : for (dof_id_type i=0; i != pmax_elem_id; ++i)
1730 : {
1731 4389030 : const Elem * elem = mesh.query_elem_ptr(i);
1732 4389030 : const unsigned int my_n_nodes = elem ? elem->n_nodes() : 0;
1733 4389030 : const unsigned int my_n_edges = elem ? elem->n_edges() : 0;
1734 4389030 : const unsigned int my_n_sides = elem ? elem->n_sides() : 0;
1735 : unsigned int
1736 4389030 : n_nodes = my_n_nodes,
1737 4389030 : n_edges = my_n_edges,
1738 4389030 : n_sides = my_n_sides;
1739 :
1740 4389030 : mesh.comm().max(n_nodes);
1741 4389030 : mesh.comm().max(n_edges);
1742 4389030 : mesh.comm().max(n_sides);
1743 :
1744 4389030 : if (elem)
1745 : {
1746 4311263 : libmesh_assert_equal_to(my_n_nodes, n_nodes);
1747 4311263 : libmesh_assert_equal_to(my_n_edges, n_edges);
1748 4311263 : libmesh_assert_equal_to(my_n_sides, n_sides);
1749 : }
1750 :
1751 : // Let's test all IDs on the element with one communication
1752 : // rather than n_nodes + n_edges + n_sides communications, to
1753 : // cut down on latency in dbg modes.
1754 8778060 : std::vector<boundary_id_type> all_bcids;
1755 :
1756 33650182 : for (unsigned int n=0; n != n_nodes; ++n)
1757 : {
1758 58522304 : std::vector<boundary_id_type> bcids;
1759 29261152 : if (elem)
1760 : {
1761 29097358 : boundary_info.boundary_ids(elem->node_ptr(n), bcids);
1762 :
1763 : // Ordering of boundary ids shouldn't matter
1764 29097358 : std::sort(bcids.begin(), bcids.end());
1765 : }
1766 : // libmesh_assert(mesh.comm().semiverify (elem ? &bcids : nullptr));
1767 :
1768 29261152 : all_bcids.insert(all_bcids.end(), bcids.begin(),
1769 58522304 : bcids.end());
1770 : // Separator
1771 29261152 : all_bcids.push_back(BoundaryInfo::invalid_id);
1772 : }
1773 :
1774 27801122 : for (unsigned short e=0; e != n_edges; ++e)
1775 : {
1776 46824184 : std::vector<boundary_id_type> bcids;
1777 :
1778 23412092 : if (elem)
1779 : {
1780 23310828 : boundary_info.edge_boundary_ids(elem, e, bcids);
1781 :
1782 : // Ordering of boundary ids shouldn't matter
1783 23310828 : std::sort(bcids.begin(), bcids.end());
1784 : }
1785 :
1786 : // libmesh_assert(mesh.comm().semiverify (elem ? &bcids : nullptr));
1787 :
1788 23412092 : all_bcids.insert(all_bcids.end(), bcids.begin(),
1789 46824184 : bcids.end());
1790 : // Separator
1791 23412092 : all_bcids.push_back(BoundaryInfo::invalid_id);
1792 :
1793 23412092 : if (elem)
1794 : {
1795 23310828 : boundary_info.raw_edge_boundary_ids(elem, e, bcids);
1796 :
1797 : // Ordering of boundary ids shouldn't matter
1798 23310828 : std::sort(bcids.begin(), bcids.end());
1799 :
1800 23310828 : all_bcids.insert(all_bcids.end(), bcids.begin(),
1801 46621656 : bcids.end());
1802 : // Separator
1803 23310828 : all_bcids.push_back(BoundaryInfo::invalid_id);
1804 : }
1805 :
1806 : // libmesh_assert(mesh.comm().semiverify (elem ? &bcids : nullptr));
1807 : }
1808 :
1809 21801870 : for (unsigned short s=0; s != n_sides; ++s)
1810 : {
1811 34825680 : std::vector<boundary_id_type> bcids;
1812 :
1813 17412840 : if (elem)
1814 : {
1815 17338884 : boundary_info.boundary_ids(elem, s, bcids);
1816 :
1817 : // Ordering of boundary ids shouldn't matter
1818 17338884 : std::sort(bcids.begin(), bcids.end());
1819 :
1820 17338884 : all_bcids.insert(all_bcids.end(), bcids.begin(),
1821 34677768 : bcids.end());
1822 : // Separator
1823 17338884 : all_bcids.push_back(BoundaryInfo::invalid_id);
1824 : }
1825 :
1826 : // libmesh_assert(mesh.comm().semiverify (elem ? &bcids : nullptr));
1827 :
1828 17412840 : if (elem)
1829 : {
1830 17338884 : boundary_info.raw_boundary_ids(elem, s, bcids);
1831 :
1832 : // Ordering of boundary ids shouldn't matter
1833 17338884 : std::sort(bcids.begin(), bcids.end());
1834 :
1835 17338884 : all_bcids.insert(all_bcids.end(), bcids.begin(),
1836 34677768 : bcids.end());
1837 : // Separator
1838 17338884 : all_bcids.push_back(BoundaryInfo::invalid_id);
1839 : }
1840 :
1841 : // libmesh_assert(mesh.comm().semiverify (elem ? &bcids : nullptr));
1842 : }
1843 :
1844 13167090 : for (unsigned short sf=0; sf != 2; ++sf)
1845 : {
1846 17556120 : std::vector<boundary_id_type> bcids;
1847 :
1848 8778060 : if (elem)
1849 : {
1850 8622526 : boundary_info.shellface_boundary_ids(elem, sf, bcids);
1851 :
1852 : // Ordering of boundary ids shouldn't matter
1853 8622526 : std::sort(bcids.begin(), bcids.end());
1854 :
1855 8622526 : all_bcids.insert(all_bcids.end(), bcids.begin(),
1856 17245052 : bcids.end());
1857 : // Separator
1858 8622526 : all_bcids.push_back(BoundaryInfo::invalid_id);
1859 : }
1860 :
1861 : // libmesh_assert(mesh.comm().semiverify (elem ? &bcids : nullptr));
1862 :
1863 8778060 : if (elem)
1864 : {
1865 8622526 : boundary_info.raw_shellface_boundary_ids(elem, sf, bcids);
1866 :
1867 : // Ordering of boundary ids shouldn't matter
1868 8622526 : std::sort(bcids.begin(), bcids.end());
1869 :
1870 8622526 : all_bcids.insert(all_bcids.end(), bcids.begin(),
1871 17245052 : bcids.end());
1872 : // Separator
1873 8622526 : all_bcids.push_back(BoundaryInfo::invalid_id);
1874 : }
1875 :
1876 : // libmesh_assert(mesh.comm().semiverify (elem ? &bcids : nullptr));
1877 : }
1878 :
1879 4389030 : libmesh_assert(mesh.comm().semiverify
1880 : (elem ? &all_bcids : nullptr));
1881 : }
1882 : }
1883 :
1884 :
1885 7652 : void libmesh_assert_valid_dof_ids(const MeshBase & mesh, unsigned int sysnum)
1886 : {
1887 7652 : LOG_SCOPE("libmesh_assert_valid_dof_ids()", "MeshTools");
1888 :
1889 7652 : if (mesh.n_processors() == 1)
1890 0 : return;
1891 :
1892 7652 : libmesh_parallel_only(mesh.comm());
1893 :
1894 7652 : dof_id_type pmax_elem_id = mesh.max_elem_id();
1895 7652 : mesh.comm().max(pmax_elem_id);
1896 :
1897 973224 : for (dof_id_type i=0; i != pmax_elem_id; ++i)
1898 965572 : assert_semiverify_dofobj(mesh.comm(),
1899 965572 : mesh.query_elem_ptr(i),
1900 : sysnum);
1901 :
1902 7652 : dof_id_type pmax_node_id = mesh.max_node_id();
1903 7652 : mesh.comm().max(pmax_node_id);
1904 :
1905 1828510 : for (dof_id_type i=0; i != pmax_node_id; ++i)
1906 1820858 : assert_semiverify_dofobj(mesh.comm(),
1907 1820858 : mesh.query_node_ptr(i),
1908 : sysnum);
1909 : }
1910 :
1911 :
1912 : #ifdef LIBMESH_ENABLE_UNIQUE_ID
1913 25512 : void libmesh_assert_valid_unique_ids(const MeshBase & mesh)
1914 : {
1915 51024 : LOG_SCOPE("libmesh_assert_valid_unique_ids()", "MeshTools");
1916 :
1917 25512 : libmesh_parallel_only(mesh.comm());
1918 :
1919 : // First collect all the unique_ids we can see and make sure there's
1920 : // no duplicates
1921 51024 : std::unordered_set<unique_id_type> semilocal_unique_ids;
1922 :
1923 2914753 : for (auto const & elem : mesh.active_element_ptr_range())
1924 : {
1925 2889241 : libmesh_assert (!semilocal_unique_ids.count(elem->unique_id()));
1926 2889241 : semilocal_unique_ids.insert(elem->unique_id());
1927 : }
1928 :
1929 5484895 : for (auto const & node : mesh.node_ptr_range())
1930 : {
1931 5459383 : libmesh_assert (!semilocal_unique_ids.count(node->unique_id()));
1932 5459383 : semilocal_unique_ids.insert(node->unique_id());
1933 : }
1934 :
1935 : // Then make sure elements are all in sync and remote elements don't
1936 : // duplicate semilocal
1937 :
1938 25512 : dof_id_type pmax_elem_id = mesh.max_elem_id();
1939 25512 : mesh.comm().max(pmax_elem_id);
1940 :
1941 3457324 : for (dof_id_type i=0; i != pmax_elem_id; ++i)
1942 : {
1943 3431812 : const Elem * elem = mesh.query_elem_ptr(i);
1944 3431812 : assert_dofobj_unique_id(mesh.comm(), elem, semilocal_unique_ids);
1945 : }
1946 :
1947 : // Then make sure nodes are all in sync and remote elements don't
1948 : // duplicate semilocal
1949 :
1950 25512 : dof_id_type pmax_node_id = mesh.max_node_id();
1951 25512 : mesh.comm().max(pmax_node_id);
1952 :
1953 5791154 : for (dof_id_type i=0; i != pmax_node_id; ++i)
1954 : {
1955 5765642 : const Node * node = mesh.query_node_ptr(i);
1956 5765642 : assert_dofobj_unique_id(mesh.comm(), node, semilocal_unique_ids);
1957 : }
1958 25512 : }
1959 : #endif
1960 :
1961 0 : void libmesh_assert_consistent_distributed(const MeshBase & mesh)
1962 : {
1963 0 : libmesh_parallel_only(mesh.comm());
1964 :
1965 0 : dof_id_type parallel_max_elem_id = mesh.max_elem_id();
1966 0 : mesh.comm().max(parallel_max_elem_id);
1967 :
1968 0 : for (dof_id_type i=0; i != parallel_max_elem_id; ++i)
1969 : {
1970 0 : const Elem * elem = mesh.query_elem_ptr(i);
1971 : processor_id_type pid =
1972 0 : elem ? elem->processor_id() : DofObject::invalid_processor_id;
1973 0 : mesh.comm().min(pid);
1974 0 : libmesh_assert(elem || pid != mesh.processor_id());
1975 : }
1976 :
1977 0 : dof_id_type parallel_max_node_id = mesh.max_node_id();
1978 0 : mesh.comm().max(parallel_max_node_id);
1979 :
1980 0 : for (dof_id_type i=0; i != parallel_max_node_id; ++i)
1981 : {
1982 0 : const Node * node = mesh.query_node_ptr(i);
1983 : processor_id_type pid =
1984 0 : node ? node->processor_id() : DofObject::invalid_processor_id;
1985 0 : mesh.comm().min(pid);
1986 0 : libmesh_assert(node || pid != mesh.processor_id());
1987 : }
1988 0 : }
1989 :
1990 :
1991 0 : void libmesh_assert_consistent_distributed_nodes(const MeshBase & mesh)
1992 : {
1993 0 : libmesh_parallel_only(mesh.comm());
1994 0 : auto locator = mesh.sub_point_locator();
1995 :
1996 0 : dof_id_type parallel_max_elem_id = mesh.max_elem_id();
1997 0 : mesh.comm().max(parallel_max_elem_id);
1998 :
1999 0 : for (dof_id_type i=0; i != parallel_max_elem_id; ++i)
2000 : {
2001 0 : const Elem * elem = mesh.query_elem_ptr(i);
2002 :
2003 0 : const unsigned int my_n_nodes = elem ? elem->n_nodes() : 0;
2004 0 : unsigned int n_nodes = my_n_nodes;
2005 0 : mesh.comm().max(n_nodes);
2006 :
2007 0 : if (n_nodes)
2008 0 : libmesh_assert(mesh.comm().semiverify(elem ? &my_n_nodes : nullptr));
2009 :
2010 0 : for (unsigned int n=0; n != n_nodes; ++n)
2011 : {
2012 0 : const Node * node = elem ? elem->node_ptr(n) : nullptr;
2013 : processor_id_type pid =
2014 0 : node ? node->processor_id() : DofObject::invalid_processor_id;
2015 0 : mesh.comm().min(pid);
2016 0 : libmesh_assert(node || pid != mesh.processor_id());
2017 : }
2018 : }
2019 0 : }
2020 :
2021 :
2022 :
2023 : template <>
2024 17694 : void libmesh_assert_parallel_consistent_procids<Elem>(const MeshBase & mesh)
2025 : {
2026 17694 : LOG_SCOPE("libmesh_assert_parallel_consistent_procids()", "MeshTools");
2027 :
2028 17694 : if (mesh.n_processors() == 1)
2029 0 : return;
2030 :
2031 17694 : libmesh_parallel_only(mesh.comm());
2032 :
2033 : // Some code (looking at you, stitch_meshes) modifies DofObject ids
2034 : // without keeping max_elem_id()/max_node_id() consistent, but
2035 : // that's done in a safe way for performance reasons, so we'll play
2036 : // along and just figure out new max ids ourselves.
2037 17694 : dof_id_type parallel_max_elem_id = 0;
2038 2642334 : for (const auto & elem : mesh.element_ptr_range())
2039 2624640 : parallel_max_elem_id = std::max<dof_id_type>(parallel_max_elem_id,
2040 2624640 : elem->id()+1);
2041 17694 : mesh.comm().max(parallel_max_elem_id);
2042 :
2043 : // Check processor ids for consistency between processors
2044 :
2045 2711048 : for (dof_id_type i=0; i != parallel_max_elem_id; ++i)
2046 : {
2047 2693354 : const Elem * elem = mesh.query_elem_ptr(i);
2048 :
2049 : processor_id_type min_id =
2050 2624640 : elem ? elem->processor_id() :
2051 2693354 : std::numeric_limits<processor_id_type>::max();
2052 2693354 : mesh.comm().min(min_id);
2053 :
2054 : processor_id_type max_id =
2055 2624640 : elem ? elem->processor_id() :
2056 2693354 : std::numeric_limits<processor_id_type>::min();
2057 2693354 : mesh.comm().max(max_id);
2058 :
2059 2693354 : if (elem)
2060 : {
2061 2624640 : libmesh_assert_equal_to (min_id, elem->processor_id());
2062 2624640 : libmesh_assert_equal_to (max_id, elem->processor_id());
2063 : }
2064 :
2065 2693354 : if (min_id == mesh.processor_id())
2066 1272943 : libmesh_assert(elem);
2067 : }
2068 : }
2069 :
2070 :
2071 :
2072 76 : void libmesh_assert_parallel_consistent_new_node_procids(const MeshBase & mesh)
2073 : {
2074 76 : LOG_SCOPE("libmesh_assert_parallel_consistent_new_node_procids()", "MeshTools");
2075 :
2076 76 : if (mesh.n_processors() == 1)
2077 0 : return;
2078 :
2079 76 : libmesh_parallel_only(mesh.comm());
2080 :
2081 : // We want this test to hit every node when called even after nodes
2082 : // have been added asynchronously but before everything has been
2083 : // renumbered.
2084 76 : dof_id_type parallel_max_elem_id = mesh.max_elem_id();
2085 76 : mesh.comm().max(parallel_max_elem_id);
2086 :
2087 152 : std::vector<bool> elem_touched_by_anyone(parallel_max_elem_id, false);
2088 :
2089 43248 : for (dof_id_type i=0; i != parallel_max_elem_id; ++i)
2090 : {
2091 43172 : const Elem * elem = mesh.query_elem_ptr(i);
2092 :
2093 43172 : const unsigned int my_n_nodes = elem ? elem->n_nodes() : 0;
2094 43172 : unsigned int n_nodes = my_n_nodes;
2095 43172 : mesh.comm().max(n_nodes);
2096 :
2097 43172 : if (n_nodes)
2098 15840 : libmesh_assert(mesh.comm().semiverify(elem ? &my_n_nodes : nullptr));
2099 :
2100 173404 : for (unsigned int n=0; n != n_nodes; ++n)
2101 : {
2102 130232 : const Node * node = elem ? elem->node_ptr(n) : nullptr;
2103 130232 : const processor_id_type pid = node ? node->processor_id() : 0;
2104 130232 : libmesh_assert(mesh.comm().semiverify (node ? &pid : nullptr));
2105 : }
2106 : }
2107 : }
2108 :
2109 : template <>
2110 27188 : void libmesh_assert_parallel_consistent_procids<Node>(const MeshBase & mesh)
2111 : {
2112 27188 : LOG_SCOPE("libmesh_assert_parallel_consistent_procids()", "MeshTools");
2113 :
2114 27188 : if (mesh.n_processors() == 1)
2115 0 : return;
2116 :
2117 27188 : libmesh_parallel_only(mesh.comm());
2118 :
2119 : // We want this test to be valid even when called even after nodes
2120 : // have been added asynchronously but before they're renumbered
2121 : //
2122 : // Plus, some code (looking at you, stitch_meshes) modifies
2123 : // DofObject ids without keeping max_elem_id()/max_node_id()
2124 : // consistent, but that's done in a safe way for performance
2125 : // reasons, so we'll play along and just figure out new max ids
2126 : // ourselves.
2127 27188 : dof_id_type parallel_max_node_id = 0;
2128 7529771 : for (const auto & node : mesh.node_ptr_range())
2129 7502583 : parallel_max_node_id = std::max<dof_id_type>(parallel_max_node_id,
2130 7502583 : node->id()+1);
2131 27188 : mesh.comm().max(parallel_max_node_id);
2132 :
2133 54376 : std::vector<bool> node_touched_by_anyone(parallel_max_node_id, false);
2134 :
2135 2269975 : for (const auto & elem : as_range(mesh.local_elements_begin(),
2136 4539950 : mesh.local_elements_end()))
2137 : {
2138 2242787 : libmesh_assert (elem);
2139 :
2140 17008104 : for (auto & node : elem->node_ref_range())
2141 : {
2142 14765317 : dof_id_type nodeid = node.id();
2143 14765317 : node_touched_by_anyone[nodeid] = true;
2144 : }
2145 : }
2146 27188 : mesh.comm().max(node_touched_by_anyone);
2147 :
2148 : // Check processor ids for consistency between processors
2149 : // on any node an element touches
2150 7949022 : for (dof_id_type i=0; i != parallel_max_node_id; ++i)
2151 : {
2152 7921834 : if (!node_touched_by_anyone[i])
2153 421536 : continue;
2154 :
2155 7500298 : const Node * node = mesh.query_node_ptr(i);
2156 7500298 : const processor_id_type pid = node ? node->processor_id() : 0;
2157 :
2158 7500298 : libmesh_assert(mesh.comm().semiverify (node ? &pid : nullptr));
2159 : }
2160 : }
2161 :
2162 :
2163 :
2164 : #ifdef LIBMESH_ENABLE_AMR
2165 0 : void libmesh_assert_valid_refinement_flags(const MeshBase & mesh)
2166 : {
2167 0 : LOG_SCOPE("libmesh_assert_valid_refinement_flags()", "MeshTools");
2168 :
2169 0 : libmesh_parallel_only(mesh.comm());
2170 0 : if (mesh.n_processors() == 1)
2171 0 : return;
2172 :
2173 0 : dof_id_type pmax_elem_id = mesh.max_elem_id();
2174 0 : mesh.comm().max(pmax_elem_id);
2175 :
2176 0 : std::vector<unsigned char> my_elem_h_state(pmax_elem_id, 255);
2177 0 : std::vector<unsigned char> my_elem_p_state(pmax_elem_id, 255);
2178 :
2179 0 : for (const auto & elem : mesh.element_ptr_range())
2180 : {
2181 0 : libmesh_assert (elem);
2182 0 : dof_id_type elemid = elem->id();
2183 :
2184 0 : my_elem_h_state[elemid] =
2185 0 : static_cast<unsigned char>(elem->refinement_flag());
2186 :
2187 0 : my_elem_p_state[elemid] =
2188 0 : static_cast<unsigned char>(elem->p_refinement_flag());
2189 : }
2190 0 : std::vector<unsigned char> min_elem_h_state(my_elem_h_state);
2191 0 : mesh.comm().min(min_elem_h_state);
2192 :
2193 0 : std::vector<unsigned char> min_elem_p_state(my_elem_p_state);
2194 0 : mesh.comm().min(min_elem_p_state);
2195 :
2196 0 : for (dof_id_type i=0; i!= pmax_elem_id; ++i)
2197 : {
2198 0 : libmesh_assert(my_elem_h_state[i] == 255 ||
2199 : my_elem_h_state[i] == min_elem_h_state[i]);
2200 0 : libmesh_assert(my_elem_p_state[i] == 255 ||
2201 : my_elem_p_state[i] == min_elem_p_state[i]);
2202 : }
2203 : }
2204 : #else
2205 : void libmesh_assert_valid_refinement_flags(const MeshBase &)
2206 : {
2207 : }
2208 : #endif // LIBMESH_ENABLE_AMR
2209 :
2210 :
2211 :
2212 15162 : void libmesh_assert_valid_neighbors(const MeshBase & mesh,
2213 : bool assert_valid_remote_elems)
2214 : {
2215 15162 : LOG_SCOPE("libmesh_assert_valid_neighbors()", "MeshTools");
2216 :
2217 2677812 : for (const auto & elem : mesh.element_ptr_range())
2218 : {
2219 2662650 : libmesh_assert (elem);
2220 2662650 : elem->libmesh_assert_valid_neighbors();
2221 : }
2222 :
2223 15162 : if (mesh.n_processors() == 1)
2224 574 : return;
2225 :
2226 14588 : libmesh_parallel_only(mesh.comm());
2227 :
2228 14588 : dof_id_type pmax_elem_id = mesh.max_elem_id();
2229 14588 : mesh.comm().max(pmax_elem_id);
2230 :
2231 2779384 : for (dof_id_type i=0; i != pmax_elem_id; ++i)
2232 : {
2233 2764796 : const Elem * elem = mesh.query_elem_ptr(i);
2234 :
2235 2764796 : const unsigned int my_n_neigh = elem ? elem->n_neighbors() : 0;
2236 2764796 : unsigned int n_neigh = my_n_neigh;
2237 2764796 : mesh.comm().max(n_neigh);
2238 2764796 : if (elem)
2239 2660502 : libmesh_assert_equal_to (my_n_neigh, n_neigh);
2240 :
2241 13327706 : for (unsigned int n = 0; n != n_neigh; ++n)
2242 : {
2243 10562910 : dof_id_type my_neighbor = DofObject::invalid_id;
2244 10562910 : dof_id_type * p_my_neighbor = nullptr;
2245 :
2246 : // If we have a non-remote_elem neighbor link, then we can
2247 : // verify it.
2248 10562910 : if (elem && elem->neighbor_ptr(n) != remote_elem)
2249 : {
2250 10485156 : p_my_neighbor = &my_neighbor;
2251 10485156 : if (elem->neighbor_ptr(n))
2252 9923316 : my_neighbor = elem->neighbor_ptr(n)->id();
2253 :
2254 : // But wait - if we haven't set remote_elem links yet then
2255 : // some nullptr links on ghost elements might be
2256 : // future-remote_elem links, so we can't verify those.
2257 20977216 : if (!assert_valid_remote_elems &&
2258 10486770 : !elem->neighbor_ptr(n) &&
2259 1614 : elem->processor_id() != mesh.processor_id())
2260 753 : p_my_neighbor = nullptr;
2261 : }
2262 10562910 : libmesh_assert(mesh.comm().semiverify(p_my_neighbor));
2263 : }
2264 : }
2265 : }
2266 : #endif // DEBUG
2267 :
2268 :
2269 :
2270 : // Functors for correct_node_proc_ids
2271 : namespace {
2272 :
2273 : typedef std::unordered_map<dof_id_type, processor_id_type> proc_id_map_type;
2274 :
2275 : struct SyncNodeSet
2276 : {
2277 : typedef unsigned char datum; // bool but without bit twiddling issues
2278 :
2279 158 : SyncNodeSet(std::unordered_set<const Node *> & _set,
2280 8347 : MeshBase & _mesh) :
2281 8347 : node_set(_set), mesh(_mesh) {}
2282 :
2283 : std::unordered_set<const Node *> & node_set;
2284 :
2285 : MeshBase & mesh;
2286 :
2287 : // ------------------------------------------------------------
2288 37825 : void gather_data (const std::vector<dof_id_type> & ids,
2289 : std::vector<datum> & data)
2290 : {
2291 : // Find whether each requested node belongs in the set
2292 37825 : data.resize(ids.size());
2293 :
2294 1830683 : for (auto i : index_range(ids))
2295 : {
2296 1792858 : const dof_id_type id = ids[i];
2297 :
2298 : // We'd better have every node we're asked for
2299 1792858 : Node * node = mesh.node_ptr(id);
2300 :
2301 : // Return if the node is in the set.
2302 2888844 : data[i] = node_set.count(node);
2303 : }
2304 37825 : }
2305 :
2306 : // ------------------------------------------------------------
2307 37825 : bool act_on_data (const std::vector<dof_id_type> & ids,
2308 : const std::vector<datum> in_set)
2309 : {
2310 143 : bool data_changed = false;
2311 :
2312 : // Add nodes we've been informed of to our own set
2313 1830683 : for (auto i : index_range(ids))
2314 : {
2315 1792858 : if (in_set[i])
2316 : {
2317 1162999 : Node * node = mesh.node_ptr(ids[i]);
2318 1162999 : if (!node_set.count(node))
2319 : {
2320 256 : node_set.insert(node);
2321 256 : data_changed = true;
2322 : }
2323 : }
2324 : }
2325 :
2326 37825 : return data_changed;
2327 : }
2328 : };
2329 :
2330 :
2331 8031 : struct NodesNotInSet
2332 : {
2333 158 : NodesNotInSet(const std::unordered_set<const Node *> _set)
2334 8189 : : node_set(_set) {}
2335 :
2336 473220 : bool operator() (const Node * node) const
2337 : {
2338 946440 : if (node_set.count(node))
2339 280716 : return false;
2340 192504 : return true;
2341 : }
2342 :
2343 : const std::unordered_set<const Node *> node_set;
2344 : };
2345 :
2346 :
2347 : struct SyncProcIdsFromMap
2348 : {
2349 : typedef processor_id_type datum;
2350 :
2351 194 : SyncProcIdsFromMap(const proc_id_map_type & _map,
2352 34711 : MeshBase & _mesh) :
2353 34711 : new_proc_ids(_map), mesh(_mesh) {}
2354 :
2355 : const proc_id_map_type & new_proc_ids;
2356 :
2357 : MeshBase & mesh;
2358 :
2359 : // ------------------------------------------------------------
2360 121429 : void gather_data (const std::vector<dof_id_type> & ids,
2361 : std::vector<datum> & data)
2362 : {
2363 : // Find the new processor id of each requested node
2364 121429 : data.resize(ids.size());
2365 :
2366 6983853 : for (auto i : index_range(ids))
2367 : {
2368 6862424 : const dof_id_type id = ids[i];
2369 :
2370 : // Return the node's new processor id if it has one, or its
2371 : // old processor id if not.
2372 6862424 : if (const auto it = new_proc_ids.find(id);
2373 47012 : it != new_proc_ids.end())
2374 6231344 : data[i] = it->second;
2375 : else
2376 : {
2377 : // We'd better find every node we're asked for
2378 631080 : const Node & node = mesh.node_ref(id);
2379 631080 : data[i] = node.processor_id();
2380 : }
2381 : }
2382 121429 : }
2383 :
2384 : // ------------------------------------------------------------
2385 121429 : void act_on_data (const std::vector<dof_id_type> & ids,
2386 : const std::vector<datum> proc_ids)
2387 : {
2388 : // Set the node processor ids we've now been informed of
2389 6983853 : for (auto i : index_range(ids))
2390 : {
2391 6862424 : Node & node = mesh.node_ref(ids[i]);
2392 6862424 : node.processor_id() = proc_ids[i];
2393 : }
2394 121429 : }
2395 : };
2396 : }
2397 :
2398 :
2399 :
2400 34711 : void correct_node_proc_ids (MeshBase & mesh)
2401 : {
2402 388 : LOG_SCOPE("correct_node_proc_ids()","MeshTools");
2403 :
2404 : // This function must be run on all processors at once
2405 194 : libmesh_parallel_only(mesh.comm());
2406 :
2407 : // We require all processors to agree on nodal processor ids before
2408 : // going into this algorithm.
2409 : #ifdef DEBUG
2410 194 : libmesh_assert_parallel_consistent_procids<Node>(mesh);
2411 : #endif
2412 :
2413 : // If we have any unpartitioned elements at this
2414 : // stage there is a problem
2415 194 : libmesh_assert (n_elem(mesh.unpartitioned_elements_begin(),
2416 : mesh.unpartitioned_elements_end()) == 0);
2417 :
2418 : // Fix nodes' processor ids. Coarsening may have left us with nodes
2419 : // which are no longer touched by any elements of the same processor
2420 : // id, and for DofMap to work we need to fix that.
2421 :
2422 : // This is harder now that libMesh no longer requires a distributed
2423 : // mesh to ghost all nodal neighbors: it is possible for two active
2424 : // elements on two different processors to share the same node in
2425 : // such a way that neither processor knows the others' element
2426 : // exists!
2427 :
2428 : // While we're at it, if this mesh is configured to allow
2429 : // repartitioning, we'll repartition *all* the nodes' processor ids
2430 : // using the canonical Node heuristic, to try and improve DoF load
2431 : // balancing. But if the mesh is disallowing repartitioning, we
2432 : // won't touch processor_id on any node where it's valid, regardless
2433 : // of whether or not it's canonical.
2434 194 : bool repartition_all_nodes = !mesh.skip_noncritical_partitioning();
2435 388 : std::unordered_set<const Node *> valid_nodes;
2436 :
2437 : // If we aren't allowed to repartition, then we're going to leave
2438 : // every node we can at its current processor_id, and *only*
2439 : // repartition the nodes whose current processor id is incompatible
2440 : // with DoFMap (because it doesn't touch an active element, e.g. due
2441 : // to coarsening)
2442 34711 : if (!repartition_all_nodes)
2443 : {
2444 1127692 : for (const auto & elem : mesh.active_element_ptr_range())
2445 5812313 : for (const auto & node : elem->node_ref_range())
2446 5188291 : if (elem->processor_id() == node.processor_id())
2447 4818885 : valid_nodes.insert(&node);
2448 :
2449 158 : SyncNodeSet syncv(valid_nodes, mesh);
2450 :
2451 : Parallel::sync_dofobject_data_by_id
2452 16536 : (mesh.comm(), mesh.nodes_begin(), mesh.nodes_end(), syncv);
2453 : }
2454 :
2455 : // We build up a set of compatible processor ids for each node
2456 388 : proc_id_map_type new_proc_ids;
2457 :
2458 8783466 : for (auto & elem : mesh.active_element_ptr_range())
2459 : {
2460 4395105 : processor_id_type pid = elem->processor_id();
2461 :
2462 42755747 : for (auto & node : elem->node_ref_range())
2463 : {
2464 38360642 : const dof_id_type id = node.id();
2465 38360642 : if (auto it = new_proc_ids.find(id);
2466 342932 : it == new_proc_ids.end())
2467 152592 : new_proc_ids.emplace(id, pid);
2468 : else
2469 27494590 : it->second = node.choose_processor_id(it->second, pid);
2470 : }
2471 34323 : }
2472 :
2473 : // Sort the new pids to push to each processor
2474 : std::map<processor_id_type, std::vector<std::pair<dof_id_type, processor_id_type>>>
2475 388 : ids_to_push;
2476 :
2477 24904448 : for (const auto & node : mesh.node_ptr_range())
2478 12666690 : if (const auto it = std::as_const(new_proc_ids).find(node->id());
2479 12666690 : it != new_proc_ids.end() && node->processor_id() != DofObject::invalid_processor_id)
2480 10900375 : ids_to_push[node->processor_id()].emplace_back(node->id(), /*pid=*/it->second);
2481 :
2482 : auto action_functor =
2483 156350 : [& mesh, & new_proc_ids]
2484 : (processor_id_type,
2485 11324190 : const std::vector<std::pair<dof_id_type, processor_id_type>> & data)
2486 : {
2487 11023126 : for (const auto & [id, pid] : data)
2488 : {
2489 10866052 : if (const auto it = new_proc_ids.find(id);
2490 152592 : it == new_proc_ids.end())
2491 0 : new_proc_ids.emplace(id, pid);
2492 : else
2493 : {
2494 10866052 : const Node & node = mesh.node_ref(id);
2495 10866052 : it->second = node.choose_processor_id(it->second, pid);
2496 : }
2497 : }
2498 35241 : };
2499 :
2500 : Parallel::push_parallel_vector_data
2501 34711 : (mesh.comm(), ids_to_push, action_functor);
2502 :
2503 : // Now new_proc_ids is correct for every node we used to own. Let's
2504 : // ask every other processor about the nodes they used to own. But
2505 : // first we'll need to keep track of which nodes we used to own,
2506 : // lest we get them confused with nodes we newly own.
2507 388 : std::unordered_set<Node *> ex_local_nodes;
2508 8853602 : for (auto & node : mesh.local_node_ptr_range())
2509 4527242 : if (const auto it = new_proc_ids.find(node->id());
2510 4527242 : it != new_proc_ids.end() && it->second != mesh.processor_id())
2511 34372 : ex_local_nodes.insert(node);
2512 :
2513 194 : SyncProcIdsFromMap sync(new_proc_ids, mesh);
2514 34711 : if (repartition_all_nodes)
2515 : Parallel::sync_dofobject_data_by_id
2516 52692 : (mesh.comm(), mesh.nodes_begin(), mesh.nodes_end(), sync);
2517 : else
2518 : {
2519 158 : NodesNotInSet nnis(valid_nodes);
2520 :
2521 : Parallel::sync_dofobject_data_by_id
2522 16536 : (mesh.comm(), mesh.nodes_begin(), mesh.nodes_end(), nnis, sync);
2523 : }
2524 :
2525 : // And finally let's update the nodes we used to own.
2526 39298 : for (const auto & node : ex_local_nodes)
2527 : {
2528 2103 : if (valid_nodes.count(node))
2529 2083 : continue;
2530 :
2531 2504 : const dof_id_type id = node->id();
2532 10 : const proc_id_map_type::iterator it = new_proc_ids.find(id);
2533 10 : libmesh_assert(it != new_proc_ids.end());
2534 2504 : node->processor_id() = it->second;
2535 : }
2536 :
2537 : // We should still have consistent nodal processor ids coming out of
2538 : // this algorithm, but if we're allowed to repartition the mesh then
2539 : // they should be canonically correct too.
2540 : #ifdef DEBUG
2541 194 : libmesh_assert_valid_procids<Node>(mesh);
2542 : //if (repartition_all_nodes)
2543 : // libmesh_assert_canonical_node_procids(mesh);
2544 : #endif
2545 34711 : }
2546 :
2547 :
2548 :
2549 19490 : void Private::globally_renumber_nodes_and_elements (MeshBase & mesh)
2550 : {
2551 19490 : MeshCommunication().assign_global_indices(mesh);
2552 19490 : }
2553 :
2554 : } // namespace MeshTools
2555 :
2556 : } // namespace libMesh
|