aboutsummaryrefslogtreecommitdiff
path: root/Source/ablastr/coarsen/average.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'Source/ablastr/coarsen/average.cpp')
-rw-r--r--Source/ablastr/coarsen/average.cpp114
1 files changed, 114 insertions, 0 deletions
diff --git a/Source/ablastr/coarsen/average.cpp b/Source/ablastr/coarsen/average.cpp
new file mode 100644
index 000000000..2e333867d
--- /dev/null
+++ b/Source/ablastr/coarsen/average.cpp
@@ -0,0 +1,114 @@
+/* Copyright 2022 Edoardo Zoni, Remi Lehe, Prabhat Kumar
+ *
+ * This file is part of ABLASTR.
+ *
+ * License: BSD-3-Clause-LBNL
+ */
+#include "average.H"
+
+#include "ablastr/utils/TextMsg.H"
+
+#include <AMReX_BLProfiler.H>
+#include <AMReX_BLassert.H>
+#include <AMReX_BoxArray.H>
+#include <AMReX_Config.H>
+#include <AMReX_GpuControl.H>
+#include <AMReX_GpuLaunch.H>
+#include <AMReX_IntVect.H>
+#include <AMReX_MFIter.H>
+#include <AMReX_MultiFab.H>
+
+
+namespace ablastr::coarsen::average
+{
+ void
+ Loop (
+ amrex::MultiFab & mf_dst,
+ amrex::MultiFab const & mf_src,
+ int const ncomp,
+ amrex::IntVect const ngrow,
+ amrex::IntVect const crse_ratio
+ )
+ {
+ // Staggering of source fine MultiFab and destination coarse MultiFab
+ amrex::IntVect const stag_src = mf_src.boxArray().ixType().toIntVect();
+ amrex::IntVect const stag_dst = mf_dst.boxArray().ixType().toIntVect();
+
+ // Auxiliary integer arrays (always 3D)
+ amrex::GpuArray<int, 3> sf; // staggering of source fine MultiFab
+ amrex::GpuArray<int, 3> sc; // staggering of destination coarse MultiFab
+ amrex::GpuArray<int, 3> cr; // coarsening ratio
+
+ sf[0] = stag_src[0];
+#if defined(WARPX_DIM_1D_Z)
+ sf[1] = 0;
+#else
+ sf[1] = stag_src[1];
+#endif
+#if (AMREX_SPACEDIM <= 2)
+ sf[2] = 0;
+#elif defined(WARPX_DIM_3D)
+ sf[2] = stag_src[2];
+#endif
+
+ sc[0] = stag_dst[0];
+#if defined(WARPX_DIM_1D_Z)
+ sc[1] = 0;
+#else
+ sc[1] = stag_dst[1];
+#endif
+#if (AMREX_SPACEDIM <= 2)
+ sc[2] = 0;
+#elif defined(WARPX_DIM_3D)
+ sc[2] = stag_dst[2];
+#endif
+
+ cr[0] = crse_ratio[0];
+#if defined(WARPX_DIM_1D_Z)
+ cr[1] = 1;
+#else
+ cr[1] = crse_ratio[1];
+#endif
+#if (AMREX_SPACEDIM <= 2)
+ cr[2] = 1;
+#elif defined(WARPX_DIM_3D)
+ cr[2] = crse_ratio[2];
+#endif
+
+#ifdef AMREX_USE_OMP
+#pragma omp parallel if (amrex::Gpu::notInLaunchRegion())
+#endif
+ // Loop over boxes (or tiles if not on GPU)
+ for (amrex::MFIter mfi(mf_dst, amrex::TilingIfNotGPU()); mfi.isValid(); ++mfi) {
+ // Tiles defined at the coarse level
+ amrex::Box const & bx = mfi.growntilebox(ngrow);
+ amrex::Array4<amrex::Real> const &arr_dst = mf_dst.array(mfi);
+ amrex::Array4<amrex::Real const> const &arr_src = mf_src.const_array(mfi);
+ ParallelFor(bx, ncomp,
+ [=] AMREX_GPU_DEVICE(int i, int j, int k, int n) {
+ arr_dst(i, j, k, n) = Interp(
+ arr_src, sf, sc, cr, i, j, k, n);
+ });
+ }
+ }
+
+ void
+ Coarsen (
+ amrex::MultiFab & mf_dst,
+ amrex::MultiFab const & mf_src,
+ amrex::IntVect const crse_ratio
+ )
+ {
+ BL_PROFILE("ablastr::coarsen::Coarsen()");
+
+ ABLASTR_ALWAYS_ASSERT_WITH_MESSAGE(mf_src.ixType() == mf_dst.ixType(),
+ "source MultiFab and destination MultiFab have different IndexType");
+
+ // Number of guard cells to fill on coarse patch and number of components
+ const amrex::IntVect ngrow = (mf_src.nGrowVect() + 1) / crse_ratio;
+ const int ncomp = mf_src.nComp();
+
+ Loop(mf_dst, mf_src, ncomp, ngrow, crse_ratio);
+ }
+
+} // namespace ablastr::coarsen::average