Skip to content

Commit

Permalink
Improve boundary check
Browse files Browse the repository at this point in the history
  • Loading branch information
gassmoeller committed Jan 6, 2025
1 parent 13786f7 commit 24495bd
Show file tree
Hide file tree
Showing 11 changed files with 123 additions and 94 deletions.
14 changes: 9 additions & 5 deletions include/aspect/boundary_traction/interface.h
Original file line number Diff line number Diff line change
Expand Up @@ -136,13 +136,17 @@ namespace aspect
get_active_plugin_boundary_indicators() const;

/**
* Return a list of component masks that indicate for
* each active plugin which components it is responsible for.
* Return a component masks that indicates for
* each boundary which components are prescribed by
* this manager class. All plugins that are responsible
* for this boundary use the same component mask.
* The list of plugin objects can be
* requested by calling get_active_plugins().
* requested by calling get_active_plugins() and the
* list of boundaries they are responsible for is
* returned by get_active_plugin_boundary_indicators().
*/
const std::vector<ComponentMask> &
get_active_plugin_component_masks() const;
ComponentMask
get_component_mask(const types::boundary_id boundary_id) const;

/**
* Declare the parameters of all known boundary traction plugins, as
Expand Down
14 changes: 9 additions & 5 deletions include/aspect/boundary_velocity/interface.h
Original file line number Diff line number Diff line change
Expand Up @@ -161,13 +161,17 @@ namespace aspect
get_active_plugin_boundary_indicators() const;

/**
* Return a list of component masks that indicate for
* each active plugin which components it is responsible for.
* Return a component masks that indicates for
* each boundary which components are prescribed by
* this manager class. All plugins that are responsible
* for this boundary use the same component mask.
* The list of plugin objects can be
* requested by calling get_active_plugins().
* requested by calling get_active_plugins() and the
* list of boundaries they are responsible for is
* returned by get_active_plugin_boundary_indicators().
*/
const std::vector<ComponentMask> &
get_active_plugin_component_masks() const;
ComponentMask
get_component_mask(const types::boundary_id boundary_id) const;

