/* Copyright 2022 Edoardo Zoni, Remi Lehe, David Grote, Axel Huebl * * This file is part of ABLASTR. * * License: BSD-3-Clause-LBNL */ #include "sample.H" #include "ablastr/utils/TextMsg.H" #include #include #include #include #include #include #include #include #include #include #include #include #include namespace ablastr::coarsen::sample { void Loop ( amrex::MultiFab& mf_dst, const amrex::MultiFab& mf_src, const int dcomp, const int scomp, const int ncomp, const amrex::IntVect ngrowvect, const amrex::IntVect crse_ratio ) { // Staggering of source fine MultiFab and destination coarse MultiFab const amrex::IntVect stag_src = mf_src.boxArray().ixType().toIntVect(); const amrex::IntVect stag_dst = mf_dst.boxArray().ixType().toIntVect(); if ( crse_ratio > amrex::IntVect(1) ) ABLASTR_ALWAYS_ASSERT_WITH_MESSAGE( ngrowvect == amrex::IntVect(0), "option of filling guard cells of destination MultiFab with coarsening not supported for this interpolation" ); ABLASTR_ALWAYS_ASSERT_WITH_MESSAGE( mf_src.nGrowVect() >= stag_dst-stag_src+ngrowvect, "source fine MultiFab does not have enough guard cells for this interpolation" ); // Auxiliary integer arrays (always 3D) auto sf = amrex::GpuArray{0,0,0}; // staggering of source fine MultiFab auto sc = amrex::GpuArray{0,0,0}; // staggering of destination coarse MultiFab auto cr = amrex::GpuArray{1,1,1}; // coarsening ratio for (int i=0; i const& arr_dst = mf_dst.array( mfi ); amrex::Array4 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+dcomp) = Interp( arr_src, sf, sc, cr, i, j, k, n+scomp ); } ); } } void Coarsen ( amrex::MultiFab& mf_dst, const amrex::MultiFab& mf_src, const int dcomp, const int scomp, const int ncomp, const int ngrow, const amrex::IntVect crse_ratio ) { const amrex::IntVect ngrowvect(ngrow); Coarsen(mf_dst, mf_src, dcomp, scomp, ncomp, ngrowvect, crse_ratio); } void Coarsen ( amrex::MultiFab& mf_dst, const amrex::MultiFab& mf_src, const int dcomp, const int scomp, const int ncomp, const amrex::IntVect ngrowvect, const amrex::IntVect crse_ratio ) { BL_PROFILE("sample::Coarsen()"); // Convert BoxArray of source MultiFab to staggering of destination MultiFab and coarsen it amrex::BoxArray ba_tmp = amrex::convert( mf_src.boxArray(), mf_dst.ixType().toIntVect() ); ABLASTR_ALWAYS_ASSERT_WITH_MESSAGE( ba_tmp.coarsenable( crse_ratio ), "source MultiFab converted to staggering of destination MultiFab is not coarsenable" ); ba_tmp.coarsen( crse_ratio ); if ( ba_tmp == mf_dst.boxArray() and mf_src.DistributionMap() == mf_dst.DistributionMap() ) Loop( mf_dst, mf_src, dcomp, scomp, ncomp, ngrowvect, crse_ratio ); else { // Cannot coarsen into MultiFab with different BoxArray or DistributionMapping: // 1) create temporary MultiFab on coarsened version of source BoxArray with same DistributionMapping amrex::MultiFab mf_tmp( ba_tmp, mf_src.DistributionMap(), ncomp, ngrowvect, amrex::MFInfo(), amrex::FArrayBoxFactory() ); // 2) interpolate from mf_src to mf_tmp (start writing into component 0) Loop( mf_tmp, mf_src, 0, scomp, ncomp, ngrowvect, crse_ratio ); // 3) copy from mf_tmp to mf_dst (with different BoxArray or DistributionMapping) mf_dst.ParallelCopy( mf_tmp, 0, dcomp, ncomp ); } } } // namespace ablastr::coarsen::sample