Block-Structured AMR Software Framework
Loading...
Searching...
No Matches
AMReX_ParticleIO.H
Go to the documentation of this file.
1#ifndef AMREX_PARTICLEIO_H
2#define AMREX_PARTICLEIO_H
3#include <AMReX_Config.H>
4
7#include <AMReX_VisMF.H>
8
9#include <iterator>
10
11namespace amrex {
12
13template <typename ParticleType, int NArrayReal, int NArrayInt,
14 template<class> class Allocator, class CellAssignor>
15void
16ParticleContainer_impl<ParticleType, NArrayReal, NArrayInt, Allocator, CellAssignor>
17::WriteParticleRealData (void* data, size_t size, std::ostream& os) const
18{
19 if (sizeof(typename ParticleType::RealType) == 4) {
20 writeFloatData((float*) data, size, os, ParticleRealDescriptor);
21 }
22 else if (sizeof(typename ParticleType::RealType) == 8) {
23 writeDoubleData((double*) data, size, os, ParticleRealDescriptor);
24 }
25}
26
27template <typename ParticleType, int NArrayReal, int NArrayInt,
28 template<class> class Allocator, class CellAssignor>
29void
31::ReadParticleRealData (void* data, size_t size, std::istream& is)
32{
33 if (sizeof(typename ParticleType::RealType) == 4) {
34 readFloatData((float*) data, size, is, ParticleRealDescriptor);
35 }
36 else if (sizeof(typename ParticleType::RealType) == 8) {
37 readDoubleData((double*) data, size, is, ParticleRealDescriptor);
38 }
39}
40
42 template <typename P>
44 int operator() (const P& p) const {
45 return p.id().is_valid();
46 }
47};
48
50template <typename ParticleType, int NArrayReal, int NArrayInt,
51 template<class> class Allocator, class CellAssignor>
52void
53ParticleContainer_impl<ParticleType, NArrayReal, NArrayInt, Allocator, CellAssignor>
54::Checkpoint (const std::string& dir,
55 const std::string& name, bool /*is_checkpoint*/,
56 const Vector<std::string>& real_comp_names,
57 const Vector<std::string>& int_comp_names) const
58{
59 Vector<int> write_real_comp;
60 Vector<std::string> tmp_real_comp_names;
61
62 int first_rcomp = ParticleType::is_soa_particle ? AMREX_SPACEDIM : 0;
63 for (int i = first_rcomp; i < NStructReal + NumRealComps(); ++i )
64 {
65 write_real_comp.push_back(1);
66 if (real_comp_names.empty())
67 {
68 tmp_real_comp_names.push_back(getDefaultCompNameReal<ParticleType>(i));
69 }
70 else
71 {
72 tmp_real_comp_names.push_back(real_comp_names[i-first_rcomp]);
73 }
74 }
75
76 Vector<int> write_int_comp;
77 Vector<std::string> tmp_int_comp_names;
78 for (int i = 0; i < NStructInt + NumIntComps(); ++i )
79 {
80 write_int_comp.push_back(1);
81 if (int_comp_names.empty())
82 {
83 tmp_int_comp_names.push_back(getDefaultCompNameInt<ParticleType>(i));
84 }
85 else
86 {
87 tmp_int_comp_names.push_back(int_comp_names[i]);
88 }
89 }
90
91 WriteBinaryParticleData(dir, name, write_real_comp, write_int_comp,
92 tmp_real_comp_names, tmp_int_comp_names,
93 FilterPositiveID{}, true);
94}
95
96template <typename ParticleType, int NArrayReal, int NArrayInt,
97 template<class> class Allocator, class CellAssignor>
98void
100::Checkpoint (const std::string& dir, const std::string& name,
101 const Vector<int>& write_real_comp,
102 const Vector<int>& write_int_comp,
103 const Vector<std::string>& real_comp_names,
104 const Vector<std::string>& int_comp_names) const
105{
106 WriteBinaryParticleData(dir, name, write_real_comp, write_int_comp,
107 real_comp_names, int_comp_names,
108 FilterPositiveID{}, true);
109}
110
112template <typename ParticleType, int NArrayReal, int NArrayInt,
113 template<class> class Allocator, class CellAssignor>
114void
116::WritePlotFile (const std::string& dir, const std::string& name) const
117{
118 Vector<int> write_real_comp;
119 Vector<std::string> real_comp_names;
120
121 int first_rcomp = ParticleType::is_soa_particle ? AMREX_SPACEDIM : 0;
122 for (int i = first_rcomp; i < NStructReal + NumRealComps(); ++i )
123 {
124 write_real_comp.push_back(1);
125 real_comp_names.push_back(getDefaultCompNameReal<ParticleType>(i));
126 }
127
128 Vector<int> write_int_comp;
129 Vector<std::string> int_comp_names;
130 for (int i = 0; i < NStructInt + NumIntComps(); ++i )
131 {
132 write_int_comp.push_back(1);
133 int_comp_names.push_back(getDefaultCompNameInt<ParticleType>(i));
134 }
135
136 WriteBinaryParticleData(dir, name, write_real_comp, write_int_comp,
137 real_comp_names, int_comp_names,
139}
140
142template <typename ParticleType, int NArrayReal, int NArrayInt,
143 template<class> class Allocator, class CellAssignor>
144void
146::WritePlotFile (const std::string& dir, const std::string& name,
147 const Vector<std::string>& real_comp_names,
148 const Vector<std::string>& int_comp_names) const
149{
150 if constexpr(ParticleType::is_soa_particle) {
151 AMREX_ALWAYS_ASSERT(real_comp_names.size() == NumRealComps() + NStructReal - AMREX_SPACEDIM); // pure SoA: skip positions
152 } else {
153 AMREX_ALWAYS_ASSERT(real_comp_names.size() == NumRealComps() + NStructReal);
154 }
155 AMREX_ASSERT( int_comp_names.size() == NStructInt + NumIntComps() );
156
157 Vector<int> write_real_comp;
158 int nrc = ParticleType::is_soa_particle ? NStructReal + NumRealComps() - AMREX_SPACEDIM : NStructReal + NumRealComps();
159 for (int i = 0; i < nrc; ++i) {
160 write_real_comp.push_back(1);
161 }
162
163 Vector<int> write_int_comp;
164 for (int i = 0; i < NStructInt + NumIntComps(); ++i) {
165 write_int_comp.push_back(1);
166 }
167
168 WriteBinaryParticleData(dir, name,
169 write_real_comp, write_int_comp,
170 real_comp_names, int_comp_names,
172}
173
175template <typename ParticleType, int NArrayReal, int NArrayInt,
176 template<class> class Allocator, class CellAssignor>
177void
179::WritePlotFile (const std::string& dir, const std::string& name,
180 const Vector<std::string>& real_comp_names) const
181{
182 if constexpr(ParticleType::is_soa_particle) {
183 AMREX_ALWAYS_ASSERT(real_comp_names.size() == NumRealComps() + NStructReal - AMREX_SPACEDIM); // pure SoA: skip positions
184 } else {
185 AMREX_ALWAYS_ASSERT(real_comp_names.size() == NumRealComps() + NStructReal);
186 }
187
188 Vector<int> write_real_comp;
189 int nrc = ParticleType::is_soa_particle ? NStructReal + NumRealComps() - AMREX_SPACEDIM : NStructReal + NumRealComps();
190 for (int i = 0; i < nrc; ++i) {
191 write_real_comp.push_back(1);
192 }
193
194 Vector<int> write_int_comp;
195 for (int i = 0; i < NStructInt + NumIntComps(); ++i) {
196 write_int_comp.push_back(1);
197 }
198
199 Vector<std::string> int_comp_names;
200 for (int i = 0; i < NStructInt + NumIntComps(); ++i )
201 {
202 int_comp_names.push_back(getDefaultCompNameInt<ParticleType>(i));
203 }
204
205 WriteBinaryParticleData(dir, name,
206 write_real_comp, write_int_comp,
207 real_comp_names, int_comp_names,
209}
210
212template <typename ParticleType, int NArrayReal, int NArrayInt,
213 template<class> class Allocator, class CellAssignor>
214void
216::WritePlotFile (const std::string& dir,
217 const std::string& name,
218 const Vector<int>& write_real_comp,
219 const Vector<int>& write_int_comp) const
220{
221
222 if constexpr(ParticleType::is_soa_particle) {
223 AMREX_ALWAYS_ASSERT(write_real_comp.size() == NumRealComps() + NStructReal - AMREX_SPACEDIM); // pure SoA: skip positions
224 } else {
225 AMREX_ALWAYS_ASSERT(write_real_comp.size() == NumRealComps() + NStructReal);
226 }
227 AMREX_ASSERT(write_int_comp.size() == NStructInt + NumIntComps() );
228
229 Vector<std::string> real_comp_names;
230 int first_rcomp = ParticleType::is_soa_particle ? AMREX_SPACEDIM : 0;
231 for (int i = first_rcomp; i < NStructReal + NumRealComps(); ++i )
232 {
233 real_comp_names.push_back(getDefaultCompNameReal<ParticleType>(i));
234 }
235
236 Vector<std::string> int_comp_names;
237 for (int i = 0; i < NStructInt + NumIntComps(); ++i )
238 {
239 int_comp_names.push_back(getDefaultCompNameInt<ParticleType>(i));
240 }
241
242 WriteBinaryParticleData(dir, name, write_real_comp, write_int_comp,
243 real_comp_names, int_comp_names,
245}
246
248template <typename ParticleType, int NArrayReal, int NArrayInt,
249 template<class> class Allocator, class CellAssignor>
250void
252WritePlotFile (const std::string& dir, const std::string& name,
253 const Vector<int>& write_real_comp,
254 const Vector<int>& write_int_comp,
255 const Vector<std::string>& real_comp_names,
256 const Vector<std::string>& int_comp_names) const
257{
258 BL_PROFILE("ParticleContainer::WritePlotFile()");
259
260 WriteBinaryParticleData(dir, name,
261 write_real_comp, write_int_comp,
262 real_comp_names, int_comp_names,
264}
265
267template <typename ParticleType, int NArrayReal, int NArrayInt,
268 template<class> class Allocator, class CellAssignor>
269template <class F>
270void
272::WritePlotFile (const std::string& dir, const std::string& name, F const& f) const
273{
274 Vector<int> write_real_comp;
275 Vector<std::string> real_comp_names;
276
277 int first_rcomp = ParticleType::is_soa_particle ? AMREX_SPACEDIM : 0;
278 for (int i = first_rcomp; i < NStructReal + NumRealComps(); ++i )
279 {
280 write_real_comp.push_back(1);
281 real_comp_names.push_back(getDefaultCompNameReal<ParticleType>(i));
282 }
283
284 Vector<int> write_int_comp;
285 Vector<std::string> int_comp_names;
286 for (int i = 0; i < NStructInt + NumIntComps(); ++i )
287 {
288 write_int_comp.push_back(1);
289 int_comp_names.push_back(getDefaultCompNameInt<ParticleType>(i));
290 }
291
292 WriteBinaryParticleData(dir, name, write_real_comp, write_int_comp,
293 real_comp_names, int_comp_names, f);
294}
295
297template <typename ParticleType, int NArrayReal, int NArrayInt,
298 template<class> class Allocator, class CellAssignor>
299template <class F>
300void
302::WritePlotFile (const std::string& dir, const std::string& name,
303 const Vector<std::string>& real_comp_names,
304 const Vector<std::string>& int_comp_names, F const& f) const
305{
306 if constexpr(ParticleType::is_soa_particle) {
307 AMREX_ALWAYS_ASSERT(real_comp_names.size() == NumRealComps() + NStructReal - AMREX_SPACEDIM); // pure SoA: skip positions
308 } else {
309 AMREX_ALWAYS_ASSERT(real_comp_names.size() == NumRealComps() + NStructReal);
310 }
311 AMREX_ASSERT( int_comp_names.size() == NStructInt + NumIntComps() );
312
313 Vector<int> write_real_comp;
314 int nrc = ParticleType::is_soa_particle ? NStructReal + NumRealComps() - AMREX_SPACEDIM : NStructReal + NumRealComps();
315 for (int i = 0; i < nrc; ++i) {
316 write_real_comp.push_back(1);
317 }
318
319 Vector<int> write_int_comp;
320 for (int i = 0; i < NStructInt + NumIntComps(); ++i) {
321 write_int_comp.push_back(1);
322 }
323
324 WriteBinaryParticleData(dir, name,
325 write_real_comp, write_int_comp,
326 real_comp_names, int_comp_names, f);
327}
328
330template <typename ParticleType, int NArrayReal, int NArrayInt,
331 template<class> class Allocator, class CellAssignor>
332template <class F>
333void
335::WritePlotFile (const std::string& dir, const std::string& name,
336 const Vector<std::string>& real_comp_names, F const& f) const
337{
338 if constexpr(ParticleType::is_soa_particle) {
339 AMREX_ALWAYS_ASSERT(real_comp_names.size() == NumRealComps() + NStructReal - AMREX_SPACEDIM); // pure SoA: skip positions
340 } else {
341 AMREX_ALWAYS_ASSERT(real_comp_names.size() == NumRealComps() + NStructReal);
342 }
343
344 Vector<int> write_real_comp;
345 int nrc = ParticleType::is_soa_particle ? NStructReal + NumRealComps() - AMREX_SPACEDIM : NStructReal + NumRealComps();
346 for (int i = 0; i < nrc; ++i) {
347 write_real_comp.push_back(1);
348 }
349
350 Vector<int> write_int_comp;
351 for (int i = 0; i < NStructInt + NumIntComps(); ++i) {
352 write_int_comp.push_back(1);
353 }
354
355 Vector<std::string> int_comp_names;
356 for (int i = 0; i < NStructInt + NumIntComps(); ++i )
357 {
358 int_comp_names.push_back(getDefaultCompNameInt<ParticleType>(i));
359 }
360
361 WriteBinaryParticleData(dir, name,
362 write_real_comp, write_int_comp,
363 real_comp_names, int_comp_names, f);
364}
365
367template <typename ParticleType, int NArrayReal, int NArrayInt,
368 template<class> class Allocator, class CellAssignor>
369template <class F>
370void
372::WritePlotFile (const std::string& dir,
373 const std::string& name,
374 const Vector<int>& write_real_comp,
375 const Vector<int>& write_int_comp, F const& f) const
376{
377 if constexpr(ParticleType::is_soa_particle) {
378 AMREX_ALWAYS_ASSERT(write_real_comp.size() == NumRealComps() + NStructReal - AMREX_SPACEDIM); // pure SoA: skip positions
379 } else {
380 AMREX_ALWAYS_ASSERT(write_real_comp.size() == NumRealComps() + NStructReal);
381 }
382 AMREX_ASSERT(write_int_comp.size() == NStructInt + NumIntComps() );
383
384 Vector<std::string> real_comp_names;
385 int first_rcomp = ParticleType::is_soa_particle ? AMREX_SPACEDIM : 0;
386 for (int i = first_rcomp; i < NStructReal + NumRealComps(); ++i )
387 {
388 real_comp_names.push_back(getDefaultCompNameReal<ParticleType>(i));
389 }
390
391 Vector<std::string> int_comp_names;
392 for (int i = 0; i < NStructInt + NumIntComps(); ++i )
393 {
394 int_comp_names.push_back(getDefaultCompNameInt<ParticleType>(i));
395 }
396
397 WriteBinaryParticleData(dir, name, write_real_comp, write_int_comp,
398 real_comp_names, int_comp_names, f);
399}
400
402template <typename ParticleType, int NArrayReal, int NArrayInt,
403 template<class> class Allocator, class CellAssignor>
404template <class F>
405void
407WritePlotFile (const std::string& dir, const std::string& name,
408 const Vector<int>& write_real_comp,
409 const Vector<int>& write_int_comp,
410 const Vector<std::string>& real_comp_names,
411 const Vector<std::string>& int_comp_names,
412 F const& f) const
413{
414 BL_PROFILE("ParticleContainer::WritePlotFile()");
415
416 WriteBinaryParticleData(dir, name,
417 write_real_comp, write_int_comp,
418 real_comp_names, int_comp_names, f);
419}
420
421template <typename ParticleType, int NArrayReal, int NArrayInt,
422 template<class> class Allocator, class CellAssignor>
423template <class F>
424void
426::WriteBinaryParticleData (const std::string& dir, const std::string& name,
427 const Vector<int>& write_real_comp,
428 const Vector<int>& write_int_comp,
429 const Vector<std::string>& real_comp_names,
430 const Vector<std::string>& int_comp_names,
431 F const& f, bool is_checkpoint) const
432{
433 if (AsyncOut::UseAsyncOut()) {
434 WriteBinaryParticleDataAsync(*this, dir, name,
435 write_real_comp, write_int_comp,
436 real_comp_names, int_comp_names, is_checkpoint);
437 } else
438 {
439 WriteBinaryParticleDataSync(*this, dir, name,
440 write_real_comp, write_int_comp,
441 real_comp_names, int_comp_names,
442 f, is_checkpoint);
443 }
444}
445
446template <typename ParticleType, int NArrayReal, int NArrayInt,
447 template<class> class Allocator, class CellAssignor>
448void
451{
452 if( ! usePrePost) {
453 return;
454 }
455
456 BL_PROFILE("ParticleContainer::CheckpointPre()");
457
458 const int IOProcNumber = ParallelDescriptor::IOProcessorNumber();
459 Long nparticles = 0;
460 Long maxnextid = ParticleType::NextID();
461
462 for (int lev = 0; lev < std::ssize(m_particles); lev++) {
463 const auto& pmap = m_particles[lev];
464 for (const auto& kv : pmap) {
465 const auto& aos = kv.second.GetArrayOfStructs();
466 for (int k = 0; k < aos.numParticles(); ++k) {
467 const ParticleType& p = aos[k];
468 if (p.id().is_valid()) {
469 //
470 // Only count (and checkpoint) valid particles.
471 //
472 nparticles++;
473 }
474 }
475 }
476 }
477 ParallelDescriptor::ReduceLongSum(nparticles, IOProcNumber);
478
479 ParticleType::NextID(maxnextid);
480 ParallelDescriptor::ReduceLongMax(maxnextid, IOProcNumber);
481
482 nparticlesPrePost = nparticles;
483 maxnextidPrePost = maxnextid;
484
485 nParticlesAtLevelPrePost.clear();
486 nParticlesAtLevelPrePost.resize(finestLevel() + 1, 0);
487 for(int lev(0); lev <= finestLevel(); ++lev) {
488 nParticlesAtLevelPrePost[lev] = NumberOfParticlesAtLevel(lev);
489 }
490
491 whichPrePost.clear();
492 whichPrePost.resize(finestLevel() + 1);
493 countPrePost.clear();
494 countPrePost.resize(finestLevel() + 1);
495 wherePrePost.clear();
496 wherePrePost.resize(finestLevel() + 1);
497
498 filePrefixPrePost.clear();
499 filePrefixPrePost.resize(finestLevel() + 1);
500}
501
502
503template <typename ParticleType, int NArrayReal, int NArrayInt,
504 template<class> class Allocator, class CellAssignor>
505void
508{
509 if( ! usePrePost) {
510 return;
511 }
512
513 BL_PROFILE("ParticleContainer::CheckpointPost()");
514
515 const int IOProcNumber = ParallelDescriptor::IOProcessorNumber();
516 std::ofstream HdrFile;
517 HdrFile.open(HdrFileNamePrePost.c_str(), std::ios::out | std::ios::app);
518
519 for(int lev(0); lev <= finestLevel(); ++lev) {
520 ParallelDescriptor::ReduceIntSum (whichPrePost[lev].dataPtr(), whichPrePost[lev].size(), IOProcNumber);
521 ParallelDescriptor::ReduceIntSum (countPrePost[lev].dataPtr(), countPrePost[lev].size(), IOProcNumber);
522 ParallelDescriptor::ReduceLongSum(wherePrePost[lev].dataPtr(), wherePrePost[lev].size(), IOProcNumber);
523
524
526 for(int j(0); j < whichPrePost[lev].size(); ++j) {
527 HdrFile << whichPrePost[lev][j] << ' ' << countPrePost[lev][j] << ' ' << wherePrePost[lev][j] << '\n';
528 }
529
530 const bool gotsome = (nParticlesAtLevelPrePost[lev] > 0);
531 if(gotsome && doUnlink) {
532// BL_PROFILE_VAR("PC<NNNN>::Checkpoint:unlink", unlink_post);
533 // Unlink any zero-length data files.
534 Vector<Long> cnt(nOutFilesPrePost,0);
535
536 for(int i(0), N = countPrePost[lev].size(); i < N; ++i) {
537 cnt[whichPrePost[lev][i]] += countPrePost[lev][i];
538 }
539
540 for(int i = 0; i < std::ssize(cnt); ++i) {
541 if(cnt[i] == 0) {
542 std::string FullFileName = NFilesIter::FileName(i, filePrefixPrePost[lev]);
543 FileSystem::Remove(FullFileName);
544 }
545 }
546 }
547 }
548 }
549
551 HdrFile.flush();
552 HdrFile.close();
553 if( ! HdrFile.good()) {
554 amrex::Abort("ParticleContainer::CheckpointPost(): problem writing HdrFile");
555 }
556 }
557}
558
559template <typename ParticleType, int NArrayReal, int NArrayInt,
560 template<class> class Allocator, class CellAssignor>
561void
564{
565 CheckpointPre();
566}
567
568
569template <typename ParticleType, int NArrayReal, int NArrayInt,
570 template<class> class Allocator, class CellAssignor>
571void
574{
575 CheckpointPost();
576}
577
578
579template <typename ParticleType, int NArrayReal, int NArrayInt,
580 template<class> class Allocator, class CellAssignor>
581void
583::WriteParticles (int lev, std::ofstream& ofs, int fnum,
584 Vector<int>& which, Vector<int>& count, Vector<Long>& where,
585 const Vector<int>& write_real_comp,
586 const Vector<int>& write_int_comp,
587 const Vector<std::map<std::pair<int, int>, FlagsVector>>& particle_io_flags,
588 bool is_checkpoint) const
589{
590 BL_PROFILE("ParticleContainer::WriteParticles()");
591
592 // For a each grid, the tiles it contains
593 std::map<int, Vector<int> > tile_map;
594
595 for (const auto& kv : m_particles[lev])
596 {
597 const int grid = kv.first.first;
598 const int tile = kv.first.second;
599 tile_map[grid].push_back(tile);
600 const auto& pflags = particle_io_flags[lev].at(kv.first);
601
602 // Only write out valid particles.
603 count[grid] += particle_detail::countFlags(pflags);
604 }
605
606 MFInfo info;
607 info.SetAlloc(false);
608 MultiFab state(ParticleBoxArray(lev), ParticleDistributionMap(lev), 1,0,info);
609
610 for (MFIter mfi(state); mfi.isValid(); ++mfi)
611 {
612 const int grid = mfi.index();
613
614 which[grid] = fnum;
615 where[grid] = VisMF::FileOffset(ofs);
616
617 if (count[grid] <= 0) { continue; }
618
619 Vector<int> istuff;
621 particle_detail::packIOData(istuff, rstuff, *this, lev, grid,
622 write_real_comp, write_int_comp,
623 particle_io_flags, tile_map[grid], count[grid], is_checkpoint);
624
625 writeIntData(istuff.dataPtr(), istuff.size(), ofs);
627 ofs.flush(); // Some systems require this flush() (probably due to a bug)
628 }
629
630 WriteParticleRealData(rstuff.dataPtr(), rstuff.size(), ofs);
632 ofs.flush(); // Some systems require this flush() (probably due to a bug)
633 }
634 }
635}
636
637
638template <typename ParticleType, int NArrayReal, int NArrayInt,
639 template<class> class Allocator, class CellAssignor>
640void
642::Restart (const std::string& dir, const std::string& file, bool /*is_checkpoint*/)
643{
644 Restart(dir, file);
645}
646
647template <typename ParticleType, int NArrayReal, int NArrayInt,
648 template<class> class Allocator, class CellAssignor>
649void
651::Restart (const std::string& dir, const std::string& file)
652{
653 BL_PROFILE("ParticleContainer::Restart()");
654 AMREX_ASSERT(!dir.empty());
655 AMREX_ASSERT(!file.empty());
656
657 const auto strttime = amrex::second();
658
659 int DATA_Digits_Read(5);
660 ParmParse pp("particles");
661 pp.query("datadigits_read",DATA_Digits_Read);
662
663 std::string fullname = dir;
664 if (!fullname.empty() && fullname[fullname.size()-1] != '/') {
665 fullname += '/';
666 }
667 fullname += file;
668 std::string HdrFileName = fullname;
669 if (!HdrFileName.empty() && HdrFileName[HdrFileName.size()-1] != '/') {
670 HdrFileName += '/';
671 }
672 HdrFileName += "Header";
673
674 Vector<char> fileCharPtr;
675 ParallelDescriptor::ReadAndBcastFile(HdrFileName, fileCharPtr);
676 std::string fileCharPtrString(fileCharPtr.dataPtr());
677 std::istringstream HdrFile(fileCharPtrString, std::istringstream::in);
678
679 // The metadata prefix (version through finest level) is parsed by the
680 // shared ParticleHeader, which leaves HdrFile positioned at the per-level
681 // grid table read below. See AMReX_ParticleHeader.{H,cpp}.
682 ParticleHeader header;
683 header.parse(HdrFile);
684
685 const std::string& how = header.how;
686 const bool convert_ids = header.convert_ids;
687
688 if (header.dim != AMREX_SPACEDIM) {
689 amrex::Abort("ParticleContainer::Restart(): dm != AMREX_SPACEDIM");
690 }
691
692 const int nrc = ParticleType::is_soa_particle ? NStructReal + NumRealComps() - AMREX_SPACEDIM : NStructReal + NumRealComps();
693 if (header.num_real != nrc) {
694 amrex::Abort("ParticleContainer::Restart(): nr not the expected value");
695 }
696
697 if (header.num_int != NStructInt + NumIntComps()) {
698 amrex::Abort("ParticleContainer::Restart(): ni != NStructInt");
699 }
700
701 AMREX_ASSERT(header.num_particles >= 0);
702
703 AMREX_ASSERT(header.next_id > 0);
704 ParticleType::NextID(header.next_id);
705
706 const int finest_level_in_file = header.finest_level;
707 AMREX_ASSERT(finest_level_in_file >= 0);
708
709 // Read the BoxArrays stored in the checkpoint for each level so we can
710 // drive the read using the file's layout. The container's own grids are
711 // never modified: Redistribute() will move every particle to the correct
712 // tile on whatever grids the container currently uses.
713 Vector<BoxArray> file_bas(finest_level_in_file + 1);
714 Vector<DistributionMapping> file_dms(finest_level_in_file + 1);
715 for (int lev = 0; lev <= finest_level_in_file; lev++)
716 {
717 std::string phdr_name = fullname;
718 phdr_name += "/Level_";
719 phdr_name = amrex::Concatenate(phdr_name, lev, 1);
720 phdr_name += "/Particle_H";
721
722 if (amrex::FileExists(phdr_name)) {
723 Vector<char> phdr_chars;
724 ParallelDescriptor::ReadAndBcastFile(phdr_name, phdr_chars);
725 std::string phdr_string(phdr_chars.dataPtr());
726 std::istringstream phdr_file(phdr_string, std::istringstream::in);
727 file_bas[lev].readFrom(phdr_file);
728 } else {
729 // No particles were written at this level; use a single-box BA as
730 // a placeholder so that MFIter below works uniformly.
731 file_bas[lev] = BoxArray(Geom(lev).Domain());
732 }
733 const int NReaders = MaxReaders();
734 if (ParallelDescriptor::NProcs() <= NReaders) {
735 file_dms[lev] = DistributionMapping(file_bas[lev]);
736 } else {
737 Vector<int> pmap(file_bas[lev].size());
738 for (int i = 0; i < (int)pmap.size(); ++i) { pmap[i] = i % NReaders; }
739 file_dms[lev] = DistributionMapping(pmap);
740 }
741 }
742
743 Vector<int> ngrids(finest_level_in_file+1);
744 for (int lev = 0; lev <= finest_level_in_file; lev++) {
745 HdrFile >> ngrids[lev];
746 AMREX_ASSERT(ngrids[lev] > 0);
747 }
748
749 resizeData();
750
751 if (finest_level_in_file > finestLevel()) {
752 m_particles.resize(finest_level_in_file+1);
753 }
754
755 for (int lev = 0; lev <= finest_level_in_file; lev++) {
756 Vector<int> which(ngrids[lev]);
757 Vector<int> count(ngrids[lev]);
758 Vector<Long> where(ngrids[lev]);
759 for (int i = 0; i < ngrids[lev]; i++) {
760 HdrFile >> which[i] >> count[i] >> where[i];
761 }
762
763 // Use the file's own layout to distribute reading across ranks.
764 MFInfo info;
765 info.SetAlloc(false);
766 MultiFab file_mf(file_bas[lev], file_dms[lev], 1, 0, info);
767 Vector<int> grids_to_read;
768 for (MFIter mfi(file_mf); mfi.isValid(); ++mfi) {
769 grids_to_read.push_back(mfi.index());
770 }
771
772 for(int grid : grids_to_read) {
773 if (count[grid] <= 0) { continue; }
774
775 // The file names in the header file are relative.
776 std::string name = fullname;
777
778 if (!name.empty() && name[name.size()-1] != '/') {
779 name += '/';
781
782 name += "Level_";
783 name += amrex::Concatenate("", lev, 1);
784 name += '/';
785 name += DataPrefix();
786 name += amrex::Concatenate("", which[grid], DATA_Digits_Read);
787
788 std::ifstream ParticleFile;
789
790 ParticleFile.open(name.c_str(), std::ios::in | std::ios::binary);
791
792 if (!ParticleFile.good()) {
794 }
795
796 ParticleFile.seekg(where[grid], std::ios::beg);
797
798 // Use if constexpr to avoid instantiating the mis-matched
799 // type case and triggering the static_assert on the
800 // underlying copy calls
801 if (how == "single") {
802 if constexpr (std::is_same_v<ParticleReal, float>) {
803 ReadParticles<float>(count[grid], grid, lev, ParticleFile, finest_level_in_file, convert_ids);
804 } else {
805 amrex::Error("File contains single-precision data, while AMReX is compiled with ParticleReal==double");
806 }
807 }
808 else if (how == "double") {
809 if constexpr (std::is_same_v<ParticleReal, double>) {
810 ReadParticles<double>(count[grid], grid, lev, ParticleFile, finest_level_in_file, convert_ids);
811 } else {
812 amrex::Error("File contains double-precision data, while AMReX is compiled with ParticleReal==float");
813 }
814 }
815 else {
816 std::string msg("ParticleContainer::Restart(): bad parameter: ");
817 msg += how;
818 amrex::Error(msg.c_str());
819 }
820
821 ParticleFile.close();
822
823 if (!ParticleFile.good()) {
824 amrex::Abort("ParticleContainer::Restart(): problem reading particles");
826 }
828
829 Redistribute();
830
831 AMREX_ASSERT(OK());
832
833 if (m_verbose > 1) {
834 auto stoptime = amrex::second() - strttime;
836 amrex::Print() << "ParticleContainer::Restart() time: " << stoptime << '\n';
837 }
838}
839
840// Read a batch of particles from the checkpoint file
841template <typename ParticleType, int NArrayReal, int NArrayInt,
842 template<class> class Allocator, class CellAssignor>
843template <class RTYPE>
844void
846::ReadParticles (int cnt, int grd, int lev, std::ifstream& ifs,
847 int finest_level_in_file, bool convert_ids)
848{
849 BL_PROFILE("ParticleContainer::ReadParticles()");
850 AMREX_ASSERT(cnt > 0);
851 AMREX_ASSERT(lev < std::ssize(m_particles));
853 // First read in the integer data in binary. We do not store
854 // the m_lev and m_grid data on disk. We can easily recreate
855 // that given the structure of the checkpoint file.
856 const int iChunkSize = 2 + NStructInt + NumIntComps();
857 Vector<int> istuff(std::size_t(cnt)*iChunkSize);
858 readIntData(istuff.dataPtr(), istuff.size(), ifs, FPC::NativeIntDescriptor());
859
860 // Then the real data in binary.
861 const int rChunkSize = ParticleType::is_soa_particle ? NStructReal + NumRealComps() : AMREX_SPACEDIM + NStructReal + NumRealComps();
862 Vector<RTYPE> rstuff(std::size_t(cnt)*rChunkSize);
863 ReadParticleRealData(rstuff.dataPtr(), rstuff.size(), ifs);
864
865 // Now reassemble the particles.
866 int* iptr = istuff.dataPtr();
867 RTYPE* rptr = rstuff.dataPtr();
868
870 ParticleLocData pld;
871
873 host_particles.reserve(15);
874 host_particles.resize(finest_level_in_file+1);
875
877 std::vector<Gpu::HostVector<RTYPE> > > > host_real_attribs;
878 host_real_attribs.reserve(15);
879 host_real_attribs.resize(finest_level_in_file+1);
880
882 std::vector<Gpu::HostVector<int> > > > host_int_attribs;
883 host_int_attribs.reserve(15);
884 host_int_attribs.resize(finest_level_in_file+1);
885
887 host_idcpu.reserve(15);
888 host_idcpu.resize(finest_level_in_file+1);
889
890 for (int i = 0; i < cnt; i++) {
891 // note: for pure SoA particle layouts, we do write the id, cpu and positions as a struct
892 // for backwards compatibility with readers
893 if (convert_ids) {
894 std::int32_t xi, yi;
895 std::uint32_t xu, yu;
896 xi = iptr[0];
897 yi = iptr[1];
898 std::memcpy(&xu, &xi, sizeof(xi));
899 std::memcpy(&yu, &yi, sizeof(yi));
900 ptemp.m_idcpu = ((std::uint64_t)xu) << 32 | yu;
901 } else {
902 ptemp.id() = iptr[0];
903 ptemp.cpu() = iptr[1];
904 }
905 iptr += 2;
907 for (int j = 0; j < NStructInt; j++)
908 {
909 ptemp.idata(j) = *iptr;
910 ++iptr;
911 }
912
913 AMREX_ASSERT(ptemp.id().is_valid());
914
915 AMREX_D_TERM(ptemp.pos(0) = ParticleReal(rptr[0]);,
916 ptemp.pos(1) = ParticleReal(rptr[1]);,
917 ptemp.pos(2) = ParticleReal(rptr[2]););
918
919 rptr += AMREX_SPACEDIM;
920
921 for (int j = 0; j < NStructReal; j++)
922 {
923 ptemp.rdata(j) = ParticleReal(*rptr);
924 ++rptr;
925 }
926
927 locateParticle(ptemp, pld, 0, finestLevel(), 0);
928
929 std::pair<int, int> ind(grd, pld.m_tile);
930
931 host_real_attribs[lev][ind].resize(NumRealComps());
932 host_int_attribs[lev][ind].resize(NumIntComps());
933
934 // add the struct
935 if constexpr(!ParticleType::is_soa_particle)
936 {
937 host_particles[lev][ind].push_back(ptemp);
938
939 // add the real...
940 for (int icomp = 0; icomp < NumRealComps(); icomp++) {
941 host_real_attribs[lev][ind][icomp].push_back(*rptr);
942 ++rptr;
943 }
944
945 // ... and int array data
946 for (int icomp = 0; icomp < NumIntComps(); icomp++) {
947 host_int_attribs[lev][ind][icomp].push_back(*iptr);
948 ++iptr;
949 }
950 } else {
951 host_particles[lev][ind];
952
953 for (int j = 0; j < AMREX_SPACEDIM; j++) {
954 host_real_attribs[lev][ind][j].push_back(ptemp.pos(j));
956
957 host_idcpu[lev][ind].push_back(ptemp.m_idcpu);
958
959 // read all other SoA
960 // add the real...
961 for (int icomp = AMREX_SPACEDIM; icomp < NumRealComps(); icomp++) {
962 host_real_attribs[lev][ind][icomp].push_back(*rptr);
963 ++rptr;
964 }
965
966 // ... and int array data
967 for (int icomp = 0; icomp < NumIntComps(); icomp++) {
968 host_int_attribs[lev][ind][icomp].push_back(*iptr);
969 ++iptr;
970 }
971 }
972 }
974 for (int host_lev = 0; host_lev < std::ssize(host_particles); ++host_lev)
975 {
976 for (auto& kv : host_particles[host_lev]) {
977 auto grid = kv.first.first;
978 auto tile = kv.first.second;
979 const auto& src_tile = kv.second;
980
981 auto& dst_tile = DefineAndReturnParticleTile(host_lev, grid, tile);
982 auto old_size = dst_tile.size();
983 auto new_size = old_size;
984 if constexpr(!ParticleType::is_soa_particle)
985 {
986 new_size += src_tile.size();
987 } else {
988 amrex::ignore_unused(src_tile);
989 new_size += host_real_attribs[host_lev][std::make_pair(grid,tile)][0].size();
990 }
991 dst_tile.resize(new_size);
992
993 if constexpr(!ParticleType::is_soa_particle)
994 {
995 Gpu::copyAsync(Gpu::hostToDevice, src_tile.begin(), src_tile.end(),
996 dst_tile.GetArrayOfStructs().begin() + old_size);
997 } else {
999 host_idcpu[host_lev][std::make_pair(grid,tile)].begin(),
1000 host_idcpu[host_lev][std::make_pair(grid,tile)].end(),
1001 dst_tile.GetStructOfArrays().GetIdCPUData().begin() + old_size);
1002 }
1003
1004 for (int i = 0; i < NumRealComps(); ++i) { // NOLINT(readability-misleading-indentation)
1006 host_real_attribs[host_lev][std::make_pair(grid,tile)][i].begin(),
1007 host_real_attribs[host_lev][std::make_pair(grid,tile)][i].end(),
1008 dst_tile.GetStructOfArrays().GetRealData(i).begin() + old_size);
1010
1011 for (int i = 0; i < NumIntComps(); ++i) {
1013 host_int_attribs[host_lev][std::make_pair(grid,tile)][i].begin(),
1014 host_int_attribs[host_lev][std::make_pair(grid,tile)][i].end(),
1015 dst_tile.GetStructOfArrays().GetIntData(i).begin() + old_size);
1016 }
1017 }
1018 }
1019
1021}
1022
1023template <typename ParticleType, int NArrayReal, int NArrayInt,
1024 template<class> class Allocator, class CellAssignor>
1025void
1026ParticleContainer_impl<ParticleType, NArrayReal, NArrayInt, Allocator, CellAssignor>
1027::WriteAsciiFile (const std::string& filename)
1028{
1029 BL_PROFILE("ParticleContainer::WriteAsciiFile()");
1030 AMREX_ASSERT(!filename.empty());
1031
1032 const auto strttime = amrex::second();
1033 //
1034 // Count # of valid particles.
1035 //
1036 Long nparticles = 0;
1037
1038 for (int lev = 0; lev < std::ssize(m_particles); lev++) {
1039 auto& pmap = m_particles[lev];
1040 for (const auto& kv : pmap) {
1041 const auto& aos = kv.second.GetArrayOfStructs();
1042 auto np = aos.numParticles();
1043 Gpu::HostVector<ParticleType> host_aos(np);
1044 Gpu::copy(Gpu::deviceToHost, aos.begin(), aos.begin() + np, host_aos.begin());
1045 for (int k = 0; k < np; ++k) {
1046 const ParticleType& p = host_aos[k];
1047 if (p.id().is_valid()) {
1048 //
1049 // Only count (and checkpoint) valid particles.
1050 //
1051 nparticles++;
1052 }
1053 }
1054 }
1055 }
1056
1057 //
1058 // And send count to I/O processor.
1059 //
1061
1063 {
1064 //
1065 // Have I/O processor open file and write out particle metadata.
1066 //
1067 std::ofstream File;
1068
1069 File.open(filename.c_str(), std::ios::out|std::ios::trunc);
1070
1071 if (!File.good()) {
1072 amrex::FileOpenFailed(filename);
1073 }
1074
1075 File << nparticles << '\n';
1076 File << NStructReal << '\n';
1077 File << NStructInt << '\n';
1078 File << NumRealComps() << '\n';
1079 File << NumIntComps() << '\n';
1080
1081 File.flush();
1082
1083 File.close();
1084
1085 if (!File.good()) {
1086 amrex::Abort("ParticleContainer::WriteAsciiFile(): problem writing file");
1087 }
1088 }
1089
1091
1092 const int MyProc = ParallelDescriptor::MyProc();
1093
1094 for (int proc = 0; proc < ParallelDescriptor::NProcs(); proc++)
1095 {
1096 if (MyProc == proc)
1097 {
1098 //
1099 // Each CPU opens the file for appending and adds its particles.
1100 //
1102
1103 std::ofstream File;
1104
1105 File.rdbuf()->pubsetbuf(io_buffer.dataPtr(), io_buffer.size());
1106
1107 File.open(filename.c_str(), std::ios::out|std::ios::app);
1108
1109 File.precision(15);
1110
1111 if (!File.good()) {
1112 amrex::FileOpenFailed(filename);
1113 }
1114
1115 for (int lev = 0; lev < std::ssize(m_particles); lev++) {
1116 auto& pmap = m_particles[lev];
1117 for (const auto& kv : pmap) {
1118 ParticleTile<ParticleType, NArrayReal, NArrayInt,
1120 pinned_ptile.define(NumRuntimeRealComps(), NumRuntimeIntComps(),
1121 nullptr, nullptr, The_Pinned_Arena());
1122 pinned_ptile.resize(kv.second.numParticles());
1123 amrex::copyParticles(pinned_ptile, kv.second);
1124 const auto& host_aos = pinned_ptile.GetArrayOfStructs();
1125 const auto& host_soa = pinned_ptile.GetStructOfArrays();
1126
1127 auto np = host_aos.numParticles();
1128 for (int index = 0; index < np; ++index) {
1129 const ParticleType* it = &host_aos[index];
1130 if (it->id().is_valid()) {
1131
1132 // write out the particle struct first...
1133 AMREX_D_TERM(File << it->pos(0) << ' ',
1134 << it->pos(1) << ' ',
1135 << it->pos(2) << ' ');
1136
1137 for (int i = 0; i < NStructReal; i++) {
1138 File << it->rdata(i) << ' ';
1139 }
1140
1141 File << it->id() << ' ';
1142 File << it->cpu() << ' ';
1143
1144 for (int i = 0; i < NStructInt; i++) {
1145 File << it->idata(i) << ' ';
1146 }
1147
1148 // then the particle attributes.
1149 for (int i = 0; i < NumRealComps(); i++) {
1150 File << host_soa.GetRealData(i)[index] << ' ';
1151 }
1152
1153 for (int i = 0; i < NumIntComps(); i++) {
1154 File << host_soa.GetIntData(i)[index] << ' ';
1155 }
1156
1157 File << '\n';
1158 }
1159 }
1160 }
1161 }
1162
1163 File.flush();
1164
1165 File.close();
1166
1167 if (!File.good()) {
1168 amrex::Abort("ParticleContainer::WriteAsciiFile(): problem writing file");
1169 }
1170
1171 }
1172
1174 }
1175
1176 if (m_verbose > 1)
1177 {
1178 auto stoptime = amrex::second() - strttime;
1179
1181
1182 amrex::Print() << "ParticleContainer::WriteAsciiFile() time: " << stoptime << '\n';
1183 }
1184}
1185
1186}
1187
1188#endif /*AMREX_PARTICLEIO_H*/
#define BL_PROFILE(a)
Definition AMReX_BLProfiler.H:551
#define AMREX_ASSERT(EX)
Definition AMReX_BLassert.H:38
#define AMREX_ALWAYS_ASSERT(EX)
Definition AMReX_BLassert.H:50
#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
A collection of Boxes stored in an Array.
Definition AMReX_BoxArray.H:564
Calculates the distribution of FABs to MPI processes.
Definition AMReX_DistributionMapping.H:43
static const IntDescriptor & NativeIntDescriptor()
Returns a constant reference to an IntDescriptor describing the native "int" under which AMReX was co...
Definition AMReX_FPC.cpp:76
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
const std::string & FileName() const
Definition AMReX_NFiles.H:160
Dynamically allocated vector for trivially copyable data.
Definition AMReX_PODVector.H:308
iterator begin() noexcept
Definition AMReX_PODVector.H:674
Parse Parameters From Command Line and Input Files.
Definition AMReX_ParmParse.H:351
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:1947
A distributed container for Particles sorted onto the levels, grids, and tiles of a block-structured ...
Definition AMReX_ParticleContainer.H:149
T_ParticleType ParticleType
Definition AMReX_ParticleContainer.H:151
Definition AMReX_GpuAllocators.H:156
This class provides the user with a few print options.
Definition AMReX_Print.H:35
This class is a thin wrapper around std::vector. Unlike vector, Vector::operator[] provides bound che...
Definition AMReX_Vector.H:29
T * dataPtr() noexcept
get access to the underlying data pointer
Definition AMReX_Vector.H:50
Long size() const noexcept
Definition AMReX_Vector.H:54
static constexpr int IO_Buffer_Size
We try to do I/O with buffers of this size.
Definition AMReX_VisMFBuffer.H:16
static Long FileOffset(std::ostream &os)
The file offset of the passed ostream.
Definition AMReX_VisMF.cpp:580
static bool GetNoFlushAfterWrite()
Definition AMReX_VisMF.H:284
amrex_particle_real ParticleReal
Floating Point Type for Particles.
Definition AMReX_REAL.H:90
amrex_long Long
Definition AMReX_INT.H:30
void WritePlotFile(const std::string &dir, const std::string &name) const
This version of WritePlotFile writes all components and assigns component names.
Definition AMReX_ParticleIO.H:116
Arena * The_Pinned_Arena()
Definition AMReX_Arena.cpp:860
int MyProc() noexcept
Definition AMReX_ParallelDescriptor.H:128
void Barrier(const std::string &)
Definition AMReX_ParallelDescriptor.cpp:1215
void ReduceIntSum(int &)
Definition AMReX_ParallelDescriptor.cpp:1265
void ReadAndBcastFile(const std::string &filename, Vector< char > &charBuf, bool bExitOnError, const MPI_Comm &comm)
Definition AMReX_ParallelDescriptor.cpp:1495
int NProcs() noexcept
Definition AMReX_ParallelDescriptor.H:255
void ReduceLongSum(Long &)
Definition AMReX_ParallelDescriptor.cpp:1236
int IOProcessorNumber() noexcept
The MPI rank number of the I/O Processor (probably rank 0). This rank is usually used to write to std...
Definition AMReX_ParallelDescriptor.H:279
void ReduceLongMax(Long &)
Definition AMReX_ParallelDescriptor.cpp:1237
bool IOProcessor() noexcept
Is this CPU the I/O Processor? To get the rank number, call IOProcessorNumber()
Definition AMReX_ParallelDescriptor.H:289
bool UseAsyncOut()
Definition AMReX_AsyncOut.cpp:70
bool Remove(std::string const &filename)
Remove a file, symbolic link, or empty directory.
Definition AMReX_FileSystem.cpp:41
void copy(HostToDevice, InIter begin, InIter end, OutIter result) noexcept
A host-to-device copy routine. Note this is just a wrapper around memcpy, so it assumes contiguous st...
Definition AMReX_GpuContainers.H:128
void copyAsync(HostToDevice, InIter begin, InIter end, OutIter result) noexcept
A host-to-device copy routine. Note this is just a wrapper around memcpy, so it assumes contiguous st...
Definition AMReX_GpuContainers.H:228
static constexpr DeviceToHost deviceToHost
Definition AMReX_GpuContainers.H:106
static constexpr HostToDevice hostToDevice
Definition AMReX_GpuContainers.H:105
void streamSynchronize() noexcept
Definition AMReX_GpuDevice.H:310
std::string const & name()
Definition AMReX_Machine.cpp:46
void ReduceRealMax(Vector< std::reference_wrapper< Real > > const &)
Definition AMReX_ParallelDescriptor.cpp:1228
Definition AMReX_Amr.cpp:50
void writeIntData(const From *data, std::size_t size, std::ostream &os, const amrex::IntDescriptor &id)
Definition AMReX_IntConv.H:23
__host__ __device__ void ignore_unused(const Ts &...)
This shuts up the compiler about unused variables.
Definition AMReX.H:139
void FileOpenFailed(const std::string &file)
Output a message and abort when couldn't open the file.
Definition AMReX_Utility.cpp:137
void WriteBinaryParticleDataAsync(PC const &pc, const std::string &dir, const std::string &name, const Vector< int > &write_real_comp, const Vector< int > &write_int_comp, const Vector< std::string > &real_comp_names, const Vector< std::string > &int_comp_names, bool is_checkpoint)
Definition AMReX_WriteBinaryParticleData.H:786
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
__host__ __device__ Dim3 begin(BoxND< dim > const &box) noexcept
Definition AMReX_Box.H:2018
void readFloatData(float *data, std::size_t size, std::istream &is, const RealDescriptor &rd)
Definition AMReX_VectorIO.cpp:136
void writeFloatData(const float *data, std::size_t size, std::ostream &os, const RealDescriptor &rd=FPC::Native32RealDescriptor())
Definition AMReX_VectorIO.cpp:130
bool FileExists(const std::string &filename)
Check if a file already exists. Return true if the filename is an existing file, directory,...
Definition AMReX_Utility.cpp:145
void WriteBinaryParticleDataSync(PC const &pc, const std::string &dir, const std::string &name, const Vector< int > &write_real_comp, const Vector< int > &write_int_comp, const Vector< std::string > &real_comp_names, const Vector< std::string > &int_comp_names, F const &f, bool is_checkpoint)
Definition AMReX_WriteBinaryParticleData.H:486
std::string Concatenate(const std::string &root, int num, int mindigits)
Returns rootNNNN where NNNN == num.
Definition AMReX_String.cpp:35
double second() noexcept
Definition AMReX_Utility.cpp:940
void writeDoubleData(const double *data, std::size_t size, std::ostream &os, const RealDescriptor &rd=FPC::Native64RealDescriptor())
Definition AMReX_VectorIO.cpp:142
void Error(const std::string &msg)
Print out message to cerr and exit via amrex::Abort().
Definition AMReX.cpp:235
void readDoubleData(double *data, std::size_t size, std::istream &is, const RealDescriptor &rd)
Definition AMReX_VectorIO.cpp:148
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
__host__ __device__ Dim3 end(BoxND< dim > const &box) noexcept
Definition AMReX_Box.H:2028
void readIntData(To *data, std::size_t size, std::istream &is, const amrex::IntDescriptor &id)
Definition AMReX_IntConv.H:38
Definition AMReX_ParticleIO.H:41
__host__ __device__ int operator()(const P &p) const
Definition AMReX_ParticleIO.H:44
FabArray memory allocation information.
Definition AMReX_FabArray.H:68
MFInfo & SetAlloc(bool a) noexcept
Definition AMReX_FabArray.H:75
uint64_t m_idcpu
Definition AMReX_Particle.H:359
Metadata prefix of an AMReX particle "Header" file.
Definition AMReX_ParticleHeader.H:33
int num_int
number of integer components
Definition AMReX_ParticleHeader.H:47
Long next_id
the next particle id to hand out (maxnextid)
Definition AMReX_ParticleHeader.H:55
int dim
AMREX_SPACEDIM the file was written with.
Definition AMReX_ParticleHeader.H:41
bool convert_ids
whether particle ids need to be converted (Version_Two_Dot_One and later)
Definition AMReX_ParticleHeader.H:39
void parse(std::istream &is)
Parse the metadata prefix from an input stream.
Definition AMReX_ParticleHeader.cpp:16
int finest_level
finest level present in the file
Definition AMReX_ParticleHeader.H:57
std::string how
precision the particle data was written in: "single" or "double"
Definition AMReX_ParticleHeader.H:37
int num_real
number of real components (pure SoA: excludes the AMREX_SPACEDIM positions)
Definition AMReX_ParticleHeader.H:43
Long num_particles
total number of particles in the file
Definition AMReX_ParticleHeader.H:53
__host__ __device__ bool is_valid() const noexcept
Definition AMReX_Particle.H:252
A struct used for storing a particle's position in the AMR hierarchy.
Definition AMReX_ParticleContainer.H:93
int m_tile
Definition AMReX_ParticleContainer.H:96
Definition AMReX_ParticleTile.H:764
The struct used to store particles.
Definition AMReX_Particle.H:405
__host__ __device__ RealVect pos() const &
Definition AMReX_Particle.H:456
__host__ __device__ int & idata(int index) &
Definition AMReX_Particle.H:538
__host__ __device__ ParticleCPUWrapper cpu() &
Definition AMReX_Particle.H:424
__host__ __device__ RealType & rdata(int index) &
Definition AMReX_Particle.H:473
__host__ __device__ ParticleIDWrapper id() &
Definition AMReX_Particle.H:427