/**
* Return a set of boundary indicators for which boundary
Expand Down
26 changes: 23 additions & 3 deletions source/boundary_traction/interface.cc
Original file line number Diff line number Diff line change
Expand Up @@ -132,10 +132,30 @@ namespace aspect


template <int dim>
const std::vector<ComponentMask> &
Manager<dim>::get_active_plugin_component_masks() const
ComponentMask
Manager<dim>::get_component_mask(const types::boundary_id boundary_id) const
{
return component_masks;
Assert(prescribed_traction_boundary_indicators.find(boundary_id) != prescribed_traction_boundary_indicators.end(),
ExcMessage("The boundary traction manager class was asked for the "
"component mask of boundary indicator <"
+
Utilities::int_to_string(boundary_id)
+
"> with symbolic name <"
+
this->get_geometry_model().translate_id_to_symbol_name(boundary_id)
+
">, but this boundary is not part of the active boundary traction plugins."));

// Since all component masks of plugins at the same boundary are identical, we can use
// the component mask of the first plugin we find that is responsible for this boundary.
for (unsigned int i=0; i<boundary_indicators.size(); ++i)
if (boundary_indicators[i] == boundary_id)
return component_masks[i];

// We should never get here if plugins and boundary indicators were set up correctly.
AssertThrow(false, ExcInternalError());
return ComponentMask();
}


Expand Down
26 changes: 23 additions & 3 deletions source/boundary_velocity/interface.cc
Original file line number Diff line number Diff line change
Expand Up @@ -129,10 +129,30 @@ namespace aspect


template <int dim>
const std::vector<ComponentMask> &
Manager<dim>::get_active_plugin_component_masks() const
ComponentMask
Manager<dim>::get_component_mask(const types::boundary_id boundary_id) const
{
return component_masks;
Assert(prescribed_velocity_boundary_indicators.find(boundary_id) != prescribed_velocity_boundary_indicators.end(),
ExcMessage("The boundary velocity manager class was asked for the "
"component mask of boundary indicator <"
+
Utilities::int_to_string(boundary_id)
+
"> with symbolic name <"
+
this->get_geometry_model().translate_id_to_symbol_name(boundary_id)
+
">, but this boundary is not part of the active boundary velocity plugins."));

// Since all component masks of plugins at the same boundary are identical, we can use
// the component mask of the first plugin we find that is responsible for this boundary.
for (unsigned int i=0; i<boundary_indicators.size(); ++i)
if (boundary_indicators[i] == boundary_id)
return component_masks[i];

// We should never get here if plugins and boundary indicators were set up correctly.
AssertThrow(false, ExcInternalError());
return ComponentMask();
}


Expand Down
5 changes: 2 additions & 3 deletions source/simulator/assemblers/stokes.cc
Original file line number Diff line number Diff line change
Expand Up @@ -942,10 +942,9 @@ namespace aspect

const typename DoFHandler<dim>::face_iterator face = scratch.cell->face(scratch.face_number);

const auto &traction_bis = this->get_boundary_traction_manager().get_active_plugin_boundary_indicators();
const auto &traction_bis = this->get_boundary_traction_manager().get_prescribed_boundary_traction_indicators();

if (std::find(traction_bis.begin(), traction_bis.end(), face->boundary_id())
!= traction_bis.end())
if (traction_bis.find(face->boundary_id()) != traction_bis.end())
{
for (unsigned int q=0; q<scratch.face_finite_element_values.n_quadrature_points; ++q)
{
Expand Down
4 changes: 2 additions & 2 deletions source/simulator/assembly.cc
Original file line number Diff line number Diff line change
Expand Up @@ -126,7 +126,7 @@ namespace aspect
" defined that handles this formulation."));

// add the terms for traction boundary conditions
if (!boundary_traction_manager.get_active_plugins().empty())
if (!boundary_traction_manager.get_prescribed_boundary_traction_indicators().empty())
{
assemblers->stokes_system_on_boundary_face.push_back(
std::make_unique<aspect::Assemblers::StokesBoundaryTraction<dim>>());
Expand Down Expand Up @@ -763,7 +763,7 @@ namespace aspect
= (
// see if we need to assemble traction boundary conditions.
// only if so do we actually need to have an FEFaceValues object
!boundary_traction_manager.get_active_plugins().empty()
!boundary_traction_manager.get_prescribed_boundary_traction_indicators().empty()
?
update_values |
update_quadrature_points |
Expand Down
28 changes: 1 addition & 27 deletions source/simulator/core.cc
Original file line number Diff line number Diff line change
Expand Up @@ -1375,32 +1375,6 @@ namespace aspect
// for the prescribed velocity fields
boundary_velocity_manager.update();

// translate from component mask per plugin to component mask per boundary id
std::map<types::boundary_id, ComponentMask> boundary_component_masks;
for (unsigned int i=0; i<boundary_velocity_manager.get_active_plugins().size(); ++i)
{
const types::boundary_id boundary_id = boundary_velocity_manager.get_active_plugin_boundary_indicators()[i];
const auto &component_mask = boundary_velocity_manager.get_active_plugin_component_masks()[i];

const auto it = boundary_component_masks.find(boundary_id);
if (it != boundary_component_masks.end())
{
AssertThrow (it->second == component_mask,
ExcMessage("Boundary indicator <"
+
Utilities::int_to_string(boundary_id)
+
"> with symbolic name <"
+
geometry_model->translate_id_to_symbol_name (boundary_id)
+
"> is listed as having different component masks "
"for different boundary velocity plugins. This is not allowed."));
}
else
boundary_component_masks[boundary_id] = component_mask;
}

// put boundary conditions into constraints object for each boundary
for (const auto boundary_id: boundary_velocity_manager.get_prescribed_boundary_velocity_indicators())
{
Expand All @@ -1421,7 +1395,7 @@ namespace aspect
boundary_id,
vel,
constraints,
boundary_component_masks.at(boundary_id));
boundary_velocity_manager.get_component_mask(boundary_id));
}
}

Expand Down
78 changes: 44 additions & 34 deletions source/simulator/helper_functions.cc
Original file line number Diff line number Diff line change
Expand Up @@ -2419,38 +2419,42 @@ namespace aspect
boundary_indicator_lists.emplace_back(boundary_velocity_manager.get_prescribed_boundary_velocity_indicators());
boundary_indicator_lists.emplace_back(boundary_traction_manager.get_prescribed_boundary_traction_indicators());

// Make sure that each combination of boundary velocity and boundary traction plugin
// Make sure that each combination of boundary velocity and boundary traction condition
// either refers to different boundary indicators or to different components
const auto &velocity_boundary_ids = boundary_velocity_manager.get_active_plugin_boundary_indicators();
const auto &traction_boundary_ids = boundary_traction_manager.get_active_plugin_boundary_indicators();

for (unsigned int i=0; i<velocity_boundary_ids.size(); ++i)
for (unsigned int j=0; j<traction_boundary_ids.size(); ++j)
{
if (velocity_boundary_ids[i] == traction_boundary_ids[j])
{
// if boundary ids are identical, make sure that the components are different
AssertThrow((boundary_velocity_manager.get_active_plugin_component_masks()[i] &
boundary_traction_manager.get_active_plugin_component_masks()[j]) ==
ComponentMask(introspection.n_components, false),
ExcMessage("Boundary indicator <"
+
Utilities::int_to_string(velocity_boundary_ids[i])
+
"> with symbolic name <"
+
geometry_model->translate_id_to_symbol_name (velocity_boundary_ids[i])
+
"> is listed as having both "
"velocity and traction boundary conditions in the input file."));

// we have ensured the prescribed velocity and prescribed traction boundary conditions
// are compatible. In order to check them against the other boundary conditions, we
// need to remove the boundary indicator from one of the lists to make sure it only
// appears in one of them. We choose to remove the boundary indicator from the traction list.
boundary_indicator_lists[3].erase(traction_boundary_ids[j]);
}
}
for (const auto velocity_boundary_id: boundary_indicator_lists[2])
{
bool found_compatible_duplicate_boundary_id = false;
for (const auto traction_boundary_id: boundary_indicator_lists[3])
{
if (velocity_boundary_id == traction_boundary_id)
{
// if boundary ids are identical, make sure that the components are different
AssertThrow((boundary_velocity_manager.get_component_mask(velocity_boundary_id) &
boundary_traction_manager.get_component_mask(traction_boundary_id)) ==
ComponentMask(introspection.n_components, false),
ExcMessage("Boundary indicator <"
+
Utilities::int_to_string(velocity_boundary_id)
+
"> with symbolic name <"
+
geometry_model->translate_id_to_symbol_name (velocity_boundary_id)
+
"> is listed as having both "
"velocity and traction boundary conditions in the input file."));

found_compatible_duplicate_boundary_id = true;
}
}
// we have ensured the prescribed velocity and prescribed traction boundary conditions
// for the current boundary id are compatible. In order to check them against the other
// boundary conditions, we need to remove the boundary indicator from one of the lists
// to make sure it only appears in one of them. We choose to remove the boundary
// indicator from the traction list. We cannot do that in the loop above, because it
// invalidates the range of the loop.
if (found_compatible_duplicate_boundary_id)
boundary_indicator_lists[3].erase(velocity_boundary_id);
}

// for each combination of velocity boundary indicator lists, make sure that the
// intersection is empty
Expand Down Expand Up @@ -2503,6 +2507,8 @@ namespace aspect
"> is listed as having more "
"than one type of temperature or heat flux boundary condition in the input file."));

boundary_indicator_lists.emplace_back(boundary_composition_manager.get_fixed_composition_boundary_indicators());

// Check that the periodic boundaries do not have other boundary conditions set
using periodic_boundary_set
= std::set<std::pair<std::pair<types::boundary_id, types::boundary_id>, unsigned int>>;
Expand Down Expand Up @@ -2548,16 +2554,20 @@ namespace aspect
for (const auto &p : list)
AssertThrow (all_boundary_indicators.find (p)
!= all_boundary_indicators.end(),
ExcMessage ("One of the boundary indicators listed in the input file "
"is not used by the geometry model."));
ExcMessage ("Boundary indicator <"
+
Utilities::int_to_string(p)
+
"> is listed for a boundary condition, but is not used by the geometry model."));

if (parameters.nonlinear_solver == NonlinearSolver::single_Advection_no_Stokes)
{
// make sure that there are no listed velocity boundary conditions
for (unsigned int i=0; i<4; ++i)
AssertThrow (boundary_indicator_lists[i].empty(),
ExcMessage ("With the solver scheme `single Advection, no Stokes', "
"one cannot set boundary conditions for velocity."));
"one cannot set boundary conditions for velocity or traction, "
"but a boundary condition has been set."));
}
}

Expand Down
7 changes: 3 additions & 4 deletions source/simulator/melt.cc
Original file line number Diff line number Diff line change
Expand Up @@ -1039,10 +1039,9 @@ namespace aspect
Assert(face_no != numbers::invalid_unsigned_int,ExcInternalError());

const typename DoFHandler<dim>::face_iterator face = cell->face(face_no);
const auto &traction_bis = this->get_boundary_traction_manager().get_active_plugin_boundary_indicators();
const auto &traction_bis = this->get_boundary_traction_manager().get_prescribed_boundary_traction_indicators();

if (std::find(traction_bis.begin(), traction_bis.end(), face->boundary_id())
!= traction_bis.end())
if (traction_bis.find(face->boundary_id()) != traction_bis.end())
{
scratch.face_finite_element_values.reinit (cell, face_no);

Expand Down Expand Up @@ -1696,7 +1695,7 @@ namespace aspect
std::make_unique<aspect::Assemblers::MeltStokesSystemBoundary<dim>>());

// add the terms for traction boundary conditions
if (!this->get_boundary_traction_manager().get_active_plugins().empty())
if (!this->get_boundary_traction_manager().get_prescribed_boundary_traction_indicators().empty())
{
assemblers.stokes_system_on_boundary_face.push_back(
std::make_unique<Assemblers::MeltBoundaryTraction<dim>> ());
Expand Down
2 changes: 1 addition & 1 deletion source/simulator/newton.cc
Original file line number Diff line number Diff line change
Expand Up @@ -96,7 +96,7 @@ namespace aspect
" defined that handles this formulation."));

// add the terms for traction boundary conditions
if (!this->get_boundary_traction_manager().get_active_plugins().empty())
if (!this->get_boundary_traction_manager().get_prescribed_boundary_traction_indicators().empty())
{
assemblers.stokes_system_on_boundary_face.push_back(
std::make_unique<aspect::Assemblers::StokesBoundaryTraction<dim>>());
Expand Down
13 changes: 6 additions & 7 deletions source/simulator/stokes_matrix_free.cc
Original file line number Diff line number Diff line change
Expand Up @@ -2509,24 +2509,23 @@ namespace aspect
mg_constrained_dofs_A_block.initialize(dof_handler_v);

std::set<types::boundary_id> dirichlet_boundary = sim.boundary_velocity_manager.get_zero_boundary_velocity_indicators();
for (unsigned int i=0; i<sim.boundary_velocity_manager.get_active_plugins().size(); ++i)
for (const auto boundary_id: sim.boundary_velocity_manager.get_prescribed_boundary_velocity_indicators())
{
const types::boundary_id bdryid = sim.boundary_velocity_manager.get_active_plugin_boundary_indicators()[i];
const ComponentMask component_mask = sim.boundary_velocity_manager.get_active_plugin_component_masks()[i];
const ComponentMask component_mask = sim.boundary_velocity_manager.get_component_mask(boundary_id);

if (component_mask != ComponentMask(sim.introspection.n_components, false))
{
ComponentMask mask(fe_v.n_components(), false);
ComponentMask velocity_mask(fe_v.n_components(), false);

for (unsigned int i=0; i<dim; ++i)
mask.set(i, component_mask[sim.introspection.component_indices.velocities[i]]);
velocity_mask.set(i, component_mask[sim.introspection.component_indices.velocities[i]]);

mg_constrained_dofs_A_block.make_zero_boundary_constraints(dof_handler_v, {bdryid}, mask);
mg_constrained_dofs_A_block.make_zero_boundary_constraints(dof_handler_v, {boundary_id}, velocity_mask);
}
else
{
// no mask given: add at the end
dirichlet_boundary.insert(bdryid);
dirichlet_boundary.insert(boundary_id);
}
}

Expand Down

0 comments on commit 24495bd

Please sign in to comment.