14template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
15 template<
class>
class Allocator,
class CellAssignor>
19 num_real_comm_comps = 0;
20 int comm_comps_start = 0;
21 if constexpr (!ParticleType::is_soa_particle) {
22 comm_comps_start += AMREX_SPACEDIM + NStructReal;
24 for (
int i = comm_comps_start; i < comm_comps_start + NumRealComps(); ++i) {
25 if (h_redistribute_real_comp[i]) {++num_real_comm_comps;}
28 num_int_comm_comps = 0;
29 comm_comps_start = 2 + NStructInt;
30 for (
int i = comm_comps_start; i < comm_comps_start + NumIntComps(); ++i) {
31 if (h_redistribute_int_comp[i]) {++num_int_comm_comps;}
34 if constexpr (ParticleType::is_soa_particle) {
35 particle_size =
sizeof(uint64_t);
39 superparticle_size = particle_size +
40 num_real_comm_comps*
sizeof(
ParticleReal) + num_int_comm_comps*
sizeof(
int);
43template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
44 template<
class>
class Allocator,
class CellAssignor>
48 levelDirectoriesCreated =
false;
53 if constexpr (ParticleType::is_soa_particle)
55 h_redistribute_real_comp.resize(NArrayReal,
true);
57 h_redistribute_real_comp.resize(AMREX_SPACEDIM + NStructReal + NArrayReal,
true);
59 h_redistribute_int_comp.resize(2 + NStructInt + NArrayInt,
true);
64 m_soa_rdata_names.clear();
65 for (
int i=0; i<NArrayReal; ++i)
67 m_soa_rdata_names.push_back(getDefaultCompNameReal<ParticleType>(i));
69 m_soa_idata_names.clear();
70 for (
int i=0; i<NArrayInt; ++i)
72 m_soa_idata_names.push_back(getDefaultCompNameInt<ParticleType>(i));
75 static bool initialized =
false;
78 static_assert(
sizeof(ParticleType)%
sizeof(RealType) == 0,
79 "sizeof ParticleType is not a multiple of sizeof RealType");
81 ParmParse
pp(
"particles");
83 Vector<int> tilesize(AMREX_SPACEDIM);
84 if (
pp.
queryarr(
"tile_size", tilesize, 0, AMREX_SPACEDIM)) {
85 for (
int i=0; i<AMREX_SPACEDIM; ++i) { tile_size[i] = tilesize[i]; }
88 static_assert(std::is_standard_layout_v<ParticleType>,
89 "Particle type must be standard layout");
93 pp.
query(
"use_prepost", usePrePost);
94 pp.
query(
"do_unlink", doUnlink);
95 pp.
queryAdd(
"do_mem_efficient_sort", memEfficientSort);
96 pp.
queryAdd(
"use_comms_arena", use_comms_arena);
103 typename ParticleType,
106 template<
class>
class Allocator,
111 std::vector<std::string>
const & rdata_name, std::vector<std::string>
const & idata_name
118 std::set<std::string>
const unique_r_names(rdata_name.begin(), rdata_name.end());
119 std::set<std::string>
const unique_i_names(idata_name.begin(), idata_name.end());
123 for (
int i=0; i<NArrayReal; ++i)
125 m_soa_rdata_names.at(i) = rdata_name.at(i);
127 for (
int i=0; i<NArrayInt; ++i)
129 m_soa_idata_names.at(i) = idata_name.at(i);
134 typename ParticleType,
137 template<
class>
class Allocator,
143 return std::find(m_soa_rdata_names.begin(), m_soa_rdata_names.end(), name) != std::end(m_soa_rdata_names);
146template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
147 template<
class>
class Allocator,
class CellAssignor>
151 return std::find(m_soa_idata_names.begin(), m_soa_idata_names.end(), name) != std::end(m_soa_idata_names);
155 typename ParticleType,
158 template<
class>
class Allocator,
164 const auto it = std::find(m_soa_rdata_names.begin(), m_soa_rdata_names.end(), name);
166 if (it == m_soa_rdata_names.end())
168 throw std::runtime_error(
"GetRealCompIndex: Component " + name +
" does not exist!");
172 return std::distance(m_soa_rdata_names.begin(), it);
177 typename ParticleType,
180 template<
class>
class Allocator,
186 const auto it = std::find(m_soa_idata_names.begin(), m_soa_idata_names.end(), name);
188 if (it == m_soa_idata_names.end())
190 throw std::runtime_error(
"GetIntCompIndex: Component " + name +
" does not exist!");
194 return std::distance(m_soa_idata_names.begin(), it);
198template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
199 template<
class>
class Allocator,
class CellAssignor>
200template <
typename P,
typename Assignor>
205 const auto& domain = geom.
Domain();
209 return Assignor{}(p, plo, dxi, domain);
212template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
213 template<
class>
class Allocator,
class CellAssignor>
222 int local_grid)
const
228 lev_max = finestLevel();
233 AMREX_ASSERT(nGrow == 0 || (nGrow >= 0 && lev_min == lev_max));
235 std::vector< std::pair<int, Box> > isects;
237 for (
int lev = lev_max; lev >= lev_min; lev--) {
238 const IntVect& iv = Index(p, lev);
239 if (lev == pld.
m_lev) {
253 const BoxArray& ba = ParticleBoxArray(lev);
256 if (local_grid < 0) {
257 bool findfirst = (nGrow == 0) ?
true :
false;
259 grid = isects.empty() ? -1 : isects[0].first;
260 if (nGrow > 0 && std::ssize(isects) > 1) {
261 for (
auto & isect : isects) {
263 for (
int dir = 0; dir < AMREX_SPACEDIM; ++dir) {
275 grid = (*redistribute_mask_ptr)[local_grid](iv, 0);
293template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
294 template<
class>
class Allocator,
class CellAssignor>
302 int local_grid)
const
307 if (!Geom(0).isAnyPeriodic()) {
return false; }
310 lev_max = finestLevel();
318 p_prime.
pos(1) = p.pos(1);,
319 p_prime.
pos(2) = p.pos(2));
320 if (PeriodicShift(p_prime)) {
321 std::vector< std::pair<int,Box> > isects;
322 for (
int lev = lev_max; lev >= lev_min; lev--) {
326 const BoxArray& ba = ParticleBoxArray(lev);
329 if (local_grid < 0) {
332 grid = isects.empty() ? -1 : isects[0].first;
335 if (ba[local_grid].contains(iv))
342 grid = isects.empty() ? -1 : isects[0].first;
345 grid = (*redistribute_mask_ptr)[local_grid](Index(p, lev), 0);
352 p.pos(1) = p_prime.
pos(1);,
353 p.pos(2) = p_prime.
pos(2););
372template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
373 template<
class>
class Allocator,
class CellAssignor>
379 const auto& geom = Geom(0);
380 const auto plo = geom.ProbLoArray();
381 const auto phi = geom.ProbHiArray();
382 const auto rlo = geom.ProbLoArrayInParticleReal();
383 const auto rhi = geom.ProbHiArrayInParticleReal();
384 const auto is_per = geom.isPeriodicArray();
389template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
390 template<
class>
class Allocator,
class CellAssignor>
400 bool ok = Where(p, pld);
402 if (!ok && Geom(0).isAnyPeriodic())
413 amrex::AllPrint()<<
"Invalidating out-of-domain particle: " << p <<
'\n';
418 p.id().make_invalid();
424template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
425 template<
class>
class Allocator,
class CellAssignor>
430 m_particles.reserve(maxLevel()+1);
433template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
434 template<
class>
class Allocator,
class CellAssignor>
439 int nlevs = std::max(0, finestLevel()+1);
440 m_particles.resize(nlevs);
443template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
444 template<
class>
class Allocator,
class CellAssignor>
448 int lev_min,
int lev_max,
int nGrow,
int local_grid)
const
451 if (Geom(0).outsideRoundoffDomain(
AMREX_D_DECL(p.pos(0), p.pos(1), p.pos(2))))
454 success = EnforcePeriodicWhere(p, pld, lev_min, lev_max, local_grid);
455 if (!success && lev_min == 0)
458 p.id().make_invalid();
464 success = Where(p, pld, lev_min, lev_max, 0, local_grid);
469 success = (nGrow > 0) && Where(p, pld, lev_min, lev_min, nGrow);
475 amrex::Abort(
"ParticleContainer::locateParticle(): invalid particle.");
479template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
480 template<
class>
class Allocator,
class CellAssignor>
485 for (
int lev = 0; lev <= finestLevel(); lev++) {
486 nparticles += NumberOfParticlesAtLevel(lev,only_valid,
true);
494template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
495 template<
class>
class Allocator,
class CellAssignor>
499 AMREX_ASSERT(lev >= 0 && lev < std::ssize(m_particles));
502 ParticleDistributionMap(lev));
506 int gid = pti.index();
509 const auto& ptile = ParticlesAt(lev, pti);
510 const int np = ptile.numParticles();
511 auto const ptd = ptile.getConstParticleTileData();
515 using ReduceTuple =
typename decltype(reduce_data)::Type;
517 reduce_op.
eval(np, reduce_data,
520 return (ptd.id(i).is_valid()) ? 1 : 0;
523 int np_valid = amrex::get<0>(reduce_data.value(reduce_op));
524 np_per_grid_local[gid] += np_valid;
527 np_per_grid_local[gid] += pti.numParticles();
534 for (ParConstIterType pti(*
this, lev); pti.isValid(); ++pti)
536 nparticles[pti.index()] = np_per_grid_local[pti.index()];
556 if (level < 0 || level >= std::ssize(m_particles)) {
return nparticles; }
561 using ReduceTuple =
typename decltype(reduce_data)::Type;
563 for (
const auto& kv : GetParticles(level)) {
564 const auto& ptile = kv.second;
565 auto const ptd = ptile.getConstParticleTileData();
567 reduce_op.
eval(ptile.numParticles(), reduce_data,
570 return (ptd.id(i).is_valid()) ? 1 : 0;
574 nparticles =
static_cast<Long>(amrex::get<0>(reduce_data.
value(reduce_op)));
577 for (
const auto& kv : GetParticles(level)) {
578 const auto& ptile = kv.second;
579 nparticles += ptile.numParticles();
592template <std::
integral I>
593requires (
sizeof(I) >=
sizeof(
Long))
603 for (ParConstIterType pti(*
this, lev); pti.isValid(); ++pti)
605 int gid = pti.index();
606 mem[gid] +=
static_cast<I
>(pti.capacity());
614template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
615 template<
class>
class Allocator,
class CellAssignor>
622 for (
unsigned lev = 0; lev < m_particles.size(); lev++) {
623 const auto& pmap = m_particles[lev];
624 for (
const auto& kv : pmap) {
625 const auto& ptile = kv.second;
626 cnt += ptile.numParticles();
630 Long mn = cnt, mx = mn;
633 const Long sz =
sizeof(ParticleType)+NumRealComps()*
sizeof(
ParticleReal)+NumIntComps()*
sizeof(
int);
642 amrex::Print() <<
"ParticleContainer spread across MPI nodes - bytes (num particles): [Min: "
650 <<
" (" << cnt <<
")]\n";
655 return {mn*sz, mx*sz, cnt*sz};
658template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
659 template<
class>
class Allocator,
class CellAssignor>
666 for (
unsigned lev = 0; lev < m_particles.size(); lev++) {
667 const auto& pmap = m_particles[lev];
668 for (
const auto& kv : pmap) {
669 const auto& ptile = kv.second;
670 cnt += ptile.capacity();
674 Long mn = cnt, mx = mn;
685 amrex::Print() <<
"ParticleContainer spread across MPI nodes - bytes: [Min: "
696 return {mn, mx, cnt};
699template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
700 template<
class>
class Allocator,
class CellAssignor>
704 for (
unsigned lev = 0; lev < m_particles.size(); lev++) {
705 auto& pmap = m_particles[lev];
706 for (
auto& kv : pmap) {
707 auto& ptile = kv.second;
708 ptile.shrink_to_fit();
719template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
720 template<
class>
class Allocator,
class CellAssignor>
727 if (m_particles.empty()) {
return; }
728 AMREX_ASSERT(lev >= 0 && lev < std::ssize(m_particles));
731 const auto& geom = Geom(lev);
732 const auto plo = geom.ProbLoArray();
733 const auto dxi = geom.InvCellSizeArray();
734 const auto domain = geom.Domain();
736 [=]
AMREX_GPU_DEVICE (
const typename ParticleTileType::ConstParticleTileDataType& ptd,
int ip,
739 const auto p = ptd[ip];
741 IntVect iv = assignor(p, plo, dxi, domain);
746template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
747 template<
class>
class Allocator,
class CellAssignor>
751 BL_PROFILE(
"ParticleContainer::IncrementWithTotal(lev)");
753 return TotalNumberOfParticles(
true, local);
756template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
757 template<
class>
class Allocator,
class CellAssignor>
761 BL_PROFILE(
"ParticleContainer::RemoveParticlesAtLevel()");
762 if (level >= std::ssize(this->m_particles)) {
return; }
764 if (!this->m_particles[level].empty())
770template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
771 template<
class>
class Allocator,
class CellAssignor>
775 BL_PROFILE(
"ParticleContainer::RemoveParticlesNotAtFinestLevel()");
776 AMREX_ASSERT(this->finestLevel()+1 == std::ssize(this->m_particles));
780 for (
unsigned lev = 0; lev < m_particles.size() - 1; ++lev) {
781 auto& pmap = m_particles[lev];
783 for (
auto& kv : pmap) {
784 const auto& pbx = kv.second;
785 cnt += pbx.numParticles();
794 if (this->m_verbose > 1 && cnt > 0) {
796 <<
" particles not in finest level\n";
812 template <
typename SrcData>
814 int operator() (
const SrcData& src,
int src_i)
const noexcept
823 template <
typename DstData,
typename SrcData>
826 int src_i,
int dst_i)
const noexcept
836template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
837 template<
class>
class Allocator,
class CellAssignor>
839ParticleContainer_impl<ParticleType, NArrayReal, NArrayInt, Allocator, CellAssignor>
840::CreateVirtualParticles (
int level,
AoS& virts)
const
843 CreateVirtualParticles(level, ptile);
844 ptile.GetArrayOfStructs().swap(virts);
847template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
848 template<
class>
class Allocator,
class CellAssignor>
853 BL_PROFILE(
"ParticleContainer::CreateVirtualParticles()");
857 if (level >= std::ssize(m_particles)) {
861 std::string
const& aggregation_type = AggregationType();
862 int aggregation_buffer = AggregationBuffer();
864 if (aggregation_type ==
"None")
866 auto virts_offset = virts.numParticles();
869 const auto& src_tile = ParticlesAt(level, pti);
871 auto np = src_tile.numParticles();
872 virts.resize(virts_offset+np);
877 if (aggregation_type ==
"Cell")
880 int nComp = AMREX_SPACEDIM + NStructReal + NArrayReal;
883 MultiFab mf(ParticleBoxArray(level), ParticleDistributionMap(level),
nComp, nGhost);
885 nComp = 1 + NStructInt + NArrayInt;
886 iMultiFab imf(ParticleBoxArray(level), ParticleDistributionMap(level),
nComp, nGhost);
888 const auto& geom = Geom(level);
889 const auto plo = geom.ProbLoArray();
890 const auto dxi = geom.InvCellSizeArray();
891 const auto domain = geom.Domain();
894 bl_buffer.
complementIn(Geom(level).Domain(), ParticleBoxArray(level));
895 BoxArray buffer(std::move(bl_buffer));
896 buffer.
grow(aggregation_buffer);
899 locator.
build(buffer, geom, do_tiling, tile_size);
909 if(assign_buffer_grid(iv).first == -1)
912 for (
int i = 0; i < NArrayReal; ++i)
917 for (
int i = 0; i < AMREX_SPACEDIM; ++i)
922 for (
int i = 1; i < NStructReal; ++i)
927 for (
int i = 0; i < 1; ++i)
943 if(assign_buffer_grid(iv).first == -1)
946 if(partData(iv,0)==0)
950 for (
int i = 0; i < NStructInt; ++i)
954 for (
int i = 0; i < NArrayInt; ++i)
963 virts.resize(imf.
sum(0));
968 const auto bx = mfi.tilebox();
969 const auto partData = mf.
array(mfi);
970 const auto imf_arr = imf.
array(mfi);
973 auto *offsets_ptr = offsets.
dataPtr();
975 auto dst = virts.getParticleTileData();
978 if(imf_arr(i,j,k,0)!=0)
980 const auto idx = last_offset + offsets_ptr[
990 p.rdata(0) =
static_cast<ParticleReal>(partData(i,j,k,AMREX_SPACEDIM));;
992 for (
int n = 0; n < AMREX_SPACEDIM; ++n)
994 p.pos(n) =
static_cast<ParticleReal>(partData(i,j,k,n) / p.rdata(0));
997 for (
int n = 1; n < NStructReal; ++n)
999 p.rdata(n) =
static_cast<ParticleReal>(partData(i,j,k,AMREX_SPACEDIM+n) / p.rdata(0));
1002 for (
int n = 0; n < NArrayReal; ++n)
1004 dst.rdata(n)[idx] =
static_cast<ParticleReal>(partData(i,j,k,AMREX_SPACEDIM+NStructReal+n));
1007 for (
int n = 0; n < NStructInt; ++n)
1009 p.idata(n) = imf_arr(i,j,k,1+n);
1012 for (
int n = 0; n < NArrayInt; ++n)
1014 dst.idata(n)[idx] = imf_arr(i,j,k,1+NStructInt+n);
1019 last_offset+=next_offset;
1024 auto virts_offset = last_offset;
1027 const auto& src_tile = ParticlesAt(level, pti);
1029 auto np = src_tile.numParticles();
1030 virts.resize(virts_offset+np);
1034 virts.resize(virts_offset);
1053 template <
typename SrcData>
1059 const auto p_boxes = amrex::get<0>(tup_min);
1060 const auto p_boxes_max = amrex::get<0>(tup_max);
1061 const auto p_levs_max = amrex::get<2>(tup_max);
1062 return p_boxes_max >=0 && p_boxes ==
m_gid && p_levs_max ==
m_lev_max;
1069 template <
typename DstData,
typename SrcData>
1072 int src_i,
int dst_i)
const noexcept
1081template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1082 template<
class>
class Allocator,
class CellAssignor>
1084ParticleContainer_impl<ParticleType, NArrayReal, NArrayInt, Allocator, CellAssignor>
1085::CreateGhostParticles (
int level,
int nGrow,
AoS& ghosts)
const
1088 CreateGhostParticles(level, nGrow, ptile);
1089 ptile.GetArrayOfStructs().swap(ghosts);
1092template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1093 template<
class>
class Allocator,
class CellAssignor>
1098 BL_PROFILE(
"ParticleContainer::CreateGhostParticles()");
1102 if (level >= std::ssize(m_particles)) {
1106 if (! m_particle_locator.isValid(GetParGDB(), do_tiling, tile_size)) {
1107 m_particle_locator.build(GetParGDB(), do_tiling, tile_size);
1110 m_particle_locator.setGeometry(GetParGDB());
1112 auto ghost_offset = ghosts.numParticles();
1115 const auto& src_tile = ParticlesAt(level, pti);
1116 int gid = pti.index();
1118 auto np = src_tile.numParticles();
1119 ghosts.resize(ghost_offset+np);
1122 ghosts.resize(ghost_offset);
1126template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1127 template<
class>
class Allocator,
class CellAssignor>
1132 BL_PROFILE(
"ParticleContainer::clearParticles()");
1134 for (
int lev = 0; lev < std::ssize(m_particles); ++lev)
1136 for (
auto& kv : m_particles[lev]) { kv.second.resize(0); }
1137 particle_detail::clearEmptyEntries(m_particles[lev]);
1141template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1142 template<
class>
class Allocator,
class CellAssignor>
1143template <
class PCType>
1149 using PData =
typename ParticleTileType::ConstParticleTileDataType;
1153template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1154 template<
class>
class Allocator,
class CellAssignor>
1155template <
class PCType>
1159addParticles (const PCType& other, bool local)
1161 using PData =
typename ParticleTileType::ConstParticleTileDataType;
1165template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1166 template<
class>
class Allocator,
class CellAssignor>
1167template <
class F,
class PCType>
1173 BL_PROFILE(
"ParticleContainer::copyParticles");
1175 addParticles(other, std::forward<F>(f), local);
1178template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1179 template<
class>
class Allocator,
class CellAssignor>
1180template <
class F,
class PCType>
1184addParticles (const PCType& other,
F const& f, bool local)
1186 BL_PROFILE(
"ParticleContainer::addParticles");
1189 for (
int lev = 0; lev < other.numLevels(); ++lev)
1192 const auto& plevel_other = other.GetParticles(lev);
1193 for(
MFIter mfi = other.MakeMFIter(lev); mfi.isValid(); ++mfi)
1195 auto index = std::make_pair(mfi.index(), mfi.LocalTileIndex());
1196 if(!plevel_other.contains(index)) {
continue; }
1198 DefineAndReturnParticleTile(lev, mfi.index(), mfi.LocalTileIndex());
1203#pragma omp parallel if (Gpu::notInLaunchRegion())
1205 for (
int lev = 0; lev < other.numLevels(); ++lev)
1207 const auto& plevel_other = other.GetParticles(lev);
1208 for(
MFIter mfi = other.MakeMFIter(lev); mfi.isValid(); ++mfi)
1210 auto index = std::make_pair(mfi.index(), mfi.LocalTileIndex());
1211 if(!plevel_other.contains(index)) {
continue; }
1214 auto& ptile = ParticlesAt(lev, mfi.index(), mfi.LocalTileIndex());
1215 const auto& ptile_other = plevel_other.at(index);
1216 auto np = ptile_other.numParticles();
1217 if (np == 0) {
continue; }
1219 auto dst_index = ptile.numParticles();
1220 ptile.resize(dst_index + np);
1223 static_cast<decltype(np)
>(0), dst_index, np);
1225 ptile.resize(dst_index + count);
1229 if (! local) { Redistribute(); }
1235template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1244 !is_rtsoa_pc || NumRuntimeRealComps() >= AMREX_SPACEDIM,
1245 "ParticleContainer with RTSoA requires at least AMREX_SPACEDIM "
1246 "runtime real components for positions"
1249 Redistribute_impl(lev_min, lev_max, nGrow, local, remove_negative);
1254template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1255 template<
class>
class Allocator,
class CellAssignor>
1256template <
class index_type>
1261 auto& ptile = ParticlesAt(lev, mfi);
1262 const size_t np = ptile.numParticles();
1263 const size_t np_total = np + ptile.numNeighborParticles();
1265 if (memEfficientSort) {
1269 ptile_tmp.define(m_num_runtime_real, m_num_runtime_int,
1270 &m_soa_rdata_names, &m_soa_idata_names, arena());
1271 ptile_tmp.resize(np_total);
1276 ptile.swap(ptile_tmp);
1280template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1281 template<
class>
class Allocator,
class CellAssignor>
1294 BL_PROFILE(
"ParticleContainer::SortParticlesByBin()");
1298 for (
int lev = 0; lev < numLevels(); ++lev)
1303 const auto domain = geom.
Domain();
1307 auto& ptile = ParticlesAt(lev, mfi);
1308 const size_t np = ptile.numParticles();
1310 const Box& box = mfi.validbox();
1314 m_bins.build(np, ptile.getParticleTileData(), ntiles,
1316 .bin_size = bin_size, .box = box});
1322template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1323 template<
class>
class Allocator,
class CellAssignor>
1328 BL_PROFILE(
"ParticleContainer::SortParticlesForDeposition()");
1330 for (
int lev = 0; lev < numLevels(); ++lev)
1336 const auto& ptile = ParticlesAt(lev, mfi);
1337 const size_t np = ptile.numParticles();
1339 const Box& box = mfi.validbox();
1341 using index_type =
typename decltype(m_bins)::index_type;
1343 PermutationForDeposition<index_type>(perm, np, ptile, box, geom, idx_type);
1349template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1350 template<
class>
class Allocator,
class CellAssignor>
1355 int lev,
int gid,
int tid,
1356 int lev_min,
int lev_max,
int nGrow,
int local,
1357 bool remove_negative,
int myproc,
1365 auto ptd = src_tile.getParticleTileData();
1368 Long last = src_tile.numParticles() - 1;
1371 while (pindex <= last) {
1372 decltype(
auto) p = ptd[pindex];
1375 if (!ptd.id(pindex).is_valid()) {
1376 if (remove_negative) {
1378 correctCellVectors(last, pindex, gid, p);
1389 if (lev == lev_max && pld.
m_tile == tid && pld.
m_grid == gid && pld.
m_lev == lev && who == myproc) {
1390 const auto iv = Index(p, lev);
1398 locateParticle(p, pld, lev_min, lev_max, nGrow, local ? gid : -1);
1399 particlePostLocate(p, pld, lev);
1402 if (!ptd.id(pindex).is_valid()) {
1404 correctCellVectors(last, pindex, gid, p);
1411 if (pld.
m_lev != lev || pld.
m_grid != gid || pld.
m_tile != tid || who != myproc) {
1414 correctCellVectors(last, pindex, gid, p);
1415 boxes[num_move] = pld.
m_grid;
1416 levels[num_move] = pld.
m_lev;
1417 tiles[num_move] = pld.
m_tile;
1418 src_indices[num_move] =
static_cast<int>(last);
1430 src_indices.
resize(num_move);
1431 periodic_shift.
resize(num_move);
1433 return static_cast<int>(pindex);
1439template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1440 template<
class>
class Allocator,
class CellAssignor>
1442ParticleContainer_impl<ParticleType, NArrayReal, NArrayInt, Allocator, CellAssignor>
1443::Redistribute_impl (
int lev_min,
int lev_max,
int nGrow,
int local,
bool remove_negative)
1447 BL_PROFILE(
"ParticleContainer::Redistribute_impl()");
1450 int theEffectiveFinestLevel = m_gdb->finestLevel();
1451 while (!m_gdb->LevelDefined(theEffectiveFinestLevel)) { theEffectiveFinestLevel--; }
1453 if (std::ssize(m_particles) < theEffectiveFinestLevel+1) {
1455 amrex::Print() <<
"ParticleContainer::Redistribute() resizing containers from "
1456 << m_particles.size() <<
" to "
1457 << theEffectiveFinestLevel + 1 <<
'\n';
1459 m_particles.resize(theEffectiveFinestLevel+1);
1460 m_dummy_mf.resize(theEffectiveFinestLevel+1);
1463 for (
int lev = 0; lev < theEffectiveFinestLevel+1; ++lev) { RedefineDummyMF(lev); }
1465 int finest_lev_particles;
1466 if (lev_max == -1) {
1467 lev_max = theEffectiveFinestLevel;
1468 finest_lev_particles = m_particles.size() - 1;
1470 finest_lev_particles = lev_max;
1474 this->defineBufferMap();
1476#ifndef AMREX_USE_GPU
1477 if (local > 0) { BuildRedistributeMask(0, local); }
1479 if (! m_particle_locator.isValid(GetParGDB(), do_tiling, tile_size)) { m_particle_locator.build(GetParGDB(), do_tiling, tile_size); }
1480 m_particle_locator.setGeometry(GetParGDB());
1485 int num_levels = finest_lev_particles + 1;
1486 op.setNumLevels(num_levels);
1487 Vector<std::map<std::pair<int, int>,
int> > new_sizes(num_levels);
1488#ifndef AMREX_USE_GPU
1492 auto assign_grid = m_particle_locator.getGridAssignor();
1493 const auto plo = Geom(0).ProbLoArray();
1494 const auto phi = Geom(0).ProbHiArray();
1495 const auto rlo = Geom(0).ProbLoArrayInParticleReal();
1496 const auto rhi = Geom(0).ProbHiArrayInParticleReal();
1497 const auto is_per = Geom(0).isPeriodicArray();
1500#if defined(AMREX_USE_OMP) || defined(AMREX_USE_GPU)
1505#ifndef AMREX_USE_GPU
1512 std::size_t num_tiles = 0;
1513 for (
int lev = lev_min; lev <= finest_lev_particles; ++lev) {
1514 for (
auto const& kv : m_particles[lev]) {
1515 if (kv.second.numParticles() != 0) {
1520 grid_tile_ids.reserve(num_tiles);
1521 ptile_ptrs.reserve(num_tiles);
1522 plevs.reserve(num_tiles);
1523 new_size_ptrs.reserve(num_tiles);
1524#ifndef AMREX_USE_GPU
1525 box_ptrs.reserve(num_tiles);
1526 level_ptrs.reserve(num_tiles);
1527 tile_ptrs.reserve(num_tiles);
1528 src_index_ptrs.reserve(num_tiles);
1529 periodic_shift_ptrs.reserve(num_tiles);
1531 for (
int lev = lev_min; lev <= finest_lev_particles; lev++) {
1532 auto& pmap = m_particles[lev];
1533 for (
auto& kv : pmap)
1535 const auto np = kv.second.numParticles();
1536 if (np == 0) {
continue; }
1538 grid_tile_ids.push_back(kv.first);
1539 ptile_ptrs.push_back(&(kv.second));
1540 plevs.push_back(lev);
1541 auto index = std::make_pair(kv.first.first, kv.first.second);
1542 auto& new_size = new_sizes[lev][index];
1544 new_size_ptrs.push_back(&new_size);
1545#ifndef AMREX_USE_GPU
1546 op.resize(kv.first.first, kv.first.second, lev,
static_cast<int>(np));
1547 auto& boxes = op.m_boxes[lev][index];
1548 auto& levels = op.m_levels[lev][index];
1549 auto& tiles = op.m_tiles[lev][index];
1550 auto& src_indices = op.m_src_indices[lev][index];
1551 auto& periodic_shift = op.m_periodic_shift[lev][index];
1552 box_ptrs.push_back(&boxes);
1553 level_ptrs.push_back(&levels);
1554 tile_ptrs.push_back(&tiles);
1555 src_index_ptrs.push_back(&src_indices);
1556 periodic_shift_ptrs.push_back(&periodic_shift);
1562#if defined(AMREX_USE_OMP) || defined(AMREX_USE_GPU)
1564#pragma omp parallel for
1566 for (
int pmap_it = 0; pmap_it < static_cast<int>(ptile_ptrs.
size()); ++pmap_it)
1568 int lev = plevs[pmap_it];
1569 int gid = grid_tile_ids[pmap_it].first;
1570 int tid = grid_tile_ids[pmap_it].second;
1571 auto& src_tile = *ptile_ptrs[pmap_it];
1572 const size_t np = src_tile.numParticles();
1573 if (np == 0) {
continue;}
1575#ifndef AMREX_USE_GPU
1576 auto& boxes = *box_ptrs[pmap_it];
1577 auto& levels = *level_ptrs[pmap_it];
1578 auto& tiles = *tile_ptrs[pmap_it];
1579 auto& src_indices = *src_index_ptrs[pmap_it];
1580 auto& periodic_shift = *periodic_shift_ptrs[pmap_it];
1581 *new_size_ptrs[pmap_it] = hostPartitionTile(src_tile,
1583 lev_min, lev_max, nGrow, local,
1584 remove_negative, myproc,
1585 boxes, levels, tiles,
1586 src_indices, periodic_shift);
1589 std::forward<CellAssignor>(CellAssignor{}),
1591 plo, phi, rlo, rhi, is_per, lev, gid, tid,
1592 lev_min, lev_max, nGrow, remove_negative);
1594 int num_move = np - num_stay;
1595 *new_size_ptrs[pmap_it] = num_stay;
1596 op.resize(gid, tid, lev, num_move);
1598 auto index = std::make_pair(gid, tid);
1599 auto* p_boxes = op.m_boxes[lev][index].dataPtr();
1600 auto* p_levs = op.m_levels[lev][index].dataPtr();
1601 auto* p_tiles = op.m_tiles[lev][index].dataPtr();
1602 auto* p_src_indices = op.m_src_indices[lev][index].dataPtr();
1603 auto* p_periodic_shift = op.m_periodic_shift[lev][index].dataPtr();
1604 auto ptd = src_tile.getParticleTileData();
1608 const auto p = ptd[i + num_stay];
1610 if (!p.id().is_valid())
1618 const auto tup = assign_grid(p, lev_min, lev_max, nGrow,
1619 std::forward<CellAssignor>(CellAssignor{}));
1620 p_boxes[i] = amrex::get<0>(tup);
1621 p_tiles[i] = amrex::get<1>(tup);
1622 p_levs[i] = amrex::get<2>(tup);
1625 p_src_indices[i] = i+num_stay;
1630 for (
int lev = lev_min; lev <= finest_lev_particles; ++lev) {
1631 auto& pmap = m_particles[lev];
1632 for (
auto& kv : pmap)
1634 auto& src_tile = kv.second;
1635 const auto np = src_tile.numParticles();
1636 if (np == 0) {
continue; }
1638 int gid = kv.first.first;
1639 int tid = kv.first.second;
1640 auto index = std::make_pair(gid, tid);
1641 auto& new_size = new_sizes[lev][index];
1644 op.resize(gid, tid, lev,
static_cast<int>(np));
1645 auto& boxes = op.m_boxes[lev][index];
1646 auto& levels = op.m_levels[lev][index];
1647 auto& tiles = op.m_tiles[lev][index];
1648 auto& src_indices = op.m_src_indices[lev][index];
1649 auto& periodic_shift = op.m_periodic_shift[lev][index];
1650 new_size = hostPartitionTile(src_tile,
1652 lev_min, lev_max, nGrow, local,
1653 remove_negative, myproc,
1654 boxes, levels, tiles,
1655 src_indices, periodic_shift);
1661 ParticleCopyPlan plan;
1663 plan.build(*
this, op, h_redistribute_int_comp,
1664 h_redistribute_real_comp, local);
1670 if (use_comms_arena) {
1678 for (
int lev = lev_min; lev <= lev_max; ++lev)
1680 auto& plev = m_particles[lev];
1681 for (
auto& kv : plev)
1683 int gid = kv.first.first;
1684 int tid = kv.first.second;
1685 auto index = std::make_pair(gid, tid);
1686 auto& tile = plev[index];
1687 tile.resize(new_sizes[lev][index]);
1691 for (
int lev = lev_min; lev <= lev_max; lev++)
1693 particle_detail::clearEmptyEntries(m_particles[lev]);
1696 if (std::ssize(m_particles) > theEffectiveFinestLevel+1) {
1697 if (m_verbose > 0) {
1698 amrex::Print() <<
"ParticleContainer::Redistribute() resizing m_particles from "
1699 << m_particles.size() <<
" to " << theEffectiveFinestLevel+1 <<
'\n';
1703 m_particles.resize(theEffectiveFinestLevel + 1);
1704 m_dummy_mf.resize(theEffectiveFinestLevel + 1);
1709 plan.buildMPIFinish(BufferMap());
1711 this->ReserveForRedistribute(plan);
1712 unpackBuffer(*
this, plan, snd_buffer, RedistributeUnpackPolicy());
1714 unpackRemotes(*
this, plan, rcv_buffer, RedistributeUnpackPolicy());
1719 Gpu::PinnedVector<char> pinned_snd_buffer;
1720 Gpu::PinnedVector<char> pinned_rcv_buffer;
1722 if (snd_buffer.arena()->isPinned()) {
1723 plan.buildMPIFinish(BufferMap());
1727 pinned_snd_buffer.resize(snd_buffer.
size());
1729 plan.buildMPIFinish(BufferMap());
1734 this->ReserveForRedistribute(plan);
1736 rcv_buffer.
resize(pinned_rcv_buffer.size());
1737 unpackBuffer(*
this, plan, snd_buffer, RedistributeUnpackPolicy());
1740 unpackRemotes(*
this, plan, rcv_buffer, RedistributeUnpackPolicy());
1747template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1748 template<
class>
class Allocator,
class CellAssignor>
1750ParticleContainer_impl<ParticleType, NArrayReal, NArrayInt, Allocator, CellAssignor>
1753 BL_PROFILE(
"ParticleContainer::ReserveForRedistribute()");
1755 std::map<ParticleTileType*, int> addsizes;
1757 for (
int lev = 0; lev < this->BufferMap().numLevels(); ++lev) {
1758 for (
MFIter mfi = this->MakeMFIter(lev); mfi.
isValid(); ++mfi) {
1759 int gid = mfi.index();
1760 int tid = mfi.LocalTileIndex();
1761 auto& tile = this->DefineAndReturnParticleTile(lev, gid, tid);
1762 int num_copies = plan.
m_box_counts_h[this->BufferMap().gridAndTileAndLevToBucket(gid, tid, lev)];
1763 if (num_copies > 0) {
1764 addsizes[&tile] += num_copies;
1775 auto& tile = this->DefineAndReturnParticleTile(lev, gid, tid);
1776 addsizes[&tile] += copy_size;
1780 ParticleTileType::reserve(addsizes);
1783template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1784 template<
class>
class Allocator,
class CellAssignor>
1790 if (lev_max == -1) {
1791 lev_max = finestLevel();
1797template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1798 template<
class>
class Allocator,
class CellAssignor>
1804 ptile.GetArrayOfStructs().swap(particles);
1805 AddParticlesAtLevel(ptile, level, nGrow);
1808template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1809 template<
class>
class Allocator,
class CellAssignor>
1814 BL_PROFILE(
"ParticleContainer::AddParticlesAtLevel()");
1816 if (std::ssize(m_particles) < level+1)
1820 amrex::Print() <<
"ParticleContainer::AddParticlesAtLevel resizing m_particles from "
1821 << m_particles.size()
1825 m_particles.resize(level+1);
1826 m_dummy_mf.resize(level+1);
1827 for (
int lev = 0; lev < level+1; ++lev) {
1828 RedefineDummyMF(lev);
1832 auto& ptile = DefineAndReturnParticleTile(level, 0, 0);
1833 int old_np = ptile.size();
1834 int num_to_add = particles.size();
1835 int new_np = old_np + num_to_add;
1836 ptile.resize(new_np);
1838 Redistribute(level, level, nGrow);
1839 particles.resize(0);
1843template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1844 template<
class>
class Allocator,
class CellAssignor>
1851 int particle_lvl_offset)
const
1853 BL_PROFILE(
"ParticleContainer::AssignCellDensitySingleLevel()");
1855 if (rho_index != 0) {
amrex::Abort(
"AssignCellDensitySingleLevel only works if rho_index = 0"); }
1859 if (OnSameGrids(lev, mf_to_be_filled)) {
1862 mf_pointer = &mf_to_be_filled;
1867 mf_pointer =
new MultiFab(ParticleBoxArray(lev),
1868 ParticleDistributionMap(lev),
1869 ncomp, mf_to_be_filled.
nGrow());
1876 if (mf_pointer->
nGrow() < 1) {
1877 amrex::Error(
"Must have at least one ghost cell when in AssignCellDensitySingleLevel");
1882 const auto dxi = Geom(lev).InvCellSizeArray();
1883 const auto plo = Geom(lev).ProbLoArray();
1884 const auto pdxi = Geom(lev + particle_lvl_offset).InvCellSizeArray();
1886 if (Geom(lev).isAnyPeriodic() && ! Geom(lev).isAllPeriodic())
1888 amrex::Error(
"AssignCellDensitySingleLevel: problem must be periodic in no or all directions");
1895#pragma omp parallel if (Gpu::notInLaunchRegion())
1900 const Long np = pti.numParticles();
1901 auto ptd = pti.GetParticleTile().getConstParticleTileData();
1903 auto rhoarr = fab.
array();
1906 if (Gpu::notInLaunchRegion())
1908 tile_box = pti.tilebox();
1910 local_rho.
resize(tile_box,ncomp);
1911 local_rho.
setVal<RunOn::Host>(0.0);
1912 rhoarr = local_rho.
array();
1916 if (particle_lvl_offset == 0)
1921 amrex_deposit_cic(p, ncomp, rhoarr, plo, dxi);
1929 amrex_deposit_particle_dx_cic(p, ncomp, rhoarr, plo, dxi, pdxi);
1934 if (Gpu::notInLaunchRegion())
1936 fab.
atomicAdd<RunOn::Host>(local_rho, tile_box, tile_box, 0, 0, ncomp);
1947 for (
int n = 1; n < ncomp; n++)
1951 (*mf_pointer)[mfi].protected_divide<RunOn::Device>((*mf_pointer)[mfi],0,n,1);
1958 const Real* dx = Geom(lev).CellSize();
1965 if (mf_pointer != &mf_to_be_filled)
1967 mf_to_be_filled.
ParallelCopy(*mf_pointer,0,0,ncomp,0,0);
1975 ParallelReduce::Max(stoptime, ParallelContext::IOProcessorNumberSub(),
1976 ParallelContext::CommunicatorSub());
1978 amrex::Print() <<
"ParticleContainer::AssignCellDensitySingleLevel) time: "
1979 << stoptime <<
'\n';
1983template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
1984 template<
class>
class Allocator,
class CellAssignor>
1989 int old_size = m_num_runtime_real;
1991 m_runtime_comps_defined = (new_size > 0);
1992 m_num_runtime_real = new_size;
1993 int cur_size = h_redistribute_real_comp.size();
1994 h_redistribute_real_comp.resize(cur_size-old_size+new_size, communicate);
1997 for (
int lev = 0; lev < numLevels(); ++lev) {
1999 auto& tile = DefineAndReturnParticleTile(lev, pti);
2000 auto np = tile.numParticles();
2001 if (np > 0 && new_size > old_size) {
2002 auto& soa = tile.GetStructOfArrays();
2009template <
typename ParticleType,
int NArrayReal,
int NArrayInt,
2010 template<
class>
class Allocator,
class CellAssignor>
2015 int old_size = m_num_runtime_int;
2017 m_runtime_comps_defined = (new_size > 0);
2018 m_num_runtime_int = new_size;
2019 int cur_size = h_redistribute_int_comp.size();
2020 h_redistribute_int_comp.resize(cur_size-old_size+new_size, communicate);
2023 for (
int lev = 0; lev < numLevels(); ++lev) {
2025 auto& tile = DefineAndReturnParticleTile(lev, pti);
2026 auto np = tile.numParticles();
2027 if (np > 0 && new_size > old_size) {
2028 auto& soa = tile.GetStructOfArrays();
#define BL_PROFILE_VAR_START(vname)
Definition AMReX_BLProfiler.H:562
#define BL_PROFILE(a)
Definition AMReX_BLProfiler.H:551
#define BL_PROFILE_VAR_STOP(vname)
Definition AMReX_BLProfiler.H:563
#define BL_PROFILE_SYNC_STOP()
Definition AMReX_BLProfiler.H:645
#define BL_PROFILE_SYNC_START_TIMED(fname)
Definition AMReX_BLProfiler.H:644
#define BL_PROFILE_VAR_NS(fname, vname)
Definition AMReX_BLProfiler.H:561
#define AMREX_ALWAYS_ASSERT_WITH_MESSAGE(EX, MSG)
Definition AMReX_BLassert.H:49
#define AMREX_ASSERT(EX)
Definition AMReX_BLassert.H:38
#define AMREX_FORCE_INLINE
Definition AMReX_Extension.H:119
#define AMREX_HOST_DEVICE_FOR_1D(...)
Definition AMReX_GpuLaunchMacrosC.nolint.H:105
#define AMREX_GPU_DEVICE
Definition AMReX_GpuQualifiers.H:18
#define AMREX_GPU_HOST_DEVICE
Definition AMReX_GpuQualifiers.H:20
amrex::ParmParse pp
Input file parser instance for the given namespace.
Definition AMReX_HypreIJIface.cpp:15
#define AMREX_D_TERM(a, b, c)
Definition AMReX_SPACE.H:172
#define AMREX_D_DECL(a, b, c)
Definition AMReX_SPACE.H:171
Print on all processors of the default communicator.
Definition AMReX_Print.H:113
BaseFab< T > & atomicAdd(const BaseFab< T > &x) noexcept
Atomic FAB addition (a[i] <- a[i] + b[i]).
Definition AMReX_BaseFab.H:2487
Array4< T const > array() const noexcept
Definition AMReX_BaseFab.H:387
void setVal(T const &x, const Box &bx, int dcomp, int ncomp) noexcept
The setVal functions set sub-regions in the BaseFab to a constant value. This most general form speci...
Definition AMReX_BaseFab.H:1404
A collection of Boxes stored in an Array.
Definition AMReX_BoxArray.H:564
IndexType ixType() const noexcept
Return index type of this BoxArray.
Definition AMReX_BoxArray.H:854
BoxArray & grow(int n)
Grow each Box in the BoxArray by the specified amount.
Definition AMReX_BoxArray.cpp:706
std::vector< std::pair< int, Box > > intersections(const Box &bx) const
Return intersections of Box and BoxArray.
Definition AMReX_BoxArray.cpp:1186
Box getCellCenteredBox(int index) const noexcept
Return cell-centered box at element index of this BoxArray.
Definition AMReX_BoxArray.H:744
static bool SameRefs(const BoxArray &lhs, const BoxArray &rhs)
whether two BoxArrays share the same data
Definition AMReX_BoxArray.H:837
A class for managing a List of Boxes that share a common IndexType. This class implements operations ...
Definition AMReX_BoxList.H:52
BoxList & complementIn(const Box &b, const BoxList &bl)
Definition AMReX_BoxList.cpp:307
__host__ __device__ BoxND & grow(int i) noexcept
Definition AMReX_Box.H:649
__host__ __device__ bool contains(const IntVectND< dim > &p) const noexcept
Return true if argument is contained within BoxND.
Definition AMReX_Box.H:216
GpuArray< Real, 3 > InvCellSizeArray() const noexcept
Definition AMReX_CoordSys.H:87
A Fortran Array of REALs.
Definition AMReX_FArrayBox.H:231
void resize(const Box &b, int N=1, Arena *ar=nullptr)
For debugging purposes we hide BaseFab version and do some extra work.
Definition AMReX_FArrayBox.cpp:178
int size() const noexcept
Return the number of FABs in the FabArray.
Definition AMReX_FabArrayBase.H:110
int nGrow(int direction=0) const noexcept
Return the grow factor that defines the region of definition.
Definition AMReX_FabArrayBase.H:78
const DistributionMapping & DistributionMap() const noexcept
Return constant reference to associated DistributionMapping.
Definition AMReX_FabArrayBase.H:131
const BoxArray & boxArray() const noexcept
Return a constant reference to the BoxArray that defines the valid region associated with this FabArr...
Definition AMReX_FabArrayBase.H:95
void ParallelCopy(const FabArray< FAB > &src, const Periodicity &period=Periodicity::NonPeriodic(), CpOp op=FabArrayBase::COPY)
Definition AMReX_FabArray.H:873
void setVal(value_type val)
Set all components in the entire region of each FAB to val.
Definition AMReX_FabArray.H:2705
Array4< typename FabArray< FAB >::value_type const > array(const MFIter &mfi) const noexcept
Definition AMReX_FabArray.H:561
Rectangular problem domain geometry.
Definition AMReX_Geometry.H:75
const Box & Domain() const noexcept
Returns our rectangular domain.
Definition AMReX_Geometry.H:216
GpuArray< Real, 3 > ProbLoArray() const noexcept
Definition AMReX_Geometry.H:192
static void streamSynchronize() noexcept
Definition AMReX_GpuDevice.cpp:856
__host__ __device__ bool cellCentered() const noexcept
True if the IndexTypeND is CELL based in all directions.
Definition AMReX_IndexType.H:102
__host__ static __device__ constexpr IntVectND< dim > TheZeroVector() noexcept
This static member function returns a reference to a constant IntVectND object, all of whose dim argu...
Definition AMReX_IntVect.H:771
a one-thingy-per-box distributed object
Definition AMReX_LayoutData.H:13
Iterator for looping ever tiles and boxes of amrex::FabArray based containers.
Definition AMReX_MFIter.H:88
bool isValid() const noexcept
Is the iterator valid i.e. is it associated with a FAB?
Definition AMReX_MFIter.H:172
A collection (stored as an array) of FArrayBox objects.
Definition AMReX_MultiFab.H:40
void mult(Real val, int comp, int num_comp, int nghost=0)
Scales the value of each cell in the specified subregion of the MultiFab by the scalar val (a[i] <- a...
Definition AMReX_MultiFab.cpp:1417
Dynamically allocated vector for trivially copyable data.
Definition AMReX_PODVector.H:308
size_type size() const noexcept
Definition AMReX_PODVector.H:648
void resize(size_type a_new_size, GrowthStrategy strategy=GrowthStrategy::Poisson)
Definition AMReX_PODVector.H:728
T * dataPtr() noexcept
Definition AMReX_PODVector.H:670
Definition AMReX_ParIter.H:147
Definition AMReX_ParIter.H:118
int queryAdd(std::string_view name, T &ref)
If name is found, the value in the ParmParse database will be stored in the ref argument....
Definition AMReX_ParmParse.H:1047
int queryarr(std::string_view name, std::vector< int > &ref, int start_ix=FIRST, int num_val=ALL) const
Same as queryktharr() but searches for last occurrence of name.
Definition AMReX_ParmParse.cpp:2034
int query(std::string_view name, bool &ref, int ival=FIRST) const
Same as querykth() but searches for the last occurrence of name.
Definition AMReX_ParmParse.cpp:1948
virtual void reserveData()
Definition AMReX_ParticleContainerBase.cpp:41
virtual void resizeData()
Definition AMReX_ParticleContainerBase.cpp:46
A distributed container for Particles sorted onto the levels, grids, and tiles of a block-structured ...
Definition AMReX_ParticleContainer.H:149
IntVect Index(const P &p, int lev) const
Definition AMReX_ParticleContainerI.H:202
std::map< std::pair< int, int >, ParticleTileType > ParticleLevel
Definition AMReX_ParticleContainer.H:196
void SetSoACompileTimeNames(std::vector< std::string > const &rdata_name, std::vector< std::string > const &idata_name)
Definition AMReX_ParticleContainerI.H:110
typename ParticleTileType::AoS AoS
Definition AMReX_ParticleContainer.H:199
void SetParticleSize()
Definition AMReX_ParticleContainerI.H:17
void RemoveParticlesAtLevel(int level)
The Following methods are for managing Virtual and Ghost Particles.
Definition AMReX_ParticleContainerI.H:759
void clearParticles()
Clear all the particles in this container. This does not free memory.
Definition AMReX_ParticleContainerI.H:1130
void Increment(MultiFab &mf, int level)
Definition AMReX_ParticleContainerI.H:722
bool HasIntComp(std::string const &name)
Definition AMReX_ParticleContainerI.H:149
void ShrinkToFit()
Definition AMReX_ParticleContainerI.H:702
bool HasRealComp(std::string const &name)
Definition AMReX_ParticleContainerI.H:141
void resizeData() override
This resizes the vector of dummy MultiFabs used by the ParticleContainer for the current number of le...
Definition AMReX_ParticleContainerI.H:436
Long IncrementWithTotal(MultiFab &mf, int level, bool local=false)
Definition AMReX_ParticleContainerI.H:749
Long NumberOfParticlesAtLevel(int level, bool only_valid=true, bool only_local=false) const
Returns # of particles at specified the level.
Definition AMReX_ParticleContainerI.H:552
void SortParticlesByCell()
Sort the particles on each tile by cell, using Fortran ordering.
Definition AMReX_ParticleContainerI.H:1283
int GetRealCompIndex(std::string const &name)
Definition AMReX_ParticleContainerI.H:162
int GetIntCompIndex(std::string const &name)
Definition AMReX_ParticleContainerI.H:184
void RemoveParticlesNotAtFinestLevel()
Definition AMReX_ParticleContainerI.H:773
Vector< Long > NumberOfParticlesInGrid(int level, bool only_valid=true, bool only_local=false) const
Definition AMReX_ParticleContainerI.H:497
ParticleLocData Reset(ParticleType &prt, bool update, bool verbose=true, ParticleLocData pld=ParticleLocData()) const
Updates a particle's location (Where), tries to periodic shift any particles that have left the domai...
Definition AMReX_ParticleContainerI.H:393
std::conditional_t< is_rtsoa_pc, ParticleTileRT< typename ParticleType::RealType, typename ParticleType::IntType >, ParticleTile< ParticleType, NArrayReal, NArrayInt, Allocator > > ParticleTileType
Definition AMReX_ParticleContainer.H:191
Long TotalNumberOfParticles(bool only_valid=true, bool only_local=false) const
Returns # of particles at all levels.
Definition AMReX_ParticleContainerI.H:482
void reserveData() override
This reserves data in the vector of dummy MultiFabs used by the ParticleContainer for the maximum num...
Definition AMReX_ParticleContainerI.H:427
T_ParticleType ParticleType
Definition AMReX_ParticleContainer.H:151
Definition AMReX_ParticleLocator.H:123
void build(const BoxArray &ba, const Geometry &geom, bool a_do_tiling=false, const IntVect &a_tile_size=IntVect(1024000, 1024000, 1024000))
Definition AMReX_ParticleLocator.H:130
AssignGrid< BinIteratorFactory > getGridAssignor() const noexcept
Definition AMReX_ParticleLocator.H:206
This class provides the user with a few print options.
Definition AMReX_Print.H:35
Definition AMReX_Reduce.H:438
Type value()
Definition AMReX_Reduce.H:473
Definition AMReX_Reduce.H:597
void eval(MF const &mf, IntVect const &nghost, D &reduce_data, F &&f)
Definition AMReX_Reduce.H:731
This class is a thin wrapper around std::vector. Unlike vector, Vector::operator[] provides bound che...
Definition AMReX_Vector.H:29
Long size() const noexcept
Definition AMReX_Vector.H:54
A Collection of IArrayBoxes.
Definition AMReX_iMultiFab.H:34
Long sum(int comp, int nghost=0, bool local=false) const
Returns the sum in component comp.
Definition AMReX_iMultiFab.cpp:454
amrex_real Real
Floating Point Type for Fields.
Definition AMReX_REAL.H:79
amrex_particle_real ParticleReal
Floating Point Type for Particles.
Definition AMReX_REAL.H:90
amrex_long Long
Definition AMReX_INT.H:30
T ExclusiveSum(N n, T const *in, T *out, RetSum a_ret_sum=retSum)
Exclusive sum.
Definition AMReX_Scan.H:1040
void SumBoundary(const Periodicity &period=Periodicity::NonPeriodic(), bool deterministic=false)
Sum values in overlapped cells.
Definition AMReX_FabArray.H:3655
__host__ __device__ BoxND< dim > grow(const BoxND< dim > &b, int i) noexcept
Grow BoxND in all directions by given amount.
Definition AMReX_Box.H:1289
Arena * The_Comms_Arena()
Definition AMReX_Arena.cpp:880
void Min(KeyValuePair< K, V > &vi, int root, MPI_Comm comm)
Definition AMReX_ParallelReduce.H:287
void Max(KeyValuePair< K, V > &vi, int root, MPI_Comm comm)
Definition AMReX_ParallelReduce.H:254
void Sum(T &v, MPI_Comm comm)
Definition AMReX_ParallelReduce.H:221
void Sum(T &v, int root, MPI_Comm comm)
Definition AMReX_ParallelReduce.H:352
__host__ __device__ AMREX_FORCE_INLINE void AddNoRet(T *sum, T value) noexcept
Definition AMReX_GpuAtomic.H:283
void streamSynchronize() noexcept
Definition AMReX_GpuDevice.H:310
void dtoh_memcpy_async(void *p_h, const void *p_d, const std::size_t sz) noexcept
Definition AMReX_GpuDevice.H:435
void htod_memcpy_async(void *p_d, const void *p_h, const std::size_t sz) noexcept
Definition AMReX_GpuDevice.H:421
void QueueReduction(Func f)
Definition AMReX_Lazy.cpp:7
constexpr Long GhostParticleID
Definition AMReX_Particle.H:19
constexpr Long VirtualParticleID
Definition AMReX_Particle.H:20
MPI_Comm CommunicatorSub() noexcept
sub-communicator for current frame
Definition AMReX_ParallelContext.H:70
int MyProcSub() noexcept
my sub-rank in current frame
Definition AMReX_ParallelContext.H:76
int IOProcessorNumberSub() noexcept
IO sub-rank in current frame.
Definition AMReX_ParallelContext.H:78
bool UseGpuAwareMpi()
Definition AMReX_ParallelDescriptor.H:113
void Bcast(void *, int, MPI_Datatype, int, MPI_Comm)
Definition AMReX_ParallelDescriptor.cpp:1295
void GatherLayoutDataToVector(const LayoutData< T > &sendbuf, Vector< T > &recvbuf, int root)
Gather LayoutData values to a vector on root.
Definition AMReX_ParallelDescriptor.H:1277
static constexpr RetSum retSum
Definition AMReX_Scan.H:33
Definition AMReX_Amr.cpp:50
void gatherParticles(PTile &dst, const PTile &src, N np, const Index *inds)
Gather particles copies particles into contiguous order from an arbitrary order. Specifically,...
Definition AMReX_ParticleTransformation.H:730
int nComp(FabArrayBase const &fa)
Definition AMReX_FabArrayBase.cpp:2852
__host__ __device__ void swapParticle(const ParticleTileData< T_ParticleType, NAR, NAI > &dst, const ParticleTileData< T_ParticleType, NAR, NAI > &src, int src_i, int dst_i) noexcept
A general single particle swapping routine that can run on the GPU.
Definition AMReX_ParticleTransformation.H:120
__host__ __device__ int getTileIndex(const IntVect &iv, const Box &box, const bool a_do_tiling, const IntVect &a_tile_size, Box &tbx)
Definition AMReX_ParticleUtil.H:191
__host__ __device__ void copyParticle(const ParticleTileData< T_ParticleType, NAR, NAI > &dst, const ConstParticleTileData< T_ParticleType, NAR, NAI > &src, int src_i, int dst_i) noexcept
A general single particle copying routine that can run on the GPU.
Definition AMReX_ParticleTransformation.H:32
void copyParticles(DstTile &dst, const SrcTile &src) noexcept
Copy particles from src to dst. This version copies all the particles, writing them to the beginning ...
Definition AMReX_ParticleTransformation.H:222
void communicateParticlesStart(const PC &pc, ParticleCopyPlan &plan, const SndBuffer &snd_buffer, RcvBuffer &rcv_buffer)
Definition AMReX_ParticleCommunication.H:909
int partitionParticlesByDest(PTile &ptile, const PLocator &ploc, CellAssignor const &assignor, const ParticleBufferMap &pmap, const GpuArray< Real, 3 > &plo, const GpuArray< Real, 3 > &phi, const GpuArray< ParticleReal, 3 > &rlo, const GpuArray< ParticleReal, 3 > &rhi, const GpuArray< int, 3 > &is_per, int lev, int gid, int tid, int lev_min, int lev_max, int nGrow, bool remove_negative)
Definition AMReX_ParticleUtil.H:654
void ReorderParticles(PTile &ptile, const index_type *permutations)
Reorder particles on the tile ptile using a the permutations array.
Definition AMReX_ParticleUtil.H:953
__host__ __device__ bool enforcePeriodic(P &p, amrex::GpuArray< amrex::Real, 3 > const &plo, amrex::GpuArray< amrex::Real, 3 > const &phi, amrex::GpuArray< amrex::ParticleReal, 3 > const &rlo, amrex::GpuArray< amrex::ParticleReal, 3 > const &rhi, amrex::GpuArray< int, 3 > const &is_per) noexcept
Definition AMReX_ParticleUtil.H:423
void ParallelFor(TypeList< CTOs... > ctos, std::array< int, sizeof...(CTOs)> const &runtime_options, T N, F &&f)
Definition AMReX_CTOParallelForImpl.H:202
__host__ __device__ int numTilesInBox(const Box &box, const bool a_do_tiling, const IntVect &a_tile_size)
Definition AMReX_ParticleUtil.H:239
BoxND< 3 > Box
Box is an alias for amrex::BoxND instantiated with AMREX_SPACEDIM.
Definition AMReX_BaseFwd.H:30
void ParticleToMesh(PC const &pc, const Vector< MultiFab * > &mf, int lev_min, int lev_max, F &&f, bool zero_out_input=true, bool vol_weight=true)
Deposit particles onto a hierarchy of MultiFabs.
Definition AMReX_AmrParticles.H:189
void unpackBuffer(PC &pc, const ParticleCopyPlan &plan, const Buffer &snd_buffer, UnpackPolicy const &policy)
Definition AMReX_ParticleCommunication.H:782
double second() noexcept
Definition AMReX_Utility.cpp:940
void packBuffer(const PC &pc, const ParticleCopyOp &op, const ParticleCopyPlan &plan, Buffer &snd_buffer)
Definition AMReX_ParticleCommunication.H:582
void communicateParticlesFinish(const ParticleCopyPlan &plan)
Definition AMReX_ParticleCommunication.cpp:445
Index filterAndTransformParticles(DstTile &dst, const SrcTile &src, Index *mask, F const &f, Index src_start, Index dst_start) noexcept
Conditionally copy particles from src to dst based on the value of mask. A transformation will also b...
Definition AMReX_ParticleTransformation.H:513
IntVectND< 3 > IntVect
IntVect is an alias for amrex::IntVectND instantiated with AMREX_SPACEDIM.
Definition AMReX_BaseFwd.H:33
void unpackRemotes(PC &pc, const ParticleCopyPlan &plan, Buffer &rcv_buffer, UnpackPolicy const &policy)
Definition AMReX_ParticleCommunication.H:1009
Index filterParticles(DstTile &dst, const SrcTile &src, const Index *mask) noexcept
Conditionally copy particles from src to dst based on the value of mask.
Definition AMReX_ParticleTransformation.H:389
void Error(const std::string &msg)
Print out message to cerr and exit via amrex::Abort().
Definition AMReX.cpp:235
int numParticlesOutOfRange(Iterator const &pti, int nGrow)
Returns the number of particles that are more than nGrow cells from the box correspond to the input i...
Definition AMReX_ParticleUtil.H:36
int Verbose() noexcept
Definition AMReX.cpp:181
void Abort(const std::string &msg)
Print out message to cerr and exit via abort().
Definition AMReX.cpp:241
const int[]
Definition AMReX_BLProfiler.cpp:1664
void transformParticles(DstTile &dst, const SrcTile &src, F &&f) noexcept
Apply the function f to all the particles in src, writing the result to dst. This version does all th...
Definition AMReX_ParticleTransformation.H:273
__host__ __device__ IntVect getParticleCell(P const &p, amrex::GpuArray< amrex::Real, 3 > const &plo, amrex::GpuArray< amrex::Real, 3 > const &dxi) noexcept
Returns the cell index for a given particle using the provided lower bounds and cell sizes.
Definition AMReX_ParticleUtil.H:343
Definition AMReX_ParticleLocator.H:248
A multidimensional array accessor.
Definition AMReX_Array4.H:285
Definition AMReX_ParticleContainerI.H:1040
amrex::AmrAssignGrid< amrex::DenseBinIteratorFactory< amrex::Box > > m_assign_grid
Definition AMReX_ParticleContainerI.H:1043
int m_lev_max
Definition AMReX_ParticleContainerI.H:1042
AssignGridFilter(amrex::AmrAssignGrid< amrex::DenseBinIteratorFactory< amrex::Box > > assign_grid, int gid, int level, int nGrow)
This filters based on matching grids.
Definition AMReX_ParticleContainerI.H:1049
int m_nGrow
Definition AMReX_ParticleContainerI.H:1042
int m_lev_min
Definition AMReX_ParticleContainerI.H:1042
AMREX_GPU_HOST_DEVICE int operator()(const SrcData &src, int src_i) const noexcept
Definition AMReX_ParticleContainerI.H:1055
int m_gid
Definition AMReX_ParticleContainerI.H:1042
Definition AMReX_ParticleLocator.H:17
Definition AMReX_ParticleUtil.H:396
Definition AMReX_DenseBins.H:32
Definition AMReX_ParticleContainerI.H:801
Box m_domain
Definition AMReX_ParticleContainerI.H:805
GpuArray< Real, 3 > m_dxi
Definition AMReX_ParticleContainerI.H:804
FilterVirt(const amrex::AssignGrid< amrex::DenseBinIteratorFactory< amrex::Box > > &assign_buffer_grid, const GpuArray< Real, 3 > &plo, const GpuArray< Real, 3 > &dxi, const Box &domain)
Definition AMReX_ParticleContainerI.H:807
AMREX_GPU_HOST_DEVICE int operator()(const SrcData &src, int src_i) const noexcept
Definition AMReX_ParticleContainerI.H:814
amrex::AssignGrid< amrex::DenseBinIteratorFactory< amrex::Box > > m_assign_buffer_grid
Definition AMReX_ParticleContainerI.H:803
GpuArray< Real, 3 > m_plo
Definition AMReX_ParticleContainerI.H:804
Definition AMReX_ParticleUtil.H:310
Fixed-size array that can be used on GPU.
Definition AMReX_Array.H:43
Definition AMReX_GpuControl.H:180
Definition AMReX_TypeTraits.H:90
Definition AMReX_ParticleCommunication.H:92
Vector< int > m_rcv_box_ids
Definition AMReX_ParticleCommunication.H:366
Vector< int > m_rcv_box_counts
Definition AMReX_ParticleCommunication.H:364
Vector< int > m_rcv_box_levs
Definition AMReX_ParticleCommunication.H:369
Vector< int > m_rcv_box_tids
Definition AMReX_ParticleCommunication.H:367
int m_nrcvs
Definition AMReX_ParticleCommunication.H:372
Gpu::HostVector< unsigned int > m_box_counts_h
Definition AMReX_ParticleCommunication.H:361
A struct used for storing a particle's position in the AMR hierarchy.
Definition AMReX_ParticleContainer.H:93
Box m_grown_gridbox
Definition AMReX_ParticleContainer.H:100
IntVect m_cell
Definition AMReX_ParticleContainer.H:97
int m_grid
Definition AMReX_ParticleContainer.H:95
int m_tile
Definition AMReX_ParticleContainer.H:96
int m_lev
Definition AMReX_ParticleContainer.H:94
Box m_tilebox
Definition AMReX_ParticleContainer.H:99
Box m_gridbox
Definition AMReX_ParticleContainer.H:98
The struct used to store particles.
Definition AMReX_Particle.H:405
__host__ __device__ RealVect pos() const &
Definition AMReX_Particle.H:456