From f966020f0475980f9e5a08a51b723572174ecd28 Mon Sep 17 00:00:00 2001 From: jdolence Date: Wed, 10 Jul 2024 10:05:48 -0400 Subject: [PATCH 01/16] Initial refactor and addition of spherical. Compiles at least. --- example/poisson/poisson_package.cpp | 8 +- src/CMakeLists.txt | 2 + src/coordinates/coordinates.hpp | 1 + src/coordinates/uniform_cartesian.hpp | 311 +--------------------- src/coordinates/uniform_coordinates.hpp | 326 ++++++++++++++++++++++++ src/coordinates/uniform_spherical.hpp | 155 +++++++++++ src/interface/swarm_device_context.hpp | 6 +- src/prolong_restrict/pr_ops.hpp | 6 +- tst/unit/CMakeLists.txt | 1 + tst/unit/test_coordinates.cpp | 94 +++++++ 10 files changed, 595 insertions(+), 315 deletions(-) create mode 100644 src/coordinates/uniform_coordinates.hpp create mode 100644 src/coordinates/uniform_spherical.hpp create mode 100644 tst/unit/test_coordinates.cpp diff --git a/example/poisson/poisson_package.cpp b/example/poisson/poisson_package.cpp index e3f5bf75d3ae..00cd5071af37 100644 --- a/example/poisson/poisson_package.cpp +++ b/example/poisson/poisson_package.cpp @@ -160,9 +160,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."); } @@ -231,9 +231,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 1ea2e8bcd0bc..dc264cb73fcb 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -115,6 +115,8 @@ add_library(parthenon coordinates/coordinates.hpp coordinates/uniform_cartesian.hpp + coordinates/uniform_coordinates.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..e89a008f8da0 100644 --- a/src/coordinates/coordinates.hpp +++ b/src/coordinates/coordinates.hpp @@ -16,6 +16,7 @@ #include "config.hpp" #include "uniform_cartesian.hpp" +#include "uniform_spherical.hpp" namespace parthenon { diff --git a/src/coordinates/uniform_cartesian.hpp b/src/coordinates/uniform_cartesian.hpp index 04cec9d41974..32f38bd6a9b5 100644 --- a/src/coordinates/uniform_cartesian.hpp +++ b/src/coordinates/uniform_cartesian.hpp @@ -13,319 +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_; } - - 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..cc6fd2948e9b --- /dev/null +++ b/src/coordinates/uniform_coordinates.hpp @@ -0,0 +1,326 @@ +//======================================================================================== +// (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]; + } + + 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) { + 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 hx(const Real x1, const Real x2, const Real x3) const { + if constexpr (dir > 0 && dir < 4) return 1.0; + PARTHENON_FAIL("Unknown dir"); + 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); + return dx_[dir - 1]; + } + + //---------------------------------------- + // 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_spherical.hpp b/src/coordinates/uniform_spherical.hpp new file mode 100644 index 000000000000..81b9cc99801b --- /dev/null +++ b/src/coordinates/uniform_spherical.hpp @@ -0,0 +1,155 @@ +//======================================================================================== +// (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: + 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); + // only the index along dir is relevant in Xc, so offsetting all three is OK + 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 hx(const Real r, const Real th, const Real phi) const { + if (dir == X1DIR) { + return 1.0; + } else if constexpr (dir == X2DIR) { + return r; + } else if constexpr (dir == X3DIR) { + return r * std::sin(th); + } else { + PARTHENON_FAIL("Unknown dir"); + } + 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 ed958126f36b..66a33aab6c0f 100644 --- a/src/interface/swarm_device_context.hpp +++ b/src/interface/swarm_device_context.hpp @@ -92,14 +92,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 ccf1e025c521..bc87f33f630a 100644 --- a/src/prolong_restrict/pr_ops.hpp +++ b/src/prolong_restrict/pr_ops.hpp @@ -443,9 +443,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/tst/unit/CMakeLists.txt b/tst/unit/CMakeLists.txt index b44e6a7986da..595c62447a00 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..c1e35fbb7c9f --- /dev/null +++ b/tst/unit/test_coordinates.cpp @@ -0,0 +1,94 @@ +//======================================================================================== +// (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_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::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(); + std::cout << "cxmin = " << cxmin[0] << " " << "nghost = " << parthenon::Globals::nghost << std::endl; + 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; +} \ No newline at end of file From ad63ab74f0991a3fe9b57517d7136bee42c9ed71 Mon Sep 17 00:00:00 2001 From: jdolence Date: Wed, 10 Jul 2024 10:48:30 -0400 Subject: [PATCH 02/16] small update to make COORDINATE_TYPE settable --- src/CMakeLists.txt | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index dc264cb73fcb..7295c3adc388 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -90,7 +90,10 @@ 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 +if (NOT COORDINATE_TYPE) + set(COORDINATE_TYPE UniformCartesian) +endif() +message(STATUS "COORDINATE_TYPE = " ${COORDINATE_TYPE}) configure_file(config.hpp.in generated/config.hpp @ONLY) From d3045bf09639a497902181245d64604f7545d808 Mon Sep 17 00:00:00 2001 From: jdolence Date: Wed, 10 Jul 2024 12:15:52 -0400 Subject: [PATCH 03/16] fix bug and add using statements to pick up overloads in base --- src/coordinates/uniform_coordinates.hpp | 4 ++-- src/coordinates/uniform_spherical.hpp | 3 +++ 2 files changed, 5 insertions(+), 2 deletions(-) diff --git a/src/coordinates/uniform_coordinates.hpp b/src/coordinates/uniform_coordinates.hpp index cc6fd2948e9b..2c7b962ae054 100644 --- a/src/coordinates/uniform_coordinates.hpp +++ b/src/coordinates/uniform_coordinates.hpp @@ -80,8 +80,8 @@ class UniformCoordinates { 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); + 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; } diff --git a/src/coordinates/uniform_spherical.hpp b/src/coordinates/uniform_spherical.hpp index 81b9cc99801b..c4ae69e24170 100644 --- a/src/coordinates/uniform_spherical.hpp +++ b/src/coordinates/uniform_spherical.hpp @@ -20,6 +20,9 @@ namespace parthenon { class UniformSpherical : public UniformCoordinates { using base_t = UniformCoordinates; public: + using base_t::Dxc; + using base_t::Xc; + using base_t::Volume; UniformSpherical() = default; UniformSpherical(const RegionSize &rs, ParameterInput *pin) : UniformCoordinates(rs, pin) {} From eeaf8ee68c3437402d4b9ea1fc5c9260a8244768 Mon Sep 17 00:00:00 2001 From: jdolence Date: Wed, 10 Jul 2024 14:14:28 -0400 Subject: [PATCH 04/16] CellWidth --- src/coordinates/uniform_coordinates.hpp | 9 ++++++++- src/coordinates/uniform_spherical.hpp | 15 ++++++++++++++- 2 files changed, 22 insertions(+), 2 deletions(-) diff --git a/src/coordinates/uniform_coordinates.hpp b/src/coordinates/uniform_coordinates.hpp index 2c7b962ae054..0a516d401a40 100644 --- a/src/coordinates/uniform_coordinates.hpp +++ b/src/coordinates/uniform_coordinates.hpp @@ -62,6 +62,10 @@ class UniformCoordinates { 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 { @@ -191,7 +195,10 @@ class UniformCoordinates { } KOKKOS_FORCEINLINE_FUNCTION Real CellWidth(const int dir, const int k, const int j, const int i) const { assert(dir > 0 && dir < 4); - return dx_[dir - 1]; + 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; } //---------------------------------------- diff --git a/src/coordinates/uniform_spherical.hpp b/src/coordinates/uniform_spherical.hpp index c4ae69e24170..1a2a3f26da57 100644 --- a/src/coordinates/uniform_spherical.hpp +++ b/src/coordinates/uniform_spherical.hpp @@ -22,6 +22,7 @@ class UniformSpherical : public UniformCoordinates { public: using base_t::Dxc; using base_t::Xc; + using base_t::CellWidth; using base_t::Volume; UniformSpherical() = default; UniformSpherical(const RegionSize &rs, ParameterInput *pin) @@ -36,7 +37,6 @@ class UniformSpherical : public UniformCoordinates { template KOKKOS_FORCEINLINE_FUNCTION Real Dxc(const int idx) const { static_assert(dir > 0 && dir < 4); - // only the index along dir is relevant in Xc, so offsetting all three is OK return Xc(idx) - Xc(idx-1); } @@ -79,6 +79,19 @@ class UniformSpherical : public UniformCoordinates { 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 //---------------------------------------- From 0175b54b7133456fcaee48d9df00f783eb639cef Mon Sep 17 00:00:00 2001 From: jdolence Date: Wed, 10 Jul 2024 16:13:06 -0400 Subject: [PATCH 05/16] Scale --- src/coordinates/uniform_coordinates.hpp | 14 ++++++++++++-- src/coordinates/uniform_spherical.hpp | 23 +++++++++++++---------- 2 files changed, 25 insertions(+), 12 deletions(-) diff --git a/src/coordinates/uniform_coordinates.hpp b/src/coordinates/uniform_coordinates.hpp index 0a516d401a40..62501e5fd645 100644 --- a/src/coordinates/uniform_coordinates.hpp +++ b/src/coordinates/uniform_coordinates.hpp @@ -177,14 +177,24 @@ class UniformCoordinates { return 0.0; } - template + template KOKKOS_FORCEINLINE_FUNCTION - Real hx(const Real x1, const Real x2, const Real x3) const { + 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) { + 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 //---------------------------------------- diff --git a/src/coordinates/uniform_spherical.hpp b/src/coordinates/uniform_spherical.hpp index 1a2a3f26da57..6f8cae6dc938 100644 --- a/src/coordinates/uniform_spherical.hpp +++ b/src/coordinates/uniform_spherical.hpp @@ -22,6 +22,7 @@ class UniformSpherical : public UniformCoordinates { public: using base_t::Dxc; using base_t::Xc; + using base_t::Scale; using base_t::CellWidth; using base_t::Volume; UniformSpherical() = default; @@ -64,17 +65,19 @@ class UniformSpherical : public UniformCoordinates { return base_t::Xc(idx); } - template + template KOKKOS_FORCEINLINE_FUNCTION - Real hx(const Real r, const Real th, const Real phi) const { - if (dir == X1DIR) { - return 1.0; - } else if constexpr (dir == X2DIR) { - return r; - } else if constexpr (dir == X3DIR) { - return r * std::sin(th); - } else { - PARTHENON_FAIL("Unknown dir"); + 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; } From 2250dc51437e872b9414455f081ad4131c25f750 Mon Sep 17 00:00:00 2001 From: jdolence Date: Tue, 30 Jul 2024 10:23:07 -0400 Subject: [PATCH 06/16] add accessors for i and j in index split --- src/utils/index_split.hpp | 10 ++++++++++ 1 file changed, 10 insertions(+) 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; From f231157832fc0c1ec457bb923fdab145a219cb5e Mon Sep 17 00:00:00 2001 From: jdolence Date: Tue, 30 Jul 2024 11:16:51 -0400 Subject: [PATCH 07/16] missing const --- src/coordinates/uniform_coordinates.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/coordinates/uniform_coordinates.hpp b/src/coordinates/uniform_coordinates.hpp index 62501e5fd645..f552b8970f0f 100644 --- a/src/coordinates/uniform_coordinates.hpp +++ b/src/coordinates/uniform_coordinates.hpp @@ -94,7 +94,7 @@ class UniformCoordinates { // Dxf: Distance between cell faces //---------------------------------------- template - KOKKOS_FORCEINLINE_FUNCTION Real Dxf(const int idx) { + KOKKOS_FORCEINLINE_FUNCTION Real Dxf(const int idx) const { static_assert(dir > 0 && dir < 4); return dx_[dir - 1]; } From 17fdc733c1c2dcfbb98aa143343aea7a7298a95a Mon Sep 17 00:00:00 2001 From: jdolence Date: Tue, 30 Jul 2024 11:27:06 -0400 Subject: [PATCH 08/16] another missing const --- src/coordinates/uniform_coordinates.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/coordinates/uniform_coordinates.hpp b/src/coordinates/uniform_coordinates.hpp index f552b8970f0f..f3a5d2340101 100644 --- a/src/coordinates/uniform_coordinates.hpp +++ b/src/coordinates/uniform_coordinates.hpp @@ -187,7 +187,7 @@ class UniformCoordinates { template KOKKOS_FORCEINLINE_FUNCTION - Real Scale(const int dir, const int k, const int j, const int i) { + 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); From 51994952e12a290f59d8503cda4505a5c2e372de Mon Sep 17 00:00:00 2001 From: jdolence Date: Tue, 30 Jul 2024 12:47:05 -0400 Subject: [PATCH 09/16] typos --- src/coordinates/uniform_coordinates.hpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/coordinates/uniform_coordinates.hpp b/src/coordinates/uniform_coordinates.hpp index f3a5d2340101..93dd6a1f5ebe 100644 --- a/src/coordinates/uniform_coordinates.hpp +++ b/src/coordinates/uniform_coordinates.hpp @@ -189,9 +189,9 @@ class UniformCoordinates { 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); + 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; } From c09858fd04d0f2dde78c14ab0a23e435d1ea90f8 Mon Sep 17 00:00:00 2001 From: jdolence Date: Tue, 7 Jan 2025 11:08:31 -0500 Subject: [PATCH 10/16] add missing using statement for Dxc --- src/coordinates/uniform_cartesian.hpp | 1 + 1 file changed, 1 insertion(+) diff --git a/src/coordinates/uniform_cartesian.hpp b/src/coordinates/uniform_cartesian.hpp index c0e1700d27b9..83e6e75967bc 100644 --- a/src/coordinates/uniform_cartesian.hpp +++ b/src/coordinates/uniform_cartesian.hpp @@ -19,6 +19,7 @@ namespace parthenon { class UniformCartesian : public UniformCoordinates { public: + using UniformCoordinates::Dxc; UniformCartesian() = default; UniformCartesian(const RegionSize &rs, ParameterInput *pin) : UniformCoordinates(rs, pin) {} From 7b401223f83a80e77d8b04b75886ea9038c39b0e Mon Sep 17 00:00:00 2001 From: jdolence Date: Tue, 7 Jan 2025 11:38:34 -0500 Subject: [PATCH 11/16] is the derived CellVolume really necessary? why did I do this? --- src/coordinates/uniform_cartesian.hpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/coordinates/uniform_cartesian.hpp b/src/coordinates/uniform_cartesian.hpp index 83e6e75967bc..77d2a9ed5ad3 100644 --- a/src/coordinates/uniform_cartesian.hpp +++ b/src/coordinates/uniform_cartesian.hpp @@ -208,10 +208,10 @@ class UniformCartesian : public UniformCoordinates { //---------------------------------------- // CellVolume //---------------------------------------- - template - KOKKOS_FORCEINLINE_FUNCTION Real CellVolume(Args... args) const { - return cell_volume_; - } + //template + //KOKKOS_FORCEINLINE_FUNCTION Real CellVolume(Args... args) const { + //return cell_volume_; + //} //---------------------------------------- // Generalized volume From bd0ac382f17c39ff4ab017902cea6f271624fd21 Mon Sep 17 00:00:00 2001 From: jdolence Date: Tue, 7 Jan 2025 12:28:56 -0500 Subject: [PATCH 12/16] undo damage from utterly careless merge of develop --- src/coordinates/uniform_cartesian.hpp | 271 -------------------------- 1 file changed, 271 deletions(-) diff --git a/src/coordinates/uniform_cartesian.hpp b/src/coordinates/uniform_cartesian.hpp index 77d2a9ed5ad3..32f38bd6a9b5 100644 --- a/src/coordinates/uniform_cartesian.hpp +++ b/src/coordinates/uniform_cartesian.hpp @@ -19,283 +19,12 @@ namespace parthenon { class UniformCartesian : public UniformCoordinates { public: - using UniformCoordinates::Dxc; UniformCartesian() = default; UniformCartesian(const RegionSize &rs, ParameterInput *pin) : UniformCoordinates(rs, pin) {} UniformCartesian(const UniformCartesian &src, int coarsen) : UniformCoordinates(src, coarsen) {} - //---------------------------------------- - // 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]; - } - - //---------------------------------------- - // 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"; private: }; From 45705b18ac518c69270d1e8525b8149e35d6e1d8 Mon Sep 17 00:00:00 2001 From: jdolence Date: Wed, 8 Jan 2025 10:29:33 -0500 Subject: [PATCH 13/16] add cmake machinery to set coordinates --- src/CMakeLists.txt | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index d37ee12a8582..363f6ba0a621 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -90,8 +90,16 @@ set(COMPILED_WITH ${CMAKE_CXX_COMPILER}) set(COMPILER_COMMAND "") # TODO: Put something more descriptive here set(COMPILER_FLAGS "") # TODO: Put something more descriptive here -if (NOT COORDINATE_TYPE) +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}) From 26a63f569a892c2a3ccf9b9ea49717b0d768f526 Mon Sep 17 00:00:00 2001 From: jdolence Date: Tue, 14 Jan 2025 12:27:28 -0500 Subject: [PATCH 14/16] add cylindrical and utility to check coordinates --- src/coordinates/coordinates.hpp | 5 + src/coordinates/uniform_cylindrical.hpp | 152 ++++++++++++++++++++++++ 2 files changed, 157 insertions(+) create mode 100644 src/coordinates/uniform_cylindrical.hpp diff --git a/src/coordinates/coordinates.hpp b/src/coordinates/coordinates.hpp index e89a008f8da0..2cd71969bce2 100644 --- a/src/coordinates/coordinates.hpp +++ b/src/coordinates/coordinates.hpp @@ -22,6 +22,11 @@ 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_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_ From 27c8bdb87e88d74d705d89932b0d0038cf56295f Mon Sep 17 00:00:00 2001 From: jdolence Date: Tue, 14 Jan 2025 12:28:46 -0500 Subject: [PATCH 15/16] update cmake --- src/CMakeLists.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index 363f6ba0a621..65c5b959e757 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -130,6 +130,7 @@ 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 From 0cb6f0e1896830141cff340ebc0eb5fb41d11e7b Mon Sep 17 00:00:00 2001 From: jdolence Date: Tue, 14 Jan 2025 12:53:25 -0500 Subject: [PATCH 16/16] add tests of cylindrical --- src/coordinates/coordinates.hpp | 1 + tst/unit/test_coordinates.cpp | 46 ++++++++++++++++++++++++++++++++- 2 files changed, 46 insertions(+), 1 deletion(-) diff --git a/src/coordinates/coordinates.hpp b/src/coordinates/coordinates.hpp index 2cd71969bce2..475232b86611 100644 --- a/src/coordinates/coordinates.hpp +++ b/src/coordinates/coordinates.hpp @@ -16,6 +16,7 @@ #include "config.hpp" #include "uniform_cartesian.hpp" +#include "uniform_cylindrical.hpp" #include "uniform_spherical.hpp" namespace parthenon { diff --git a/tst/unit/test_coordinates.cpp b/tst/unit/test_coordinates.cpp index c1e35fbb7c9f..e2a98dbd8a25 100644 --- a/tst/unit/test_coordinates.cpp +++ b/tst/unit/test_coordinates.cpp @@ -15,6 +15,7 @@ #include #include "coordinates/uniform_cartesian.hpp" +#include "coordinates/uniform_cylindrical.hpp" #include "coordinates/uniform_spherical.hpp" #include "basic_types.hpp" @@ -28,6 +29,7 @@ using parthenon::X1DIR; using parthenon::X2DIR; using parthenon::X3DIR; using parthenon::UniformCartesian; +using parthenon::UniformCylindrical; using parthenon::UniformSpherical; #include @@ -66,7 +68,6 @@ TEST_CASE("Checking UniformSpherical") { const auto istart = c.GetStartIndex(); Real dr = (xmax[0] - xmin[0]) / nx[0]; const auto cxmin = c.GetXmin(); - std::cout << "cxmin = " << cxmin[0] << " " << "nghost = " << parthenon::Globals::nghost << std::endl; 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; @@ -91,4 +92,47 @@ TEST_CASE("Checking UniformSpherical") { 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