Line data Source code
1 : #ifndef AMREX_ML_LINOP_H_
2 : #define AMREX_ML_LINOP_H_
3 : #include <AMReX_Config.H>
4 :
5 : #if defined(AMREX_USE_HYPRE) && (AMREX_SPACEDIM > 1)
6 : #include <AMReX_Hypre.H>
7 : #include <AMReX_HypreNodeLap.H>
8 : #endif
9 :
10 : #if defined(AMREX_USE_PETSC) && (AMREX_SPACEDIM > 1)
11 : #include <AMReX_PETSc.H>
12 : #endif
13 :
14 : #ifdef AMREX_USE_EB
15 : #include <AMReX_EBMultiFabUtil.H>
16 : #include <AMReX_MultiCutFab.H>
17 : #endif
18 :
19 : #include <AMReX_Any.H>
20 : #include <AMReX_BndryRegister.H>
21 : #include <AMReX_FabDataType.H>
22 : #include <AMReX_MLMGBndry.H>
23 : #include <AMReX_MultiFabUtil.H>
24 :
25 : #include <algorithm>
26 : #include <string>
27 :
28 : namespace amrex {
29 :
30 : enum class BottomSolver : int {
31 : Default, smoother, bicgstab, cg, bicgcg, cgbicg, hypre, petsc
32 : };
33 :
34 : struct LPInfo
35 : {
36 : bool do_agglomeration = true;
37 : bool do_consolidation = true;
38 : bool do_semicoarsening = false;
39 : int agg_grid_size = -1;
40 : int con_grid_size = -1;
41 : int con_ratio = 2;
42 : int con_strategy = 3;
43 : bool has_metric_term = true;
44 : int max_coarsening_level = 30;
45 : int max_semicoarsening_level = 0;
46 : int semicoarsening_direction = -1;
47 : int hidden_direction = -1;
48 :
49 : LPInfo& setAgglomeration (bool x) noexcept { do_agglomeration = x; return *this; }
50 : LPInfo& setConsolidation (bool x) noexcept { do_consolidation = x; return *this; }
51 : LPInfo& setSemicoarsening (bool x) noexcept { do_semicoarsening = x; return *this; }
52 : LPInfo& setAgglomerationGridSize (int x) noexcept { agg_grid_size = x; return *this; }
53 : LPInfo& setConsolidationGridSize (int x) noexcept { con_grid_size = x; return *this; }
54 : LPInfo& setConsolidationRatio (int x) noexcept { con_ratio = x; return *this; }
55 : LPInfo& setConsolidationStrategy (int x) noexcept { con_strategy = x; return *this; }
56 : LPInfo& setMetricTerm (bool x) noexcept { has_metric_term = x; return *this; }
57 0 : LPInfo& setMaxCoarseningLevel (int n) noexcept { max_coarsening_level = n; return *this; }
58 : LPInfo& setMaxSemicoarseningLevel (int n) noexcept { max_semicoarsening_level = n; return *this; }
59 : LPInfo& setSemicoarseningDirection (int n) noexcept { semicoarsening_direction = n; return *this; }
60 : LPInfo& setHiddenDirection (int n) noexcept { hidden_direction = n; return *this; }
61 :
62 : [[nodiscard]] bool hasHiddenDimension () const noexcept {
63 : return hidden_direction >=0 && hidden_direction < AMREX_SPACEDIM;
64 : }
65 :
66 : static constexpr int getDefaultAgglomerationGridSize () {
67 : #ifdef AMREX_USE_GPU
68 : return 32;
69 : #else
70 : return AMREX_D_PICK(32, 16, 8);
71 : #endif
72 : }
73 :
74 : static constexpr int getDefaultConsolidationGridSize () {
75 : #ifdef AMREX_USE_GPU
76 : return 32;
77 : #else
78 : return AMREX_D_PICK(32, 16, 8);
79 : #endif
80 : }
81 : };
82 :
83 : struct LinOpEnumType
84 : {
85 : enum struct BCMode { Homogeneous, Inhomogeneous };
86 : enum struct StateMode { Solution, Correction };
87 : enum struct Location { FaceCenter, FaceCentroid, CellCenter, CellCentroid };
88 : };
89 :
90 : template <typename T> class MLMGT;
91 : template <typename T> class MLCGSolverT;
92 : template <typename T> class MLPoissonT;
93 : template <typename T> class MLABecLaplacianT;
94 : template <typename T> class GMRESMLMGT;
95 :
96 : template <typename MF>
97 : class MLLinOpT
98 : {
99 : public:
100 :
101 : template <typename T> friend class MLMGT;
102 : template <typename T> friend class MLCGSolverT;
103 : template <typename T> friend class MLPoissonT;
104 : template <typename T> friend class MLABecLaplacianT;
105 : template <typename T> friend class GMRESMLMGT;
106 :
107 : using MFType = MF;
108 : using FAB = typename FabDataType<MF>::fab_type;
109 : using RT = typename FabDataType<MF>::value_type;
110 :
111 : using BCType = LinOpBCType;
112 : using BCMode = LinOpEnumType::BCMode;
113 : using StateMode = LinOpEnumType::StateMode;
114 : using Location = LinOpEnumType::Location;
115 :
116 : MLLinOpT () = default;
117 126 : virtual ~MLLinOpT () = default;
118 :
119 : MLLinOpT (const MLLinOpT<MF>&) = delete;
120 : MLLinOpT (MLLinOpT<MF>&&) = delete;
121 : MLLinOpT<MF>& operator= (const MLLinOpT<MF>&) = delete;
122 : MLLinOpT<MF>& operator= (MLLinOpT<MF>&&) = delete;
123 :
124 : void define (const Vector<Geometry>& a_geom,
125 : const Vector<BoxArray>& a_grids,
126 : const Vector<DistributionMapping>& a_dmap,
127 : const LPInfo& a_info,
128 : const Vector<FabFactory<FAB> const*>& a_factory,
129 : bool eb_limit_coarsening = true);
130 :
131 : [[nodiscard]] virtual std::string name () const { return std::string("Unspecified"); }
132 :
133 : /**
134 : * \brief Boundary of the whole domain.
135 : *
136 : * This functions must be called, and must be called before other bc
137 : * functions. This version is for single-component solve or when all the
138 : * components have the same BC types.
139 : *
140 : * \param lobc lower boundaries
141 : * \param hibc upper boundaries
142 : */
143 : void setDomainBC (const Array<BCType,AMREX_SPACEDIM>& lobc,
144 : const Array<BCType,AMREX_SPACEDIM>& hibc) noexcept;
145 :
146 : /**
147 : * \brief Boundary of the whole domain.
148 : *
149 : * This functions must be called, and must be called before other bc
150 : * functions. This version is for multi-component solve.
151 : *
152 : * \param lobc lower boundaries
153 : * \param hibc upper boundaries
154 : */
155 : void setDomainBC (const Vector<Array<BCType,AMREX_SPACEDIM> >& lobc,
156 : const Vector<Array<BCType,AMREX_SPACEDIM> >& hibc) noexcept;
157 :
158 : /**
159 : * \brief Set location of domain boundaries.
160 : *
161 : * By default, domain BC is on the domain face. If that's the
162 : * case, this function doesn't need to be called. However, one
163 : * could use this function to set non-zero domain BC locations.
164 : * Note all values should be >= 0. If this function is called,
165 : * it MUST be called before setLevelBC.
166 : */
167 : void setDomainBCLoc (const Array<Real,AMREX_SPACEDIM>& lo_bcloc,
168 : const Array<Real,AMREX_SPACEDIM>& hi_bcloc) noexcept;
169 :
170 : /**
171 : * \brief Needs coarse data for bc?
172 : *
173 : * If the lowest level grids does not cover the entire domain, coarse
174 : * level data are needed for supplying Dirichlet bc at coarse/fine
175 : * boundary, even when the domain bc is not Dirichlet.
176 : */
177 : [[nodiscard]] bool needsCoarseDataForBC () const noexcept { return m_needs_coarse_data_for_bc; }
178 :
179 : /**
180 : * \brief Set coarse/fine boundary conditions. For cell-centered solves
181 : * only.
182 : *
183 : * If we want to do a linear solve where the boundary conditions on the
184 : * coarsest AMR level of the solve come from a coarser level (e.g. the
185 : * base AMR level of the solve is > 0 and does not cover the entire
186 : * domain), we must explicitly provide the coarser data. Boundary
187 : * conditions from a coarser level are Dirichlet by default. The MultiFab
188 : * crse does not need to have ghost cells and is at a coarser resolution
189 : * than the coarsest AMR level of the solve; it is used to supply
190 : * (interpolated) boundary conditions for the solve. NOTE: If this is
191 : * called, it must be called before `setLevelBC`. If crse is nullptr,
192 : * then the bc values are assumed to be zero. The coarse/fine BC type
193 : * can be changed to homogeneous Neumann by the bc_type argument. In that
194 : * case, use nullptr for the crse argument.
195 : *
196 : * \param crse the coarse AMR level data
197 : * \param crse_ratio the coarsening ratio between fine and coarse AMR levels.
198 : * \param bc_type optional. It's Dirichlet by default, and can be Neumann.
199 : */
200 : void setCoarseFineBC (const MF* crse, int crse_ratio,
201 : LinOpBCType bc_type = LinOpBCType::Dirichlet) noexcept;
202 :
203 : void setCoarseFineBC (const MF* crse, IntVect const& crse_ratio,
204 : LinOpBCType bc_type = LinOpBCType::Dirichlet) noexcept;
205 :
206 : template <typename AMF, std::enable_if_t<!std::is_same_v<MF,AMF>,int> = 0>
207 : void setCoarseFineBC (const AMF* crse, int crse_ratio,
208 : LinOpBCType bc_type = LinOpBCType::Dirichlet) noexcept;
209 :
210 : template <typename AMF, std::enable_if_t<!std::is_same_v<MF,AMF>,int> = 0>
211 : void setCoarseFineBC (const AMF* crse, IntVect const& crse_ratio,
212 : LinOpBCType bc_type = LinOpBCType::Dirichlet) noexcept;
213 :
214 : /**
215 : * \brief Set boundary conditions for given level. For cell-centered
216 : * solves only.
217 : *
218 : * This must be called for each level. Argument `levelbcdata` is used to
219 : * supply Dirichlet or Neumann bc at the physical domain; if those data
220 : * are homogeneous we can pass nullptr instead of levelbcdata.
221 : * Regardless, this function must be called. If used, the MultiFab
222 : * levelbcdata must have one ghost cell. Only the data outside the
223 : * physical domain will be used. It is assumed that the data in those
224 : * ghost cells outside the domain live exactly on the face of the
225 : * physical domain. Argument `amrlev` is relative level such that the
226 : * lowest to the solver is always 0. The optional arguments
227 : * robinbc_[a|b|f] provide Robin boundary condition `a*phi + b*dphi/dn =
228 : * f`. Note that `d./dn` is `d./dx` at the upper boundary and `-d./dx`
229 : * at the lower boundary, for Robin BC. However, for inhomogeneous
230 : * Neumann BC, the value in leveldata is assumed to be `d./dx`.
231 : */
232 : virtual void setLevelBC (int /*amrlev*/, const MF* /*levelbcdata*/,
233 : const MF* /*robinbc_a*/ = nullptr,
234 : const MF* /*robinbc_b*/ = nullptr,
235 : const MF* /*robinbc_f*/ = nullptr) = 0;
236 :
237 : template <typename AMF, std::enable_if_t<!std::is_same_v<MF,AMF>,int> = 0>
238 : void setLevelBC (int amrlev, const AMF* levelbcdata,
239 : const AMF* robinbc_a = nullptr,
240 : const AMF* robinbc_b = nullptr,
241 : const AMF* robinbc_f = nullptr);
242 :
243 : //! Set verbosity
244 : void setVerbose (int v) noexcept { verbose = v; }
245 :
246 : //! Set order of interpolation at coarse/fine boundary
247 : void setMaxOrder (int o) noexcept { maxorder = o; }
248 : //! Get order of interpolation at coarse/fine boundary
249 : [[nodiscard]] int getMaxOrder () const noexcept { return maxorder; }
250 :
251 : //! Set the flag for whether the solver should try to make singular
252 : //! problem solvable, which is on by default.
253 : void setEnforceSingularSolvable (bool o) noexcept { enforceSingularSolvable = o; }
254 : //! Get the flag for whether the solver should try to make singular
255 : //! problem solvable.
256 : [[nodiscard]] bool getEnforceSingularSolvable () const noexcept { return enforceSingularSolvable; }
257 :
258 : [[nodiscard]] virtual BottomSolver getDefaultBottomSolver () const { return BottomSolver::bicgstab; }
259 :
260 : //! Return number of components
261 : [[nodiscard]] virtual int getNComp () const { return 1; }
262 :
263 : [[nodiscard]] virtual int getNGrow (int /*a_lev*/ = 0, int /*mg_lev*/ = 0) const { return 0; }
264 :
265 : //! Does it need update if it's reused?
266 : [[nodiscard]] virtual bool needsUpdate () const { return false; }
267 : //! Update for reuse.
268 : virtual void update () {}
269 :
270 : /**
271 : * \brief Restriction onto coarse MG level
272 : *
273 : * \param amrlev AMR level
274 : * \param cmglev coarse MG level
275 : * \param crse coarse data. This is the output.
276 : * \param fine fine data. This is the input. Some operators might need to fill ghost cells.
277 : * This is why it's not a const reference.
278 : */
279 : virtual void restriction (int amrlev, int cmglev, MF& crse, MF& fine) const = 0;
280 :
281 : /**
282 : * \brief Add interpolated coarse MG level data to fine MG level data
283 : *
284 : * \param amrlev AMR level
285 : * \param fmglev fine MG level
286 : * \param crse coarse data.
287 : * \param fine fine data.
288 : */
289 : virtual void interpolation (int amrlev, int fmglev, MF& fine, const MF& crse) const = 0;
290 :
291 : /**
292 : * \brief Overwrite fine MG level data with interpolated coarse data.
293 : *
294 : * \param amrlev AMR level
295 : * \param fmglev fine MG level
296 : * \param fine fine MG level data
297 : * \param crse coarse MG level data
298 : */
299 : virtual void interpAssign (int amrlev, int fmglev, MF& fine, MF& crse) const
300 : {
301 : amrex::ignore_unused(amrlev, fmglev, fine, crse);
302 : amrex::Abort("MLLinOpT::interpAssign: Must be implemented for FMG cycle");
303 : }
304 :
305 : /**
306 : * \brief Interpolation between AMR levels
307 : *
308 : * \param famrlev fine AMR level
309 : * \param fine fine level data
310 : * \param crse coarse level data
311 : * \param nghost number of ghost cells
312 : */
313 : virtual void interpolationAmr (int famrlev, MF& fine, const MF& crse,
314 : IntVect const& nghost) const
315 : {
316 : amrex::ignore_unused(famrlev, fine, crse, nghost);
317 : amrex::Abort("MLLinOpT::interpolationAmr: Must be implemented for composite solves across multiple AMR levels");
318 : }
319 :
320 : /**
321 : * \brief Average-down data from fine AMR level to coarse AMR level.
322 : *
323 : * \param camrlev coarse AMR level
324 : * \param crse_sol solutoin on coarse AMR level
325 : * \param crse_rhs RHS on coarse AMR level
326 : * \param fine_sol solution on fine AMR level
327 : * \param fine_rhs RHS on fine AMR level
328 : */
329 : virtual void averageDownSolutionRHS (int camrlev, MF& crse_sol, MF& crse_rhs,
330 : const MF& fine_sol, const MF& fine_rhs)
331 : {
332 : amrex::ignore_unused(camrlev, crse_sol, crse_rhs, fine_sol, fine_rhs);
333 : amrex::Abort("MLLinOpT::averageDownSolutionRHS: Must be implemented for composite solves across multiple AMR levels");
334 : }
335 :
336 : /**
337 : * \brief Apply the linear operator, out = L(in)
338 : *
339 : * \param amrlev AMR level
340 : * \param mglev MG level
341 : * \param out output
342 : * \param in input
343 : * \param bc_mode Is the BC homogeneous or inhomogeneous?
344 : * \param s_mode Are data data solution or correction?
345 : * \param bndry object for handling coarse/fine and physical boundaries
346 : */
347 : virtual void apply (int amrlev, int mglev, MF& out, MF& in, BCMode bc_mode,
348 : StateMode s_mode, const MLMGBndryT<MF>* bndry=nullptr) const = 0;
349 :
350 : /**
351 : * \brief Smooth
352 : *
353 : * \param amrlev AMR level
354 : * \param mglev MG level
355 : * \param sol unknowns
356 : * \param rhs RHS
357 : * \param skip_fillboundary flag controlling whether ghost cell filling can be skipped.
358 : */
359 : virtual void smooth (int amrlev, int mglev, MF& sol, const MF& rhs,
360 : bool skip_fillboundary=false) const = 0;
361 :
362 : //! Divide mf by the diagonal component of the operator. Used by bicgstab.
363 : virtual void normalize (int amrlev, int mglev, MF& mf) const {
364 : amrex::ignore_unused(amrlev, mglev, mf);
365 : }
366 :
367 : /**
368 : * \brief Compute residual for solution
369 : *
370 : * \param amrlev AMR level
371 : * \param resid residual
372 : * \param x solution
373 : * \param b RHS
374 : * \param crse_bc_data optional argument providing BC at coarse/fine boundary.
375 : */
376 : virtual void solutionResidual (int amrlev, MF& resid, MF& x, const MF& b,
377 : const MF* crse_bcdata=nullptr) = 0;
378 :
379 : /**
380 : * \brief Compute residual for the residual-correction form, resid = b - L(x)
381 : *
382 : * \param amrlev AMR level
383 : * \param mglev MG level
384 : * \param resid residual
385 : * \param x unknown in the residual-correction form
386 : * \param b RHS in the residual-correction form
387 : * \param bc_mode Is the BC homogeneous or inhomogeneous?
388 : * \param crse_bc_data optional argument providing BC at coarse/fine boundary.
389 : */
390 : virtual void correctionResidual (int amrlev, int mglev, MF& resid, MF& x, const MF& b,
391 : BCMode bc_mode, const MF* crse_bcdata=nullptr) = 0;
392 :
393 : /**
394 : * \brief Reflux at AMR coarse/fine boundary
395 : *
396 : * \param crse_amrlev coarse AMR level
397 : * \param res coarse level residual
398 : * \param crse_sol coarse level solution
399 : * \param crse_rhs coarse level RHS
400 : * \param fine_res fine level residual
401 : * \param fine_sol fine level solution
402 : * \param fine_rhs fine level RHS
403 : */
404 : virtual void reflux (int crse_amrlev,
405 : MF& res, const MF& crse_sol, const MF& crse_rhs,
406 : MF& fine_res, MF& fine_sol, const MF& fine_rhs) const
407 : {
408 : amrex::ignore_unused(crse_amrlev, res, crse_sol, crse_rhs, fine_res,
409 : fine_sol, fine_rhs);
410 : amrex::Abort("MLLinOpT::reflux: Must be implemented for composite solves across multiple AMR levels");
411 : }
412 :
413 : /**
414 : * \brief Compute fluxes
415 : *
416 : * \param amrlev AMR level
417 : * \param fluxes fluxes
418 : * \param sol solution
419 : * \param loc location of the fluxes
420 : */
421 : virtual void compFlux (int /*amrlev*/, const Array<MF*,AMREX_SPACEDIM>& /*fluxes*/,
422 : MF& /*sol*/, Location /*loc*/) const
423 : {
424 : amrex::Abort("AMReX_MLLinOp::compFlux::How did we get here?");
425 : }
426 :
427 : /**
428 : * \brief Compute gradients of the solution
429 : *
430 : * \param amrlev AMR level
431 : * \param grad grad(sol)
432 : * \param sol solution
433 : * \param loc location of the gradients
434 : */
435 : virtual void compGrad (int /*amrlev*/, const Array<MF*,AMREX_SPACEDIM>& /*grad*/,
436 : MF& /*sol*/, Location /*loc*/) const
437 : {
438 : amrex::Abort("AMReX_MLLinOp::compGrad::How did we get here?");
439 : }
440 :
441 : //! apply metric terms if there are any
442 : virtual void applyMetricTerm (int /*amrlev*/, int /*mglev*/, MF& /*rhs*/) const {}
443 : //! unapply metric terms if there are any
444 : virtual void unapplyMetricTerm (int /*amrlev*/, int /*mglev*/, MF& /*rhs*/) const {}
445 :
446 : //! This is needed for our nodal projection solver
447 : virtual void unimposeNeumannBC (int /*amrlev*/, MF& /*rhs*/) const {}
448 :
449 : //! Extra terms introduced when we treat inhomogeneous Nuemann BC as homogeneous.
450 : virtual void applyInhomogNeumannTerm (int /*amrlev*/, MF& /*rhs*/) const {}
451 :
452 : //! for overset solver only
453 : virtual void applyOverset (int /*amlev*/, MF& /*rhs*/) const {}
454 :
455 : //! scale RHS to fix solvability
456 : virtual void scaleRHS (int /*amrlev*/, MF& /*rhs*/) const {}
457 :
458 : //! get offset for fixing solvability
459 : virtual Vector<RT> getSolvabilityOffset (int /*amrlev*/, int /*mglev*/,
460 : MF const& /*rhs*/) const { return {}; }
461 :
462 : //! fix solvability by subtracting offset from RHS
463 : virtual void fixSolvabilityByOffset (int /*amrlev*/, int /*mglev*/, MF& /*rhs*/,
464 : Vector<RT> const& /*offset*/) const {}
465 :
466 : virtual void prepareForSolve () = 0;
467 :
468 : virtual void prepareForGMRES () {}
469 :
470 : //! For GMRES to work, this might need to be implemented to mask out
471 : //! Dirichlet nodes or cells (e.g., EB covered cells or overset cells)
472 : virtual void setDirichletNodesToZero (int /*amrlev*/, int /*mglev*/,
473 : MF& /*mf*/) const
474 : {
475 : amrex::Warning("This function might need to be implemented for GMRES to work with this LinOp.");
476 : }
477 :
478 : //! Is it singular on given AMR level?
479 : [[nodiscard]] virtual bool isSingular (int amrlev) const = 0;
480 : //! Is the bottom of MG singular?
481 : [[nodiscard]] virtual bool isBottomSingular () const = 0;
482 :
483 : //! x dot y, used by the bottom solver
484 : virtual RT xdoty (int amrlev, int mglev, const MF& x, const MF& y, bool local) const = 0;
485 :
486 : virtual std::unique_ptr<MLLinOpT<MF>> makeNLinOp (int /*grid_size*/) const
487 : {
488 : amrex::Abort("MLLinOp::makeNLinOp: N-Solve not supported");
489 : return nullptr;
490 : }
491 :
492 : virtual void getFluxes (const Vector<Array<MF*,AMREX_SPACEDIM> >& /*a_flux*/,
493 : const Vector<MF*>& /*a_sol*/,
494 : Location /*a_loc*/) const {
495 : amrex::Abort("MLLinOp::getFluxes: How did we get here?");
496 : }
497 : virtual void getFluxes (const Vector<MF*>& /*a_flux*/,
498 : const Vector<MF*>& /*a_sol*/) const {
499 : amrex::Abort("MLLinOp::getFluxes: How did we get here?");
500 : }
501 :
502 : #ifdef AMREX_USE_EB
503 : virtual void getEBFluxes (const Vector<MF*>& /*a_flux*/,
504 : const Vector<MF*>& /*a_sol*/) const {
505 : amrex::Abort("MLLinOp::getEBFluxes: How did we get here?");
506 : }
507 : #endif
508 :
509 : #if defined(AMREX_USE_HYPRE) && (AMREX_SPACEDIM > 1)
510 : [[nodiscard]] virtual std::unique_ptr<Hypre> makeHypre (Hypre::Interface /*hypre_interface*/) const {
511 : amrex::Abort("MLLinOp::makeHypre: How did we get here?");
512 : return {nullptr};
513 : }
514 : [[nodiscard]] virtual std::unique_ptr<HypreNodeLap> makeHypreNodeLap(
515 : int /*bottom_verbose*/,
516 : const std::string& /* options_namespace */) const
517 : {
518 : amrex::Abort("MLLinOp::makeHypreNodeLap: How did we get here?");
519 : return {nullptr};
520 : }
521 : #endif
522 :
523 : #if defined(AMREX_USE_PETSC) && (AMREX_SPACEDIM > 1)
524 : [[nodiscard]] virtual std::unique_ptr<PETScABecLap> makePETSc () const {
525 : amrex::Abort("MLLinOp::makePETSc: How did we get here?");
526 : return {nullptr};
527 : }
528 : #endif
529 :
530 : [[nodiscard]] virtual bool supportNSolve () const { return false; }
531 :
532 : virtual void copyNSolveSolution (MF&, MF const&) const {}
533 :
534 : virtual void postSolve (Vector<MF>& /*sol*/) const {}
535 :
536 : [[nodiscard]] virtual RT normInf (int amrlev, MF const& mf, bool local) const = 0;
537 :
538 : virtual void averageDownAndSync (Vector<MF>& sol) const = 0;
539 :
540 : virtual void avgDownResAmr (int clev, MF& cres, MF const& fres) const
541 : {
542 : amrex::ignore_unused(clev, cres, fres);
543 : amrex::Abort("MLLinOpT::avgDownResAmr: Must be implemented for composite solves across multiple AMR levels");
544 : }
545 :
546 : // This function is needed for FMG cycle, but not V-cycle.
547 : virtual void avgDownResMG (int clev, MF& cres, MF const& fres) const;
548 :
549 : [[nodiscard]] bool isMFIterSafe (int amrlev, int mglev1, int mglev2) const;
550 :
551 : //! Return the number of AMR levels
552 : [[nodiscard]] int NAMRLevels () const noexcept { return m_num_amr_levels; }
553 :
554 : //! Return the number of MG levels at given AMR level
555 : [[nodiscard]] int NMGLevels (int amrlev) const noexcept { return m_num_mg_levels[amrlev]; }
556 :
557 : [[nodiscard]] const Geometry& Geom (int amr_lev, int mglev=0) const noexcept { return m_geom[amr_lev][mglev]; }
558 :
559 : // BC
560 : Vector<Array<BCType, AMREX_SPACEDIM> > m_lobc;
561 : Vector<Array<BCType, AMREX_SPACEDIM> > m_hibc;
562 : // Need to save the original copy because we change the BC type to
563 : // Neumann for inhomogeneous Neumann and Robin.
564 : Vector<Array<BCType, AMREX_SPACEDIM> > m_lobc_orig;
565 : Vector<Array<BCType, AMREX_SPACEDIM> > m_hibc_orig;
566 :
567 : protected:
568 :
569 : static constexpr int mg_coarsen_ratio = 2;
570 : static constexpr int mg_box_min_width = 2;
571 : #ifdef AMREX_USE_EB
572 : static constexpr int mg_domain_min_width = 4;
573 : #else
574 : static constexpr int mg_domain_min_width = 2;
575 : #endif
576 :
577 : LPInfo info;
578 :
579 : int verbose = 0;
580 :
581 : int maxorder = 3;
582 :
583 : bool enforceSingularSolvable = true;
584 :
585 : int m_num_amr_levels = 0;
586 : Vector<int> m_amr_ref_ratio;
587 :
588 : Vector<int> m_num_mg_levels;
589 : const MLLinOpT<MF>* m_parent = nullptr;
590 :
591 : IntVect m_ixtype;
592 :
593 : bool m_do_agglomeration = false;
594 : bool m_do_consolidation = false;
595 :
596 : bool m_do_semicoarsening = false;
597 : Vector<IntVect> mg_coarsen_ratio_vec;
598 :
599 : //! first Vector is for amr level and second is mg level
600 : Vector<Vector<Geometry> > m_geom;
601 : Vector<Vector<BoxArray> > m_grids;
602 : Vector<Vector<DistributionMapping> > m_dmap;
603 : Vector<Vector<std::unique_ptr<FabFactory<FAB> > > > m_factory;
604 : Vector<int> m_domain_covered;
605 :
606 : MPI_Comm m_default_comm = MPI_COMM_NULL;
607 : MPI_Comm m_bottom_comm = MPI_COMM_NULL;
608 : struct CommContainer {
609 : MPI_Comm comm;
610 : CommContainer (MPI_Comm m) noexcept : comm(m) {}
611 : CommContainer (const CommContainer&) = delete;
612 : CommContainer (CommContainer&&) = delete;
613 : void operator= (const CommContainer&) = delete;
614 : void operator= (CommContainer&&) = delete;
615 : ~CommContainer () { // NOLINT(modernize-use-equals-default)
616 : #ifdef BL_USE_MPI
617 : if (comm != MPI_COMM_NULL) { MPI_Comm_free(&comm); }
618 : #endif
619 : }
620 : };
621 : std::unique_ptr<CommContainer> m_raii_comm;
622 :
623 : Array<Real, AMREX_SPACEDIM> m_domain_bloc_lo {{AMREX_D_DECL(0._rt,0._rt,0._rt)}};
624 : Array<Real, AMREX_SPACEDIM> m_domain_bloc_hi {{AMREX_D_DECL(0._rt,0._rt,0._rt)}};
625 :
626 : bool m_needs_coarse_data_for_bc = false;
627 : LinOpBCType m_coarse_fine_bc_type = LinOpBCType::Dirichlet;
628 : IntVect m_coarse_data_crse_ratio = IntVect(-1);
629 : RealVect m_coarse_bc_loc;
630 : const MF* m_coarse_data_for_bc = nullptr;
631 : MF m_coarse_data_for_bc_raii;
632 :
633 : //! Return AMR refinement ratios
634 : [[nodiscard]] const Vector<int>& AMRRefRatio () const noexcept { return m_amr_ref_ratio; }
635 :
636 : //! Return AMR refinement ratio at given AMR level
637 : [[nodiscard]] int AMRRefRatio (int amr_lev) const noexcept { return m_amr_ref_ratio[amr_lev]; }
638 :
639 : [[nodiscard]] FabFactory<FAB> const* Factory (int amr_lev, int mglev=0) const noexcept {
640 : return m_factory[amr_lev][mglev].get();
641 : }
642 :
643 : [[nodiscard]] GpuArray<BCType,AMREX_SPACEDIM> LoBC (int icomp = 0) const noexcept {
644 : return GpuArray<BCType,AMREX_SPACEDIM>{{AMREX_D_DECL(m_lobc[icomp][0],
645 : m_lobc[icomp][1],
646 : m_lobc[icomp][2])}};
647 : }
648 : [[nodiscard]] GpuArray<BCType,AMREX_SPACEDIM> HiBC (int icomp = 0) const noexcept {
649 : return GpuArray<BCType,AMREX_SPACEDIM>{{AMREX_D_DECL(m_hibc[icomp][0],
650 : m_hibc[icomp][1],
651 : m_hibc[icomp][2])}};
652 : }
653 :
654 : [[nodiscard]] bool hasBC (BCType bct) const noexcept;
655 : [[nodiscard]] bool hasInhomogNeumannBC () const noexcept;
656 : [[nodiscard]] bool hasRobinBC () const noexcept;
657 :
658 : [[nodiscard]] virtual bool supportRobinBC () const noexcept { return false; }
659 : [[nodiscard]] virtual bool supportInhomogNeumannBC () const noexcept { return false; }
660 :
661 : #ifdef BL_USE_MPI
662 : [[nodiscard]] bool isBottomActive () const noexcept { return m_bottom_comm != MPI_COMM_NULL; }
663 : #else
664 : [[nodiscard]] bool isBottomActive () const noexcept { return true; }
665 : #endif
666 : [[nodiscard]] MPI_Comm BottomCommunicator () const noexcept { return m_bottom_comm; }
667 : [[nodiscard]] MPI_Comm Communicator () const noexcept { return m_default_comm; }
668 :
669 : void setCoarseFineBCLocation (const RealVect& cloc) noexcept { m_coarse_bc_loc = cloc; }
670 :
671 : [[nodiscard]] bool doAgglomeration () const noexcept { return m_do_agglomeration; }
672 : [[nodiscard]] bool doConsolidation () const noexcept { return m_do_consolidation; }
673 : [[nodiscard]] bool doSemicoarsening () const noexcept { return m_do_semicoarsening; }
674 :
675 : [[nodiscard]] bool isCellCentered () const noexcept { return m_ixtype == 0; }
676 :
677 : [[nodiscard]] virtual IntVect getNGrowVectRestriction () const {
678 : return isCellCentered() ? IntVect(0) : IntVect(1);
679 : }
680 :
681 : virtual void make (Vector<Vector<MF> >& mf, IntVect const& ng) const;
682 :
683 : [[nodiscard]] virtual MF make (int amrlev, int mglev, IntVect const& ng) const;
684 :
685 : [[nodiscard]] virtual MF makeAlias (MF const& mf) const;
686 :
687 : [[nodiscard]] virtual MF makeCoarseMG (int amrlev, int mglev, IntVect const& ng) const;
688 :
689 : [[nodiscard]] virtual MF makeCoarseAmr (int famrlev, IntVect const& ng) const;
690 :
691 : [[nodiscard]] virtual std::unique_ptr<FabFactory<FAB> > makeFactory (int /*amrlev*/, int /*mglev*/) const {
692 : return std::make_unique<DefaultFabFactory<FAB>>();
693 : }
694 :
695 : virtual void resizeMultiGrid (int new_size);
696 :
697 : [[nodiscard]] bool hasHiddenDimension () const noexcept { return info.hasHiddenDimension(); }
698 : [[nodiscard]] int hiddenDirection () const noexcept { return info.hidden_direction; }
699 : [[nodiscard]] Box compactify (Box const& b) const noexcept;
700 :
701 : template <typename T>
702 : [[nodiscard]] Array4<T> compactify (Array4<T> const& a) const noexcept
703 : {
704 : if (info.hidden_direction == 0) {
705 : return Array4<T>(a.dataPtr(), {a.begin.y,a.begin.z,0}, {a.end.y,a.end.z,1}, a.nComp());
706 : } else if (info.hidden_direction == 1) {
707 : return Array4<T>(a.dataPtr(), {a.begin.x,a.begin.z,0}, {a.end.x,a.end.z,1}, a.nComp());
708 : } else if (info.hidden_direction == 2) {
709 : return Array4<T>(a.dataPtr(), {a.begin.x,a.begin.y,0}, {a.end.x,a.end.y,1}, a.nComp());
710 : } else {
711 : return a;
712 : }
713 : }
714 :
715 : template <typename T>
716 : [[nodiscard]] T get_d0 (T const& dx, T const& dy, T const&) const noexcept
717 : {
718 : if (info.hidden_direction == 0) {
719 : return dy;
720 : } else {
721 : return dx;
722 : }
723 : }
724 :
725 : template <typename T>
726 : [[nodiscard]] T get_d1 (T const&, T const& dy, T const& dz) const noexcept
727 : {
728 : if (info.hidden_direction == 0 || info.hidden_direction == 1) {
729 : return dz;
730 : } else {
731 : return dy;
732 : }
733 : }
734 :
735 : private:
736 :
737 : void defineGrids (const Vector<Geometry>& a_geom,
738 : const Vector<BoxArray>& a_grids,
739 : const Vector<DistributionMapping>& a_dmap,
740 : const Vector<FabFactory<FAB> const*>& a_factory);
741 : void defineBC ();
742 : static void makeAgglomeratedDMap (const Vector<BoxArray>& ba, Vector<DistributionMapping>& dm);
743 : static void makeConsolidatedDMap (const Vector<BoxArray>& ba, Vector<DistributionMapping>& dm,
744 : int ratio, int strategy);
745 : [[nodiscard]] MPI_Comm makeSubCommunicator (const DistributionMapping& dm);
746 :
747 : virtual void checkPoint (std::string const& /*file_name*/) const {
748 : amrex::Abort("MLLinOp:checkPoint: not implemented");
749 : }
750 :
751 : Vector<std::unique_ptr<MF>> levelbc_raii;
752 : Vector<std::unique_ptr<MF>> robin_a_raii;
753 : Vector<std::unique_ptr<MF>> robin_b_raii;
754 : Vector<std::unique_ptr<MF>> robin_f_raii;
755 : };
756 :
757 : template <typename MF>
758 : void
759 : MLLinOpT<MF>::define (const Vector<Geometry>& a_geom,
760 : const Vector<BoxArray>& a_grids,
761 : const Vector<DistributionMapping>& a_dmap,
762 : const LPInfo& a_info,
763 : const Vector<FabFactory<FAB> const*>& a_factory,
764 : [[maybe_unused]] bool eb_limit_coarsening)
765 : {
766 : BL_PROFILE("MLLinOp::define()");
767 :
768 : info = a_info;
769 : #ifdef AMREX_USE_GPU
770 : if (Gpu::notInLaunchRegion())
771 : {
772 : if (info.agg_grid_size <= 0) { info.agg_grid_size = AMREX_D_PICK(32, 16, 8); }
773 : if (info.con_grid_size <= 0) { info.con_grid_size = AMREX_D_PICK(32, 16, 8); }
774 : }
775 : else
776 : #endif
777 : {
778 : if (info.agg_grid_size <= 0) { info.agg_grid_size = LPInfo::getDefaultAgglomerationGridSize(); }
779 : if (info.con_grid_size <= 0) { info.con_grid_size = LPInfo::getDefaultConsolidationGridSize(); }
780 : }
781 :
782 : #ifdef AMREX_USE_EB
783 : if (!a_factory.empty() && eb_limit_coarsening) {
784 : const auto *f = dynamic_cast<EBFArrayBoxFactory const*>(a_factory[0]);
785 : if (f) {
786 : info.max_coarsening_level = std::min(info.max_coarsening_level,
787 : f->maxCoarseningLevel());
788 : }
789 : }
790 : #endif
791 : defineGrids(a_geom, a_grids, a_dmap, a_factory);
792 : defineBC();
793 : }
794 :
795 : template <typename MF>
796 : void
797 : MLLinOpT<MF>::defineGrids (const Vector<Geometry>& a_geom,
798 : const Vector<BoxArray>& a_grids,
799 : const Vector<DistributionMapping>& a_dmap,
800 : const Vector<FabFactory<FAB> const*>& a_factory)
801 : {
802 : BL_PROFILE("MLLinOp::defineGrids()");
803 :
804 : m_num_amr_levels = 0;
805 : for (int amrlev = 0; amrlev < a_geom.size(); amrlev++) {
806 : if (!a_grids[amrlev].empty()) {
807 : m_num_amr_levels++;
808 : }
809 : }
810 :
811 : m_amr_ref_ratio.resize(m_num_amr_levels);
812 : m_num_mg_levels.resize(m_num_amr_levels);
813 :
814 : m_geom.resize(m_num_amr_levels);
815 : m_grids.resize(m_num_amr_levels);
816 : m_dmap.resize(m_num_amr_levels);
817 : m_factory.resize(m_num_amr_levels);
818 :
819 : m_default_comm = ParallelContext::CommunicatorSub();
820 :
821 : const RealBox& rb = a_geom[0].ProbDomain();
822 : const int coord = a_geom[0].Coord();
823 : const Array<int,AMREX_SPACEDIM>& is_per = a_geom[0].isPeriodic();
824 :
825 : IntVect mg_coarsen_ratio_v(mg_coarsen_ratio);
826 : IntVect mg_box_min_width_v(mg_box_min_width);
827 : IntVect mg_domain_min_width_v(mg_domain_min_width);
828 : if (hasHiddenDimension()) {
829 : AMREX_ASSERT_WITH_MESSAGE(AMREX_SPACEDIM == 3 && m_num_amr_levels == 1,
830 : "Hidden direction only supported for 3d level solve");
831 : mg_coarsen_ratio_v[info.hidden_direction] = 1;
832 : mg_box_min_width_v[info.hidden_direction] = 0;
833 : mg_domain_min_width_v[info.hidden_direction] = 0;
834 : }
835 :
836 : // fine amr levels
837 : for (int amrlev = m_num_amr_levels-1; amrlev > 0; --amrlev)
838 : {
839 : m_num_mg_levels[amrlev] = 1;
840 : m_geom[amrlev].push_back(a_geom[amrlev]);
841 : m_grids[amrlev].push_back(a_grids[amrlev]);
842 : m_dmap[amrlev].push_back(a_dmap[amrlev]);
843 : if (amrlev < a_factory.size()) {
844 : m_factory[amrlev].emplace_back(a_factory[amrlev]->clone());
845 : } else {
846 : m_factory[amrlev].push_back(std::make_unique<DefaultFabFactory<FAB>>());
847 : }
848 :
849 : IntVect rr = mg_coarsen_ratio_v;
850 : const Box& dom = a_geom[amrlev].Domain();
851 : for (int i = 0; i < 2; ++i)
852 : {
853 : if (!dom.coarsenable(rr)) { amrex::Abort("MLLinOp: Uncoarsenable domain"); }
854 :
855 : const Box& cdom = amrex::coarsen(dom,rr);
856 : if (cdom == a_geom[amrlev-1].Domain()) { break; }
857 :
858 : ++(m_num_mg_levels[amrlev]);
859 :
860 : m_geom[amrlev].emplace_back(cdom, rb, coord, is_per);
861 :
862 : m_grids[amrlev].push_back(a_grids[amrlev]);
863 : AMREX_ASSERT(m_grids[amrlev].back().coarsenable(rr));
864 : m_grids[amrlev].back().coarsen(rr);
865 :
866 : m_dmap[amrlev].push_back(a_dmap[amrlev]);
867 :
868 : rr *= mg_coarsen_ratio_v;
869 : }
870 :
871 : #if (AMREX_SPACEDIM > 1)
872 : if (hasHiddenDimension()) {
873 : m_amr_ref_ratio[amrlev-1] = rr[AMREX_SPACEDIM-info.hidden_direction];
874 : } else
875 : #endif
876 : {
877 : m_amr_ref_ratio[amrlev-1] = rr[0];
878 : }
879 : }
880 :
881 : // coarsest amr level
882 : m_num_mg_levels[0] = 1;
883 : m_geom[0].push_back(a_geom[0]);
884 : m_grids[0].push_back(a_grids[0]);
885 : m_dmap[0].push_back(a_dmap[0]);
886 : if (!a_factory.empty()) {
887 : m_factory[0].emplace_back(a_factory[0]->clone());
888 : } else {
889 : m_factory[0].push_back(std::make_unique<DefaultFabFactory<FAB>>());
890 : }
891 :
892 : m_domain_covered.resize(m_num_amr_levels, false);
893 : auto npts0 = m_grids[0][0].numPts();
894 : m_domain_covered[0] = (npts0 == compactify(m_geom[0][0].Domain()).numPts());
895 : for (int amrlev = 1; amrlev < m_num_amr_levels; ++amrlev)
896 : {
897 : if (!m_domain_covered[amrlev-1]) { break; }
898 : m_domain_covered[amrlev] = (m_grids[amrlev][0].numPts() ==
899 : compactify(m_geom[amrlev][0].Domain()).numPts());
900 : }
901 :
902 : Box aggbox;
903 : bool aggable = false;
904 :
905 : if (m_grids[0][0].size() > 1 && info.do_agglomeration)
906 : {
907 : if (m_domain_covered[0])
908 : {
909 : aggbox = m_geom[0][0].Domain();
910 : if (hasHiddenDimension()) {
911 : aggbox.makeSlab(hiddenDirection(), m_grids[0][0][0].smallEnd(hiddenDirection()));
912 : }
913 : aggable = true;
914 : }
915 : else
916 : {
917 : aggbox = m_grids[0][0].minimalBox();
918 : aggable = (aggbox.numPts() == npts0);
919 : }
920 : }
921 :
922 : bool agged = false;
923 : bool coned = false;
924 : int agg_lev = 0, con_lev = 0;
925 :
926 : AMREX_ALWAYS_ASSERT( ! (info.do_semicoarsening && info.hasHiddenDimension())
927 : && info.semicoarsening_direction >= -1
928 : && info.semicoarsening_direction < AMREX_SPACEDIM );
929 :
930 : if (info.do_agglomeration && aggable)
931 : {
932 : Box dbx = m_geom[0][0].Domain();
933 : Box bbx = aggbox;
934 : Real const nbxs = static_cast<Real>(m_grids[0][0].size());
935 : Real const threshold_npts = static_cast<Real>(AMREX_D_TERM(info.agg_grid_size,
936 : *info.agg_grid_size,
937 : *info.agg_grid_size));
938 : Vector<Box> domainboxes{dbx};
939 : Vector<Box> boundboxes{bbx};
940 : Vector<int> agg_flag{false};
941 : Vector<IntVect> accum_coarsen_ratio{IntVect(1)};
942 : int numsclevs = 0;
943 :
944 : for (int lev = 0; lev < info.max_coarsening_level; ++lev)
945 : {
946 : IntVect rr_level = mg_coarsen_ratio_v;
947 : bool const do_semicoarsening_level = info.do_semicoarsening
948 : && numsclevs < info.max_semicoarsening_level;
949 : if (do_semicoarsening_level
950 : && info.semicoarsening_direction != -1)
951 : {
952 : rr_level[info.semicoarsening_direction] = 1;
953 : }
954 : IntVect is_coarsenable;
955 : for (int idim = 0; idim < AMREX_SPACEDIM; ++idim) {
956 : IntVect rr_dir(1);
957 : rr_dir[idim] = rr_level[idim];
958 : is_coarsenable[idim] = dbx.coarsenable(rr_dir, mg_domain_min_width_v)
959 : && bbx.coarsenable(rr_dir, mg_box_min_width_v);
960 : if (!is_coarsenable[idim] && do_semicoarsening_level
961 : && info.semicoarsening_direction == -1)
962 : {
963 : is_coarsenable[idim] = true;
964 : rr_level[idim] = 1;
965 : }
966 : }
967 : if (is_coarsenable != IntVect(1) || rr_level == IntVect(1)) {
968 : break;
969 : }
970 : if (do_semicoarsening_level && info.semicoarsening_direction == -1) {
971 : // make sure there is at most one direction that is not coarsened
972 : int n_ones = AMREX_D_TERM( static_cast<int>(rr_level[0] == 1),
973 : + static_cast<int>(rr_level[1] == 1),
974 : + static_cast<int>(rr_level[2] == 1));
975 : if (n_ones > 1) { break; }
976 : }
977 : if (rr_level != mg_coarsen_ratio_v) {
978 : ++numsclevs;
979 : }
980 :
981 : accum_coarsen_ratio.push_back(accum_coarsen_ratio.back()*rr_level);
982 : domainboxes.push_back(dbx.coarsen(rr_level));
983 : boundboxes.push_back(bbx.coarsen(rr_level));
984 : bool to_agg = (bbx.d_numPts() / nbxs) < 0.999*threshold_npts;
985 : agg_flag.push_back(to_agg);
986 : }
987 :
988 : for (int lev = 1, nlevs = static_cast<int>(domainboxes.size()); lev < nlevs; ++lev) {
989 : if (!agged && !agg_flag[lev] &&
990 : a_grids[0].coarsenable(accum_coarsen_ratio[lev], mg_box_min_width_v))
991 : {
992 : m_grids[0].push_back(amrex::coarsen(a_grids[0], accum_coarsen_ratio[lev]));
993 : m_dmap[0].push_back(a_dmap[0]);
994 : } else {
995 : IntVect cr = domainboxes[lev-1].length() / domainboxes[lev].length();
996 : if (!m_grids[0].back().coarsenable(cr)) {
997 : break; // average_down would fail if fine boxarray is not coarsenable.
998 : }
999 : m_grids[0].emplace_back(boundboxes[lev]);
1000 : IntVect max_grid_size(info.agg_grid_size);
1001 : if (info.do_semicoarsening && info.max_semicoarsening_level >= lev
1002 : && info.semicoarsening_direction != -1)
1003 : {
1004 : IntVect blen = amrex::enclosedCells(boundboxes[lev]).size();
1005 : AMREX_D_TERM(int mgs_0 = (max_grid_size[0]+blen[0]-1) / blen[0];,
1006 : int mgs_1 = (max_grid_size[1]+blen[1]-1) / blen[1];,
1007 : int mgs_2 = (max_grid_size[2]+blen[2]-1) / blen[2]);
1008 : max_grid_size[info.semicoarsening_direction]
1009 : *= AMREX_D_TERM(mgs_0, *mgs_1, *mgs_2);
1010 : }
1011 : m_grids[0].back().maxSize(max_grid_size);
1012 : m_dmap[0].push_back(DistributionMapping());
1013 : if (!agged) {
1014 : agged = true;
1015 : agg_lev = lev;
1016 : }
1017 : }
1018 : m_geom[0].emplace_back(domainboxes[lev],rb,coord,is_per);
1019 : }
1020 : }
1021 : else
1022 : {
1023 : Long consolidation_threshold = 0;
1024 : Real avg_npts = 0.0;
1025 : if (info.do_consolidation) {
1026 : avg_npts = static_cast<Real>(a_grids[0].d_numPts()) / static_cast<Real>(ParallelContext::NProcsSub());
1027 : consolidation_threshold = AMREX_D_TERM(Long(info.con_grid_size),
1028 : *info.con_grid_size,
1029 : *info.con_grid_size);
1030 : }
1031 :
1032 : Box const& dom0 = a_geom[0].Domain();
1033 : IntVect rr_vec(1);
1034 : int numsclevs = 0;
1035 : for (int lev = 0; lev < info.max_coarsening_level; ++lev)
1036 : {
1037 : IntVect rr_level = mg_coarsen_ratio_v;
1038 : bool do_semicoarsening_level = info.do_semicoarsening
1039 : && numsclevs < info.max_semicoarsening_level;
1040 : if (do_semicoarsening_level
1041 : && info.semicoarsening_direction != -1)
1042 : {
1043 : rr_level[info.semicoarsening_direction] = 1;
1044 : }
1045 : IntVect is_coarsenable;
1046 : for (int idim = 0; idim < AMREX_SPACEDIM; ++idim) {
1047 : IntVect rr_dir(1);
1048 : rr_dir[idim] = rr_vec[idim] * rr_level[idim];
1049 : is_coarsenable[idim] = dom0.coarsenable(rr_dir, mg_domain_min_width_v)
1050 : && a_grids[0].coarsenable(rr_dir, mg_box_min_width_v);
1051 : if (!is_coarsenable[idim] && do_semicoarsening_level
1052 : && info.semicoarsening_direction == -1)
1053 : {
1054 : is_coarsenable[idim] = true;
1055 : rr_level[idim] = 1;
1056 : }
1057 : }
1058 : if (is_coarsenable != IntVect(1) || rr_level == IntVect(1)) {
1059 : break;
1060 : }
1061 : if (do_semicoarsening_level && info.semicoarsening_direction == -1) {
1062 : // make sure there is at most one direction that is not coarsened
1063 : int n_ones = AMREX_D_TERM( static_cast<int>(rr_level[0] == 1),
1064 : + static_cast<int>(rr_level[1] == 1),
1065 : + static_cast<int>(rr_level[2] == 1));
1066 : if (n_ones > 1) { break; }
1067 : }
1068 : if (rr_level != mg_coarsen_ratio_v) {
1069 : ++numsclevs;
1070 : }
1071 : rr_vec *= rr_level;
1072 :
1073 : m_geom[0].emplace_back(amrex::coarsen(dom0, rr_vec), rb, coord, is_per);
1074 : m_grids[0].push_back(amrex::coarsen(a_grids[0], rr_vec));
1075 :
1076 : if (info.do_consolidation)
1077 : {
1078 : if (avg_npts/static_cast<Real>(AMREX_D_TERM(rr_vec[0], *rr_vec[1], *rr_vec[2]))
1079 : < Real(0.999)*static_cast<Real>(consolidation_threshold))
1080 : {
1081 : coned = true;
1082 : con_lev = m_dmap[0].size();
1083 : m_dmap[0].push_back(DistributionMapping());
1084 : }
1085 : else
1086 : {
1087 : m_dmap[0].push_back(m_dmap[0].back());
1088 : }
1089 : }
1090 : else
1091 : {
1092 : m_dmap[0].push_back(a_dmap[0]);
1093 : }
1094 : }
1095 : }
1096 :
1097 : m_num_mg_levels[0] = m_grids[0].size();
1098 :
1099 : for (int mglev = 0; mglev < m_num_mg_levels[0] - 1; mglev++){
1100 : const Box& fine_domain = m_geom[0][mglev].Domain();
1101 : const Box& crse_domain = m_geom[0][mglev+1].Domain();
1102 : mg_coarsen_ratio_vec.push_back(fine_domain.length()/crse_domain.length());
1103 : }
1104 :
1105 : for (int amrlev = 0; amrlev < m_num_amr_levels; ++amrlev) {
1106 : if (AMRRefRatio(amrlev) == 4 && mg_coarsen_ratio_vec.empty()) {
1107 : mg_coarsen_ratio_vec.push_back(IntVect(2));
1108 : }
1109 : }
1110 :
1111 : if (agged)
1112 : {
1113 : makeAgglomeratedDMap(m_grids[0], m_dmap[0]);
1114 : }
1115 : else if (coned)
1116 : {
1117 : makeConsolidatedDMap(m_grids[0], m_dmap[0], info.con_ratio, info.con_strategy);
1118 : }
1119 :
1120 : if (agged || coned)
1121 : {
1122 : m_bottom_comm = makeSubCommunicator(m_dmap[0].back());
1123 : }
1124 : else
1125 : {
1126 : m_bottom_comm = m_default_comm;
1127 : }
1128 :
1129 : m_do_agglomeration = agged;
1130 : m_do_consolidation = coned;
1131 :
1132 : if (verbose > 1) {
1133 : if (agged) {
1134 : Print() << "MLLinOp::defineGrids(): agglomerated AMR level 0 starting at MG level "
1135 : << agg_lev << " of " << m_num_mg_levels[0] << "\n";
1136 : } else if (coned) {
1137 : Print() << "MLLinOp::defineGrids(): consolidated AMR level 0 starting at MG level "
1138 : << con_lev << " of " << m_num_mg_levels[0]
1139 : << " (ratio = " << info.con_ratio << ")" << "\n";
1140 : } else {
1141 : Print() << "MLLinOp::defineGrids(): no agglomeration or consolidation of AMR level 0\n";
1142 : }
1143 : }
1144 :
1145 : for (int amrlev = 0; amrlev < m_num_amr_levels; ++amrlev)
1146 : {
1147 : for (int mglev = 1; mglev < m_num_mg_levels[amrlev]; ++mglev)
1148 : {
1149 : m_factory[amrlev].emplace_back(makeFactory(amrlev,mglev));
1150 : }
1151 : }
1152 :
1153 : for (int amrlev = 1; amrlev < m_num_amr_levels; ++amrlev)
1154 : {
1155 : AMREX_ASSERT_WITH_MESSAGE(m_grids[amrlev][0].coarsenable(m_amr_ref_ratio[amrlev-1]),
1156 : "MLLinOp: grids not coarsenable between AMR levels");
1157 : }
1158 : }
1159 :
1160 : template <typename MF>
1161 : void
1162 : MLLinOpT<MF>::defineBC ()
1163 : {
1164 : m_needs_coarse_data_for_bc = !m_domain_covered[0];
1165 :
1166 : levelbc_raii.resize(m_num_amr_levels);
1167 : robin_a_raii.resize(m_num_amr_levels);
1168 : robin_b_raii.resize(m_num_amr_levels);
1169 : robin_f_raii.resize(m_num_amr_levels);
1170 : }
1171 :
1172 : template <typename MF>
1173 : void
1174 : MLLinOpT<MF>::setDomainBC (const Array<BCType,AMREX_SPACEDIM>& a_lobc,
1175 : const Array<BCType,AMREX_SPACEDIM>& a_hibc) noexcept
1176 : {
1177 : const int ncomp = getNComp();
1178 : setDomainBC(Vector<Array<BCType,AMREX_SPACEDIM> >(ncomp,a_lobc),
1179 : Vector<Array<BCType,AMREX_SPACEDIM> >(ncomp,a_hibc));
1180 : }
1181 :
1182 : template <typename MF>
1183 : void
1184 : MLLinOpT<MF>::setDomainBC (const Vector<Array<BCType,AMREX_SPACEDIM> >& a_lobc,
1185 : const Vector<Array<BCType,AMREX_SPACEDIM> >& a_hibc) noexcept
1186 : {
1187 : const int ncomp = getNComp();
1188 : AMREX_ASSERT_WITH_MESSAGE(ncomp == a_lobc.size() && ncomp == a_hibc.size(),
1189 : "MLLinOp::setDomainBC: wrong size");
1190 : m_lobc = a_lobc;
1191 : m_hibc = a_hibc;
1192 : m_lobc_orig = m_lobc;
1193 : m_hibc_orig = m_hibc;
1194 : for (int icomp = 0; icomp < ncomp; ++icomp) {
1195 : for (int idim = 0; idim < AMREX_SPACEDIM; ++idim) {
1196 : if (m_geom[0][0].isPeriodic(idim)) {
1197 : AMREX_ALWAYS_ASSERT(m_lobc[icomp][idim] == BCType::Periodic &&
1198 : m_hibc[icomp][idim] == BCType::Periodic);
1199 : } else {
1200 : AMREX_ALWAYS_ASSERT(m_lobc[icomp][idim] != BCType::Periodic &&
1201 : m_hibc[icomp][idim] != BCType::Periodic);
1202 : }
1203 :
1204 : if (m_lobc[icomp][idim] == LinOpBCType::inhomogNeumann ||
1205 : m_lobc[icomp][idim] == LinOpBCType::Robin)
1206 : {
1207 : m_lobc[icomp][idim] = LinOpBCType::Neumann;
1208 : }
1209 :
1210 : if (m_hibc[icomp][idim] == LinOpBCType::inhomogNeumann ||
1211 : m_hibc[icomp][idim] == LinOpBCType::Robin)
1212 : {
1213 : m_hibc[icomp][idim] = LinOpBCType::Neumann;
1214 : }
1215 : }
1216 : }
1217 :
1218 : if (hasHiddenDimension()) {
1219 : const int hd = hiddenDirection();
1220 : for (int n = 0; n < ncomp; ++n) {
1221 : m_lobc[n][hd] = LinOpBCType::Neumann;
1222 : m_hibc[n][hd] = LinOpBCType::Neumann;
1223 : }
1224 : }
1225 :
1226 : if (hasInhomogNeumannBC() && !supportInhomogNeumannBC()) {
1227 : amrex::Abort("Inhomogeneous Neumann BC not supported");
1228 : }
1229 : if (hasRobinBC() && !supportRobinBC()) {
1230 : amrex::Abort("Robin BC not supported");
1231 : }
1232 : }
1233 :
1234 : template <typename MF>
1235 : bool
1236 : MLLinOpT<MF>::hasBC (BCType bct) const noexcept
1237 : {
1238 : int ncomp = m_lobc_orig.size();
1239 : for (int n = 0; n < ncomp; ++n) {
1240 : for (int idim = 0; idim <AMREX_SPACEDIM; ++idim) {
1241 : if (m_lobc_orig[n][idim] == bct || m_hibc_orig[n][idim] == bct) {
1242 : return true;
1243 : }
1244 : }
1245 : }
1246 : return false;
1247 : }
1248 :
1249 : template <typename MF>
1250 : bool
1251 : MLLinOpT<MF>::hasInhomogNeumannBC () const noexcept
1252 : {
1253 : return hasBC(BCType::inhomogNeumann);
1254 : }
1255 :
1256 : template <typename MF>
1257 : bool
1258 : MLLinOpT<MF>::hasRobinBC () const noexcept
1259 : {
1260 : return hasBC(BCType::Robin);
1261 : }
1262 :
1263 : template <typename MF>
1264 : Box
1265 : MLLinOpT<MF>::compactify (Box const& b) const noexcept
1266 : {
1267 : #if (AMREX_SPACEDIM == 3)
1268 : if (info.hasHiddenDimension()) {
1269 : const auto& lo = b.smallEnd();
1270 : const auto& hi = b.bigEnd();
1271 : if (info.hidden_direction == 0) {
1272 : return Box(IntVect(lo[1],lo[2],0), IntVect(hi[1],hi[2],0), b.ixType());
1273 : } else if (info.hidden_direction == 1) {
1274 : return Box(IntVect(lo[0],lo[2],0), IntVect(hi[0],hi[2],0), b.ixType());
1275 : } else {
1276 : return Box(IntVect(lo[0],lo[1],0), IntVect(hi[0],hi[1],0), b.ixType());
1277 : }
1278 : } else
1279 : #endif
1280 : {
1281 : return b;
1282 : }
1283 : }
1284 :
1285 : template <typename MF>
1286 : void
1287 : MLLinOpT<MF>::makeAgglomeratedDMap (const Vector<BoxArray>& ba,
1288 : Vector<DistributionMapping>& dm)
1289 : {
1290 : BL_PROFILE("MLLinOp::makeAgglomeratedDMap");
1291 :
1292 : BL_ASSERT(!dm[0].empty());
1293 : for (int i = 1, N=static_cast<int>(ba.size()); i < N; ++i)
1294 : {
1295 : if (dm[i].empty())
1296 : {
1297 : const std::vector< std::vector<int> >& sfc = DistributionMapping::makeSFC(ba[i]);
1298 :
1299 : const int nprocs = ParallelContext::NProcsSub();
1300 : AMREX_ASSERT(static_cast<int>(sfc.size()) == nprocs);
1301 :
1302 : Vector<int> pmap(ba[i].size());
1303 : for (int iproc = 0; iproc < nprocs; ++iproc) {
1304 : int grank = ParallelContext::local_to_global_rank(iproc);
1305 : for (int ibox : sfc[iproc]) {
1306 : pmap[ibox] = grank;
1307 : }
1308 : }
1309 : dm[i].define(std::move(pmap));
1310 : }
1311 : }
1312 : }
1313 :
1314 : template <typename MF>
1315 : void
1316 : MLLinOpT<MF>::makeConsolidatedDMap (const Vector<BoxArray>& ba,
1317 : Vector<DistributionMapping>& dm,
1318 : int ratio, int strategy)
1319 : {
1320 : BL_PROFILE("MLLinOp::makeConsolidatedDMap()");
1321 :
1322 : int factor = 1;
1323 : BL_ASSERT(!dm[0].empty());
1324 : for (int i = 1, N=static_cast<int>(ba.size()); i < N; ++i)
1325 : {
1326 : if (dm[i].empty())
1327 : {
1328 : factor *= ratio;
1329 :
1330 : const int nprocs = ParallelContext::NProcsSub();
1331 : const auto& pmap_fine = dm[i-1].ProcessorMap();
1332 : Vector<int> pmap(pmap_fine.size());
1333 : ParallelContext::global_to_local_rank(pmap.data(), pmap_fine.data(), static_cast<int>(pmap.size()));
1334 : if (strategy == 1) {
1335 : for (auto& x: pmap) {
1336 : x /= ratio;
1337 : }
1338 : } else if (strategy == 2) {
1339 : int nprocs_con = static_cast<int>(std::ceil(static_cast<Real>(nprocs)
1340 : / static_cast<Real>(factor)));
1341 : for (auto& x: pmap) {
1342 : auto d = std::div(x,nprocs_con);
1343 : x = d.rem;
1344 : }
1345 : } else if (strategy == 3) {
1346 : if (factor == ratio) {
1347 : const std::vector< std::vector<int> >& sfc = DistributionMapping::makeSFC(ba[i]);
1348 : for (int iproc = 0; iproc < nprocs; ++iproc) {
1349 : for (int ibox : sfc[iproc]) {
1350 : pmap[ibox] = iproc;
1351 : }
1352 : }
1353 : }
1354 : for (auto& x: pmap) {
1355 : x /= ratio;
1356 : }
1357 : }
1358 :
1359 : if (ParallelContext::CommunicatorSub() == ParallelDescriptor::Communicator()) {
1360 : dm[i].define(std::move(pmap));
1361 : } else {
1362 : Vector<int> pmap_g(pmap.size());
1363 : ParallelContext::local_to_global_rank(pmap_g.data(), pmap.data(), static_cast<int>(pmap.size()));
1364 : dm[i].define(std::move(pmap_g));
1365 : }
1366 : }
1367 : }
1368 : }
1369 :
1370 : template <typename MF>
1371 : MPI_Comm
1372 : MLLinOpT<MF>::makeSubCommunicator (const DistributionMapping& dm)
1373 : {
1374 : BL_PROFILE("MLLinOp::makeSubCommunicator()");
1375 :
1376 : #ifdef BL_USE_MPI
1377 :
1378 : Vector<int> newgrp_ranks = dm.ProcessorMap();
1379 : std::sort(newgrp_ranks.begin(), newgrp_ranks.end());
1380 : auto last = std::unique(newgrp_ranks.begin(), newgrp_ranks.end());
1381 : newgrp_ranks.erase(last, newgrp_ranks.end());
1382 :
1383 : MPI_Comm newcomm;
1384 : MPI_Group defgrp, newgrp;
1385 : MPI_Comm_group(m_default_comm, &defgrp);
1386 : if (ParallelContext::CommunicatorSub() == ParallelDescriptor::Communicator()) {
1387 : MPI_Group_incl(defgrp, static_cast<int>(newgrp_ranks.size()), newgrp_ranks.data(), &newgrp);
1388 : } else {
1389 : Vector<int> local_newgrp_ranks(newgrp_ranks.size());
1390 : ParallelContext::global_to_local_rank(local_newgrp_ranks.data(),
1391 : newgrp_ranks.data(), static_cast<int>(newgrp_ranks.size()));
1392 : MPI_Group_incl(defgrp, static_cast<int>(local_newgrp_ranks.size()), local_newgrp_ranks.data(), &newgrp);
1393 : }
1394 :
1395 : MPI_Comm_create(m_default_comm, newgrp, &newcomm);
1396 :
1397 : m_raii_comm = std::make_unique<CommContainer>(newcomm);
1398 :
1399 : MPI_Group_free(&defgrp);
1400 : MPI_Group_free(&newgrp);
1401 :
1402 : return newcomm;
1403 : #else
1404 : amrex::ignore_unused(dm);
1405 : return m_default_comm;
1406 : #endif
1407 : }
1408 :
1409 : template <typename MF>
1410 : void
1411 : MLLinOpT<MF>::setDomainBCLoc (const Array<Real,AMREX_SPACEDIM>& lo_bcloc,
1412 : const Array<Real,AMREX_SPACEDIM>& hi_bcloc) noexcept
1413 : {
1414 : m_domain_bloc_lo = lo_bcloc;
1415 : m_domain_bloc_hi = hi_bcloc;
1416 : }
1417 :
1418 : template <typename MF>
1419 : void
1420 : MLLinOpT<MF>::setCoarseFineBC (const MF* crse, int crse_ratio,
1421 : LinOpBCType bc_type) noexcept
1422 : {
1423 : setCoarseFineBC(crse, IntVect(crse_ratio), bc_type);
1424 : }
1425 :
1426 : template <typename MF>
1427 : void
1428 : MLLinOpT<MF>::setCoarseFineBC (const MF* crse, IntVect const& crse_ratio,
1429 : LinOpBCType bc_type) noexcept
1430 : {
1431 : m_coarse_data_for_bc = crse;
1432 : m_coarse_data_crse_ratio = crse_ratio;
1433 : m_coarse_fine_bc_type = bc_type;
1434 : }
1435 :
1436 : template <typename MF>
1437 : template <typename AMF, std::enable_if_t<!std::is_same_v<MF,AMF>,int>>
1438 : void
1439 : MLLinOpT<MF>::setCoarseFineBC (const AMF* crse, int crse_ratio,
1440 : LinOpBCType bc_type) noexcept
1441 : {
1442 : setCoarseFineBC(crse, IntVect(crse_ratio), bc_type);
1443 : }
1444 :
1445 : template <typename MF>
1446 : template <typename AMF, std::enable_if_t<!std::is_same_v<MF,AMF>,int>>
1447 : void
1448 : MLLinOpT<MF>::setCoarseFineBC (const AMF* crse, IntVect const& crse_ratio,
1449 : LinOpBCType bc_type) noexcept
1450 : {
1451 : m_coarse_data_for_bc_raii = MF(crse->boxArray(), crse->DistributionMap(),
1452 : crse->nComp(), crse->nGrowVect());
1453 : m_coarse_data_for_bc_raii.LocalCopy(*crse, 0, 0, crse->nComp(),
1454 : crse->nGrowVect());
1455 : m_coarse_data_for_bc = &m_coarse_data_for_bc_raii;
1456 : m_coarse_data_crse_ratio = crse_ratio;
1457 : m_coarse_fine_bc_type = bc_type;
1458 : }
1459 :
1460 : template <typename MF>
1461 : void
1462 : MLLinOpT<MF>::make (Vector<Vector<MF> >& mf, IntVect const& ng) const
1463 : {
1464 : mf.clear();
1465 : mf.resize(m_num_amr_levels);
1466 : for (int alev = 0; alev < m_num_amr_levels; ++alev) {
1467 : mf[alev].resize(m_num_mg_levels[alev]);
1468 : for (int mlev = 0; mlev < m_num_mg_levels[alev]; ++mlev) {
1469 : mf[alev][mlev] = make(alev, mlev, ng);
1470 : }
1471 : }
1472 : }
1473 :
1474 : template <typename MF>
1475 : MF
1476 : MLLinOpT<MF>::make (int amrlev, int mglev, IntVect const& ng) const
1477 : {
1478 : if constexpr (IsMultiFabLike_v<MF>) {
1479 : return MF(amrex::convert(m_grids[amrlev][mglev], m_ixtype),
1480 : m_dmap[amrlev][mglev], getNComp(), ng, MFInfo(),
1481 : *m_factory[amrlev][mglev]);
1482 : } else {
1483 : amrex::ignore_unused(amrlev, mglev, ng);
1484 : amrex::Abort("MLLinOpT::make: how did we get here?");
1485 : return {};
1486 : }
1487 : }
1488 :
1489 : template <typename MF>
1490 : MF
1491 : MLLinOpT<MF>::makeAlias (MF const& mf) const
1492 : {
1493 : if constexpr (IsMultiFabLike_v<MF>) {
1494 : return MF(mf, amrex::make_alias, 0, mf.nComp());
1495 : } else {
1496 : amrex::ignore_unused(mf);
1497 : amrex::Abort("MLLinOpT::makeAlias: how did we get here?");
1498 : return {};
1499 : }
1500 : }
1501 :
1502 : template <typename MF>
1503 : MF
1504 : MLLinOpT<MF>::makeCoarseMG (int amrlev, int mglev, IntVect const& ng) const
1505 : {
1506 : if constexpr (IsMultiFabLike_v<MF>) {
1507 : BoxArray cba = m_grids[amrlev][mglev];
1508 : IntVect ratio = (amrlev > 0) ? IntVect(2) : mg_coarsen_ratio_vec[mglev];
1509 : cba.coarsen(ratio);
1510 : cba.convert(m_ixtype);
1511 : return MF(cba, m_dmap[amrlev][mglev], getNComp(), ng);
1512 : } else {
1513 : amrex::ignore_unused(amrlev, mglev, ng);
1514 : amrex::Abort("MLLinOpT::makeCoarseMG: how did we get here?");
1515 : return {};
1516 : }
1517 : }
1518 :
1519 : template <typename MF>
1520 : MF
1521 : MLLinOpT<MF>::makeCoarseAmr (int famrlev, IntVect const& ng) const
1522 : {
1523 : if constexpr (IsMultiFabLike_v<MF>) {
1524 : BoxArray cba = m_grids[famrlev][0];
1525 : IntVect ratio(AMRRefRatio(famrlev-1));
1526 : cba.coarsen(ratio);
1527 : cba.convert(m_ixtype);
1528 : return MF(cba, m_dmap[famrlev][0], getNComp(), ng);
1529 : } else {
1530 : amrex::ignore_unused(famrlev, ng);
1531 : amrex::Abort("MLLinOpT::makeCoarseAmr: how did we get here?");
1532 : return {};
1533 : }
1534 : }
1535 :
1536 : template <typename MF>
1537 : void
1538 : MLLinOpT<MF>::resizeMultiGrid (int new_size)
1539 : {
1540 : if (new_size <= 0 || new_size >= m_num_mg_levels[0]) { return; }
1541 :
1542 : m_num_mg_levels[0] = new_size;
1543 :
1544 : m_geom[0].resize(new_size);
1545 : m_grids[0].resize(new_size);
1546 : m_dmap[0].resize(new_size);
1547 : m_factory[0].resize(new_size);
1548 :
1549 : if (m_bottom_comm != m_default_comm) {
1550 : m_bottom_comm = makeSubCommunicator(m_dmap[0].back());
1551 : }
1552 : }
1553 :
1554 : template <typename MF>
1555 : void
1556 : MLLinOpT<MF>::avgDownResMG (int clev, MF& cres, MF const& fres) const
1557 : {
1558 : amrex::ignore_unused(clev, cres, fres);
1559 : if constexpr (amrex::IsFabArray<MF>::value) {
1560 : const int ncomp = this->getNComp();
1561 : #ifdef AMREX_USE_EB
1562 : if (!fres.isAllRegular()) {
1563 : if constexpr (std::is_same<MF,MultiFab>()) {
1564 : amrex::EB_average_down(fres, cres, 0, ncomp,
1565 : mg_coarsen_ratio_vec[clev-1]);
1566 : } else {
1567 : amrex::Abort("EB_average_down only works with MultiFab");
1568 : }
1569 : } else
1570 : #endif
1571 : {
1572 : amrex::average_down(fres, cres, 0, ncomp, mg_coarsen_ratio_vec[clev-1]);
1573 : }
1574 : } else {
1575 : amrex::Abort("For non-FabArray, MLLinOpT<MF>::avgDownResMG should be overridden.");
1576 : }
1577 : }
1578 :
1579 : template <typename MF>
1580 : bool
1581 : MLLinOpT<MF>::isMFIterSafe (int amrlev, int mglev1, int mglev2) const
1582 : {
1583 : return m_dmap[amrlev][mglev1] == m_dmap[amrlev][mglev2]
1584 : && BoxArray::SameRefs(m_grids[amrlev][mglev1], m_grids[amrlev][mglev2]);
1585 : }
1586 :
1587 : template <typename MF>
1588 : template <typename AMF, std::enable_if_t<!std::is_same_v<MF,AMF>,int>>
1589 : void
1590 : MLLinOpT<MF>::setLevelBC (int amrlev, const AMF* levelbcdata,
1591 : const AMF* robinbc_a, const AMF* robinbc_b,
1592 : const AMF* robinbc_f)
1593 : {
1594 : const int ncomp = this->getNComp();
1595 : if (levelbcdata) {
1596 : levelbc_raii[amrlev] = std::make_unique<MF>(levelbcdata->boxArray(),
1597 : levelbcdata->DistributionMap(),
1598 : ncomp, levelbcdata->nGrowVect());
1599 : levelbc_raii[amrlev]->LocalCopy(*levelbcdata, 0, 0, ncomp,
1600 : levelbcdata->nGrowVect());
1601 : } else {
1602 : levelbc_raii[amrlev].reset();
1603 : }
1604 :
1605 : if (robinbc_a) {
1606 : robin_a_raii[amrlev] = std::make_unique<MF>(robinbc_a->boxArray(),
1607 : robinbc_a->DistributionMap(),
1608 : ncomp, robinbc_a->nGrowVect());
1609 : robin_a_raii[amrlev]->LocalCopy(*robinbc_a, 0, 0, ncomp,
1610 : robinbc_a->nGrowVect());
1611 : } else {
1612 : robin_a_raii[amrlev].reset();
1613 : }
1614 :
1615 : if (robinbc_b) {
1616 : robin_b_raii[amrlev] = std::make_unique<MF>(robinbc_b->boxArray(),
1617 : robinbc_b->DistributionMap(),
1618 : ncomp, robinbc_b->nGrowVect());
1619 : robin_b_raii[amrlev]->LocalCopy(*robinbc_b, 0, 0, ncomp,
1620 : robinbc_b->nGrowVect());
1621 : } else {
1622 : robin_b_raii[amrlev].reset();
1623 : }
1624 :
1625 : if (robinbc_f) {
1626 : robin_f_raii[amrlev] = std::make_unique<MF>(robinbc_f->boxArray(),
1627 : robinbc_f->DistributionMap(),
1628 : ncomp, robinbc_f->nGrowVect());
1629 : robin_f_raii[amrlev]->LocalCopy(*robinbc_f, 0, 0, ncomp,
1630 : robinbc_f->nGrowVect());
1631 : } else {
1632 : robin_f_raii[amrlev].reset();
1633 : }
1634 :
1635 : this->setLevelBC(amrlev, levelbc_raii[amrlev].get(), robin_a_raii[amrlev].get(),
1636 : robin_b_raii[amrlev].get(), robin_f_raii[amrlev].get());
1637 : }
1638 :
1639 : extern template class MLLinOpT<MultiFab>;
1640 :
1641 : using MLLinOp = MLLinOpT<MultiFab>;
1642 :
1643 : }
1644 :
1645 : #endif
|