1#ifndef AMREX_FFT_OPENBC_SOLVER_H_
2#define AMREX_FFT_OPENBC_SOLVER_H_
9template <
typename T = Real>
23 [[nodiscard]]
Box const&
Domain ()
const {
return m_domain; }
26 static Box make_grown_domain (
Box const& domain,
Info const& info);
32 std::unique_ptr<R2C<T>> m_r2c_green;
39#if (AMREX_SPACEDIM == 3)
40 if (info.twod_mode) { len[2] = 0; }
54#if (AMREX_SPACEDIM == 3)
56 auto gdom = make_grown_domain(domain,m_info);
57 gdom.enclosedCells(2);
62 gdom.setBig(2, nprocs-1);
63 m_r2c_green = std::make_unique<R2C<T>>(gdom,m_info);
64 auto [sd, ord] = m_r2c_green->getSpectralData();
70 auto [sd, ord] = m_r2c.getSpectralData();
72 m_G_fft.define(sd->boxArray(), sd->DistributionMap(), 1, 0);
82 auto* infab = m_info.twod_mode ? detail::get_fab(m_r2c_green->m_rx)
83 : detail::get_fab(m_r2c.m_rx);
84 auto const& lo = m_domain.smallEnd();
85 auto const& lo3 = lo.dim3();
86 auto const& len = m_domain.length3d();
88 auto const& a = infab->array();
89 auto box = infab->box();
91 int ndims = m_info.twod_mode ? AMREX_SPACEDIM-1 : AMREX_SPACEDIM;
92 for (
int idim = 0; idim < ndims; ++idim) {
93 if (box.smallEnd(idim) == lo[idim] && box.length(idim) == 2*len[idim]) {
94 box.growHi(idim, -len[idim]+1);
103 if (i == len[0] || j == len[1] || k == len[2]) {
107 auto jj = (j > len[1]) ? 2*len[1]-j : j;
108 auto kk = (k > len[2]) ? 2*len[2]-k : k;
109 G = greens_function(ii+lo3.x,jj+lo3.y,kk+lo3.z);
111 for (
int koff = 0; koff < nimages[2]; ++koff) {
112 int k2 = (koff == 0) ? k : 2*len[2]-k;
113 if ((k2 == 2*len[2]) || (koff == 1 && k == len[2])) {
116 for (
int joff = 0; joff < nimages[1]; ++joff) {
117 int j2 = (joff == 0) ? j : 2*len[1]-j;
118 if ((j2 == 2*len[1]) || (joff == 1 && j == len[1])) {
121 for (
int ioff = 0; ioff < nimages[0]; ++ioff) {
122 int i2 = (ioff == 0) ? i : 2*len[0]-i;
123 if ((i2 == 2*len[0]) || (ioff == 1 && i == len[0])) {
126 a(i2+lo3.x,j2+lo3.y,k2+lo3.z) = G;
133 if (m_info.twod_mode) {
134 m_r2c_green->forward(m_r2c_green->m_rx);
136 m_r2c.forward(m_r2c.m_rx);
139 if (!m_info.twod_mode) {
140 auto [sd, ord] = m_r2c.getSpectralData();
142 auto const* srcfab = detail::get_fab(*sd);
144 auto* dstfab = detail::get_fab(m_G_fft);
152 m_r2c.prepare_openbc();
161 auto& inmf = m_r2c.m_rx;
163 inmf.ParallelCopy(rho, 0, 0, 1);
165 m_r2c.m_openbc_half = !m_info.twod_mode;
167 m_r2c.m_openbc_half =
false;
169 auto scaling_factor = m_r2c.scalingFactor();
171 auto const* gfab = detail::get_fab(m_G_fft);
173 auto [sd, ord] = m_r2c.getSpectralData();
175 auto* rhofab = detail::get_fab(*sd);
177 auto*
pdst = rhofab->dataPtr();
178 auto const* psrc = gfab->dataPtr();
179 Box const& rhobox = rhofab->box();
180#if (AMREX_SPACEDIM == 3)
182 if (m_info.twod_mode) {
191#if (AMREX_SPACEDIM == 3)
192 Long isrc = i % leng;
196 pdst[i] *= psrc[isrc] * scaling_factor;
199 amrex::Abort(
"FFT::OpenBCSolver::solve: how did this happen?");
203 m_r2c.m_openbc_half = !m_info.twod_mode;
204 m_r2c.backward_doit(phi, phi.nGrowVect());
205 m_r2c.m_openbc_half =
false;
#define BL_PROFILE(a)
Definition AMReX_BLProfiler.H:551
#define AMREX_ASSERT(EX)
Definition AMReX_BLassert.H:38
#define AMREX_GPU_DEVICE
Definition AMReX_GpuQualifiers.H:18
Real * pdst
Definition AMReX_HypreMLABecLap.cpp:1090
__host__ __device__ const IntVectND< dim > & bigEnd() const &noexcept
Return the inclusive upper bound of the box.
Definition AMReX_Box.H:123
__host__ __device__ Long numPts() const noexcept
Return the number of points contained in the BoxND.
Definition AMReX_Box.H:356
__host__ __device__ IntVectND< dim > length() const noexcept
Return the length of the BoxND.
Definition AMReX_Box.H:154
__host__ __device__ IndexTypeND< dim > ixType() const noexcept
Return the indexing type.
Definition AMReX_Box.H:135
__host__ __device__ const IntVectND< dim > & smallEnd() const &noexcept
Return the inclusive lower bound of the box.
Definition AMReX_Box.H:111
Definition AMReX_FFT_OpenBCSolver.H:11
Box const & Domain() const
Definition AMReX_FFT_OpenBCSolver.H:23
typename R2C< T >::MF MF
Definition AMReX_FFT_OpenBCSolver.H:13
void solve(MF &phi, MF const &rho)
Definition AMReX_FFT_OpenBCSolver.H:157
void setGreensFunction(F const &greens_function)
Definition AMReX_FFT_OpenBCSolver.H:78
typename R2C< T >::cMF cMF
Definition AMReX_FFT_OpenBCSolver.H:14
OpenBCSolver(Box const &domain, Info const &info=Info{})
Definition AMReX_FFT_OpenBCSolver.H:48
Parallel Discrete Fourier Transform.
Definition AMReX_FFT_R2C.H:41
std::conditional_t< C, cMF, std::conditional_t< std::is_same_v< T, Real >, MultiFab, FabArray< BaseFab< T > > > > MF
Definition AMReX_FFT_R2C.H:46
Open Boundary Poisson Solver.
Definition AMReX_OpenBC.H:63
amrex_long Long
Definition AMReX_INT.H:30
void ParallelForOMP(T n, L const &f) noexcept
Performance-portable kernel launch function with optional OpenMP threading.
Definition AMReX_GpuLaunch.H:243
Definition AMReX_FFT_Helper.H:46
DomainStrategy
Definition AMReX_FFT_Helper.H:50
void dtod_memcpy_async(void *p_d_dst, const void *p_d_src, const std::size_t sz) noexcept
Definition AMReX_GpuDevice.H:329
int NProcsSub() noexcept
number of ranks in current frame
Definition AMReX_ParallelContext.H:74
@ make_alias
Definition AMReX_MakeType.H:7
__host__ __device__ void ignore_unused(const Ts &...)
This shuts up the compiler about unused variables.
Definition AMReX.H:138
BoxND< 3 > Box
Box is an alias for amrex::BoxND instantiated with AMREX_SPACEDIM.
Definition AMReX_BaseFwd.H:27
void Abort(const std::string &msg)
Print out message to cerr and exit via abort().
Definition AMReX.cpp:230
Definition AMReX_FFT_Helper.H:58
bool twod_mode
Definition AMReX_FFT_Helper.H:69
int nprocs
Max number of processes to use.
Definition AMReX_FFT_Helper.H:78
Fixed-size array that can be used on GPU.
Definition AMReX_Array.H:40