-
Notifications
You must be signed in to change notification settings - Fork 15
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Reflecting boundary conditions #58
Changes from 4 commits
5318562
c837b4f
582cb03
8ec8d28
cd307a4
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -4,6 +4,7 @@ | |
add_executable( | ||
athenaPK | ||
main.cpp | ||
bc.cpp | ||
eos/adiabatic_glmmhd.cpp | ||
units.hpp | ||
eos/adiabatic_hydro.cpp | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,54 @@ | ||
//======================================================================================== | ||
// AthenaPK - a performance portable block structured AMR astrophysical MHD code. | ||
// Copyright (c) 2021, Athena-Parthenon Collaboration. All rights reserved. | ||
// Licensed under the 3-clause BSD License, see LICENSE file for details | ||
//======================================================================================== | ||
//! \file bc.cpp | ||
// \brief Custom boundary conditions for AthenaPK | ||
// | ||
// Computes reflecting boundary conditions using AthenaPK's cons variable pack. | ||
//======================================================================================== | ||
|
||
#include "bc.hpp" | ||
|
||
// Parthenon headers | ||
#include "main.hpp" | ||
#include "mesh/mesh.hpp" | ||
|
||
using parthenon::Real; | ||
|
||
void ReflectingInnerX3(std::shared_ptr<parthenon::MeshBlockData<Real>> &mbd, | ||
bool coarse) { | ||
std::shared_ptr<parthenon::MeshBlock> pmb = mbd->GetBlockPointer(); | ||
auto cons_pack = mbd->PackVariables(std::vector<std::string>{"cons"}, coarse); | ||
|
||
// loop over vars in cons_pack | ||
const auto nvar = cons_pack.GetDim(4); | ||
for (int n = 0; n < nvar; ++n) { | ||
bool is_normal_dir = false; | ||
if (n == IM3) { | ||
is_normal_dir = true; | ||
} | ||
parthenon::IndexRange nv{n, n}; | ||
ApplyBC<parthenon::X3DIR, BCSide::Inner, BCType::Reflect>(pmb.get(), cons_pack, nv, | ||
is_normal_dir, coarse); | ||
} | ||
} | ||
|
||
void ReflectingOuterX3(std::shared_ptr<parthenon::MeshBlockData<Real>> &mbd, | ||
bool coarse) { | ||
std::shared_ptr<parthenon::MeshBlock> pmb = mbd->GetBlockPointer(); | ||
auto cons_pack = mbd->PackVariables(std::vector<std::string>{"cons"}, coarse); | ||
|
||
// loop over vars in cons_pack | ||
const auto nvar = cons_pack.GetDim(4); | ||
for (int n = 0; n < nvar; ++n) { | ||
bool is_normal_dir = false; | ||
if (n == IM3) { | ||
is_normal_dir = true; | ||
} | ||
parthenon::IndexRange nv{n, n}; | ||
ApplyBC<parthenon::X3DIR, BCSide::Outer, BCType::Reflect>(pmb.get(), cons_pack, nv, | ||
is_normal_dir, coarse); | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,123 @@ | ||
#ifndef BC_HPP_ | ||
#define BC_HPP_ | ||
//======================================================================================== | ||
// AthenaPK - a performance portable block structured AMR astrophysical MHD code. | ||
// Copyright (c) 2021, Athena-Parthenon Collaboration. All rights reserved. | ||
// Licensed under the 3-clause BSD License, see LICENSE file for details | ||
//======================================================================================== | ||
//! \file bc.hpp | ||
// \brief Custom boundary conditions for AthenaPK | ||
// | ||
// Computes reflecting boundary conditions using AthenaPK's cons variable pack. | ||
//======================================================================================== | ||
|
||
#include "bvals/bvals.hpp" | ||
#include "mesh/meshblock.hpp" | ||
|
||
using parthenon::Real; | ||
|
||
void ReflectingInnerX3(std::shared_ptr<parthenon::MeshBlockData<Real>> &mbd, bool coarse); | ||
void ReflectingOuterX3(std::shared_ptr<parthenon::MeshBlockData<Real>> &mbd, bool coarse); | ||
|
||
// Function for checking boundary flags: is this a domain or internal bound? | ||
// | ||
inline bool IsDomainBound(parthenon::MeshBlock *pmb, parthenon::BoundaryFace face) { | ||
return !(pmb->boundary_flag[face] == parthenon::BoundaryFlag::block || | ||
pmb->boundary_flag[face] == parthenon::BoundaryFlag::periodic); | ||
} | ||
|
||
// Get zones which are inside the physical domain, i.e. set by computation or MPI halo | ||
// sync, not by problem boundary conditions. | ||
// | ||
inline auto GetPhysicalZones(parthenon::MeshBlock *pmb, parthenon::IndexShape &bounds) | ||
-> std::tuple<parthenon::IndexRange, parthenon::IndexRange, parthenon::IndexRange> { | ||
return std::tuple<parthenon::IndexRange, parthenon::IndexRange, parthenon::IndexRange>{ | ||
parthenon::IndexRange{IsDomainBound(pmb, parthenon::BoundaryFace::inner_x1) | ||
? bounds.is(parthenon::IndexDomain::interior) | ||
: bounds.is(parthenon::IndexDomain::entire), | ||
IsDomainBound(pmb, parthenon::BoundaryFace::outer_x1) | ||
? bounds.ie(parthenon::IndexDomain::interior) | ||
: bounds.ie(parthenon::IndexDomain::entire)}, | ||
parthenon::IndexRange{IsDomainBound(pmb, parthenon::BoundaryFace::inner_x2) | ||
? bounds.js(parthenon::IndexDomain::interior) | ||
: bounds.js(parthenon::IndexDomain::entire), | ||
IsDomainBound(pmb, parthenon::BoundaryFace::outer_x2) | ||
? bounds.je(parthenon::IndexDomain::interior) | ||
: bounds.je(parthenon::IndexDomain::entire)}, | ||
parthenon::IndexRange{IsDomainBound(pmb, parthenon::BoundaryFace::inner_x3) | ||
? bounds.ks(parthenon::IndexDomain::interior) | ||
: bounds.ks(parthenon::IndexDomain::entire), | ||
IsDomainBound(pmb, parthenon::BoundaryFace::outer_x3) | ||
? bounds.ke(parthenon::IndexDomain::interior) | ||
: bounds.ke(parthenon::IndexDomain::entire)}}; | ||
} | ||
Comment on lines
+22
to
+53
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Is this code actually used somewhere? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This is used in my precipitator pgen. I could move it there if you don't want to include it here. |
||
|
||
enum class BCSide { Inner, Outer }; | ||
enum class BCType { Outflow, Reflect }; | ||
|
||
template <parthenon::CoordinateDirection DIR, BCSide SIDE, BCType TYPE> | ||
void ApplyBC(parthenon::MeshBlock *pmb, parthenon::VariablePack<Real> &q, | ||
parthenon::IndexRange &nvar, const bool is_normal, const bool coarse) { | ||
// convenient shorthands | ||
constexpr bool X1 = (DIR == parthenon::X1DIR); | ||
constexpr bool X2 = (DIR == parthenon::X2DIR); | ||
constexpr bool X3 = (DIR == parthenon::X3DIR); | ||
constexpr bool INNER = (SIDE == BCSide::Inner); | ||
|
||
constexpr parthenon::BoundaryFace bface = | ||
INNER ? (X1 ? parthenon::BoundaryFace::inner_x1 | ||
: (X2 ? parthenon::BoundaryFace::inner_x2 | ||
: parthenon::BoundaryFace::inner_x3)) | ||
: (X1 ? parthenon::BoundaryFace::outer_x1 | ||
: (X2 ? parthenon::BoundaryFace::outer_x2 | ||
: parthenon::BoundaryFace::outer_x3)); | ||
|
||
// check that we are actually on a physical boundary | ||
if (!IsDomainBound(pmb, bface)) { | ||
return; | ||
} | ||
|
||
const auto &bounds = coarse ? pmb->c_cellbounds : pmb->cellbounds; | ||
|
||
const auto &range = X1 ? bounds.GetBoundsI(parthenon::IndexDomain::interior) | ||
: (X2 ? bounds.GetBoundsJ(parthenon::IndexDomain::interior) | ||
: bounds.GetBoundsK(parthenon::IndexDomain::interior)); | ||
const int ref = INNER ? range.s : range.e; | ||
|
||
std::string label = (TYPE == BCType::Reflect ? "Reflect" : "Outflow"); | ||
label += (INNER ? "Inner" : "Outer"); | ||
label += "X" + std::to_string(DIR); | ||
Comment on lines
+87
to
+89
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. No need for outflow here but maybe add "AthenaPK" to the label. |
||
|
||
constexpr parthenon::IndexDomain domain = | ||
INNER ? (X1 ? parthenon::IndexDomain::inner_x1 | ||
: (X2 ? parthenon::IndexDomain::inner_x2 | ||
: parthenon::IndexDomain::inner_x3)) | ||
: (X1 ? parthenon::IndexDomain::outer_x1 | ||
: (X2 ? parthenon::IndexDomain::outer_x2 | ||
: parthenon::IndexDomain::outer_x3)); | ||
|
||
// used for reflections | ||
const int offset = 2 * ref + (INNER ? -1 : 1); | ||
|
||
pmb->par_for_bndry( | ||
label, nvar, domain, coarse, | ||
KOKKOS_LAMBDA(const int &l, const int &k, const int &j, const int &i) { | ||
if (!q.IsAllocated(l)) return; | ||
if (TYPE == BCType::Reflect) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We can probably skip this |
||
q(l, k, j, i) = | ||
(is_normal ? -1.0 : 1.0) * | ||
q(l, X3 ? offset - k : k, X2 ? offset - j : j, X1 ? offset - i : i); | ||
} else { | ||
q(l, k, j, i) = q(l, X3 ? ref : k, X2 ? ref : j, X1 ? ref : i); | ||
} | ||
}); | ||
} | ||
|
||
template <parthenon::CoordinateDirection DIR, BCSide SIDE, BCType TYPE> | ||
void ApplyBC(parthenon::MeshBlock *pmb, parthenon::VariablePack<Real> &q, bool is_normal, | ||
bool coarse = false) { | ||
auto nvar = parthenon::IndexRange{0, q.GetDim(4) - 1}; | ||
ApplyBC<DIR, SIDE, TYPE>(pmb, q, nvar, is_normal, coarse); | ||
} | ||
|
||
#endif // BC_HPP_ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Any reason to not template this (for different dims) already now?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I can do that.