1#ifndef __PARTICLES_CONTAINER_HPP__
2#define __PARTICLES_CONTAINER_HPP__
4#include "Kokkos_Macros.hpp"
5#include <Kokkos_Core.hpp>
6#include <biocma_cst_config.hpp>
8#include <common/common.hpp>
9#include <common/env_var.hpp>
10#include <common/has_serialize.hpp>
12#include <mc/alias.hpp>
13#include <mc/prng/prng.hpp>
14#include <mc/traits.hpp>
16#include <Kokkos_Sort.hpp>
17#include <sorting/Kokkos_BinSortPublicAPI.hpp>
18#include <sorting/impl/Kokkos_CopyOpsForBinSortImpl.hpp>
19#include <sorting/impl/Kokkos_SortByKeyImpl.hpp>
32 template <
class Archive>
void serialize(Archive& ar)
62 std::size_t n_particle,
63 std::size_t _n_samples);
87 KOKKOS_INLINE_FUNCTION
auto sample(
const std::size_t idx,
88 const std::size_t i_sample)
const
90 return random(idx, i_sample);
102 template <
typename CviewType>
103 KOKKOS_INLINE_FUNCTION
void
105 const CviewType& contributions)
const
109 "ModelType: Constapply_weight()");
113 auto access = contributions.access();
117 const int rel = i -
begin;
118 access(rel, pos) += weight *
model(idx, i);
135 [[nodiscard]] KOKKOS_INLINE_FUNCTION
bool
141 [[nodiscard]] KOKKOS_INLINE_FUNCTION
double
160 template <
class Archive>
void save(Archive& ar)
const;
165 template <
class Archive>
void load(Archive& ar);
185 [[nodiscard]] std::size_t
capacity() const noexcept;
205 [[nodiscard]] KOKKOS_INLINE_FUNCTION std::
size_t n_particles() const;
244 void _resize(std::size_t new_size,
bool force =
false);
262 using TeamPolicy = Kokkos::TeamPolicy<ComputeSpace>;
263 using TeamMember = TeamPolicy::member_type;
303 template <ModelType M>
struct CompactParticlesFunctor
307 M::SelfParticle _model,
310 std::size_t _to_remove,
311 std::size_t _last_used_index)
312 : status(std::move(_status)), model(std::move(_model)),
313 position(std::move(_position)), ages(std::move(_ages)),
314 offset(
"offset"), to_remove(_to_remove),
315 last_used_index(_last_used_index)
318 Kokkos::deep_copy(offset, 0);
321 KOKKOS_INLINE_FUNCTION
void
322 operator()(
const int i, std::size_t& update,
const bool final)
const
336 const std::size_t scan_index = update;
339 update += is_inactive ? 1 : 0;
342 if (
final && is_inactive && scan_index < to_remove)
345 const auto inactive_slot = i;
351 auto replacement_index =
352 last_used_index - Kokkos::atomic_fetch_add(&offset(), 1);
354 replacement_index ==
static_cast<std::size_t
>(inactive_slot))
357 last_used_index - Kokkos::atomic_fetch_add(&offset(), 1);
365 Kokkos::atomic_exchange(&position(inactive_slot),
366 position(replacement_index));
369 for (std::size_t i_properties = 0; i_properties < M::n_var;
372 model(inactive_slot, i_properties) =
373 model(replacement_index, i_properties);
375 ages(inactive_slot, 0) = ages(replacement_index, 0);
376 ages(inactive_slot, 1) = ages(replacement_index, 1);
381 M::SelfParticle model;
384 Kokkos::View<std::size_t, ComputeSpace> offset;
385 std::size_t to_remove;
386 std::size_t last_used_index;
405 template <ModelType M>
struct InsertFunctor
407 InsertFunctor(std::size_t _original_size,
408 M::SelfParticle _model,
411 M::SelfParticle _buffer_model,
413 : original_size(_original_size), model(std::move(_model)),
414 ages(std::move(_ages)), position(std::move(_position)),
415 buffer_model(std::move(_buffer_model)),
416 buffer_position(std::move(_buffer_position))
433 KOKKOS_INLINE_FUNCTION
434 void operator()(
const TeamMember& team)
const
436 auto range = M::n_var;
437 const int i = team.league_rank();
439 Kokkos::parallel_for(
440 Kokkos::TeamVectorRange(team, range),
442 { model(original_size + i, j) = buffer_model(i, j); });
443 position(original_size + i) = buffer_position(i);
448 ages(original_size + i, 0) = 0;
449 ages(original_size + i, 1) = 0;
452 std::size_t original_size;
453 M::SelfParticle model;
456 M::SelfParticle buffer_model;
462 template <ModelType Model>
463 [[nodiscard]] KOKKOS_INLINE_FUNCTION std::size_t
469 template <ModelType Model>
470 [[nodiscard]] std::size_t
482 template <ModelType Model>
483 [[maybe_unused]] [[nodiscard]]
auto
490 template <ModelType Model>
497 template <ModelType Model>
503 template <ModelType Model>
504 template <
class Archive>
513 deserialize_view(ar,
status);
514 deserialize_view(ar,
model);
515 deserialize_view(ar,
ages);
517 Kokkos::printf(
"ParticlesContainer::load: Check if load_tuning_constant "
518 "works with different value");
524 template <ModelType Model>
525 template <
class Archive>
533 serialize_view(ar,
status);
534 serialize_view(ar,
model);
535 serialize_view(ar,
ages);
538 template <ModelType Model>
541 const std::size_t dead)
547 const auto _threshold = std::max(
550 rt_params.dead_particle_ratio_threshold));
559 template <ModelType Model>
560 KOKKOS_INLINE_FUNCTION
bool
562 std::size_t idx1)
const
566 const auto idx2 = Kokkos::atomic_fetch_add(&
buffer_index(), 1);
577 PROFILE_SECTION(
"ParticlesContainer::merge_buffer")
584 _resize(original_size + n_add_item);
585 Kokkos::parallel_for(
"insert_merge",
586 TeamPolicy(n_add_item, Kokkos::AUTO, Model::n_var),
587 InsertFunctor<Model>(original_size,
599 template <ModelType Model>
602 PROFILE_SECTION(
"ParticlesContainer::_resize")
611 const auto new_allocated_size =
static_cast<std::size_t
>(std::ceil(
612 static_cast<double>(new_size) *
rt_params.allocation_factor));
619 Kokkos::resize(
model,
643 template <ModelType Model>
646 PROFILE_SECTION(
"ParticlesContainer::__allocate_buffer__")
648 const auto buffer_ratio =
rt_params.buffer_ratio;
649 if (
static_cast<double>(buffer_size) /
653 buffer_size =
static_cast<std::size_t
>(
665 template <ModelType M>
667 std::size_t n_particle,
668 std::size_t _n_samples)
669 :
model(Kokkos::view_alloc(Kokkos::WithoutInitializing,
"particle_model"),
672 position(Kokkos::view_alloc(Kokkos::WithoutInitializing,
673 "particle_position"),
676 Kokkos::view_alloc(Kokkos::WithoutInitializing,
"particle_status"),
679 Kokkos::view_alloc(Kokkos::WithoutInitializing,
"particle_weigth"),
681 ages(Kokkos::view_alloc(Kokkos::WithoutInitializing,
"particle_age"),
684 Kokkos::view_alloc(Kokkos::WithoutInitializing,
"particle_random"),
702 auto bounds = M::get_bounds();
703 begin = bounds.begin;
707 template <ModelType M>
713 template <ModelType M>
717 PROFILE_SECTION(
"ParticlesContainer::remove_inactive_particles")
731 throw std::runtime_error(
732 "remove_inactive_particles: Error in kernel cannot remove more "
733 "element than existing");
741 Kokkos::parallel_scan(
744 CompactParticlesFunctor<M>(
768 template <ModelType M>
769 [[nodiscard]] KOKKOS_INLINE_FUNCTION
double
782 template <ModelType M>
RuntimeParameters rt_params
Definition particles_container.hpp:246
void update_and_remove_inactive(std::size_t out, std::size_t dead)
Definition particles_container.hpp:540
void change_nsample(const std::size_t new_n_sample)
Definition particles_container.hpp:172
double get_allocation_factor() const noexcept
Get the allocation factor.
Definition particles_container.hpp:492
~ParticlesContainer()=default
Default destructor.
void force_remove_dead()
Clean all the non-idle particle even if number smaller than threshold.
Definition particles_container.hpp:476
KOKKOS_INLINE_FUNCTION auto sample(const std::size_t idx, const std::size_t i_sample) const
Definition particles_container.hpp:87
Model::SelfParticle model
Definition particles_container.hpp:79
KOKKOS_INLINE_FUNCTION std::size_t n_particles() const
Definition particles_container.hpp:464
ParticleWeigths weights
Definition particles_container.hpp:82
int end
Definition particles_container.hpp:249
KOKKOS_INLINE_FUNCTION bool handle_division(const MC::pool_type &random_pool, std::size_t idx1) const
Attempts to spawn a new particle by performing division on the specified particle.
Definition particles_container.hpp:561
ParticleSamples random
Definition particles_container.hpp:84
Kokkos::View< uint64_t, Kokkos::SharedSpace > buffer_index
Definition particles_container.hpp:238
void _resize(std::size_t new_size, bool force=false)
Definition particles_container.hpp:600
void remove_inactive_particles(std::size_t to_remove)
Definition particles_container.hpp:714
void __allocate_buffer__()
Definition particles_container.hpp:644
ParticlesContainer()
Definition particles_container.hpp:708
ParticlesContainer & operator=(ParticlesContainer &&)=default
std::size_t get_inactive() const noexcept
Definition particles_container.hpp:471
std::size_t capacity() const noexcept
Returns the total number of elements the container can hold without reallocating.
Definition particles_container.hpp:498
uint64_t n_used_elements
Definition particles_container.hpp:240
auto get_buffer_index() const
Definition particles_container.hpp:484
KOKKOS_INLINE_FUNCTION void get_contributions(const std::size_t idx, const CviewType &contributions) const
Get the contribution if particle at index idx.
Definition particles_container.hpp:104
void merge_buffer()
Insert particle buffer into the main container.
Definition particles_container.hpp:575
Model::SelfParticle buffer_model
Definition particles_container.hpp:236
KOKKOS_INLINE_FUNCTION double get_weight(std::size_t idx) const
Return the particle weight.
Definition particles_container.hpp:770
std::size_t n_samples
Definition particles_container.hpp:245
ParticlesContainer(RuntimeParameters rt_param, std::size_t n_particle, std::size_t _n_samples)
Alias for the model used by the container.
Definition particles_container.hpp:666
ParticlePositions buffer_position
Definition particles_container.hpp:237
MC::ParticlePositions position
Definition particles_container.hpp:80
void save(Archive &ar) const
Save data into ar for serialization.
Definition particles_container.hpp:526
std::size_t inactive_counter
Definition particles_container.hpp:241
ParticleAges ages
Definition particles_container.hpp:83
Model UsedModel
Definition particles_container.hpp:55
ParticlesContainer & operator=(const ParticlesContainer &)=default
void _sort(size_t n_c)
Definition particles_container.hpp:789
ParticlesContainer(const ParticlesContainer &)=default
Default copy and move constructors and assignment operators.
void load(Archive &ar)
Load data from ar for deserialization.
Definition particles_container.hpp:505
ParticlesContainer(ParticlesContainer &&)=default
std::size_t n_allocated_elements
Definition particles_container.hpp:239
void change_runtime(RuntimeParameters &¶meters) noexcept
Definition particles_container.hpp:784
int begin
Definition particles_container.hpp:248
KOKKOS_INLINE_FUNCTION auto samples()
Definition particles_container.hpp:93
MC::ParticleStatus status
Definition particles_container.hpp:81
Concept to check if a model type has uniform_weight
Definition traits.hpp:183
Namespace that contains classes and structures related to Monte Carlo (MC) simulations.
Definition alias.hpp:14
Kokkos::View< double *, ComputeSpace > ParticleWeigths
Definition alias.hpp:72
Kokkos::View< Status *, ComputeSpace > ParticleStatus
Definition alias.hpp:71
Kokkos::View< uint64_t *, ComputeSpace > ParticlePositions
Definition alias.hpp:70
@ Idle
Definition alias.hpp:58
ParticleAgesBase< ComputeSpace > ParticleAges
Definition alias.hpp:73
gen_pool_type< Kokkos::DefaultExecutionSpace > pool_type
Definition alias.hpp:38
Kokkos:: View< Kokkos::Experimental::half_t **, Kokkos::LayoutRight, ComputeSpace > ParticleSamples
Definition alias.hpp:75
Definition particles_container.hpp:25
double shink_ratio
Definition particles_container.hpp:29
void serialize(Archive &ar)
Definition particles_container.hpp:32
double buffer_ratio
Definition particles_container.hpp:27
double dead_particle_ratio_threshold
Definition particles_container.hpp:30
uint64_t minimum_dead_particle_removal
Definition particles_container.hpp:26
double allocation_factor
Definition particles_container.hpp:28