diff --git a/example/poisson/poisson_package.cpp b/example/poisson/poisson_package.cpp index 3f1376be9707..4edf7686412b 100644 --- a/example/poisson/poisson_package.cpp +++ b/example/poisson/poisson_package.cpp @@ -161,9 +161,9 @@ TaskStatus SumMass(T *u, Real *reduce_sum) { const parthenon::Coordinates_t &coords = GetCoords(pm); const int ndim = v.GetNdim(); - const Real dx = coords.Dxc(); + const Real dx = coords.Dxc(0, 0, 0); for (int i = X2DIR; i <= ndim; i++) { - const Real dy = coords.DxcFA(i); + const Real dy = coords.Dxc(i, 0, 0, 0); PARTHENON_REQUIRE_THROWS(dx == dy, "SumMass requires that DX be equal in all directions."); } @@ -232,9 +232,9 @@ TaskStatus UpdatePhi(T *u, T *du) { const parthenon::Coordinates_t &coords = GetCoords(pm); const int ndim = v.GetNdim(); - const Real dx = coords.Dxc(); + const Real dx = coords.Dxc(0, 0, 0); for (int i = X2DIR; i <= ndim; i++) { - const Real dy = coords.DxcFA(i); + const Real dy = coords.Dxc(i, 0, 0, 0); PARTHENON_REQUIRE_THROWS(dx == dy, "UpdatePhi requires that DX be equal in all directions."); } diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index 825014341aef..65c5b959e757 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -90,7 +90,18 @@ set(COMPILED_WITH ${CMAKE_CXX_COMPILER}) set(COMPILER_COMMAND "") # TODO: Put something more descriptive here set(COMPILER_FLAGS "") # TODO: Put something more descriptive here -set(COORDINATE_TYPE UniformCartesian) # TODO: Make this an option when more are available +set(PARTHENON_COORDINATES "Cartesian" CACHE STRING "Set the coordinate system from {Cartesian, Cylindrical, Spherical} with Cartesian the default") + +if (${PARTHENON_COORDINATES} STREQUAL "Cartesian") + set(COORDINATE_TYPE UniformCartesian) +elseif(${PARTHENON_COORDINATES} STREQUAL "Cylindrical") + set(COORDINATE_TYPE UniformCylindrical) +elseif(${PARTHENON_COORDINATES} STREQUAL "Spherical") + set(COORDINATE_TYPE UniformSpherical) +else() + message(FATAL_ERROR "Invalid setting for COORDINATES") +endif() +message(STATUS "COORDINATE_TYPE = " ${COORDINATE_TYPE}) configure_file(config.hpp.in generated/config.hpp @ONLY) @@ -118,6 +129,9 @@ add_library(parthenon coordinates/coordinates.hpp coordinates/uniform_cartesian.hpp + coordinates/uniform_coordinates.hpp + coordinates/uniform_cylindrical.hpp + coordinates/uniform_spherical.hpp driver/driver.cpp driver/driver.hpp diff --git a/src/coordinates/coordinates.hpp b/src/coordinates/coordinates.hpp index d1290deea425..475232b86611 100644 --- a/src/coordinates/coordinates.hpp +++ b/src/coordinates/coordinates.hpp @@ -16,11 +16,18 @@ #include "config.hpp" #include "uniform_cartesian.hpp" +#include "uniform_cylindrical.hpp" +#include "uniform_spherical.hpp" namespace parthenon { using Coordinates_t = COORDINATE_TYPE; +template +constexpr bool IsCoord() { + return std::is_same_v; +} + } // namespace parthenon #endif // COORDINATES_COORDINATES_HPP_ diff --git a/src/coordinates/uniform_cartesian.hpp b/src/coordinates/uniform_cartesian.hpp index 90e98447f0df..32f38bd6a9b5 100644 --- a/src/coordinates/uniform_cartesian.hpp +++ b/src/coordinates/uniform_cartesian.hpp @@ -13,320 +13,20 @@ #ifndef COORDINATES_UNIFORM_CARTESIAN_HPP_ #define COORDINATES_UNIFORM_CARTESIAN_HPP_ -#include -#include - -#include "basic_types.hpp" -#include "defs.hpp" -#include "globals.hpp" - -#include +#include "uniform_coordinates.hpp" namespace parthenon { -class UniformCartesian { +class UniformCartesian : public UniformCoordinates { public: UniformCartesian() = default; - UniformCartesian(const RegionSize &rs, ParameterInput *pin) { - for (auto &dir : {X1DIR, X2DIR, X3DIR}) { - dx_[dir - 1] = (rs.xmax(dir) - rs.xmin(dir)) / rs.nx(dir); - istart_[dir - 1] = (!rs.symmetry(dir) ? Globals::nghost : 0); - xmin_[dir - 1] = rs.xmin(dir) - istart_[dir - 1] * dx_[dir - 1]; - } - area_[0] = dx_[1] * dx_[2]; - area_[1] = dx_[0] * dx_[2]; - area_[2] = dx_[0] * dx_[1]; - cell_volume_ = dx_[0] * dx_[1] * dx_[2]; - } + UniformCartesian(const RegionSize &rs, ParameterInput *pin) + : UniformCoordinates(rs, pin) {} UniformCartesian(const UniformCartesian &src, int coarsen) - : istart_(src.GetStartIndex()) { - dx_ = src.Dx_(); - xmin_ = src.GetXmin(); - xmin_[0] += istart_[0] * dx_[0] * (1 - coarsen); - xmin_[1] += istart_[1] * dx_[1] * (1 - coarsen); - xmin_[2] += istart_[2] * dx_[2] * (1 - coarsen); - dx_[0] *= coarsen; - dx_[1] *= (istart_[1] > 0 ? coarsen : 1); - dx_[2] *= (istart_[2] > 0 ? coarsen : 1); - area_[0] = dx_[1] * dx_[2]; - area_[1] = dx_[0] * dx_[2]; - area_[2] = dx_[0] * dx_[1]; - cell_volume_ = dx_[0] * dx_[1] * dx_[2]; - } - - //---------------------------------------- - // Dxc: Distance between cell centers - //---------------------------------------- - template - KOKKOS_FORCEINLINE_FUNCTION Real Dxc() const { - assert(dir > 0 && dir < 4); - return dx_[dir - 1]; - } - template - KOKKOS_FORCEINLINE_FUNCTION Real Dxc(Args... args) const { - assert(dir > 0 && dir < 4); - return dx_[dir - 1]; - } - template - KOKKOS_FORCEINLINE_FUNCTION Real DxcFA(const int dir, Args... args) const { - assert(dir > 0 && dir < 4); - return dx_[dir - 1]; - } - - //---------------------------------------- - // Dxf: Distance between cell faces - //---------------------------------------- - template - KOKKOS_FORCEINLINE_FUNCTION Real Dxf(Args... args) const { - assert(dir > 0 && dir < 4 && face > 0 && face < 4); - return dx_[face - 1]; - } - template - KOKKOS_FORCEINLINE_FUNCTION Real Dxf(Args... args) const { - assert(dir > 0 && dir < 4); - return dx_[dir - 1]; - } - - //---------------------------------------- - // Xf: Positions at cell centers - //---------------------------------------- - template - KOKKOS_FORCEINLINE_FUNCTION Real Xc(const int idx) const { - assert(dir > 0 && dir < 4); - return xmin_[dir - 1] + (idx + 0.5) * dx_[dir - 1]; - } - template - KOKKOS_FORCEINLINE_FUNCTION Real Xc(const int k, const int j, const int i) const { - assert(dir > 0 && dir < 4); - switch (dir) { - case 1: - return Xc(i); - case 2: - return Xc(j); - case 3: - return Xc(k); - default: - PARTHENON_FAIL("Unknown dir"); - return 0; // To appease compiler - } - } - - //---------------------------------------- - // Xf: Positions on Faces - //---------------------------------------- - template - KOKKOS_FORCEINLINE_FUNCTION Real Xf(const int idx) const { - assert(dir > 0 && dir < 4 && face > 0 && face < 4); - // Return position in direction "dir" along index "idx" on face "face" - if constexpr (dir == face) { - return xmin_[dir - 1] + idx * dx_[dir - 1]; - } else { - return Xc(idx); - } - } - template - KOKKOS_FORCEINLINE_FUNCTION Real Xf(const int k, const int j, const int i) const { - assert(dir > 0 && dir < 4); - switch (dir) { - case 1: - return Xf(i); - case 2: - return Xf(j); - case 3: - return Xf(k); - default: - PARTHENON_FAIL("Unknown dir"); - return 0; // To appease compiler - } - } - - template - KOKKOS_FORCEINLINE_FUNCTION Real Xf(const int idx) const { - assert(dir > 0 && dir < 4); - // Return position in direction "dir" along index "idx" on face "dir" - return xmin_[dir - 1] + idx * dx_[dir - 1]; - } - template - KOKKOS_FORCEINLINE_FUNCTION Real Xf(const int k, const int j, const int i) const { - assert(dir > 0 && dir < 4); - switch (dir) { - case 1: - return Xf(i); - case 2: - return Xf(j); - case 3: - return Xf(k); - default: - PARTHENON_FAIL("Unknown dir"); - return 0; // To appease compiler - } - } - - template - KOKKOS_FORCEINLINE_FUNCTION Real X(const int idx) const { - if constexpr (dir == X1DIR && TopologicalOffsetI(el)) { - return xmin_[dir - 1] + idx * dx_[dir - 1]; // idx - 1/2 - } else if constexpr (dir == X2DIR && TopologicalOffsetJ(el)) { - return xmin_[dir - 1] + idx * dx_[dir - 1]; // idx - 1/2 - } else if constexpr (dir == X3DIR && TopologicalOffsetK(el)) { - return xmin_[dir - 1] + idx * dx_[dir - 1]; // idx - 1/2 - } else { - return xmin_[dir - 1] + (idx + 0.5) * dx_[dir - 1]; // idx - } - return 0; // This should never be reached, but w/o it some compilers generate warnings - } - - template - KOKKOS_FORCEINLINE_FUNCTION Real X(const int k, const int j, const int i) const { - assert(dir > 0 && dir < 4); - switch (dir) { - case 1: - return X(i); - case 2: - return X(j); - case 3: - return X(k); - default: - PARTHENON_FAIL("Unknown dir"); - return 0; // To appease compiler - } - } - - //---------------------------------------- - // CellWidth: Width of cells at cell centers - //---------------------------------------- - template - KOKKOS_FORCEINLINE_FUNCTION Real CellWidth(Args... args) const { - assert(dir > 0 && dir < 4); - return dx_[dir - 1]; - } - template - KOKKOS_FORCEINLINE_FUNCTION Real CellWidthFA(const int dir, Args... args) const { - assert(dir > 0 && dir < 4); - return dx_[dir - 1]; - } + : UniformCoordinates(src, coarsen) {} - //---------------------------------------- - // EdgeLength: Length of cell edges - //---------------------------------------- - template - KOKKOS_FORCEINLINE_FUNCTION Real EdgeLength(Args... args) const { - assert(dir > 0 && dir < 4); - return CellWidth(); - } - template - KOKKOS_FORCEINLINE_FUNCTION Real EdgeLengthFA(const int dir, Args... args) const { - return CellWidthFA(dir); - } - - //---------------------------------------- - // FaceArea: Area of cell areas - //---------------------------------------- - template - KOKKOS_FORCEINLINE_FUNCTION Real FaceArea(Args... args) const { - assert(dir > 0 && dir < 4); - return area_[dir - 1]; - } - template - KOKKOS_FORCEINLINE_FUNCTION Real FaceAreaFA(const int dir, Args... args) const { - assert(dir > 0 && dir < 4); - return area_[dir - 1]; - } - - //---------------------------------------- - // CellVolume - //---------------------------------------- - template - KOKKOS_FORCEINLINE_FUNCTION Real CellVolume(Args... args) const { - return cell_volume_; - } - - //---------------------------------------- - // Generalized volume - //---------------------------------------- - template - KOKKOS_FORCEINLINE_FUNCTION Real Volume(Args... args) const { - using TE = TopologicalElement; - if constexpr (el == TE::CC) { - return cell_volume_; - } else if constexpr (el == TE::F1) { - return area_[X1DIR - 1]; - } else if constexpr (el == TE::F2) { - return area_[X2DIR - 1]; - } else if constexpr (el == TE::F3) { - return area_[X3DIR - 1]; - } else if constexpr (el == TE::E1) { - return dx_[X1DIR - 1]; - } else if constexpr (el == TE::E2) { - return dx_[X2DIR - 1]; - } else if constexpr (el == TE::E3) { - return dx_[X3DIR - 1]; - } else if constexpr (el == TE::NN) { - return 1.0; - } - PARTHENON_FAIL("If you reach this point, someone has added a new value to the the " - "TopologicalElement enum."); - return 0.0; - } - - template - KOKKOS_FORCEINLINE_FUNCTION Real Volume(CellLevel cl, TopologicalElement el, - Args... args) const { - using TE = TopologicalElement; - if (cl == CellLevel::same) { - if (el == TE::CC) { - return cell_volume_; - } else if (el == TE::F1) { - return area_[X1DIR - 1]; - } else if (el == TE::F2) { - return area_[X2DIR - 1]; - } else if (el == TE::F3) { - return area_[X3DIR - 1]; - } else if (el == TE::E1) { - return dx_[X1DIR - 1]; - } else if (el == TE::E2) { - return dx_[X2DIR - 1]; - } else if (el == TE::E3) { - return dx_[X3DIR - 1]; - } else if (el == TE::NN) { - return 1.0; - } - } else if (cl == CellLevel::fine) { - if (el == TE::CC) { - return cell_volume_ / 8.0; - } else if (el == TE::F1) { - return area_[X1DIR - 1] / 4.0; - } else if (el == TE::F2) { - return area_[X2DIR - 1] / 4.0; - } else if (el == TE::F3) { - return area_[X3DIR - 1] / 4.0; - } else if (el == TE::E1) { - return dx_[X1DIR - 1] / 2.0; - } else if (el == TE::E2) { - return dx_[X2DIR - 1] / 2.0; - } else if (el == TE::E3) { - return dx_[X3DIR - 1] / 2.0; - } else if (el == TE::NN) { - return 1.0; - } - } - PARTHENON_FAIL("If you reach this point, someone has added a new value to the the " - "TopologicalElement enum."); - return 0.0; - } - - const std::array &GetXmin() const { return xmin_; } - const std::array &GetStartIndex() const { return istart_; } - const char *Name() const { return name_; } - static const char *StaticName() { return name_; } - - private: - std::array istart_; - std::array xmin_, dx_, area_; - Real cell_volume_; constexpr static const char *name_ = "UniformCartesian"; - - const std::array &Dx_() const { return dx_; } + private: }; } // namespace parthenon diff --git a/src/coordinates/uniform_coordinates.hpp b/src/coordinates/uniform_coordinates.hpp new file mode 100644 index 000000000000..93dd6a1f5ebe --- /dev/null +++ b/src/coordinates/uniform_coordinates.hpp @@ -0,0 +1,343 @@ +//======================================================================================== +// (C) (or copyright) 2024. Triad National Security, LLC. All rights reserved. +// +// This program was produced under U.S. Government contract 89233218CNA000001 for Los +// Alamos National Laboratory (LANL), which is operated by Triad National Security, LLC +// for the U.S. Department of Energy/National Nuclear Security Administration. All rights +// in the program are reserved by Triad National Security, LLC, and the U.S. Department +// of Energy/National Nuclear Security Administration. The Government is granted for +// itself and others acting on its behalf a nonexclusive, paid-up, irrevocable worldwide +// license in this material to reproduce, prepare derivative works, distribute copies to +// the public, perform publicly and display publicly, and to permit others to do so. +//======================================================================================== +#ifndef COORDINATES_UNIFORM_COORDINATES_HPP_ +#define COORDINATES_UNIFORM_COORDINATES_HPP_ + +#include +#include + +#include "basic_types.hpp" +#include "defs.hpp" +#include "globals.hpp" + +#include + +namespace parthenon { + +template +class UniformCoordinates { + public: + UniformCoordinates() = default; + UniformCoordinates(const RegionSize &rs, ParameterInput *pin) { + for (auto &dir : {X1DIR, X2DIR, X3DIR}) { + dx_[dir - 1] = (rs.xmax(dir) - rs.xmin(dir)) / rs.nx(dir); + istart_[dir - 1] = (!rs.symmetry(dir) ? Globals::nghost : 0); + xmin_[dir - 1] = rs.xmin(dir) - istart_[dir - 1] * dx_[dir - 1]; + } + area_[0] = dx_[1] * dx_[2]; + area_[1] = dx_[0] * dx_[2]; + area_[2] = dx_[0] * dx_[1]; + cell_volume_ = dx_[0] * dx_[1] * dx_[2]; + } + UniformCoordinates(const UniformCoordinates &src, int coarsen) + : istart_(src.GetStartIndex()) { + dx_ = src.Dx_(); + xmin_ = src.GetXmin(); + xmin_[0] += istart_[0] * dx_[0] * (1 - coarsen); + xmin_[1] += istart_[1] * dx_[1] * (1 - coarsen); + xmin_[2] += istart_[2] * dx_[2] * (1 - coarsen); + dx_[0] *= coarsen; + dx_[1] *= (istart_[1] > 0 ? coarsen : 1); + dx_[2] *= (istart_[2] > 0 ? coarsen : 1); + area_[0] = dx_[1] * dx_[2]; + area_[1] = dx_[0] * dx_[2]; + area_[2] = dx_[0] * dx_[1]; + cell_volume_ = dx_[0] * dx_[1] * dx_[2]; + } + //---------------------------------------- + // Dxc: Distance between cell centers + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real Dx() const { + static_assert(dir > 0 && dir < 4); + return dx_[dir - 1]; + } + KOKKOS_FORCEINLINE_FUNCTION Real Dx(const int dir) const { + assert(dir > 0 && dir < 4); + return dx_[dir - 1]; + } + + template + KOKKOS_FORCEINLINE_FUNCTION Real Dxc(const int idx) const { + static_assert(dir > 0 && dir < 4); + return dx_[dir - 1]; + } + template + KOKKOS_FORCEINLINE_FUNCTION Real Dxc(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR) return static_cast(this)->template Dxc(i); + else if constexpr (dir == X2DIR) return static_cast(this)->template Dxc(j); + else if constexpr (dir == X3DIR) return static_cast(this)->template Dxc(k); + PARTHENON_FAIL("Unknown dir."); + return 0.0; + } + KOKKOS_FORCEINLINE_FUNCTION Real Dxc(const int dir, const int k, const int j, const int i) const { + assert(dir > 0 && dir < 4); + if (dir == X1DIR) return static_cast(this)->template Dxc(k, j, i); + else if (dir == X2DIR) return static_cast(this)->template Dxc(k, j, i); + else if (dir == X3DIR) return static_cast(this)->template Dxc(k, j, i); + PARTHENON_FAIL("Unknown dir."); + return 0.0; + } + + //---------------------------------------- + // Dxf: Distance between cell faces + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real Dxf(const int idx) const { + static_assert(dir > 0 && dir < 4); + return dx_[dir - 1]; + } + template + KOKKOS_FORCEINLINE_FUNCTION Real Dxf(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + return dx_[dir - 1]; + } + + //---------------------------------------- + // Xc: Positions at cell centroids + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real Xc(const int idx) const { + static_assert(dir > 0 && dir < 4); + return xmin_[dir - 1] + (idx + 0.5) * dx_[dir - 1]; + } + template + KOKKOS_FORCEINLINE_FUNCTION Real Xc(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR) return static_cast(this)->template Xc(i); + else if constexpr (dir == X2DIR) return static_cast(this)->template Xc(j); + else if constexpr (dir == X3DIR) return static_cast(this)->template Xc(k); + return 0; // To appease compiler + } + + //---------------------------------------- + // Xf: Positions on Faces + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real Xf(const int idx) const { + static_assert(dir > 0 && dir < 4); + return xmin_[dir - 1] + idx * dx_[dir - 1]; + } + template + KOKKOS_FORCEINLINE_FUNCTION Real Xf(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + static_assert(face > 0 && face < 4); + if constexpr (dir == X1DIR && dir == face) { + return Xf(i); + } else if constexpr (dir == X1DIR) { + return static_cast(this)->template Xc(k, j, i); + } else if constexpr (dir == X2DIR && dir == face) { + return Xf(j); + } else if constexpr (dir == X2DIR) { + return static_cast(this)->template Xc(k, j, i); + } else if constexpr (dir == X3DIR && dir == face) { + return Xf(k); + } else if constexpr (dir == X3DIR) { + return static_cast(this)->template Xc(k, j, i); + } + return 0; // To appease compiler + } + + template + KOKKOS_FORCEINLINE_FUNCTION Real Xf(const int k, const int j, const int i) const { + return Xf(k, j, i); + } + + template + KOKKOS_FORCEINLINE_FUNCTION Real X(const int idx) const { + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR && TopologicalOffsetI(el)) { + return xmin_[dir - 1] + idx * dx_[dir - 1]; // idx - 1/2 + } else if constexpr (dir == X2DIR && TopologicalOffsetJ(el)) { + return xmin_[dir - 1] + idx * dx_[dir - 1]; // idx - 1/2 + } else if constexpr (dir == X3DIR && TopologicalOffsetK(el)) { + return xmin_[dir - 1] + idx * dx_[dir - 1]; // idx - 1/2 + } else { + return static_cast(this)->template Xc(idx); // idx + } + return 0; // This should never be reached, but w/o it some compilers generate warnings + } + template + KOKKOS_FORCEINLINE_FUNCTION Real X(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR) return X(i); + else if constexpr (dir == X2DIR) return X(j); + else if constexpr (dir == X3DIR) return X(k); + return 0.0; + } + + template + KOKKOS_FORCEINLINE_FUNCTION + Real Scale(const int k, const int j, const int i) const { + if constexpr (dir > 0 && dir < 4) return 1.0; + PARTHENON_FAIL("Unknown dir"); + return 0.0; + } + + template + KOKKOS_FORCEINLINE_FUNCTION + Real Scale(const int dir, const int k, const int j, const int i) const { + assert(dir > 0 && dir < 4); + if (dir == X1DIR) return static_cast(this)->template Scale(k, j, i); + else if (dir == X2DIR) return static_cast(this)->template Scale(k, j, i); + else if (dir == X3DIR) return static_cast(this)->template Scale(k, j, i); + return 0.0; + } + + //---------------------------------------- + // CellWidth: Width of cells at cell centers + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real CellWidth(const int k, const int j, const int i) const { + assert(dir > 0 && dir < 4); + return dx_[dir - 1]; + } + KOKKOS_FORCEINLINE_FUNCTION Real CellWidth(const int dir, const int k, const int j, const int i) const { + assert(dir > 0 && dir < 4); + if (dir == X1DIR) return static_cast(this)->template CellWidth(k, j, i); + else if (dir == X2DIR) return static_cast(this)->template CellWidth(k, j, i); + else if (dir == X3DIR) return static_cast(this)->template CellWidth(k, j, i); + return 0.0; + } + + //---------------------------------------- + // EdgeLength: Length of cell edges + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real EdgeLength(const int k, const int j, const int i) const { + assert(dir > 0 && dir < 4); + return CellWidth(k, j, i); + } + KOKKOS_FORCEINLINE_FUNCTION Real EdgeLength(const int dir, const int k, const int j, const int i) const { + return CellWidth(dir, k, j, i); + } + + //---------------------------------------- + // FaceArea: Area of cell areas + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real FaceArea(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + return area_[dir - 1]; + } + KOKKOS_FORCEINLINE_FUNCTION Real FaceArea(const int dir, const int k, const int j, const int i) const { + assert(dir > 0 && dir < 4); + switch(dir) { + case X1DIR: + return static_cast(this)->template FaceArea(k, j, i); + case X2DIR: + return static_cast(this)->template FaceArea(k, j, i); + case X3DIR: + return static_cast(this)->template FaceArea(k, j, i); + default: + PARTHENON_FAIL("Unknown dir"); + return 0; // To appease compiler + } + } + + //---------------------------------------- + // CellVolume + //---------------------------------------- + KOKKOS_FORCEINLINE_FUNCTION Real CellVolume(const int k, const int j, const int i) const { + return cell_volume_; + } + + //---------------------------------------- + // Generalized volume + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real Volume(const int k, const int j, const int i) const { + using TE = TopologicalElement; + if constexpr (el == TE::CC) { + return static_cast(this)->CellVolume(k, j, i); + } else if constexpr (el == TE::F1) { + return static_cast(this)->template FaceArea(k, j, i); + } else if constexpr (el == TE::F2) { + return static_cast(this)->template FaceArea(k, j, i); + } else if constexpr (el == TE::F3) { + return static_cast(this)->template FaceArea(k, j, i); + } else if constexpr (el == TE::E1) { + return static_cast(this)->template EdgeLength(k, j, i); + } else if constexpr (el == TE::E2) { + return static_cast(this)->template EdgeLength(k, j, i); + } else if constexpr (el == TE::E3) { + return static_cast(this)->template EdgeLength(k, j, i); + } else if constexpr (el == TE::NN) { + return 1.0; + } + PARTHENON_FAIL("If you reach this point, someone has added a new value to the the " + "TopologicalElement enum."); + return 0.0; + } + + template + KOKKOS_FORCEINLINE_FUNCTION Real Volume(CellLevel cl, TopologicalElement el, + const int k, const int j, const int i) const { + using TE = TopologicalElement; + if (cl == CellLevel::same) { + if (el == TE::CC) { + return cell_volume_; + } else if (el == TE::F1) { + return area_[X1DIR - 1]; + } else if (el == TE::F2) { + return area_[X2DIR - 1]; + } else if (el == TE::F3) { + return area_[X3DIR - 1]; + } else if (el == TE::E1) { + return dx_[X1DIR - 1]; + } else if (el == TE::E2) { + return dx_[X2DIR - 1]; + } else if (el == TE::E3) { + return dx_[X3DIR - 1]; + } else if (el == TE::NN) { + return 1.0; + } + } else if (cl == CellLevel::fine) { + if (el == TE::CC) { + return cell_volume_ / 8.0; + } else if (el == TE::F1) { + return area_[X1DIR - 1] / 4.0; + } else if (el == TE::F2) { + return area_[X2DIR - 1] / 4.0; + } else if (el == TE::F3) { + return area_[X3DIR - 1] / 4.0; + } else if (el == TE::E1) { + return dx_[X1DIR - 1] / 2.0; + } else if (el == TE::E2) { + return dx_[X2DIR - 1] / 2.0; + } else if (el == TE::E3) { + return dx_[X3DIR - 1] / 2.0; + } else if (el == TE::NN) { + return 1.0; + } + } + PARTHENON_FAIL("If you reach this point, someone has added a new value to the the " + "TopologicalElement enum."); + return 0.0; + } + + const std::array &GetXmin() const { return xmin_; } + const std::array &GetStartIndex() const { return istart_; } + const char *Name() const { return System::name_; } + private: + std::array istart_; + std::array xmin_, dx_, area_; + Real cell_volume_; + + const std::array &Dx_() const { return dx_; } +}; + +} // namespace parthenon + +#endif // COORDINATES_UNIFORM_COORDINATES_HPP_ diff --git a/src/coordinates/uniform_cylindrical.hpp b/src/coordinates/uniform_cylindrical.hpp new file mode 100644 index 000000000000..877507058c8f --- /dev/null +++ b/src/coordinates/uniform_cylindrical.hpp @@ -0,0 +1,152 @@ +//======================================================================================== +// (C) (or copyright) 2023-2024. Triad National Security, LLC. All rights reserved. +// +// This program was produced under U.S. Government contract 89233218CNA000001 for Los +// Alamos National Laboratory (LANL), which is operated by Triad National Security, LLC +// for the U.S. Department of Energy/National Nuclear Security Administration. All rights +// in the program are reserved by Triad National Security, LLC, and the U.S. Department +// of Energy/National Nuclear Security Administration. The Government is granted for +// itself and others acting on its behalf a nonexclusive, paid-up, irrevocable worldwide +// license in this material to reproduce, prepare derivative works, distribute copies to +// the public, perform publicly and display publicly, and to permit others to do so. +//======================================================================================== +#ifndef COORDINATES_UNIFORM_CYLINDRICAL_HPP_ +#define COORDINATES_UNIFORM_CYLINDRICAL_HPP_ + +#include "uniform_coordinates.hpp" + +namespace parthenon { + +class UniformCylindrical : public UniformCoordinates { + using base_t = UniformCoordinates; + public: + using base_t::Dxc; + using base_t::Xc; + using base_t::Scale; + using base_t::CellWidth; + using base_t::Volume; + UniformCylindrical() = default; + UniformCylindrical(const RegionSize &rs, ParameterInput *pin) + : UniformCoordinates(rs, pin) {} + UniformCylindrical(const UniformCylindrical &src, int coarsen) + : UniformCoordinates(src, coarsen) {} + constexpr static const char *name_ = "UniformCylindrical"; + + //---------------------------------------- + // Dxc: Distance between cell centers + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real Dxc(const int idx) const { + static_assert(dir > 0 && dir < 4); + return Xc(idx) - Xc(idx-1); + } + + //---------------------------------------- + // Xc: Positions at cell centroids + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real Xc(const int idx) const { + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR) { + const Real r0 = Xf(idx); + const Real r0sq = r0 * r0; + const Real r1 = Xf(idx + 1); + const Real r1sq = r1 * r1; + return (2.0 / 3.0) * (r1sq * r1 - r0sq * r0) / (r1sq - r0sq); + } + return base_t::Xc(idx); + } + + template + KOKKOS_FORCEINLINE_FUNCTION + Real Scale(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + using TE = TopologicalElement; + if constexpr (dir == X1DIR || dir == X2DIR) return 1.0; + // phi direction scale factor is r + return X(k, j, i); + } + + //---------------------------------------- + // CellWidth: width of cell through the centroid + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real CellWidth(const int k, const int j, const int i) const { + using TE = TopologicalElement; + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR || dir == X2DIR) return Dx(); + // phi direction = r * dphi + return Xc(i) * Dx(); + } + + //---------------------------------------- + // EdgeLength: Length of cell edges + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real EdgeLength(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR || dir == X2DIR) { + // radial and z directions are trivial + return Dx(); + } + // phi direction + return Xf(k, j, i) * Dx(); + } + KOKKOS_FORCEINLINE_FUNCTION Real EdgeLength(const int dir, const int k, const int j, const int i) const { + assert(dir > 0 && dir < 4); + if (dir == X1DIR) return EdgeLength(k, j, i); + else if (dir == X2DIR) return EdgeLength(k, j, i); + return EdgeLength(k, j, i); + } + + //---------------------------------------- + // FaceArea: Area of cell areas + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real FaceArea(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR) { + return Xf(k, j, i) * Dx() * Dx(); + } else if constexpr (dir == X2DIR) { + Real r0 = Xf(k, j, i); + Real r1 = Xf(k, j, i+1); + return 0.5 * (r1*r1 - r0*r0) * Dx(); + } + Real r0 = Xf(k, j, i); + Real r1 = Xf(k, j, i+1); + return 0.5 * (r1*r1 - r0*r0) * Dx(); + } + + //---------------------------------------- + // CellVolume + //---------------------------------------- + KOKKOS_FORCEINLINE_FUNCTION Real CellVolume(const int k, const int j, const int i) const { + return FaceArea(k, j, i) * Dx(); + } + + KOKKOS_FORCEINLINE_FUNCTION + Real Volume(CellLevel cl, TopologicalElement el, const int k, const int j, const int i) { + using TE = TopologicalElement; + if (cl == CellLevel::same) { + if (el == TE::CC) return CellVolume(k, j, i); + else if (el == TE::F1) return FaceArea(k, j, i); + else if (el == TE::F2) return FaceArea(k, j, i); + else if (el == TE::F3) return FaceArea(k, j, i); + else if (el == TE::E1) return EdgeLength(k, j, i); + else if (el == TE::E2) return EdgeLength(k, j, i); + else if (el == TE::E3) return EdgeLength(k, j, i); + else if (el == TE::NN) return 1.0; + } else { + PARTHENON_FAIL("Have not yet implemented fine fields for UniformCylindrical coordinates."); + } + PARTHENON_FAIL("If you reach this point, someone has added a new value to the the " + "TopologicalElement enum."); + return 0.0; + } + + private: +}; + +} // namespace parthenon + +#endif // COORDINATES_UNIFORM_CYLINDRICAL_HPP_ diff --git a/src/coordinates/uniform_spherical.hpp b/src/coordinates/uniform_spherical.hpp new file mode 100644 index 000000000000..6f8cae6dc938 --- /dev/null +++ b/src/coordinates/uniform_spherical.hpp @@ -0,0 +1,174 @@ +//======================================================================================== +// (C) (or copyright) 2023-2024. Triad National Security, LLC. All rights reserved. +// +// This program was produced under U.S. Government contract 89233218CNA000001 for Los +// Alamos National Laboratory (LANL), which is operated by Triad National Security, LLC +// for the U.S. Department of Energy/National Nuclear Security Administration. All rights +// in the program are reserved by Triad National Security, LLC, and the U.S. Department +// of Energy/National Nuclear Security Administration. The Government is granted for +// itself and others acting on its behalf a nonexclusive, paid-up, irrevocable worldwide +// license in this material to reproduce, prepare derivative works, distribute copies to +// the public, perform publicly and display publicly, and to permit others to do so. +//======================================================================================== +#ifndef COORDINATES_UNIFORM_SPHERICAL_HPP_ +#define COORDINATES_UNIFORM_SPHERICAL_HPP_ + +#include "uniform_coordinates.hpp" + +namespace parthenon { + +class UniformSpherical : public UniformCoordinates { + using base_t = UniformCoordinates; + public: + using base_t::Dxc; + using base_t::Xc; + using base_t::Scale; + using base_t::CellWidth; + using base_t::Volume; + UniformSpherical() = default; + UniformSpherical(const RegionSize &rs, ParameterInput *pin) + : UniformCoordinates(rs, pin) {} + UniformSpherical(const UniformSpherical &src, int coarsen) + : UniformCoordinates(src, coarsen) {} + constexpr static const char *name_ = "UniformSpherical"; + + //---------------------------------------- + // Dxc: Distance between cell centers + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real Dxc(const int idx) const { + static_assert(dir > 0 && dir < 4); + return Xc(idx) - Xc(idx-1); + } + + //---------------------------------------- + // Xc: Positions at cell centroids + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real Xc(const int idx) const { + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR) { + const Real r0 = Xf(idx); + const Real r0sq = r0 * r0; + const Real r1 = Xf(idx + 1); + const Real r1sq = r1 * r1; + return 0.75 * (r1sq * r1sq - r0sq * r0sq) / (r1sq * r1 - r0sq * r0); + } else if constexpr (dir == X2DIR) { + const Real th0 = Xf(idx); + const Real sth0 = std::sin(th0); + const Real cth0 = std::cos(th0); + const Real th1 = Xf(idx + 1); + const Real sth1 = std::sin(th1); + const Real cth1 = std::cos(th1); + return (th0 * cth0 - th1 * cth1 - sth0 + sth1) / (cth0 - cth1); + } + return base_t::Xc(idx); + } + + template + KOKKOS_FORCEINLINE_FUNCTION + Real Scale(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + using TE = TopologicalElement; + if constexpr (dir == X1DIR) return 1.0; + else { + const Real r = X(k, j, i); + if constexpr (dir == X2DIR) return r; + else { + const Real th = X(k, j, i); + return r * std::sin(th); + } + } + return 0.0; + } + + //---------------------------------------- + // CellWidth: width of cell through the centroid + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real CellWidth(const int k, const int j, const int i) const { + using TE = TopologicalElement; + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR) return Dx(); + else if constexpr (dir == X2DIR) return Xc(i) * Dx(); + else if constexpr (dir == X3DIR) return Xc(i) * std::sin(Xc(j)) * Dx(); + return 0.0; + } + + //---------------------------------------- + // EdgeLength: Length of cell edges + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real EdgeLength(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR) { + // radial direction is trivial + return Dx(); + } else if constexpr (dir == X2DIR) { + // theta direction + return Xf(k, j, i) * Dx(); + } + // phi direction + return Xf(k, j, i) * std::sin(Xf(k, j, i)) * Dx(); + } + KOKKOS_FORCEINLINE_FUNCTION Real EdgeLength(const int dir, const int k, const int j, const int i) const { + assert(dir > 0 && dir < 4); + if (dir == X1DIR) return EdgeLength(k, j, i); + else if (dir == X2DIR) return EdgeLength(k, j, i); + return EdgeLength(k, j, i); + } + + //---------------------------------------- + // FaceArea: Area of cell areas + //---------------------------------------- + template + KOKKOS_FORCEINLINE_FUNCTION Real FaceArea(const int k, const int j, const int i) const { + static_assert(dir > 0 && dir < 4); + if constexpr (dir == X1DIR) { + Real dOmega = Dx() * (std::cos(Xf(k, j, i)) - std::cos(Xf(k, j+1, i))); + Real r = Xf(k, j, i); + return r * r * dOmega; + } else if constexpr (dir == X2DIR) { + Real r0 = Xf(k, j, i); + Real r1 = Xf(k, j, i+1); + return (r1*r1*r1 - r0*r0*r0) * std::sin(Xf(k, j, i)) * Dx() / 3.0; + } + Real r0 = Xf(k, j, i); + Real r1 = Xf(k, j, i+1); + Real dcth = std::cos(Xf(k, j, i)) - std::cos(Xf(k, j+1, i)); + return (r1*r1*r1 - r0*r0*r0) * dcth / 3.0; + } + + //---------------------------------------- + // CellVolume + //---------------------------------------- + KOKKOS_FORCEINLINE_FUNCTION Real CellVolume(const int k, const int j, const int i) const { + return FaceArea(k, j, i) * Dx(); + } + + KOKKOS_FORCEINLINE_FUNCTION + Real Volume(CellLevel cl, TopologicalElement el, const int k, const int j, const int i) { + using TE = TopologicalElement; + if (cl == CellLevel::same) { + if (el == TE::CC) return CellVolume(k, j, i); + else if (el == TE::F1) return FaceArea(k, j, i); + else if (el == TE::F2) return FaceArea(k, j, i); + else if (el == TE::F3) return FaceArea(k, j, i); + else if (el == TE::E1) return EdgeLength(k, j, i); + else if (el == TE::E2) return EdgeLength(k, j, i); + else if (el == TE::E3) return EdgeLength(k, j, i); + else if (el == TE::NN) return 1.0; + } else { + PARTHENON_FAIL("Have not yet implemented fine fields for UniformSpherical coordinates."); + } + PARTHENON_FAIL("If you reach this point, someone has added a new value to the the " + "TopologicalElement enum."); + return 0.0; + } + + private: +}; + +} // namespace parthenon + +#endif // COORDINATES_UNIFORM_SPHERICAL_HPP_ diff --git a/src/interface/swarm_device_context.hpp b/src/interface/swarm_device_context.hpp index b16daf0d0cdf..e438dcf53d02 100644 --- a/src/interface/swarm_device_context.hpp +++ b/src/interface/swarm_device_context.hpp @@ -94,14 +94,14 @@ class SwarmDeviceContext { KOKKOS_INLINE_FUNCTION void Xtoijk(const Real &x, const Real &y, const Real &z, int &i, int &j, int &k) const { i = static_cast( - std::floor((x - x_min_) / coords_.Dxc())) + + std::floor((x - x_min_) / coords_.Dx())) + ib_s_; j = (ndim_ > 1) ? static_cast(std::floor( - (y - y_min_) / coords_.Dxc())) + + (y - y_min_) / coords_.Dx())) + jb_s_ : jb_s_; k = (ndim_ > 2) ? static_cast(std::floor( - (z - z_min_) / coords_.Dxc())) + + (z - z_min_) / coords_.Dx())) + kb_s_ : kb_s_; } diff --git a/src/prolong_restrict/pr_ops.hpp b/src/prolong_restrict/pr_ops.hpp index f074f6a5d02b..ba918185feb2 100644 --- a/src/prolong_restrict/pr_ops.hpp +++ b/src/prolong_restrict/pr_ops.hpp @@ -453,9 +453,9 @@ struct ProlongateInternalTothAndRoe { const int dir1 = element_idx + 1; const int dir2 = (element_idx + 1) % 3 + 1; const int dir3 = (element_idx + 2) % 3 + 1; - const auto dx2 = std::pow(coarse_coords.DxcFA(dir1, k, j, i), 2); - const auto dy2 = std::pow(coarse_coords.DxcFA(dir2, k, j, i), 2); - const auto dz2 = std::pow(coarse_coords.DxcFA(dir3, k, j, i), 2); + const auto dx2 = std::pow(coarse_coords.Dxc(dir1, k, j, i), 2); + const auto dy2 = std::pow(coarse_coords.Dxc(dir2, k, j, i), 2); + const auto dz2 = std::pow(coarse_coords.Dxc(dir3, k, j, i), 2); Vxyz *= 0.125 * dz2 / (dx2 + dz2); Wxyz *= 0.125 * dy2 / (dx2 + dy2); diff --git a/src/utils/index_split.hpp b/src/utils/index_split.hpp index e9a5f4b441bd..4767bcff8aee 100644 --- a/src/utils/index_split.hpp +++ b/src/utils/index_split.hpp @@ -65,6 +65,16 @@ class IndexSplit { IndexRange GetInnerBounds(const IndexRange &jb, const IndexRange &ib) const { return {ib.s, (ibe_entire_ + 1) * (jb.e - jb.s + 1) - (ibe_entire_ - ib.e) - 1}; } + + KOKKOS_FORCEINLINE_FUNCTION + int get_i(const int idx) const { + return idx % (ibe_entire_ + 1); + } + KOKKOS_FORCEINLINE_FUNCTION + int get_deltaj(const int idx) const { + return idx / (ibe_entire_ + 1); + } + KOKKOS_INLINE_FUNCTION bool is_i_ghost(const int idx) const { const int ni = ibe_entire_ + 1; diff --git a/tst/unit/CMakeLists.txt b/tst/unit/CMakeLists.txt index 7d36ebb2be49..4fece68dc68b 100644 --- a/tst/unit/CMakeLists.txt +++ b/tst/unit/CMakeLists.txt @@ -18,6 +18,7 @@ list(APPEND unit_tests_SOURCES test_concepts_lite.cpp + test_coordinates.cpp test_data_collection.cpp test_taskid.cpp test_tasklist.cpp diff --git a/tst/unit/test_coordinates.cpp b/tst/unit/test_coordinates.cpp new file mode 100644 index 000000000000..e2a98dbd8a25 --- /dev/null +++ b/tst/unit/test_coordinates.cpp @@ -0,0 +1,138 @@ +//======================================================================================== +// (C) (or copyright) 2024. Triad National Security, LLC. All rights reserved. +// +// This program was produced under U.S. Government contract 89233218CNA000001 for Los +// Alamos National Laboratory (LANL), which is operated by Triad National Security, LLC +// for the U.S. Department of Energy/National Nuclear Security Administration. All rights +// in the program are reserved by Triad National Security, LLC, and the U.S. Department +// of Energy/National Nuclear Security Administration. The Government is granted for +// itself and others acting on its behalf a nonexclusive, paid-up, irrevocable worldwide +// license in this material to reproduce, prepare derivative works, distribute copies to +// the public, perform publicly and display publicly, and to permit others to do so. +//======================================================================================== + +#include +#include + +#include "coordinates/uniform_cartesian.hpp" +#include "coordinates/uniform_cylindrical.hpp" +#include "coordinates/uniform_spherical.hpp" + +#include "basic_types.hpp" +#include "defs.hpp" +#include "globals.hpp" +#include "parameter_input.hpp" +using Real = double; +using parthenon::ParameterInput; +using parthenon::RegionSize; +using parthenon::X1DIR; +using parthenon::X2DIR; +using parthenon::X3DIR; +using parthenon::UniformCartesian; +using parthenon::UniformCylindrical; +using parthenon::UniformSpherical; + +#include + +int nghost_save = 888; + +TEST_CASE("Checking UniformCartesian") { + std::array xrat{1.0, 1.0, 1.0}; + ParameterInput pin; + nghost_save = parthenon::Globals::nghost; + parthenon::Globals::nghost = 2; + GIVEN("A coordinate object") { + std::array xmin{0.1, -0.2, 0.3}; + std::array xmax{0.3, 0.1, 0.7}; + std::array nx{10, 12, 14}; + RegionSize rs(xmin, xmax, xrat, nx); + UniformCartesian c(rs, &pin); + REQUIRE(c.Dx() == (xmax[0] - xmin[0]) / nx[0]); + REQUIRE(c.Dx() == (xmax[1] - xmin[1]) / nx[1]); + REQUIRE(c.Dx() == (xmax[2] - xmin[2]) / nx[2]); + } +} + +TEST_CASE("Checking UniformSpherical") { + std::array xrat{1.0, 1.0, 1.0}; + ParameterInput pin; + parthenon::Globals::nghost = 2; + GIVEN("A coordinate object") { + const Real rout = 3.5; + const Real rin = 0.0; + std::array xmin{rin, 0.0, 0.0}; + std::array xmax{rout, M_PI, 2 * M_PI}; + std::array nx{3, 5, 8}; + RegionSize rs(xmin, xmax, xrat, nx); + UniformSpherical c(rs, &pin); + const auto istart = c.GetStartIndex(); + Real dr = (xmax[0] - xmin[0]) / nx[0]; + const auto cxmin = c.GetXmin(); + REQUIRE(std::abs(cxmin[0] - (xmin[0] - parthenon::Globals::nghost*dr)) < 1.e-14); + int i0 = 6 + istart[0]; + Real r0 = xmin[0] + (i0 - istart[0]) * dr; + REQUIRE(c.Xf(i0) == r0); + + Real area = 0.0; + for (int j = istart[1]; j < istart[1] + nx[1]; j++) { + for (int k = istart[2]; k < istart[2] + nx[2]; k++) { + area += c.FaceArea(k, j, i0); + } + } + REQUIRE(std::abs(area - 4.0 * M_PI * r0 * r0) / area < 1.e-13); + + Real volume = 0.0; + for (int k = istart[2]; k < istart[2] + nx[2]; k++) { + for (int j = istart[1]; j < istart[1] + nx[1]; j++) { + for (int i = istart[0]; i < istart[0] + nx[0]; i++) { + volume += c.CellVolume(k, j, i); + } + } + } + REQUIRE(std::abs(volume - (4.0 / 3.0 * M_PI * (std::pow(rout,3) - std::pow(rin,3)))) / volume < 1.e-13); + } + parthenon::Globals::nghost = nghost_save; +} + +TEST_CASE("Checking UniformCylindrical") { + std::array xrat{1.0, 1.0, 1.0}; + ParameterInput pin; + parthenon::Globals::nghost = 2; + GIVEN("A coordinate object") { + const Real rout = 3.5; + const Real rin = 0.0; + const Real zlo = 1.3; + const Real zhi = 2.78; + std::array xmin{rin, zlo, 0.0}; + std::array xmax{rout, zhi, 2 * M_PI}; + std::array nx{3, 5, 8}; + RegionSize rs(xmin, xmax, xrat, nx); + UniformCylindrical c(rs, &pin); + const auto istart = c.GetStartIndex(); + Real dr = (xmax[0] - xmin[0]) / nx[0]; + const auto cxmin = c.GetXmin(); + REQUIRE(std::abs(cxmin[0] - (xmin[0] - parthenon::Globals::nghost*dr)) < 1.e-14); + int i0 = 6 + istart[0]; + Real r0 = xmin[0] + (i0 - istart[0]) * dr; + REQUIRE(c.Xf(i0) == r0); + + Real area = 0.0; + for (int j = istart[1]; j < istart[1] + nx[1]; j++) { + for (int k = istart[2]; k < istart[2] + nx[2]; k++) { + area += c.FaceArea(k, j, i0); + } + } + REQUIRE(std::abs(area - 2.0 * M_PI * r0 * (zhi - zlo)) / area < 1.e-13); + + Real volume = 0.0; + for (int k = istart[2]; k < istart[2] + nx[2]; k++) { + for (int j = istart[1]; j < istart[1] + nx[1]; j++) { + for (int i = istart[0]; i < istart[0] + nx[0]; i++) { + volume += c.CellVolume(k, j, i); + } + } + } + REQUIRE(std::abs(volume - M_PI * (rout * rout - rin * rin) * (zhi - zlo)) / volume < 1.e-13); + } + parthenon::Globals::nghost = nghost_save; +} \ No newline at end of file