Block-Structured AMR Software Framework
Loading...
Searching...
No Matches
AMReX_NonLocalBC.H
Go to the documentation of this file.
1#ifndef AMREX_NONLOCAL_BC_H_
2#define AMREX_NONLOCAL_BC_H_
3#include <AMReX_Config.H>
4#include <AMReX_Concepts.H>
5#include <AMReX_TypeTraits.H>
6#include <AMReX_FabArray.H>
7#include <AMReX_FArrayBox.H>
8
9namespace amrex::NonLocalBC {
10
12// [concept.IndexMapping]
13//
14
16template <typename T, typename... Args>
17using Inverse_t = decltype(std::declval<T>().Inverse(std::declval<Args>()...));
18
20template <typename T>
21struct HasInverseMemFn : IsDetectedExact<Dim3, Inverse_t, T, Dim3> {};
22
25template <class IndexMap>
27 : Conjunction<IsCallableR<Dim3, IndexMap&, const Dim3&>,
28 HasInverseMemFn<const IndexMap&>> {};
29
31// [IndexMapping.MultiBlockIndexMapping]
32//
33
45
52 [[nodiscard]] AMREX_GPU_HOST_DEVICE Dim3 operator()(Dim3 i) const noexcept {
53 int iv[3]{i.x, i.y, i.z};
54 int iv_new[3]{};
55 for (int d = 0; d < AMREX_SPACEDIM; ++d) {
56 iv_new[d] = sign[d] * (iv[permutation[d]] - offset[d]);
57 }
58 return Dim3{.x = iv_new[0], .y = iv_new[1], .z = iv_new[2]};
59 }
60
67 [[nodiscard]] AMREX_GPU_HOST_DEVICE Dim3 Inverse(Dim3 i) const noexcept {
68 int iv_new[3]{i.x, i.y, i.z};
69 int iv[3]{};
70 for (int d = 0; d < AMREX_SPACEDIM; ++d) {
71 AMREX_ASSERT(sign[d] == 1 || sign[d] == -1);
72 iv[permutation[d]] = iv_new[d] * sign[d] + offset[d];
73 }
74 return Dim3{.x = iv[0], .y = iv[1], .z = iv[2]};
75 }
76
77 [[nodiscard]] IndexType operator()(IndexType it) const noexcept {
78 return IndexType{IntVect{AMREX_D_DECL(it[permutation[0]], it[permutation[1]], it[permutation[2]])}};
79 }
80
81 [[nodiscard]] IndexType Inverse(IndexType it) const noexcept {
82 IntVect inverse_permutation;
83 for (int i = 0; i < AMREX_SPACEDIM; ++i) {
84 inverse_permutation[permutation[i]] = i;
85 }
86 return IndexType{IntVect{AMREX_D_DECL(it[inverse_permutation[0]], it[inverse_permutation[1]], it[inverse_permutation[2]])}};
87 }
88};
89
102template <typename DTOS>
105Apply(DTOS const& dtos, const IntVect& iv)
106{
107 Dim3 i = dtos(iv.dim3());
108 return IntVect{AMREX_D_DECL(i.x, i.y, i.z)};
109}
110
120template <typename DTOS>
122Box
123Image (DTOS const& dtos, const Box& box)
124{
125 IntVect mapped_smallEnd = Apply(dtos, box.smallEnd());
126 IntVect mapped_bigEnd = Apply(dtos, box.bigEnd());
127 IntVect smallEnd;
128 IntVect bigEnd;
129 for (int d = 0; d < AMREX_SPACEDIM; ++d) {
130 smallEnd[d] = std::min(mapped_smallEnd[d], mapped_bigEnd[d]);
131 bigEnd[d] = std::max(mapped_smallEnd[d], mapped_bigEnd[d]);
132 }
133 return Box{smallEnd, bigEnd, box.ixType()};
134}
135
145template <typename DTOS>
147Box
148Image (DTOS const& dtos, const Box& box)
149{
150 // "Forget" the index type mapping and invoke Image without changing the index type.
151 Box srcbox = Image([&dtos](Dim3 d) { return dtos(d); }, box);
152 // Fix the index type of the resulting box
153 srcbox.setType(dtos(box.ixType()));
154 return srcbox;
155}
156
169template <typename DTOS>
170requires (HasInverseMemFn<DTOS>::value)
172ApplyInverse(DTOS const& dtos, const IntVect& iv)
173{
174 return Apply([&dtos](Dim3 i) { return dtos.Inverse(i); }, iv);
175}
176
186template <typename DTOS>
187requires (HasInverseMemFn<DTOS>::value && !IsCallableR<IndexType, DTOS, IndexType>::value)
188Box
189InverseImage (DTOS const& dtos, const Box& box)
190{
191 return Image([&dtos](Dim3 i) { return dtos.Inverse(i); }, box);
192}
193
203template <typename DTOS>
204requires (HasInverseMemFn<DTOS>::value && IsCallableR<IndexType, DTOS, IndexType>::value)
205Box
206InverseImage (DTOS const& dtos, const Box& box)
207{
208 return Image([&dtos](auto&& i) { return dtos.Inverse(i); }, box);
209}
210
212static_assert(IsIndexMapping<MultiBlockIndexMapping>(), // NOLINT(bugprone-throw-keyword-missing)
213 "MultiBlockIndexMapping is expected to satisfy IndexMapping");
214
216// [class.MultiBlockCommMetaData]
217//
218
225
229 template <typename DTOS>
231 MultiBlockCommMetaData(const FabArrayBase& dst, const Box& dstbox, const FabArrayBase& src,
232 const IntVect& ngrow, DTOS const& dtos);
233
237 template <typename DTOS>
240 const Box& dstbox, const BoxArray& srcba,
241 const DistributionMapping& srcdm, const IntVect& ngrow, DTOS const& dtos);
242
244
267 template <typename DTOS>
269 void define(const BoxArray& dstba, const DistributionMapping& dstdm, const Box& dstbox,
270 const BoxArray& srcba, const DistributionMapping& srcdm, const IntVect& ngrow,
271 DTOS const& dtos);
272};
273
275// [concept.FabProjection]
276//
277
279template <typename P, BaseFabType FAB>
281 : IsCallableR<typename FAB::value_type, P, Array4<const typename FAB::value_type>, Dim3, int>
282{};
283
285// [FabProjection.Identity]
286// [IndexMapping.Identity]
287
291struct Identity {
293 constexpr Dim3 operator()(Dim3 i) const noexcept { return i; }
295 [[nodiscard]] static constexpr Dim3 Inverse(Dim3 i) noexcept { return i; }
296
298 template <typename T>
299 constexpr T operator()(Array4<const T> array, Dim3 i, int comp = 0) const
300 noexcept(noexcept(array(i.x, i.y, i.z, comp))) {
301 return array(i.x, i.y, i.z, comp);
302 }
303
305 constexpr int operator()(int i) const noexcept { return i; }
306};
307static constexpr Identity identity{};
308
309static_assert(sizeof(Identity) == 1 );
310static_assert(std::is_trivially_default_constructible_v<Identity> );
311static_assert(std::is_trivially_copy_assignable_v<Identity> );
312static_assert(std::is_trivially_copy_constructible_v<Identity> );
313static_assert(IsIndexMapping<Identity>() ); // NOLINT(bugprone-throw-keyword-missing)
314static_assert(IsFabProjection<Identity, FArrayBox>() ); // NOLINT(bugprone-throw-keyword-missing)
315
317// [FabProjection.MapComponents]
318
325template <typename Base, typename Map = Identity> struct MapComponents {
326 static_assert(IsCallable<Base, Array4<const Real>, Dim3, int>::value,
327 "Base needs to be a callable function: (Array4<const T>, Dim3, i) -> auto.");
328
330 "Map needs to be a callable function: int -> int.");
331
332 Base base;
333 Map map;
334
335 template <typename T>
337 constexpr decltype(auto) operator()(Array4<const T> array, Dim3 i, int comp) const
338 noexcept(noexcept(base(array, i, map(comp))))
339 {
340 return base(array, i, map(comp));
341 }
342};
343
344static_assert(std::is_trivially_copy_assignable<MapComponents<Identity>>() ); // NOLINT(bugprone-throw-keyword-missing)
345static_assert(std::is_trivially_copy_constructible<MapComponents<Identity>>() ); // NOLINT(bugprone-throw-keyword-missing)
346static_assert(IsFabProjection<MapComponents<Identity>, FArrayBox>() ); // NOLINT(bugprone-throw-keyword-missing)
347
349// [FabProjection.MapComponents.SwapComponents]
350
356template <int I, int J> struct SwapComponents {
357 static_assert(I >= 0 && J >= 0, "I >= 0 && J >= 0");
358
362 constexpr int operator()(int i) const noexcept {
363 const int map[2] = {I, J};
364 return i == I || i == J ? map[std::size_t(i == I)] : i;
365 }
366};
367
368template <int I> struct SwapComponents<I, -1> {
369 static_assert(I >= 0, "I >= 0");
370
371 int J;
372 constexpr int operator()(int i) const noexcept {
373 const int map[2] = {I, J};
374 return i == I || i == J ? map[std::size_t(i == I)] : i;
375 }
376};
377
378template <int J> struct SwapComponents<-1, J> {
379 static_assert(J >= 0, "J >= 0");
380
381 int I;
382 constexpr int operator()(int i) const noexcept {
383 const int map[2] = {I, J};
384 return i == I || i == J ? map[std::size_t(i == I)] : i;
385 }
386};
387
388template <> struct SwapComponents<-1, -1> {
389 int I;
390 int J;
391 constexpr int operator()(int i) const noexcept {
392 const int map[2] = {I, J};
393 return i == I || i == J ? map[std::size_t(i == I)] : i;
394 }
395};
396
398
399template <int I, int J> static constexpr SwapComponents<I, J> swap_indices{};
400
401static_assert(sizeof(SwapComponents<0, 1>) == 1 );
402static_assert(sizeof(DynamicSwapComponents) == 2 * sizeof(int) );
403static_assert(sizeof(SwapComponents<0, -1>) == sizeof(int) );
404static_assert(sizeof(SwapComponents<-1, 1>) == sizeof(int) );
405static_assert(std::is_trivially_default_constructible<MapComponents<Identity, SwapComponents<0, 1>>>() ); // NOLINT(bugprone-throw-keyword-missing)
406static_assert(std::is_trivially_copy_assignable<MapComponents<Identity, SwapComponents<0, 1>>>() ); // NOLINT(bugprone-throw-keyword-missing)
407static_assert(std::is_trivially_copy_constructible<MapComponents<Identity, SwapComponents<0, 1>>>() ); // NOLINT(bugprone-throw-keyword-missing)
408static_assert(IsFabProjection<MapComponents<Identity, SwapComponents<0, 1>>, FArrayBox>() ); // NOLINT(bugprone-throw-keyword-missing)
409
410static_assert(swap_indices<0, 1>(0) == 1 );
411static_assert(swap_indices<0, 1>(1) == 0 );
412static_assert(swap_indices<0, 1>(2) == 2 );
413static_assert(DynamicSwapComponents{0, 1}(0) == 1 );
414static_assert(DynamicSwapComponents{0, 1}(1) == 0 );
415static_assert(DynamicSwapComponents{0, 1}(2) == 2 );
416
418// [class.CommData]
419// [class.CommHandler]
420
447
448#ifdef AMREX_USE_MPI
451 int n_components, std::size_t object_size, std::size_t align);
452
454void PostRecvs(CommData& recv, int mpi_tag);
455
457void PostSends(CommData& send, int mpi_tag);
458#endif
459
464#ifdef AMREX_USE_MPI
465 int mpi_tag{};
468#endif
469};
470
472// [concept.DataPacking]
473//
474template <typename... Args>
475using PrepareSendBuffers_t = decltype(PrepareSendBuffers(std::declval<Args>()...));
476
477template <typename... Args>
478using PrepareRecvBuffers_t = decltype(PrepareRecvBuffers(std::declval<Args>()...));
479
480template <typename... Args>
481using PackSendBuffers_t = decltype(PackSendBuffers(std::declval<Args>()...));
482
483template <typename... Args>
484using UnpackRecvBuffers_t = decltype(UnpackRecvBuffers(std::declval<Args>()...));
485
486template <typename... Args>
487using LocalCopy_t = decltype(LocalCopy(std::declval<Args>()...));
488
490#if defined(AMREX_USE_CUDA) && defined(_WIN32)
491template <typename DP, BaseFabType FAB> struct IsDataPacking : std::true_type {};
492#else
493template <typename DP, BaseFabType FAB>
496 IsDetected<LocalCopy_t, DP&, FabArray<FAB>&, const FabArray<FAB>&, const FabArrayBase::CopyComTagsContainer&>
497#ifdef AMREX_USE_MPI
498 ,IsDetected<PrepareSendBuffers_t, DP&, FabArray<FAB>&, const FabArray<FAB>&, CommData&, const FabArrayBase::MapOfCopyComTagContainers&>,
499 IsDetected<PrepareRecvBuffers_t, DP&, FabArray<FAB>&, const FabArray<FAB>&, CommData&, const FabArrayBase::MapOfCopyComTagContainers&>,
500 IsDetected<PackSendBuffers_t, DP&, const FabArray<FAB>&, CommData&>,
501 IsDetected<UnpackRecvBuffers_t, DP&, FabArray<FAB>&, CommData&>
502#endif
503 > {};
504#endif
505
506template <BaseFabType FAB, class DTOS = Identity, class Proj = Identity>
508void
509local_copy_cpu (FabArray<FAB>& dest, const FabArray<FAB>& src, int dcomp, int scomp, int ncomp,
510 FabArrayBase::CopyComTagsContainer const& local_tags, DTOS const& dtos = DTOS{},
511 Proj const& proj = Proj{}) noexcept;
512
513
514template <BaseFabType FAB, class DTOS = Identity, class Proj = Identity>
515requires (IsCallableR<Dim3, DTOS, Dim3>::value && IsFabProjection<Proj, FAB>::value)
516void
517unpack_recv_buffer_cpu (FabArray<FAB>& mf, int dcomp, int ncomp, Vector<char*> const& recv_data,
518 Vector<std::size_t> const& recv_size,
520 DTOS const& dtos = DTOS{}, Proj const& proj = Proj{}) noexcept;
521
522
523#ifdef AMREX_USE_GPU
524template <BaseFabType FAB, class DTOS = Identity, class Proj = Identity>
525requires (IsCallableR<Dim3, DTOS, Dim3>::value && IsFabProjection<Proj, FAB>::value)
526void
527local_copy_gpu (FabArray<FAB>& dest, const FabArray<FAB>& src, int dcomp, int scomp, int ncomp,
528 FabArrayBase::CopyComTagsContainer const& local_tags, DTOS const& dtos = DTOS{},
529 Proj const& proj = Proj{}) noexcept;
530
531
532template <BaseFabType FAB, class DTOS = Identity, class Proj = Identity>
533requires (IsCallableR<Dim3, DTOS, Dim3>::value && IsFabProjection<Proj, FAB>::value)
534void
535unpack_recv_buffer_gpu (FabArray<FAB>& mf, int scomp, int ncomp,
536 Vector<char*> const& recv_data,
537 Vector<std::size_t> const& recv_size,
539 DTOS const& dtos = DTOS{}, Proj const& proj = Proj{});
540#endif
541
543// [DataPacking.PackComponents]
544//
545// PackComponents is the simplest data packing policy.
546// This provides us with sane default behaviour that we can use when defining new
547// data packing policies.
548
555
557template <BaseFabType FAB>
558void
559LocalCopy (const PackComponents& components, FabArray<FAB>& dest, const FabArray<FAB>& src,
560 const FabArrayBase::CopyComTagsContainer& local_tags)
561{
562#ifdef AMREX_USE_GPU
563 if (Gpu::inLaunchRegion()) {
564 local_copy_gpu(dest, src, components.dest_component, components.src_component,
565 components.n_components, local_tags);
566 } else
567#endif
568 {
569 local_copy_cpu(dest, src, components.dest_component, components.src_component,
570 components.n_components, local_tags);
571 }
572}
573
574#ifdef AMREX_USE_MPI
576template <BaseFabType FAB>
577void
578PrepareSendBuffers (const PackComponents& components, FabArray<FAB>& dest, const FabArray<FAB>& src,
580{
581 using T = typename FAB::value_type;
582 ignore_unused(dest, src);
583 PrepareCommBuffers(comm, cctc, components.n_components, sizeof(T), alignof(T));
584}
585
587template <BaseFabType FAB>
588void
589PrepareRecvBuffers (const PackComponents& components, FabArray<FAB>& dest, const FabArray<FAB>& src,
591{
592 using T = typename FAB::value_type;
593 ignore_unused(dest, src);
594 PrepareCommBuffers(comm, cctc, components.n_components, sizeof(T), alignof(T));
595}
596
598template <BaseFabType FAB>
599void
600PackSendBuffers (const PackComponents& components, const FabArray<FAB>& src, CommData& send)
601{
602#ifdef AMREX_USE_GPU
603 if (Gpu::inLaunchRegion()) {
605 send.data, send.size, send.cctc, send.id);
606 } else
607#endif // AMREX_USE_GPU
608 {
610 send.data, send.size, send.cctc);
611 }
612}
613
615template <BaseFabType FAB>
616void
617UnpackRecvBuffers (const PackComponents& components, FabArray<FAB>& dest, const CommData& recv)
618{
619#ifdef AMREX_USE_GPU
620 if (Gpu::inLaunchRegion()) {
621 unpack_recv_buffer_gpu(dest, components.dest_component, components.n_components, recv.data,
622 recv.size, recv.cctc);
623 } else
624#endif // AMREX_USE_GPU
625 {
626 unpack_recv_buffer_cpu(dest, components.dest_component, components.n_components, recv.data,
627 recv.size, recv.cctc);
628 }
629}
630#endif // AMREX_USE_MPI
631
632static_assert(IsDataPacking<PackComponents, FArrayBox>(), // NOLINT(bugprone-throw-keyword-missing)
633 "PackComponents is expected to satisfy the concept DataPacking.");
634
636// [DataPacking.ApplyDtosAndProjectionOnReciever]
637//
641template <typename DTOS = Identity, typename FabProj = Identity>
643 constexpr ApplyDtosAndProjectionOnReciever() = default;
644 constexpr ApplyDtosAndProjectionOnReciever(const PackComponents& components, DTOS dtos_ = DTOS{}, FabProj proj_ = FabProj{})
645 : PackComponents(components), dtos(std::move(dtos_)), proj(std::move(proj_)) {}
646
647 DTOS dtos;
648 FabProj proj;
649
650 static_assert(IsCallableR<Dim3, DTOS, Dim3>::value, "DTOS needs to be a callable: Dim3 -> Dim3");
651 static_assert(IsFabProjection<FabProj, FArrayBox>(), "FabProj needs to be at least a projection on FArrayBox.");
652};
653
655template <BaseFabType FAB, typename DTOS, typename FabProj>
656void
658 const FabArray<FAB>& src, const FabArrayBase::CopyComTagsContainer& local_tags)
659{
660 static_assert(IsFabProjection<FabProj, FAB>(), "FabProj needs to be a projection for given FAB type.");
661#ifdef AMREX_USE_GPU
662 if (Gpu::inLaunchRegion()) {
663 local_copy_gpu(dest, src, packing.dest_component, packing.src_component,
664 packing.n_components, local_tags, packing.dtos, packing.proj);
665 } else
666#endif
667 {
668 local_copy_cpu(dest, src, packing.dest_component, packing.src_component,
669 packing.n_components, local_tags, packing.dtos, packing.proj);
670 }
671}
672
673#ifdef AMREX_USE_MPI
675template <BaseFabType FAB, typename DTOS, typename FabProj>
676void
678 FabArray<FAB>& dest, const CommData& recv)
679{
680 // If FAB is not FArrayBox we have not checked for the correct types yet.
681 static_assert(IsFabProjection<FabProj, FAB>(), "FabProj needs to be a projection for given FAB type.");
682#ifdef AMREX_USE_GPU
683 if (Gpu::inLaunchRegion()) {
684 unpack_recv_buffer_gpu(dest, packing.dest_component, packing.n_components, recv.data,
685 recv.size, recv.cctc, packing.dtos, packing.proj);
686 } else
687#endif // AMREX_USE_GPU
688 {
689 unpack_recv_buffer_cpu(dest, packing.dest_component, packing.n_components, recv.data,
690 recv.size, recv.cctc, packing.dtos, packing.proj);
691 }
692}
693#endif // AMREX_USE_MPI
694
695static_assert(IsDataPacking<ApplyDtosAndProjectionOnReciever<>, FArrayBox>(), // NOLINT(bugprone-throw-keyword-missing)
696 "ApplyDtosAndProjectionOnReciever<> is expected to satisfy the DataPacking concept.");
697
699// [ParallelCopy_nowait]
700
701static constexpr struct NoLocalCopy {} no_local_copy{};
702static constexpr struct DoLocalCopy {} do_local_copy{};
703
725template <BaseFabType FAB, typename DataPacking>
726requires (IsDataPacking<DataPacking, FAB>::value)
727#ifdef AMREX_USE_MPI
729ParallelCopy_nowait (NoLocalCopy, FabArray<FAB>& dest, const FabArray<FAB>& src,
730 const FabArrayBase::CommMetaData& cmd, const DataPacking& data_packing)
731{
732 CommHandler handler{};
733 if (ParallelContext::NProcsSub() == 1) {
734 return handler;
735 }
736 //
737 // Do this before prematurely exiting if running in parallel.
738 // Otherwise sequence numbers will not match across MPI processes.
739 //
741
742 if (cmd.m_RcvTags && !(cmd.m_RcvTags->empty())) {
743 PrepareRecvBuffers(data_packing, dest, src, handler.recv, *cmd.m_RcvTags);
744 PostRecvs(handler.recv, handler.mpi_tag);
745 }
746
747 if (cmd.m_SndTags && !(cmd.m_SndTags->empty())) {
748 PrepareSendBuffers(data_packing, dest, src, handler.send, *cmd.m_SndTags);
749 PackSendBuffers(data_packing, src, handler.send);
750 PostSends(handler.send, handler.mpi_tag);
751 }
752 return handler;
753}
754#else
755CommHandler ParallelCopy_nowait (NoLocalCopy, FabArray<FAB>&, const FabArray<FAB>&,
756 const FabArrayBase::CommMetaData&, const DataPacking&) {
757 return CommHandler{};
758}
759#endif
760
782template <BaseFabType FAB, typename DataPacking>
783requires (IsDataPacking<DataPacking, FAB>::value)
784#ifdef AMREX_USE_MPI
786#endif
787CommHandler
788ParallelCopy_nowait (FabArray<FAB>& dest, const FabArray<FAB>& src,
789 const FabArrayBase::CommMetaData& cmd, const DataPacking& data_packing)
790{
791 CommHandler comm = ParallelCopy_nowait(no_local_copy, dest, src, cmd, data_packing);
792 // Eagerly do the local work and hope for some overlap with communication
793 if (cmd.m_LocTags && !cmd.m_LocTags->empty()) {
794 LocalCopy(data_packing, dest, src, *cmd.m_LocTags);
795 }
796 return comm;
797}
799// [ParallelCopy_finish]
800
817template <BaseFabType FAB, typename DataPacking>
818requires (IsDataPacking<DataPacking, FAB>::value)
819void
820#ifdef AMREX_USE_MPI
821ParallelCopy_finish (FabArray<FAB>& dest, CommHandler handler,
822 const FabArrayBase::CommMetaData& cmd, const DataPacking& data_packing)
823{
824 // If any FabArray is empty we have nothing to do.
825 if (dest.empty()) {
826 return;
827 }
828 // Return if nothing do
829 if (ParallelContext::NProcsSub() == 1) {
830 return;
831 }
832 // Unpack receives
833 if (cmd.m_RcvTags && !(cmd.m_RcvTags->empty())) {
835#ifdef AMREX_DEBUG
836 if (!CheckRcvStats(handler.recv.stats, handler.recv.size, handler.mpi_tag)) {
837 amrex::Abort("NonLocalPC::ParallelCopy_finish failed with wrong message size");
838 }
839#endif
840 UnpackRecvBuffers(data_packing, dest, handler.recv);
841 }
842
843 // Wait for all sends to be done
844 if (cmd.m_SndTags && !(cmd.m_SndTags->empty())) {
846 }
847}
848#else
849ParallelCopy_finish (FabArray<FAB>&, CommHandler, const FabArrayBase::CommMetaData&, const DataPacking&) {}
850#endif
851
869template <BaseFabType FAB, typename DataPacking>
870requires (IsDataPacking<DataPacking, FAB>::value)
871void
872ParallelCopy_finish (DoLocalCopy, FabArray<FAB>& dest, const FabArray<FAB>& src, CommHandler handler,
873 const FabArrayBase::CommMetaData& cmd, const DataPacking& data_packing)
874{
875 // Eagerly do the local work and hope for some overlap with communication
876 if (cmd.m_LocTags && !cmd.m_LocTags->empty()) {
877 LocalCopy(data_packing, dest, src, *cmd.m_LocTags);
878 }
879 ParallelCopy_finish(dest, std::move(handler), cmd, data_packing); // NOLINT
880}
881
904template <BaseFabType FAB, typename DTOS = Identity, typename Proj = Identity>
905requires (IsCallableR<Dim3, DTOS, Dim3>::value && IsFabProjection<Proj, FAB>::value)
906void
907ParallelCopy (FabArray<FAB>& dest, const FabArray<FAB>& src, const FabArrayBase::CommMetaData& cmd,
908 SrcComp srccomp, DestComp destcomp, NumComps numcomp, DTOS const& dtos = DTOS{}, Proj const& proj = Proj{})
909{
910 PackComponents components{};
911 components.dest_component = destcomp.i;
912 components.src_component = srccomp.i;
913 components.n_components = numcomp.n;
914#if defined(__GNUC__) && !defined(__clang__)
915#pragma GCC diagnostic push
916#pragma GCC diagnostic ignored "-Wmaybe-uninitialized"
917#endif
918 ApplyDtosAndProjectionOnReciever<DTOS, Proj> packing{components, dtos, proj};
919 CommHandler handler = ParallelCopy_nowait(dest, src, cmd, packing);
920 ParallelCopy_finish(dest, std::move(handler), cmd, packing); // NOLINT
921#if defined(__GNUC__) && !defined(__clang__)
922#pragma GCC diagnostic pop
923#endif
924}
925
948template <BaseFabType FAB, typename DTOS = Identity, typename Proj = Identity>
949requires (IsCallableR<Dim3, DTOS, Dim3>::value && IsFabProjection<Proj, FAB>::value)
950void
951ParallelCopy (FabArray<FAB>& dest, const FabArray<FAB>& src, const FabArrayBase::CommMetaData& cmd,
952 int srccomp, int destcomp, int numcomp, DTOS const& dtos = DTOS{}, Proj const& proj = Proj{})
953{
954 ParallelCopy(dest, src, cmd, SrcComp(srccomp), DestComp(destcomp), NumComps(numcomp), dtos, proj);
955}
956
986template <BaseFabType FAB, typename DTOS = Identity, typename Proj = Identity>
987requires (IsIndexMapping<DTOS>::value && IsFabProjection<Proj, FAB>::value)
988MultiBlockCommMetaData
989ParallelCopy (FabArray<FAB>& dest, const Box& destbox, const FabArray<FAB>& src, SrcComp srccomp,
990 DestComp destcomp, NumComps numcomp, const IntVect& ngrow, DTOS const& dtos = DTOS{}, Proj const& proj = Proj{})
991{
992 MultiBlockCommMetaData cmd(dest, destbox, src, ngrow, dtos);
993 ParallelCopy(dest, src, cmd, srccomp, destcomp, numcomp, dtos, proj);
994 return cmd;
995}
996
1026template <BaseFabType FAB, typename DTOS = Identity, typename Proj = Identity>
1027requires (IsIndexMapping<DTOS>::value && IsFabProjection<Proj, FAB>::value)
1028MultiBlockCommMetaData
1029ParallelCopy (FabArray<FAB>& dest, const Box& destbox, const FabArray<FAB>& src, int srccomp,
1030 int destcomp, int numcomp, const IntVect& ngrow, DTOS const& dtos = DTOS{}, Proj const& proj = Proj{})
1031{
1032 return ParallelCopy(dest, destbox, src, SrcComp(srccomp), DestComp(destcomp), NumComps(numcomp), ngrow, dtos, proj);
1033}
1034
1035// Rotate90 fills the lo-x and lo-y boundary regions by rotating the data
1036// around (x=0,y=0) by 90 degrees in either direction. It also fills the
1037// corner of lo-x and lo-y boundary region by rotating the data by 180
1038// degrees.
1039
1040template <BaseFabType FAB>
1041void
1042Rotate90 (FabArray<FAB>& mf, int scomp, int ncomp, IntVect const& nghost, Box const& domain);
1043
1044template <BaseFabType FAB>
1045void
1046Rotate90 (FabArray<FAB>& mf, Box const& domain);
1047
1048// Rotate180 fills the lo-x boundary by rotating the data around
1049// (x=0,y=L_y/2) by 180 degrees.
1050
1051template <BaseFabType FAB>
1052void
1053Rotate180 (FabArray<FAB>& mf, int scomp, int ncomp, IntVect const& nghost, Box const& domain);
1054
1055template <BaseFabType FAB>
1056void
1057Rotate180 (FabArray<FAB>& mf, Box const& domain);
1058
1059// Fill the polar boundaries of the spherical coordinates (theta, phi, r).
1060// The lo-x boundary is filled with f(-x,y) = f(x,mod(y+pi,2*pi)), and
1061// the hi-x boundary is filled with f(pi+x,y) = f(pi-x,mod(y+pi,2*pi)).
1062
1063template <BaseFabType FAB>
1064void
1065FillPolar (FabArray<FAB>& mf, int scomp, int ncomp, IntVect const& nghost, Box const& domain);
1066
1067template <BaseFabType FAB>
1068void
1069FillPolar (FabArray<FAB>& mf, Box const& domain);
1070
1107template <BaseFabType FAB, typename DTOS, typename Proj = Identity>
1109[[nodiscard]]
1111FillBoundary_nowait (FabArray<FAB>& mf, const FabArrayBase::CommMetaData& cmd,
1112 int scomp, int ncomp, DTOS const& dtos,
1113 Proj const& proj = Proj{});
1114
1150template <BaseFabType FAB, typename DTOS, typename Proj = Identity>
1151requires (IsCallableR<Dim3,DTOS,Dim3>::value && IsFabProjection<Proj,FAB>::value)
1152void
1153FillBoundary_finish (CommHandler handler,
1155 int scomp, int ncomp, DTOS const& dtos,
1156 Proj const& proj = Proj{});
1157
1189template <BaseFabType FAB, typename DTOS, typename Proj = Identity>
1190requires (IsCallableR<Dim3,DTOS,Dim3>::value && IsFabProjection<Proj,FAB>::value)
1191void
1192FillBoundary (FabArray<FAB>& mf, const FabArrayBase::CommMetaData& cmd,
1193 int scomp, int ncomp, DTOS const& dtos, Proj const& proj = Proj{})
1194{
1195 BL_PROFILE("FillBoundary(cmd)");
1196 auto handler = FillBoundary_nowait(mf, cmd, scomp, ncomp, dtos, proj);
1197 FillBoundary_finish(std::move(handler), mf, cmd, scomp, ncomp, dtos, proj);
1198}
1199
1214template <BaseFabType FAB, typename DTOS>
1215requires (IsCallableR<Dim3,DTOS,Dim3>::value)
1216[[nodiscard]]
1217FabArrayBase::CommMetaData
1218makeFillBoundaryMetaData (FabArray<FAB>& mf, IntVect const& nghost,
1219 Geometry const& geom, DTOS const& dtos);
1220
1221}
1222
1223#include <AMReX_NonLocalBCImpl.H>
1224
1225namespace amrex {
1230 using NonLocalBC::SphThetaPhiRIndexMapping;
1231 using NonLocalBC::SphThetaPhiRComponentMapping;
1238}
1239
1240#endif
#define BL_PROFILE(a)
Definition AMReX_BLProfiler.H:551
#define AMREX_ASSERT(EX)
Definition AMReX_BLassert.H:38
#define AMREX_NODISCARD
Definition AMReX_Extension.H:252
#define AMREX_GPU_HOST_DEVICE
Definition AMReX_GpuQualifiers.H:20
#define AMREX_D_DECL(a, b, c)
Definition AMReX_SPACE.H:171
A collection of Boxes stored in an Array.
Definition AMReX_BoxArray.H:564
__host__ __device__ const IntVectND< dim > & bigEnd() const &noexcept
Return the inclusive upper bound of the box.
Definition AMReX_Box.H:124
__host__ __device__ IndexTypeND< dim > ixType() const noexcept
Return the indexing type.
Definition AMReX_Box.H:136
__host__ __device__ BoxND & setType(const IndexTypeND< dim > &t) noexcept
Set indexing type.
Definition AMReX_Box.H:513
__host__ __device__ const IntVectND< dim > & smallEnd() const &noexcept
Return the inclusive lower bound of the box.
Definition AMReX_Box.H:112
Calculates the distribution of FABs to MPI processes.
Definition AMReX_DistributionMapping.H:43
A Fortran Array of REALs.
Definition AMReX_FArrayBox.H:231
Base class for FabArray.
Definition AMReX_FabArrayBase.H:42
CopyComTag::CopyComTagsContainer CopyComTagsContainer
Definition AMReX_FabArrayBase.H:220
CopyComTag::MapOfCopyComTagContainers MapOfCopyComTagContainers
Definition AMReX_FabArrayBase.H:221
bool empty() const noexcept
Definition AMReX_FabArrayBase.H:89
An Array of FortranArrayBox(FAB)-like Objects.
Definition AMReX_FabArray.H:344
static void pack_send_buffer_cpu(FabArray< FAB > const &src, int scomp, int ncomp, Vector< char * > const &send_data, Vector< std::size_t > const &send_size, Vector< const CopyComTagsContainer * > const &send_cctc)
Definition AMReX_FBI.H:1373
static void pack_send_buffer_gpu(FabArray< FAB > const &src, int scomp, int ncomp, Vector< char * > const &send_data, Vector< std::size_t > const &send_size, Vector< const CopyComTagsContainer * > const &send_cctc, std::uint64_t id)
Definition AMReX_FBI.H:1113
Rectangular problem domain geometry.
Definition AMReX_Geometry.H:75
__host__ __device__ constexpr Dim3 dim3() const noexcept
Definition AMReX_IntVect.H:262
This class is a thin wrapper around std::vector. Unlike vector, Vector::operator[] provides bound che...
Definition AMReX_Vector.H:29
bool inLaunchRegion() noexcept
Definition AMReX_GpuControl.H:88
Definition AMReX_NonLocalBC.cpp:39
decltype(UnpackRecvBuffers(std::declval< Args >()...)) UnpackRecvBuffers_t
Definition AMReX_NonLocalBC.H:484
IntVect Apply(DTOS const &dtos, const IntVect &iv)
Applies the Dim3 to Dim3 mapping onto IntVects.
Definition AMReX_NonLocalBC.H:105
void Rotate90(FabArray< FAB > &mf, int scomp, int ncomp, IntVect const &nghost, Box const &domain)
decltype(std::declval< T >().Inverse(std::declval< Args >()...)) Inverse_t
Return type of an InverseImage class member function.
Definition AMReX_NonLocalBC.H:17
FabArrayBase::CommMetaData makeFillBoundaryMetaData(FabArray< FAB > &mf, IntVect const &nghost, Geometry const &geom, DTOS const &dtos)
Make metadata for FillBoundary.
void unpack_recv_buffer_cpu(FabArray< FAB > &mf, int dcomp, int ncomp, Vector< char * > const &recv_data, Vector< std::size_t > const &recv_size, Vector< FabArrayBase::CopyComTagsContainer const * > const &recv_cctc, DTOS const &dtos=DTOS{}, Proj const &proj=Proj{}) noexcept
decltype(PackSendBuffers(std::declval< Args >()...)) PackSendBuffers_t
Definition AMReX_NonLocalBC.H:481
IntVect ApplyInverse(DTOS const &dtos, const IntVect &iv)
Applies the Dim3 to Dim3 invserse mapping onto IntVects.
Definition AMReX_NonLocalBC.H:172
void PrepareCommBuffers(CommData &comm, const FabArrayBase::MapOfCopyComTagContainers &cctc, int n_components, std::size_t object_size, std::size_t align)
Fill all class member variables of comm but the request and the stats vector.
Definition AMReX_NonLocalBC.cpp:43
Box InverseImage(DTOS const &dtos, const Box &box)
Applies the inverse Dim3 to Dim3 mapping onto Boxes without changing the index type.
Definition AMReX_NonLocalBC.H:189
void PackSendBuffers(const PackComponents &components, const FabArray< FAB > &src, CommData &send)
Serializes FAB data without any knowledge of a DTOS nor a projection.
Definition AMReX_NonLocalBC.H:600
CommHandler FillBoundary_nowait(FabArray< FAB > &mf, const FabArrayBase::CommMetaData &cmd, int scomp, int ncomp, DTOS const &dtos, Proj const &proj=Proj{})
Start communication to fill boundary.
decltype(LocalCopy(std::declval< Args >()...)) LocalCopy_t
Definition AMReX_NonLocalBC.H:487
void ParallelCopy_finish(FabArray< FAB > &dest, CommHandler handler, const FabArrayBase::CommMetaData &cmd, const DataPacking &data_packing)
Definition AMReX_NonLocalBC.H:821
void Rotate180(FabArray< FAB > &mf, int scomp, int ncomp, IntVect const &nghost, Box const &domain)
void LocalCopy(const PackComponents &components, FabArray< FAB > &dest, const FabArray< FAB > &src, const FabArrayBase::CopyComTagsContainer &local_tags)
Dispatch local copies to the default behaviour that knows no DTOS nor projection.
Definition AMReX_NonLocalBC.H:559
void PostSends(CommData &send, int mpi_tag)
Initiate all sends with MPI_Isend calls associated with tag mpi_tag.
Definition AMReX_NonLocalBC.cpp:125
void local_copy_cpu(FabArray< FAB > &dest, const FabArray< FAB > &src, int dcomp, int scomp, int ncomp, FabArrayBase::CopyComTagsContainer const &local_tags, DTOS const &dtos=DTOS{}, Proj const &proj=Proj{}) noexcept
static constexpr SwapComponents< I, J > swap_indices
Definition AMReX_NonLocalBC.H:399
void PostRecvs(CommData &recv, int mpi_tag)
Initiate all recvieves with MPI_Irecv calls associated with tag mpi_tag.
Definition AMReX_NonLocalBC.cpp:109
void FillBoundary_finish(CommHandler handler, FabArray< FAB > &mf, const FabArrayBase::CommMetaData &cmd, int scomp, int ncomp, DTOS const &dtos, Proj const &proj=Proj{})
Finish communication started by FillBoundary_nowait.
void FillPolar(FabArray< FAB > &mf, int scomp, int ncomp, IntVect const &nghost, Box const &domain)
void FillBoundary(FabArray< FAB > &mf, const FabArrayBase::CommMetaData &cmd, int scomp, int ncomp, DTOS const &dtos, Proj const &proj=Proj{})
Fill ghost cells for FabArray/MultiFab.
Definition AMReX_NonLocalBC.H:1192
void UnpackRecvBuffers(const PackComponents &components, FabArray< FAB > &dest, const CommData &recv)
De-serializes FAB data without any knowledge of a DTOS nor a projection.
Definition AMReX_NonLocalBC.H:617
static constexpr struct amrex::NonLocalBC::DoLocalCopy do_local_copy
decltype(PrepareSendBuffers(std::declval< Args >()...)) PrepareSendBuffers_t
Definition AMReX_NonLocalBC.H:475
void local_copy_gpu(FabArray< FAB > &dest, const FabArray< FAB > &src, int dcomp, int scomp, int ncomp, FabArrayBase::CopyComTagsContainer const &local_tags, DTOS const &dtos=DTOS{}, Proj const &proj=Proj{}) noexcept
template MultiBlockCommMetaData ParallelCopy(FabArray< FArrayBox > &dest, const Box &destbox, const FabArray< FArrayBox > &src, int destcomp, int srccomp, int numcomp, const IntVect &ngrow, MultiBlockIndexMapping const &, Identity const &)
CommHandler ParallelCopy_nowait(NoLocalCopy, FabArray< FAB > &dest, const FabArray< FAB > &src, const FabArrayBase::CommMetaData &cmd, const DataPacking &data_packing)
Definition AMReX_NonLocalBC.H:729
static constexpr Identity identity
Definition AMReX_NonLocalBC.H:307
Box Image(DTOS const &dtos, const Box &box)
Applies the Dim3 to Dim3 mapping onto Boxes but does not change the index type.
Definition AMReX_NonLocalBC.H:123
decltype(PrepareRecvBuffers(std::declval< Args >()...)) PrepareRecvBuffers_t
Definition AMReX_NonLocalBC.H:478
void PrepareRecvBuffers(const PackComponents &components, FabArray< FAB > &dest, const FabArray< FAB > &src, CommData &comm, const FabArrayBase::MapOfCopyComTagContainers &cctc)
Calls PrepareComBuffers.
Definition AMReX_NonLocalBC.H:589
static constexpr struct amrex::NonLocalBC::NoLocalCopy no_local_copy
void PrepareSendBuffers(const PackComponents &components, FabArray< FAB > &dest, const FabArray< FAB > &src, CommData &comm, const FabArrayBase::MapOfCopyComTagContainers &cctc)
Calls PrepareComBuffers.
Definition AMReX_NonLocalBC.H:578
SwapComponents<-1, -1 > DynamicSwapComponents
Definition AMReX_NonLocalBC.H:397
void unpack_recv_buffer_gpu(FabArray< FAB > &mf, int scomp, int ncomp, Vector< char * > const &recv_data, Vector< std::size_t > const &recv_size, Vector< FabArrayBase::CopyComTagsContainer const * > const &recv_cctc, DTOS const &dtos=DTOS{}, Proj const &proj=Proj{})
int NProcsSub() noexcept
number of ranks in current frame
Definition AMReX_ParallelContext.H:74
void Waitall(Vector< MPI_Request > &, Vector< MPI_Status > &)
Definition AMReX_ParallelDescriptor.cpp:1308
int SeqNum() noexcept
Returns sequential message sequence numbers, usually used as tags for send/recv.
Definition AMReX_ParallelDescriptor.H:678
Definition AMReX_Amr.cpp:50
__host__ __device__ void ignore_unused(const Ts &...)
This shuts up the compiler about unused variables.
Definition AMReX.H:139
void FillBoundary_finish(Vector< MF * > const &mf)
Wait for outstanding FillBoundary_nowait operations launched with the vector helper to complete.
Definition AMReX_FabArrayCommI.H:1115
std::unique_ptr< char, TheFaArenaDeleter > TheFaArenaPointer
Definition AMReX_FabArray.H:106
void FillBoundary_nowait(Vector< MF * > const &mf, Vector< int > const &scomp, Vector< int > const &ncomp, Vector< IntVect > const &nghost, Vector< Periodicity > const &period, Vector< int > const &cross={})
Launch FillBoundary_nowait across a vector of FabArrays.
Definition AMReX_FabArrayCommI.H:1069
void Abort(const std::string &msg)
Print out message to cerr and exit via abort().
Definition AMReX.cpp:241
std::is_same< Expected, Detected_t< Op, Args... > > IsDetectedExact
Definition AMReX_TypeTraits.H:186
A multidimensional array accessor.
Definition AMReX_Array4.H:285
Logical traits let us combine multiple type requirements.
Definition AMReX_TypeTraits.H:232
Definition AMReX_BaseFab.H:77
int i
Definition AMReX_BaseFab.H:80
Definition AMReX_Dim3.H:13
int x
Definition AMReX_Dim3.H:13
int z
Definition AMReX_Dim3.H:13
int y
Definition AMReX_Dim3.H:13
Definition AMReX_FabArrayBase.H:472
std::unique_ptr< MapOfCopyComTagContainers > m_RcvTags
Definition AMReX_FabArrayBase.H:478
std::unique_ptr< MapOfCopyComTagContainers > m_SndTags
Definition AMReX_FabArrayBase.H:477
std::unique_ptr< CopyComTagsContainer > m_LocTags
Definition AMReX_FabArrayBase.H:476
Test if a given type T is callable with arguments of type Args...
Definition AMReX_TypeTraits.H:214
Test if a given type T is callable with arguments of type Args...
Definition AMReX_TypeTraits.H:208
This class specializes behaviour on local copies and unpacking receive buffers.
Definition AMReX_NonLocalBC.H:642
constexpr ApplyDtosAndProjectionOnReciever(const PackComponents &components, DTOS dtos_=DTOS{}, FabProj proj_=FabProj{})
Definition AMReX_NonLocalBC.H:644
FabProj proj
Definition AMReX_NonLocalBC.H:648
DTOS dtos
Definition AMReX_NonLocalBC.H:647
This class holds data buffers for either immediate MPI send or recv calls.
Definition AMReX_NonLocalBC.H:426
Vector< std::size_t > size
The size in bytes for each data transaction.
Definition AMReX_NonLocalBC.H:437
Vector< char * > data
Pointers to the_data that can be used for each single data transaction.
Definition AMReX_NonLocalBC.H:433
Vector< const FabArrayBase::CopyComTagsContainer * > cctc
For each request the copy comm tags for the corresponding data FABs.
Definition AMReX_NonLocalBC.H:443
Vector< MPI_Status > stats
For each request the corresponding MPI_status, used for debugging.
Definition AMReX_NonLocalBC.H:441
Vector< MPI_Request > request
The associated MPI_Request for each data transaction.
Definition AMReX_NonLocalBC.H:439
TheFaArenaPointer the_data
Holds 'em all in one data pointer.
Definition AMReX_NonLocalBC.H:429
std::uint64_t id
Definition AMReX_NonLocalBC.H:444
Vector< int > rank
Stores MPI ranks. For recvs it is the 'from' rank and for sends it is the 'to' rank.
Definition AMReX_NonLocalBC.H:431
Vector< std::size_t > offset
All offsets of data in the_data.
Definition AMReX_NonLocalBC.H:435
This class stores both recv and send buffers with an associated MPI tag.
Definition AMReX_NonLocalBC.H:463
CommData send
Definition AMReX_NonLocalBC.H:467
int mpi_tag
Definition AMReX_NonLocalBC.H:465
CommData recv
Definition AMReX_NonLocalBC.H:466
Definition AMReX_NonLocalBC.H:702
Type trait that tests if T has an InverseImage class member function.
Definition AMReX_NonLocalBC.H:21
This class acts as a default no-op operator.
Definition AMReX_NonLocalBC.H:291
constexpr Dim3 operator()(Dim3 i) const noexcept
The identity function for Dim3.
Definition AMReX_NonLocalBC.H:293
constexpr T operator()(Array4< const T > array, Dim3 i, int comp=0) const noexcept(noexcept(array(i.x, i.y, i.z, comp)))
Definition AMReX_NonLocalBC.H:299
constexpr int operator()(int i) const noexcept
The identity for int.
Definition AMReX_NonLocalBC.H:305
static constexpr Dim3 Inverse(Dim3 i) noexcept
The identity function for Dim3.
Definition AMReX_NonLocalBC.H:295
This type trait tests if a given type DP satisfies the DataPacking concept for type FAB.
Definition AMReX_NonLocalBC.H:503
This type trait tests if a type P is a projection for FAB.
Definition AMReX_NonLocalBC.H:282
Tests if a given type IndexMap is usable as an index mapping between two index based coordinate syste...
Definition AMReX_NonLocalBC.H:28
This class takes a projection and a component map and combines them to form a new projection.
Definition AMReX_NonLocalBC.H:325
Base base
Definition AMReX_NonLocalBC.H:332
Map map
Definition AMReX_NonLocalBC.H:333
constexpr decltype(auto) operator()(Array4< const T > array, Dim3 i, int comp) const noexcept(noexcept(base(array, i, map(comp))))
Definition AMReX_NonLocalBC.H:337
This is the index mapping based on the DTOS MultiBlockDestToSrc.
Definition AMReX_NonLocalBC.H:223
MultiBlockCommMetaData(const BoxArray &dstba, const DistributionMapping &dstdm, const Box &dstbox, const BoxArray &srcba, const DistributionMapping &srcdm, const IntVect &ngrow, DTOS const &dtos)
Build global meta data by calling the define() member function.
MultiBlockCommMetaData(const FabArrayBase &dst, const Box &dstbox, const FabArrayBase &src, const IntVect &ngrow, DTOS const &dtos)
Build global meta data by calling the define() member function.
void define(const BoxArray &dstba, const DistributionMapping &dstdm, const Box &dstbox, const BoxArray &srcba, const DistributionMapping &srcdm, const IntVect &ngrow, DTOS const &dtos)
Build global meta data that is being used to identify send and recv dependencies in communication rou...
This struct describes an affine index transformation for two coordinate systems.
Definition AMReX_NonLocalBC.H:38
IndexType operator()(IndexType it) const noexcept
Definition AMReX_NonLocalBC.H:77
IntVect sign
A vector of 1 and -1 describing the orientation in each component.
Definition AMReX_NonLocalBC.H:44
__host__ __device__ Dim3 operator()(Dim3 i) const noexcept
Applies this mapping on the index from destination space and returns an index in the source space.
Definition AMReX_NonLocalBC.H:52
__host__ __device__ Dim3 Inverse(Dim3 i) const noexcept
The inverse function is given by rearringing all above terms.
Definition AMReX_NonLocalBC.H:67
IntVect offset
The offset in the source index space.
Definition AMReX_NonLocalBC.H:42
IntVect permutation
This vector needs to be a valid permutation.
Definition AMReX_NonLocalBC.H:40
IndexType Inverse(IndexType it) const noexcept
Definition AMReX_NonLocalBC.H:81
Definition AMReX_NonLocalBC.H:701
Contains information about which components take part of the data transaction.
Definition AMReX_NonLocalBC.H:550
int dest_component
Definition AMReX_NonLocalBC.H:551
int src_component
Definition AMReX_NonLocalBC.H:552
int n_components
Definition AMReX_NonLocalBC.H:553
Definition AMReX_NonLocalBC.H:388
int J
Definition AMReX_NonLocalBC.H:390
int I
Definition AMReX_NonLocalBC.H:389
constexpr int operator()(int i) const noexcept
Definition AMReX_NonLocalBC.H:391
constexpr int operator()(int i) const noexcept
Definition AMReX_NonLocalBC.H:382
int I
Definition AMReX_NonLocalBC.H:381
int J
Definition AMReX_NonLocalBC.H:371
constexpr int operator()(int i) const noexcept
Definition AMReX_NonLocalBC.H:372
This is a permutation where only two components are swapped.
Definition AMReX_NonLocalBC.H:356
constexpr int operator()(int i) const noexcept
Swaps indices I and J.
Definition AMReX_NonLocalBC.H:362
Definition AMReX_BaseFab.H:83
int n
Definition AMReX_BaseFab.H:86
Definition AMReX_BaseFab.H:71
int i
Definition AMReX_BaseFab.H:74