23 #include "libmesh/enum_to_string.h" 24 #include "libmesh/mesh_tools.h" 25 #include "libmesh/parallel_sync.h" 26 #include "libmesh/remote_elem.h" 27 #include "libmesh/periodic_boundary.h" 28 #include "libmesh/periodic_boundaries.h" 40 params.addRangeCheckedParam<
Real>(
"ray_distance",
41 std::numeric_limits<Real>::max(),
43 "The maximum distance all Rays can travel");
45 params.addParam<
bool>(
46 "tolerate_failure",
false,
"Whether or not to tolerate a ray tracing failure");
48 MooseEnum work_buffers(
"lifo circular",
"circular");
49 params.addParam<
MooseEnum>(
"work_buffer_type", work_buffers,
"The work buffer type to use");
51 params.addParam<
bool>(
52 "ray_kernel_coverage_check",
true,
"Whether or not to perform coverage checks on RayKernels");
53 params.addParam<
bool>(
"warn_non_planar",
55 "Whether or not to produce a warning if any element faces are non-planar.");
57 params.addParam<
bool>(
58 "always_cache_traces",
60 "Whether or not to cache the Ray traces on every execution, primarily for use in output. " 61 "Warning: this can get expensive very quick with a large number of rays!");
62 params.addParam<
bool>(
"data_on_cache_traces",
64 "Whether or not to also cache the Ray's data when caching its traces");
65 params.addParam<
bool>(
"aux_data_on_cache_traces",
67 "Whether or not to also cache the Ray's aux data when caching its traces");
68 params.addParam<
bool>(
69 "segments_on_cache_traces",
71 "Whether or not to cache individual segments when trace caching is enabled. If false, we " 72 "will instead cache a segment for each part of the trace where the direction is the same. " 73 "This minimizes the number of segments requied to represent the Ray's path, but removes the " 74 "ability to show Ray field data on each segment through an element.");
76 params.addParam<
bool>(
"use_internal_sidesets",
78 "Whether or not to use internal sidesets for RayBCs in ray tracing");
80 params.addParam<
bool>(
"warn_subdomain_hmax",
82 "Whether or not to warn if the approximated hmax (constant on subdomain) " 83 "varies significantly for an element");
85 params.addParam<
bool>(
88 "Whether or not to verify the generated Rays. This includes checking their " 89 "starting information and the uniqueness of Rays before and after execution. This is also " 90 "used by derived studies for more specific verification.");
91 params.addParam<
bool>(
"verify_trace_intersections",
93 "Whether or not to verify the trace intersections in devel and dbg modes. " 94 "Trace intersections are not verified regardless of this parameter in " 95 "optimized modes (opt, oprof).");
97 params.addParam<
bool>(
"allow_other_flags_with_prekernels",
99 "Whether or not to allow the list of execution flags to have PRE_KERNELS " 100 "mixed with other flags. If this parameter is not set then if PRE_KERNELS " 101 "is provided it must be the only execution flag.");
106 params.addParamNamesToGroup(
107 "always_cache_traces data_on_cache_traces aux_data_on_cache_traces segments_on_cache_traces",
109 params.addParamNamesToGroup(
"warn_non_planar warn_subdomain_hmax",
"Tracing Warnings");
110 params.addParamNamesToGroup(
"ray_kernel_coverage_check verify_rays verify_trace_intersections",
111 "Checks and verifications");
114 params.addPrivateParam<
bool>(
"_use_ray_registration",
true);
116 params.addPrivateParam<
bool>(
"_bank_rays_on_completion",
true);
118 params.addPrivateParam<
bool>(
"_ray_dependent_subdomain_setup",
true);
121 params.addRelationshipManager(
"ElementPointNeighborLayers",
125 { rm_params.
set<
unsigned short>(
"layers") = 1; });
132 _mesh(_fe_problem.
mesh()),
136 _ray_kernel_coverage_check(getParam<bool>(
"ray_kernel_coverage_check")),
137 _warn_non_planar(getParam<bool>(
"warn_non_planar")),
138 _use_ray_registration(getParam<bool>(
"_use_ray_registration")),
139 _use_internal_sidesets(getParam<bool>(
"use_internal_sidesets")),
140 _tolerate_failure(getParam<bool>(
"tolerate_failure")),
141 _bank_rays_on_completion(getParam<bool>(
"_bank_rays_on_completion")),
142 _ray_dependent_subdomain_setup(getParam<bool>(
"_ray_dependent_subdomain_setup")),
144 _always_cache_traces(getParam<bool>(
"always_cache_traces")),
145 _data_on_cache_traces(getParam<bool>(
"data_on_cache_traces")),
146 _aux_data_on_cache_traces(getParam<bool>(
"aux_data_on_cache_traces")),
147 _segments_on_cache_traces(getParam<bool>(
"segments_on_cache_traces")),
148 _ray_max_distance(getParam<
Real>(
"ray_distance")),
149 _verify_rays(getParam<bool>(
"verify_rays")),
151 _verify_trace_intersections(getParam<bool>(
"verify_trace_intersections")),
157 declareRestartableData<
std::unordered_map<
std::string,
RayID>>(
"registered_ray_map")),
158 _reverse_registered_ray_map(
159 declareRestartableData<
std::vector<
std::string>>(
"reverse_registered_ray_map")),
165 _has_non_planar_sides(true),
166 _has_same_level_active_elems(sameLevelActiveElems()),
169 _domain_max_length(1.01 * (_b_box.
max() - _b_box.
min()).
norm()),
170 _total_volume(computeTotalVolume()),
184 _local_trace_ray_results(
TraceRay::FAILED_TRACES + 1, 0),
186 _called_initial_setup(false),
188 _elem_index_helper(_mesh.
getMesh(),
name() +
"_elem_index")
206 if (!getParam<bool>(
"allow_other_flags_with_prekernels") &&
_execute_enum.
size() > 1)
208 "PRE_KERNELS cannot be mixed with any other execution flag.\nThat is, you cannot " 210 "mix RayKernels that contribute to the Jacobian/residual with those that do not.");
213 mooseError(
"Execution on residual and Jacobian evaluation (execute_on = PRE_KERNELS)\n",
214 "is not supported for an eigenvalue solve.");
259 std::vector<RayKernelBase *> ray_kernels;
261 for (
const auto & rkb : ray_kernels)
263 mooseError(
"This study has RayKernel objects that contribute to residuals and Jacobians.",
264 "\nIn this case, the study must use the execute_on = PRE_KERNELS");
275 mooseAssert(
_num_cached[tid] == 0,
"Cached residuals/Jacobians not empty");
278 rto->residualSetup();
285 mooseAssert(
_num_cached[tid] == 0,
"Cached residuals/Jacobians not empty");
288 rto->jacobianSetup();
295 rto->timestepSetup();
309 trace_ray->meshChanged();
324 std::vector<RayKernelBase *> ray_kernels;
327 std::set<SubdomainID> ray_kernel_blocks;
328 for (
const auto & rk : ray_kernels)
329 ray_kernel_blocks.insert(rk->blockIDs().begin(), rk->blockIDs().end());
331 std::set<SubdomainID> missing;
334 ray_kernel_blocks.begin(),
335 ray_kernel_blocks.end(),
336 std::inserter(missing, missing.begin()));
340 std::ostringstream error;
341 error <<
"Subdomains { ";
342 std::copy(missing.begin(), missing.end(), std::ostream_iterator<SubdomainID>(error,
" "));
343 error <<
"} do not have RayKernels defined!";
353 std::vector<RayTracingObject *> ray_tracing_objects;
365 for (
const auto & rto : rtos)
366 for (
const auto & dep_name : rto->getRequestedItems())
369 for (
const auto & rto_search : rtos)
370 if (rto_search->name() == dep_name)
377 rto->paramError(
"depends_on",
"The ", rto->getBase(),
" '", dep_name,
"' does not exist");
390 Utility::enum_to_string(elem->type()),
391 " is not supported in ray tracing with adaptivity");
395 Utility::enum_to_string(elem->type()),
396 " is not supported in ray tracing");
404 std::vector<const RayBoundaryConditionBase *> rbc_ptrs;
406 std::vector<const PeriodicRayBC *> prbc_ptrs;
407 for (
const auto rbc_ptr : rbc_ptrs)
408 if (
const auto prbc_ptr = dynamic_cast<const PeriodicRayBC *>(rbc_ptr))
409 prbc_ptrs.push_back(prbc_ptr);
410 if (prbc_ptrs.empty())
417 std::unordered_set<dof_id_type>>>
419 for (
const auto prbc_ptr : prbc_ptrs)
421 for (
const auto & [bid, pb] : prbc_ptr->getPeriodicBoundaries())
423 const auto [it, inserted] = boundary_map.emplace(
424 std::piecewise_construct,
426 std::forward_as_tuple(prbc_ptr, pb.get(), std::unordered_set<dof_id_type>()));
428 prbc_ptr->mooseError(
"The periodic boundary '",
430 "' has been defined in both ",
431 prbc_ptr->typeAndName(),
445 for (
const auto & [elem, side_bid_pair] : sideset_map)
447 const auto [side, bid] = side_bid_pair;
448 if (
auto it = boundary_map.find(bid); it != boundary_map.end())
449 for (
const auto n : elem->nodes_on_side(side))
450 std::get<2>(it->second).insert(elem->node_ref(n).id());
454 for (
auto & bid_tuple_pair : boundary_map)
458 std::map<std::pair<boundary_id_type, boundary_id_type>,
459 std::pair<const PeriodicRayBC *, const PeriodicRayBC *>>
461 for (
auto it = boundary_map.begin(); it != boundary_map.end(); ++it)
463 const auto & [bid, tup] = *it;
464 const auto [prbc_ptr, pb, node_ids] = tup;
466 for (
auto other_it = std::next(it); other_it != boundary_map.end(); ++other_it)
468 const auto & [other_bid, other_tup] = *other_it;
471 if (pb->pairedboundary == other_bid)
474 const auto other_prbc_ptr = std::get<0>(other_tup);
475 const auto & other_node_ids = std::get<2>(other_tup);
476 for (
const auto node_id : node_ids)
477 if (other_node_ids.count(node_id))
479 if (!warn_boundaries.count(std::make_pair(other_bid, bid)))
480 warn_boundaries.emplace(std::make_pair(bid, other_bid),
481 std::make_pair(prbc_ptr, other_prbc_ptr));
487 if (warn_boundaries.size())
489 std::ostringstream oss;
490 oss << warn_boundaries.size()
491 <<
" ray tracing periodic boundaries were found to be neighbors:\n\n";
492 for (
const auto & [bids_pair, prbc_ptrs_pair] : warn_boundaries)
494 const auto [bid, paired_bid] = bids_pair;
495 const auto [prbc_ptr, paired_prbc_ptr] = prbc_ptrs_pair;
498 << paired_prbc_ptr->typeAndName() <<
")\n";
500 oss <<
"\nThe periodic propagation of rays at points where two or more periodic" 501 <<
"\nboundaries meet is not fully supported with a distributed mesh." 502 <<
"\n\nIf you encounter trace failures, you should use a replicated mesh.";
530 Elem * elem = bnd_elem->_elem;
531 const unsigned int side = bnd_elem->_side;
532 const auto bnd_id = bnd_elem->_bnd_id;
540 std::vector<RayBoundaryConditionBase *> result;
546 mooseError(
"RayBCs exist on internal sidesets that are not bounded by a different",
547 "\nsubdomain on each side.",
548 "\n\nIn order to use RayBCs on internal sidesets, said sidesets must have",
549 "\na different subdomain on each side.");
560 entry.resize(elem->
n_sides(), std::vector<BoundaryID>());
563 entry[side].push_back(bnd_id);
567 mooseError(
"RayBCs are defined on internal sidesets, but the study is not set to use ",
568 "internal sidesets during tracing.",
569 "\n\nSet the parameter use_internal_sidesets = true to enable this capability.");
590 entry.resize(elem->n_sides(), 0);
592 for (
const auto s : elem->side_index_range())
594 const auto & side =
elemSide(*elem, s);
595 if (side.n_vertices() < 4)
598 if (!side.has_affine_map())
606 "Ray tracing on non-planar faces is an approximation and may fail.\n\n",
607 "Use at your own risk! You can disable this warning by setting the\n",
608 "parameter 'warn_non_planar' to false.");
628 entry = std::max(entry, elem->hmax());
634 if (getParam<bool>(
"warn_subdomain_hmax"))
636 const auto warn_prefix =
type() +
" '" +
name() +
"': ";
637 const auto warn_suffix =
638 "\n\nRay tracing uses an approximate element size for each subdomain to scale the\n" 639 "tolerances used in computing ray intersections. This warning suggests that the\n" 640 "approximate element size is not a good approximation. This is likely due to poor\n" 641 "element aspect ratios.\n\n" 642 "This warning is only output for the first element affected.\n" 643 "To disable this warning, set warn_subdomain_hmax = false.\n";
647 const auto hmin = elem->hmin();
648 const auto hmax = elem->hmax();
651 const auto hmax_rel = hmax / max_hmax;
652 if (hmax_rel < 1.e-2 || hmax_rel > 1.e2)
654 "Element hmax varies significantly from subdomain hmax.\n",
656 "First element affected:\n",
659 const auto h_rel = max_hmax / hmin;
662 "Element hmin varies significantly from subdomain hmax.\n",
664 "First element affected:\n",
678 entry.resize(num_rays);
687 std::vector<std::string> all_ray_names;
690 all_ray_names.push_back(pair.first);
692 for (
auto & rto : rtos)
695 const auto & ray_names = rto->parameters().get<std::vector<std::string>>(
"rays");
697 const auto tid = rto->parameters().get<
THREAD_ID>(
"_tid");
701 for (
const auto & ray_name : (ray_names.empty() ? all_ray_names : ray_names))
706 "rays",
"Supplied ray '", ray_name,
"' is not a registered Ray in ",
typeAndName());
707 registration[id].insert(rto);
714 for (
const auto & rto : rtos)
715 if (rto->parameters().get<std::vector<std::string>>(
"rays").size())
718 "Rays cannot be supplied when the study does not require Ray registration.\n\n",
720 " does not require Ray registration.");
727 std::set<std::string> vars_to_be_zeroed;
728 std::vector<RayKernelBase *> ray_kernels;
730 for (
auto & rk : ray_kernels)
737 std::vector<std::string> vars_to_be_zeroed_vec(vars_to_be_zeroed.begin(),
738 vars_to_be_zeroed.end());
752 std::set<MooseVariableFEBase *> needed_moose_vars;
753 std::unordered_set<unsigned int> needed_mat_props;
759 rkb->subdomainSetup();
761 const auto & mv_deps = rkb->getMooseVariableDependencies();
762 needed_moose_vars.insert(mv_deps.begin(), mv_deps.end());
764 const auto & mp_deps = rkb->getMatPropDependencies();
765 needed_mat_props.insert(mp_deps.begin(), mp_deps.end());
769 for (
auto & var : needed_moose_vars)
770 if (var->kind() == Moose::VarKindType::VAR_AUXILIARY)
781 mooseAssert(MooseUtils::absoluteFuzzyEqual((start - end).
norm(), length),
"Invalid length");
793 if (rk->needSegmentReinit())
803 std::vector<Point> points;
804 std::vector<Real> weights;
817 std::vector<Point> & points,
818 std::vector<Real> & weights)
const 823 const Point diff = end - start;
824 const Point sum = end + start;
825 mooseAssert(MooseUtils::absoluteFuzzyEqual(length, diff.
norm()),
"Invalid length");
847 "Values should only be cached when computing Jacobian/residual");
856 Threads::spin_mutex::scoped_lock lock(
_spin_mutex);
867 Threads::spin_mutex::scoped_lock lock(
_spin_mutex);
877 TIME_SECTION(
"executeStudy", 2,
"Executing Study");
905 rto->preExecuteStudy();
922 auto generation_start_time = std::chrono::steady_clock::now();
924 TIME_SECTION(
"generateRays", 2,
"Generating Rays");
928 _generation_time = std::chrono::steady_clock::now() - generation_start_time;
937 "after generateRays()");
942 "after generateRays()");
948 TIME_SECTION(
"propagateRays", 2,
"Propagating Rays");
950 const auto propagation_start_time = std::chrono::steady_clock::now();
964 "after tracing completed");
972 "after tracing completed");
1003 type(),
" '",
name(),
"': ", failures,
" ray tracing failures were tolerated.\n");
1013 std::size_t num_entries = 0;
1031 "Should not have cached values without Jacobian/residual computation");
1054 rto->postExecuteStudy();
1081 Threads::spin_mutex::scoped_lock lock(
_spin_mutex);
1084 mooseError(
"Cannot register Ray ", (aux ?
"aux " :
""),
"data after initialSetup()");
1088 if (
find != map.end())
1089 return find->second;
1092 if (other_map.find(
name) != other_map.end())
1093 mooseError(
"Cannot register Ray aux data with name ",
1096 (aux ?
"(non-aux)" :
"aux"),
1097 " data already exists with said name.");
1100 map.emplace(
name, map.size());
1104 vector.push_back(
name);
1106 return map.size() - 1;
1109 std::vector<RayDataIndex>
1112 std::vector<RayDataIndex> indices(names.size());
1113 for (std::size_t i = 0; i < names.size(); ++i)
1121 const bool graceful)
const 1123 Threads::spin_mutex::scoped_lock lock(
_spin_mutex);
1127 if (
find != map.end())
1128 return find->second;
1134 if (other_map.find(
name) != other_map.end())
1137 "' was not found.\n\n",
1139 (aux ?
"non-aux" :
"aux"),
1140 " data with said name was found.\n",
1141 "Did you mean to use ",
1142 (aux ?
"getRayDataIndex()/getRayDataIndices()?" 1143 :
"getRayAuxDataIndex()/getRayAuxDataIndices()"),
1146 mooseError(
"Unknown Ray ", (aux ?
"aux " :
""),
"data with name ",
name);
1149 std::vector<RayDataIndex>
1152 const bool graceful)
const 1154 std::vector<RayDataIndex> indices(names.size());
1155 for (std::size_t i = 0; i < names.size(); ++i)
1163 Threads::spin_mutex::scoped_lock lock(
_spin_mutex);
1166 mooseError(
"Unknown Ray ", aux ?
"aux " :
"",
"data with index ", index);
1176 std::vector<RayDataIndex>
1188 std::vector<RayDataIndex>
1190 const bool graceful )
const 1207 std::vector<RayDataIndex>
1215 const bool graceful )
const 1220 std::vector<RayDataIndex>
1222 const bool graceful )
const 1236 std::vector<RayKernelBase *> result;
1238 return result.size();
1249 mooseError(
"Should not call getRayKernels() before initialSetup()");
1254 .condition<AttribSystem>(
"RayKernel")
1277 std::vector<RayKernelBase *> rkbs;
1285 for (
auto rkb : rkbs)
1286 if (ray_id_rtos.count(rkb))
1287 result.push_back(rkb);
1301 mooseError(
"Should not call getRayBCs() before initialSetup()");
1306 .condition<AttribSystem>(
"RayBoundaryCondition")
1316 const std::vector<TraceRayBndElement> & bnd_elems,
1323 if (bnd_elems.size() == 1)
1324 getRayBCs(result, bnd_elems[0].bnd_id, tid);
1327 std::vector<BoundaryID> bnd_ids(bnd_elems.size());
1328 for (MooseIndex(bnd_elems.size()) i = 0; i < bnd_elems.size(); ++i)
1329 bnd_ids[i] = bnd_elems[i].bnd_id;
1337 std::vector<RayBoundaryConditionBase *> rbcs;
1338 if (bnd_elems.size() == 1)
1339 getRayBCs(rbcs, bnd_elems[0].bnd_id, tid);
1342 std::vector<BoundaryID> bnd_ids(bnd_elems.size());
1343 for (MooseIndex(bnd_elems.size()) i = 0; i < bnd_elems.size(); ++i)
1344 bnd_ids[i] = bnd_elems[i].bnd_id;
1354 for (
auto rbc : rbcs)
1355 if (ray_id_rtos.count(rbc))
1356 result.push_back(rbc);
1360 std::vector<RayTracingObject *>
1363 std::vector<RayTracingObject *> result;
1368 const std::vector<std::shared_ptr<Ray>> &
1372 mooseError(
"The Ray bank is not available because the private parameter " 1373 "'_bank_rays_on_completion' is set to false.");
1375 mooseError(
"Cannot get the Ray bank during generation or propagation.");
1380 std::shared_ptr<Ray>
1385 std::shared_ptr<Ray> ray;
1386 for (
const std::shared_ptr<Ray> & possible_ray :
rayBank())
1387 if (possible_ray->id() == ray_id)
1394 unsigned int have_ray = ray ? 1 : 0;
1397 mooseError(
"Could not find a Ray with the ID ", ray_id,
" in the Ray banks.");
1400 mooseAssert(have_ray == 1,
"Multiple rays with the same ID were found in the Ray banks");
1408 const bool aux)
const 1413 Real value = ray ? (aux ? ray->auxData(index) : ray->data(index)) : 0;
1433 libmesh_parallel_only(
comm());
1435 Threads::spin_mutex::scoped_lock lock(
_spin_mutex);
1438 mooseError(
"Cannot use registerRay() with Ray registration disabled");
1442 libmesh_parallel_only(
comm());
1457 Threads::spin_mutex::scoped_lock lock(
_spin_mutex);
1460 mooseError(
"Should not use registeredRayID() with Ray registration disabled");
1464 return search->second;
1469 mooseError(
"Attempted to obtain ID of registered Ray ",
1471 ", but a Ray with said name is not registered.");
1477 Threads::spin_mutex::scoped_lock lock(
_spin_mutex);
1480 mooseError(
"Should not use registeredRayName() with Ray registration disabled");
1485 mooseError(
"Attempted to obtain name of registered Ray with ID ",
1487 ", but a Ray with said ID is not registered.");
1495 volume += elem->volume();
1500 const std::vector<std::vector<BoundaryID>> &
1506 "Internal sideset map not initialized");
1524 const std::vector<std::shared_ptr<Ray>>::const_iterator end,
1526 const std::string & error_suffix)
const 1531 std::set<RayID> local_rays;
1532 for (
const std::shared_ptr<Ray> & ray :
as_range(begin, end))
1534 mooseAssert(ray,
"Null ray");
1538 if (!local_rays.insert(ray->id()).second)
1540 for (
const std::shared_ptr<Ray> & other_ray :
as_range(begin, end))
1541 if (ray.get() != other_ray.get() && ray->id() == other_ray->id())
1548 "\n\nOffending Ray information:\n\n",
1551 other_ray->getInfo());
1559 std::map<processor_id_type, std::vector<RayID>> send_ids;
1560 if (local_rays.size())
1561 send_ids.emplace(std::piecewise_construct,
1562 std::forward_as_tuple(0),
1563 std::forward_as_tuple(local_rays.begin(), local_rays.end()));
1567 std::map<RayID, processor_id_type> global_map;
1570 const auto check_ids =
1571 [
this, &global_map, &error_suffix](
processor_id_type pid,
const std::vector<RayID> & ids)
1573 for (
const RayID id : ids)
1575 const auto emplace_pair = global_map.emplace(
id, pid);
1578 if (!emplace_pair.second)
1581 " exists on ranks ",
1582 emplace_pair.first->second,
1590 Parallel::push_parallel_vector_data(
_communicator, send_ids, check_ids);
1596 const std::vector<std::shared_ptr<Ray>>::const_iterator end,
1597 const std::string & error_suffix)
1599 std::set<const Ray *> rays;
1600 for (
const std::shared_ptr<Ray> & ray :
as_range(begin, end))
1601 if (!rays.insert(ray.get()).second)
1602 mooseError(
"Multiple shared_ptrs were found that point to the same Ray ",
1604 "\n\nOffending Ray:\n",
1612 mooseAssert(ray,
"Null ray");
1613 mooseAssert(ray->shouldContinue(),
"Ray is not continuing");
1623 for (
const std::shared_ptr<Ray> & ray : rays)
1625 mooseAssert(ray,
"Null ray");
1626 mooseAssert(ray->shouldContinue(),
"Ray is not continuing");
1638 mooseAssert(ray,
"Null ray");
1648 mooseError(
"Can only reserve in Ray buffer during generateRays()");
1656 std::unordered_map<std::pair<const Elem *, unsigned short>,
Point> & cache =
1660 const auto elem_side_pair = std::make_pair(elem, side);
1661 const auto search = cache.find(elem_side_pair);
1664 if (search == cache.end())
1668 cache.emplace(elem_side_pair, normal);
1673 return search->second;
1679 unsigned int min_level = std::numeric_limits<unsigned int>::max();
1680 unsigned int max_level = std::numeric_limits<unsigned int>::min();
1684 const auto level = elem->
level();
1685 min_level = std::min(level, min_level);
1686 max_level = std::max(level, max_level);
1692 return min_level == max_level;
1700 mooseError(
"Subdomain ", subdomain_id,
" not found in subdomain hmax map");
1701 return find->second;
1707 Real bbox_volume = 1;
1711 return MooseUtils::absoluteFuzzyEqual(bbox_volume,
totalVolume(), TOLERANCE);
1717 libmesh_parallel_only(
comm());
1719 Threads::spin_mutex::scoped_lock lock(
_spin_mutex);
1722 "Cannot be reset during generation or propagation");
1743 libmesh_parallel_only(
comm());
1745 Threads::spin_mutex::scoped_lock lock(
_spin_mutex);
1748 "Cannot be reset during generation or propagation");
1761 const unsigned short side,
1762 const Point & direction,
1766 const auto dot = normal * direction;
1770 std::shared_ptr<Ray>
1785 std::shared_ptr<Ray>
1799 std::shared_ptr<Ray>
1803 libmesh_parallel_only(
comm());
1815 std::shared_ptr<Ray>
1834 std::shared_ptr<Ray>
1842 std::shared_ptr<Ray>
bool hasRayKernels(const THREAD_ID tid)
Whether or not there are currently any active RayKernel objects.
bool currentlyGenerating() const
Whether or not the study is generating.
void getRayKernels(std::vector< RayKernelBase *> &result, SubdomainID id, THREAD_ID tid)
Fills the active RayKernels associated with this study and a block into result.
void internalSidesetSetup()
Does the setup for internal sidesets.
RayDataIndex registerRayData(const std::string &name)
Register a value to be filled in the data on a Ray with a given name.
RayData getBankedRayDataInternal(const RayID ray_id, const RayDataIndex index, const bool aux) const
Internal method for getting the value (replicated across all processors) in a Ray's data or aux data ...
void periodicBoundaryChecks()
Check for overlapping PeriodicRayBC boundaries and check for cases in which ghosting may not be suffi...
virtual void postOnSegment(const THREAD_ID tid, const std::shared_ptr< Ray > &ray)
Called at the end of a Ray segment.
const bool _tolerate_failure
Whether or not to tolerate a Ray Tracing failure.
Real totalVolume() const
Get the current total volume of the domain.
void traceableMeshChecks()
Check for if all of the element types in the mesh are supported by ray tracing.
virtual void onCompleteRay(const std::shared_ptr< Ray > &ray)
Entry point for acting on a ray when it is completed (shouldContinue() == false)
libMesh::ConstElemRange * getActiveLocalElementRange()
const std::string & getRayDataNameInternal(const RayDataIndex index, const bool aux) const
Internal method for getting the name of Ray data or Ray aux data.
KOKKOS_INLINE_FUNCTION const T * find(const T &target, const T *const begin, const T *const end)
const std::vector< std::shared_ptr< Ray > > & rayBank() const
Get the Ray bank.
static const RayDataIndex INVALID_RAY_DATA_INDEX
Invalid index into a Ray's data.
std::vector< std::vector< std::set< const RayTracingObject * > > > _threaded_ray_object_registration
Threaded storage for all of the RayTracingObjects associated with a single Ray.
std::vector< std::string > & _reverse_registered_ray_map
Map from registered Ray ID to name.
unsigned long long int _total_intersections
Total number of Ray/element intersections.
RayID registerRay(const std::string &name)
Registers a Ray with a given name.
unsigned long long int _ending_intersections
Total number of Ray/element intersections for Rays that finished on this processor.
unsigned int _max_trajectory_changes
Max number of trajectory changes for a single Ray.
unsigned long int RayID
Type for a Ray's ID.
void verifyUniqueRayIDs(const std::vector< std::shared_ptr< Ray >>::const_iterator begin, const std::vector< std::shared_ptr< Ray >>::const_iterator end, const bool global, const std::string &error_suffix) const
Verifies that the Rays in the given range have unique Ray IDs.
bool _has_same_level_active_elems
Whether or not the mesh has active elements of the same level.
bool sameLevelActiveElems() const
Determine whether or not the mesh currently has active elements that are all the same level...
virtual void initialSetup() override
void moveRayToBufferDuringTrace(std::shared_ptr< Ray > &ray, const THREAD_ID tid, const AcquireMoveDuringTraceKey &)
INTERNAL method for moving a Ray into the buffer during tracing.
RayDataIndex registerRayAuxData(const std::string &name)
Register a value to be filled in the aux data on a Ray with a given name.
Class that is used as a parameter to the public constructors/reset methods.
RayBC that enforces periodic boundaries.
void paramError(const std::string ¶m, Args... args) const
MooseMesh & _mesh
The Mesh.
T & getMesh(MooseMesh &mesh)
function to cast mesh
Attribute for the RayTracingStudy a RayTracingObject is associated with.
static InputParameters validParams()
virtual void prepare(const Elem *elem, const THREAD_ID tid) override
NumericVector< Number > & solution()
std::vector< std::vector< TraceData > > _threaded_cached_traces
The threaded storage for cached traces.
const bool _use_internal_sidesets
Whether or not to use the internal sidesets in ray tracing.
Data structure that stores information for output of a partial trace of a Ray on a processor...
std::shared_ptr< Ray > acquireRay()
User APIs for constructing Rays within the RayTracingStudy.
std::size_t rayAuxDataSize() const
The registered size of values in the Ray aux data.
std::unordered_map< SubdomainID, Real > _subdomain_hmax
The cached hmax for all elements in a subdomain.
libMesh::dof_id_type maxIndex() const
Gets the maximum index generated using this object.
libMesh::BoundingBox _loose_b_box
Loose nodal bounding box for the domain.
const std::set< BoundaryID > & getInternalSidesets() const
Gets the internal sidesets (that have RayBCs) within the local domain.
std::vector< std::vector< unsigned short > > _non_planar_sides
Non planar side data, which is for quick checking if an elem side is non-planar We use unsigned short...
RayData getBankedRayData(const RayID ray_id, const RayDataIndex index) const
Gets the data value for a banked ray with a given ID.
Base object for the RayKernel syntax.
void initialize(const libMesh::SimpleRange< libMesh::MeshBase::element_iterator > elems)
Initializes the indices in a contiguous manner for the given element range.
Real _ending_distance
Total distance traveled by Rays that end on this processor.
virtual void reinitSegment(const Elem *elem, const Point &start, const Point &end, const Real length, THREAD_ID tid)
Reinitialize objects for a Ray segment for ray tracing.
unsigned int size() const
RayDataIndex registerRayDataInternal(const std::string &name, const bool aux)
Internal method for registering Ray data or Ray aux data with a name.
const std::vector< RayKernelBase * > & currentRayKernels(THREAD_ID tid) const
Gets the current RayKernels for a thread, which are set in segmentSubdomainSetup() ...
const Parallel::Communicator & comm() const
Key that is used for restricting access to moveRayToBufferDuringTrace() and acquireRayDuringTrace().
void addAvailableFlags(const ExecFlagType &flag, Args... flags)
std::shared_ptr< Ray > acquireReplicatedRay()
Acquire a Ray from the pool of Rays within generateRays() in a replicated fashion.
std::shared_ptr< Ray > acquireRayDuringTrace(const THREAD_ID tid, const AcquireMoveDuringTraceKey &)
INTERNAL methods for acquiring a Ray during a trace in RayKernels and RayBCs.
const Parallel::Communicator & _communicator
std::shared_ptr< Ray > acquireRegisteredRay(const std::string &name)
Acquires a Ray with a given name within generateRays().
void clearActiveMaterialProperties(const THREAD_ID tid)
The following methods are specializations for using the Parallel::packed_range_* routines for a vecto...
const BoundaryInfo & get_boundary_info() const
RayID _replicated_next_ray_id
Storage for the next available replicated RayID, obtained via generateReplicatedRayID() ...
std::set< BoundaryID > _internal_sidesets
The BoundaryIDs on the local mesh that have internal RayBCs.
virtual RayID generateUniqueRayID(const THREAD_ID tid)
Generates a unique RayID to be used for a Ray.
std::vector< std::vector< RayKernelBase * > > _threaded_current_ray_kernels
The current RayKernel objects for each thread.
auto max(const L &left, const R &right)
std::vector< TraceData > _cached_traces
Storage for the cached traces.
RayDataIndex getRayAuxDataIndex(const std::string &name, const bool graceful=false) const
Gets the index associated with a registered value in the Ray aux data.
std::vector< std::shared_ptr< TraceRay > > _threaded_trace_ray
The TraceRay objects for each thread (they do the physical tracing)
virtual Assembly & assembly(const THREAD_ID tid, const unsigned int sys_num) override
Real subdomainHmax(const SubdomainID subdomain_id) const
Get the cached hmax for all elements in a subdomain.
std::vector< RayID > _threaded_next_ray_id
Storage for the next available unique RayID, obtained via generateUniqueRayID()
bool isRectangularDomain() const
Whether or not the domain is rectangular (if it is prefectly encompassed by its bounding box) ...
std::vector< std::string > _ray_aux_data_names
The names for each Ray aux data entry.
processor_id_type size() const
FEProblemBase & _fe_problem
virtual void setActiveElementalMooseVariables(const std::set< MooseVariableFEBase * > &moose_vars, const THREAD_ID tid) override
Threads::spin_mutex _spin_mutex
Spin mutex object for locks.
uint8_t processor_id_type
std::shared_ptr< Ray > getBankedRay(const RayID ray_id) const
Gets the Ray with the ID ray_id from the Ray bank.
unsigned long long int _total_processor_crossings
Total number of processor crossings.
processor_id_type n_processors() const
void nonPlanarSideSetup()
Sets up the caching of whether or not each element side is non-planar, which is stored in _non_planar...
bool verifyRays() const
Whether or not to verify if Rays have valid information before being traced.
const std::string & name() const
unsigned int RayDataIndex
Type for the index into the data and aux data on a Ray.
void coverageChecks()
Perform coverage checks (coverage of RayMaterials and RayKernels, if enabled)
void resetReplicatedRayIDs()
Resets the generation of unique replicated RayIDs accessed via generateReplicatedRayID().
Real value(unsigned n, unsigned alpha, unsigned beta, Real x)
const bool _bank_rays_on_completion
Whether or not to bank rays on completion.
void min(const T &r, T &o, Request &req) const
unsigned int _ending_max_intersections
Max number of intersections for Rays that finished on this processor.
TheWarehouse & theWarehouse() const
virtual void postExecuteStudy()
Entry point after study execution.
MooseVariableFE< Real > & variable()
Gets the variable this AuxRayKernel contributes to.
unsigned int _max_processor_crossings
Max number of processor crossings for all Rays.
const std::unique_ptr< ParallelRayStudy > _parallel_ray_study
The study that used is to actually execute (trace) the Rays.
virtual unsigned int dimension() const
std::vector< RayTracingObject * > getRayTracingObjects()
Gets all of the currently active RayTracingObjects.
virtual void residualSetup() override
static InputParameters validParams()
const Point & min() const
const bool & currentlyComputingResidual() const
boundary_id_type BoundaryID
std::vector< TheWarehouse::QueryCache< AttribSubdomains > > _threaded_cache_ray_kernel
Threaded cached subdomain query for RayKernelBase objects pertaining to this study.
virtual void execute() override
Executes the study (generates and propagates Rays)
SimpleRange< IndexType > as_range(const std::pair< IndexType, IndexType > &p)
unsigned int _max_intersections
Max number of intersections for a single Ray.
unsigned int _ending_max_trajectory_changes
Max number of trajectory changes for Rays that finished on this processor.
unsigned int _ending_max_processor_crossings
Max number of total processor crossings for Rays that finished on this processor. ...
const std::string & getRayDataName(const RayDataIndex index) const
Gets the name associated with a registered value in the Ray data.
void reserveRayBuffer(const std::size_t size)
Reserve size entires in the Ray buffer.
void zeroAuxVariables()
Zero the AuxVariables that the registered AuxRayKernels contribute to.
std::vector< std::unique_ptr< libMesh::FEBase > > _threaded_fe_face
Face FE used for computing face normals for each thread.
const std::string & type() const
void dependencyChecks()
Perform checks to see if the listed dependencies in the RayTracingObjects exist.
static InputParameters validParams()
const std::string & getRayAuxDataName(const RayDataIndex index) const
Gets the name associated with a registered value in the Ray aux data.
virtual void cacheResidual(const THREAD_ID tid) override
Basic datastructure for a ray that will traverse the mesh.
std::chrono::steady_clock::time_point _execution_start_time
Timing.
std::vector< TheWarehouse::QueryCache< AttribBoundaries > > _threaded_cache_ray_bc
Threaded cached boundary query for RayBC objects pertaining to this study.
virtual void jacobianSetup() override
std::string typeAndName() const
bool isValueSet(const std::string &value) const
unsigned int number() const
std::string stringify(const T &t)
virtual void preExecuteStudy()
Entry point before study execution.
std::unordered_map< std::string, RayDataIndex > _ray_aux_data_map
The map from Ray aux data names to index.
AuxiliarySystem & getAuxiliarySystem()
std::string getBoundaryString(const BoundaryID boundary_id) const
const bool _warn_non_planar
Whether not to warn if non-planar faces are found.
libMesh::BoundingBox _b_box
Nodal bounding box for the domain.
virtual void meshChanged() override
libMesh::dof_id_type getIndex(const libMesh::Elem *elem) const
Get the index associated with the element elem.
virtual void setCurrentSubdomainID(const Elem *elem, const THREAD_ID tid) override
virtual void generateRays()=0
Subclasses should override this to determine how to generate Rays.
RayID generateReplicatedRayID()
Generates a Ray ID that is replicated across all processors.
const std::multimap< const Elem *, std::pair< unsigned short int, boundary_id_type > > & get_sideset_map() const
const ExecFlagEnum & _execute_enum
RayDataIndex getRayDataIndexInternal(const std::string &name, const bool aux, const bool graceful) const
Internal method for getting the index of Ray data or Ray aux data.
void verifyDependenciesExist(const std::vector< RayTracingObject *> &rtos)
Verifies that the dependencies exist for a set of RayTracingObjects.
unsigned long long int _ending_processor_crossings
Total number of processor crossings for Rays that finished on this processor.
ElemIndexHelper _elem_index_helper
Helper for defining a local contiguous index for each element.
RayTracingStudy(const InputParameters ¶meters)
virtual unsigned int n_sides() const=0
std::vector< std::unordered_map< std::pair< const Elem *, unsigned short >, Point > > _threaded_cached_normals
Threaded cache for side normals that have been computed already during tracing.
TraceData & initThreadedCachedTrace(const std::shared_ptr< Ray > &ray, THREAD_ID tid)
Initialize a Ray in the threaded cached trace map to be filled with segments.
virtual const SystemBase & getSystemBase(const unsigned int sys_num) const
const Elem * neighbor_ptr(unsigned int i) const
virtual const Point & getSideNormal(const Elem *elem, const unsigned short side, const THREAD_ID tid)
Get the outward normal for a given element side.
const SubdomainID ANY_BLOCK_ID
unsigned int level() const
std::vector< RayDataIndex > getRayAuxDataIndices(const std::vector< std::string > &names, const bool graceful=false) const
Gets the indices associated with registered values in the Ray aux data.
std::unordered_map< std::string, RayDataIndex > _ray_data_map
The map from Ray data names to index.
DIE A HORRIBLE DEATH HERE typedef LIBMESH_DEFAULT_SCALAR_TYPE Real
std::vector< RayDataIndex > getRayDataIndices(const std::vector< std::string > &names, const bool graceful=false) const
Gets the indices associated with registered values in the Ray data.
bool currentlyPropagating() const
Whether or not the study is propagating (tracing Rays)
subdomain_id_type subdomain_id() const
bool _has_non_planar_sides
Whether or not the local mesh has elements with non-planar sides.
virtual void subdomainSetup(SubdomainID subdomain, const THREAD_ID tid)
const libMesh::Elem & elemSide(const libMesh::Elem &elem, const unsigned int s, const THREAD_ID tid=0)
Get an element's side pointer without excessive memory allocation.
void max(const T &r, T &o, Request &req) const
void reinitMaterials(SubdomainID blk_id, const THREAD_ID tid, bool swap_stateful=true)
void getRayBCs(std::vector< RayBoundaryConditionBase *> &result, BoundaryID id, THREAD_ID tid)
Fills the active RayBCs associated with this study and a boundary into result.
std::vector< std::size_t > _num_cached
Number of currently cached objects for Jacobian/residual for each thread.
std::chrono::steady_clock::duration _propagation_time
virtual void segmentSubdomainSetup(const SubdomainID subdomain, const THREAD_ID tid, const RayID ray_id)
Setup for on subdomain change or subdomain AND ray change during ray tracing.
void mooseWarning(Args &&... args) const
RayData getBankedRayAuxData(const RayID ray_id, const RayDataIndex index) const
Gets the data value for a banked ray with a given ID.
bool _called_initial_setup
Whether or not we've called initial setup - used to stop from late registration.
void mooseError(Args &&... args) const
void verifyUniqueRays(const std::vector< std::shared_ptr< Ray >>::const_iterator begin, const std::vector< std::shared_ptr< Ray >>::const_iterator end, const std::string &error_suffix)
Verifies that the Rays in the given range are unique.
std::vector< RayDataIndex > getRayDataIndicesInternal(const std::vector< std::string > &names, const bool aux, const bool graceful) const
Internal method for getting the indicies of Ray data or Ray aux data.
Real _total_distance
Total distance traveled by all Rays.
const processor_id_type _pid
The rank of this processor (this actually takes time to lookup - so just do it once) ...
std::unordered_map< std::string, RayID > & _registered_ray_map
Map from registered Ray name to ID.
std::vector< std::string > _ray_data_names
The names for each Ray data entry.
float RayData
Type for a Ray's data.
const std::string & registeredRayName(const RayID ray_id) const
Gets the name of a registered ray.
void moveRayToBuffer(std::shared_ptr< Ray > &ray)
Moves a ray to the buffer to be traced during generateRays().
const ExecFlagType EXEC_PRE_KERNELS
RayID registeredRayID(const std::string &name, const bool graceful=false) const
Gets the ID of a registered ray.
virtual void buildSegmentQuadrature(const Point &start, const Point &end, const Real length, std::vector< Point > &points, std::vector< Real > &weights) const
Builds quadrature points for a given segment using the _segment_qrule.
libMesh::StoredRange< MooseMesh::const_bnd_elem_iterator, const BndElement *> * getBoundaryElementRange()
const bool _use_ray_registration
Whether or not to use Ray registration.
virtual bool hasActiveElementalMooseVariables(const THREAD_ID tid) const
const bool _ray_kernel_coverage_check
Whether or not to perform coverage checks on RayKernels.
void localElemIndexSetup()
Sets up the _elem_index_helper, which is used for obtaining a contiguous index for all elements that ...
const bool & currentlyComputingJacobian() const
std::vector< std::shared_ptr< Ray > > _ray_bank
Cumulative Ray bank - stored only when _bank_rays_on_completion.
const Point & max() const
std::vector< unsigned long long int > _local_trace_ray_results
Cumulative results on this processor from the threaded TraceRay objects.
std::chrono::steady_clock::duration _generation_time
void registeredRaySetup()
Sets up the maps from Ray to associated RayTracingObjects if _use_ray_registration.
void scale(const Real factor)
void moveRaysToBuffer(std::vector< std::shared_ptr< Ray >> &rays)
Moves rays to the buffer to be traced during generateRays().
void executeStudy()
Method for executing the study so that it can be called out of the standard UO execute() ...
virtual bool isDistributedMesh() const
void prepareMaterials(const std::unordered_set< unsigned int > &consumer_needed_mat_props, const SubdomainID blk_id, const THREAD_ID tid)
Adaptivity & adaptivity()
bool hasActiveMaterialProperties(const THREAD_ID tid) const
virtual void cacheJacobian(const THREAD_ID tid) override
auto min(const L &left, const R &right)
virtual void reinitElemPhys(const Elem *elem, const std::vector< Point > &phys_points_in_elem, const THREAD_ID tid) override
std::shared_ptr< Ray > acquireUnsizedRay()
Acquire a Ray from the pool of Rays within generateRays(), without resizing the data (sizes the data ...
RayDataIndex getRayDataIndex(const std::string &name, const bool graceful=false) const
Gets the index associated with a registered value in the Ray data.
void resetUniqueRayIDs()
Resets the generation of unique RayIDs via generateUniqueRayID() to the beginning of the range...
virtual void addCachedResidual(const THREAD_ID tid) override
std::chrono::steady_clock::duration _execution_time
virtual void zeroVariables(std::vector< std::string > &vars_to_be_zeroed)
virtual void clearActiveElementalMooseVariables(const THREAD_ID tid) override
std::vector< std::unique_ptr< libMesh::QBase > > _threaded_q_face
Face quadrature used for computing face normals for each thread.
std::shared_ptr< Ray > acquireCopiedRay(const Ray &ray)
Acquires a Ray that that is copied from another Ray within generateRays().
Traces Rays through the mesh on a single processor.
std::vector< std::vector< std::vector< BoundaryID > > > _internal_sidesets_map
Internal sideset data, if internal sidesets exist (indexed with getLocalElemIndex()) ...
void subdomainHMaxSetup()
Caches the hmax for all elements in each subdomain.
bool sideIsIncoming(const Elem *const elem, const unsigned short side, const Point &direction, const THREAD_ID tid)
Whether or not side is incoming on element elem in direction direction.
static const RayID INVALID_RAY_ID
Invalid Ray ID.
std::unique_ptr< libMesh::QBase > _segment_qrule
Quadrature rule for laying points across a 1D ray segment.
bool hasInternalSidesets() const
Whether or not the local mesh has internal sidesets that have RayBCs on them.
std::size_t rayDataSize() const
The registered size of values in the Ray data.
virtual void timestepSetup() override
const std::set< SubdomainID > & meshSubdomains() const
virtual void addCachedJacobian(const THREAD_ID tid) override
virtual bool shouldCacheTrace(const std::shared_ptr< Ray > &) const
Virtual that allows for selection in if a Ray should be cached or not (only used when _cache_traces)...
void set_union(T &data, const unsigned int root_id) const
const RemoteElem * remote_elem
Real computeTotalVolume()
Helper function for computing the total domain volume.