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>
65 std::size_t n_particle,
66 std::size_t _n_samples);
91 KOKKOS_INLINE_FUNCTION
auto sample(std::size_t idx,
92 std::size_t i_sample)
const;
101 template <
typename CviewType>
102 KOKKOS_INLINE_FUNCTION
void
104 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);
136 [[nodiscard]] KOKKOS_INLINE_FUNCTION
bool
142 [[nodiscard]] KOKKOS_INLINE_FUNCTION
double
161 template <
class Archive>
void save(Archive& ar)
const;
166 template <
class Archive>
void load(Archive& ar);
178 [[nodiscard]] std::
size_t capacity() const noexcept;
198 [[nodiscard]] KOKKOS_INLINE_FUNCTION std::
size_t n_particles() const;
237 void _resize(std::size_t new_size,
bool force =
false);
255 using TeamPolicy = Kokkos::TeamPolicy<ComputeSpace>;
256 using TeamMember = TeamPolicy::member_type;
296 template <ModelType M>
struct CompactParticlesFunctor
300 M::SelfParticle _model,
303 std::size_t _to_remove,
304 std::size_t _last_used_index)
305 : status(std::move(_status)), model(std::move(_model)),
306 position(std::move(_position)), ages(std::move(_ages)),
307 offset(
"offset"), to_remove(_to_remove),
308 last_used_index(_last_used_index)
311 Kokkos::deep_copy(offset, 0);
314 KOKKOS_INLINE_FUNCTION
void
315 operator()(
const int i, std::size_t& update,
const bool final)
const
329 const std::size_t scan_index = update;
332 update += is_inactive ? 1 : 0;
335 if (
final && is_inactive && scan_index < to_remove)
338 const auto inactive_slot = i;
344 auto replacement_index
345 = last_used_index - Kokkos::atomic_fetch_add(&offset(), 1);
348 ==
static_cast<std::size_t
>(inactive_slot))
351 = last_used_index - Kokkos::atomic_fetch_add(&offset(), 1);
359 Kokkos::atomic_exchange(&position(inactive_slot),
360 position(replacement_index));
363 for (std::size_t i_properties = 0; i_properties < M::n_var;
366 model(inactive_slot, i_properties)
367 = model(replacement_index, i_properties);
369 ages(inactive_slot, 0) = ages(replacement_index, 0);
370 ages(inactive_slot, 1) = ages(replacement_index, 1);
375 M::SelfParticle model;
378 Kokkos::View<std::size_t, ComputeSpace> offset;
379 std::size_t to_remove;
380 std::size_t last_used_index;
399 template <ModelType M>
struct InsertFunctor
401 InsertFunctor(std::size_t _original_size,
402 M::SelfParticle _model,
405 M::SelfParticle _buffer_model,
407 : original_size(_original_size), model(std::move(_model)),
408 ages(std::move(_ages)), position(std::move(_position)),
409 buffer_model(std::move(_buffer_model)),
410 buffer_position(std::move(_buffer_position))
413 KOKKOS_INLINE_FUNCTION
415 operator()(
const TeamMember& team)
const
417 auto range = M::n_var;
418 const int i = team.league_rank();
420 Kokkos::parallel_for(
421 Kokkos::TeamVectorRange(team, range),
423 { model(original_size + i, j) = buffer_model(i, j); });
424 position(original_size + i) = buffer_position(i);
429 ages(original_size + i, 0) = 0;
430 ages(original_size + i, 1) = 0;
433 std::size_t original_size;
434 M::SelfParticle model;
437 M::SelfParticle buffer_model;
443 template <ModelType Model>
444 [[nodiscard]] KOKKOS_INLINE_FUNCTION std::size_t
450 template <ModelType Model>
451 [[nodiscard]] std::size_t
457 template <ModelType Model>
465 template <ModelType Model>
466 [[maybe_unused]] [[nodiscard]]
auto
473 template <ModelType Model>
480 template <ModelType Model>
481 [[nodiscard]] std::size_t
487 template <ModelType Model>
488 KOKKOS_INLINE_FUNCTION
auto
493 template <ModelType Model>
494 KOKKOS_INLINE_FUNCTION
auto
496 const std::size_t i_sample)
const
498 return random(idx, i_sample);
501 template <ModelType Model>
502 template <
class Archive>
512 deserialize_view(ar,
status);
513 deserialize_view(ar,
model);
514 deserialize_view(ar,
ages);
516 Kokkos::printf(
"ParticlesContainer::load: Check if load_tuning_constant "
517 "works with different value");
523 template <ModelType Model>
524 template <
class Archive>
533 serialize_view(ar,
status);
534 serialize_view(ar,
model);
535 serialize_view(ar,
ages);
538 template <ModelType Model>
549 template <ModelType Model>
552 const std::size_t dead)
558 const auto _threshold = std::max(
561 *
rt_params.dead_particle_ratio_threshold));
569 template <ModelType Model>
570 KOKKOS_INLINE_FUNCTION
bool
572 std::size_t idx1)
const
576 const auto idx2 = Kokkos::atomic_fetch_add(&
buffer_index(), 1);
585 template <ModelType Model>
589 PROFILE_SECTION(
"ParticlesContainer::merge_buffer")
596 _resize(original_size + n_add_item);
597 Kokkos::parallel_for(
"insert_merge",
598 TeamPolicy(n_add_item, Kokkos::AUTO, Model::n_var),
599 InsertFunctor<Model>(original_size,
611 template <ModelType Model>
615 PROFILE_SECTION(
"ParticlesContainer::_resize")
624 const auto new_allocated_size =
static_cast<std::size_t
>(std::ceil(
625 static_cast<double>(new_size) *
rt_params.allocation_factor));
632 Kokkos::resize(
model,
681 template <ModelType Model>
685 PROFILE_SECTION(
"ParticlesContainer::__allocate_buffer__")
687 const auto required_buffer_size =
static_cast<std::size_t
>(
694 Kokkos::realloc(
buffer_model, required_buffer_size, Model::n_var);
700#define alloc_without_init(name) \
701 Kokkos::view_alloc(Kokkos::WithoutInitializing, name)
704 template <ModelType M>
706 std::size_t n_particle,
707 std::size_t _n_samples)
708 :
model(alloc_without_init(
"particle_model"), 0, 0),
709 position(alloc_without_init(
"particle_position"), 0),
710 status(alloc_without_init(
"particle_status"), 0),
711 weights(alloc_without_init(
"particle_weigth"), 0),
712 ages(alloc_without_init(
"particle_age"), 0),
713 random(alloc_without_init(
"particle_random"), 0, 0),
729 const auto bounds = M::get_bounds();
730 begin = bounds.begin;
734 template <ModelType M>
740 template <ModelType M>
745 PROFILE_SECTION(
"ParticlesContainer::remove_inactive_particles")
759 throw std::runtime_error(
760 "remove_inactive_particles: Error in kernel cannot remove more "
761 "element than existing");
769 Kokkos::parallel_scan(
772 CompactParticlesFunctor<M>(
788 const bool do_shrink = n_used_elements <= static_cast<std::size_t>(
804 template <ModelType M>
805 [[nodiscard]] KOKKOS_INLINE_FUNCTION
double
818 template <ModelType M>
825 template <ModelType M>
RuntimeParameters rt_params
Definition particles_container.hpp:239
void update_and_remove_inactive(std::size_t out, std::size_t dead)
Definition particles_container.hpp:551
double get_allocation_factor() const noexcept
Get the allocation factor.
Definition particles_container.hpp:475
~ParticlesContainer()=default
Default destructor.
void force_remove_dead()
Clean all the non-idle particle even if number smaller than threshold.
Definition particles_container.hpp:459
Model::SelfParticle model
Definition particles_container.hpp:82
KOKKOS_INLINE_FUNCTION std::size_t n_particles() const
Definition particles_container.hpp:445
ParticleWeigths weights
Definition particles_container.hpp:85
int end
Definition particles_container.hpp:242
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:571
ParticleSamples random
Definition particles_container.hpp:87
Kokkos::View< uint64_t, Kokkos::SharedSpace > buffer_index
Definition particles_container.hpp:231
void _resize(std::size_t new_size, bool force=false)
Definition particles_container.hpp:613
void change_nsample(std::size_t new_n_sample)
Definition particles_container.hpp:540
void remove_inactive_particles(std::size_t to_remove)
Definition particles_container.hpp:742
void __allocate_buffer__()
Definition particles_container.hpp:683
ParticlesContainer()
Definition particles_container.hpp:735
ParticlesContainer & operator=(ParticlesContainer &&)=default
std::size_t get_inactive() const noexcept
Definition particles_container.hpp:452
std::size_t capacity() const noexcept
Definition particles_container.hpp:482
uint64_t n_used_elements
Definition particles_container.hpp:233
auto get_buffer_index() const
Definition particles_container.hpp:467
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:103
void merge_buffer()
Insert particle buffer into the main container.
Definition particles_container.hpp:587
Model::SelfParticle buffer_model
Definition particles_container.hpp:229
KOKKOS_INLINE_FUNCTION double get_weight(std::size_t idx) const
Return the particle weight.
Definition particles_container.hpp:806
std::size_t n_samples
Definition particles_container.hpp:238
KOKKOS_INLINE_FUNCTION auto sample(std::size_t idx, std::size_t i_sample) const
get the sample at idx/i_sample position
Definition particles_container.hpp:495
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:705
ParticlePositions buffer_position
Definition particles_container.hpp:230
MC::ParticlePositions position
Definition particles_container.hpp:83
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:234
ParticleAges ages
Definition particles_container.hpp:86
Model UsedModel
Definition particles_container.hpp:58
ParticlesContainer & operator=(const ParticlesContainer &)=default
void _sort(size_t n_c)
Definition particles_container.hpp:827
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:504
ParticlesContainer(ParticlesContainer &&)=default
std::size_t n_allocated_elements
Definition particles_container.hpp:232
void change_runtime(RuntimeParameters &¶meters) noexcept
Definition particles_container.hpp:820
int begin
Definition particles_container.hpp:241
KOKKOS_INLINE_FUNCTION auto samples()
get the sample view
Definition particles_container.hpp:489
MC::ParticleStatus status
Definition particles_container.hpp:84
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:15
Kokkos::View< double *, ComputeSpace > ParticleWeigths
Definition alias.hpp:75
Kokkos::View< Status *, ComputeSpace > ParticleStatus
Definition alias.hpp:74
Kokkos::View< uint64_t *, ComputeSpace > ParticlePositions
Definition alias.hpp:73
@ Idle
Definition alias.hpp:59
ParticleAgesBase< ComputeSpace > ParticleAges
Definition alias.hpp:76
gen_pool_type< Kokkos::DefaultExecutionSpace > pool_type
Definition alias.hpp:39
Kokkos:: View< Kokkos::Experimental::half_t **, Kokkos::LayoutRight, ComputeSpace > ParticleSamples
Definition alias.hpp:78
Definition particles_container.hpp:25
void serialize(Archive &ar)
Definition particles_container.hpp:34
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 shrink_ratio
Definition particles_container.hpp:29
double allocation_factor
Definition particles_container.hpp:28