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>
33 template <
class Archive>
65 std::size_t n_particle,
66 std::size_t _n_samples);
130 [[nodiscard]] KOKKOS_INLINE_FUNCTION
bool
136 [[nodiscard]] KOKKOS_INLINE_FUNCTION Model::FloatType
155 template <
class Archive>
void save(Archive& ar)
const;
160 template <
class Archive>
void load(Archive& ar);
171 [[nodiscard]] std::
size_t capacity() const noexcept;
191 [[nodiscard]] KOKKOS_INLINE_FUNCTION std::
size_t n_particles() const;
230 void _resize(std::size_t new_size,
bool force =
false);
251 using TeamPolicy = Kokkos::TeamPolicy<ComputeSpace>;
252 using TeamMember = TeamPolicy::member_type;
292 template <ModelType M>
struct CompactParticlesFunctor
296 M::SelfParticle _model,
297 M::SelfContribs _contribs,
300 std::size_t _to_remove,
301 std::size_t _last_used_index)
302 : status(std::move(_status)), model(std::move(_model)),
303 contribs(std::move(_contribs)), position(std::move(_position)),
304 ages(std::move(_ages)), offset(
"offset"), to_remove(_to_remove),
305 last_used_index(_last_used_index)
308 Kokkos::deep_copy(offset, 0);
311 KOKKOS_INLINE_FUNCTION
void
312 operator()(
const int i, std::size_t& update,
const bool final)
const
326 const std::size_t scan_index = update;
329 update += is_inactive ? 1 : 0;
332 if (
final && is_inactive && scan_index < to_remove)
335 const auto inactive_slot = i;
341 auto replacement_index
342 = last_used_index - Kokkos::atomic_fetch_add(&offset(), 1);
345 ==
static_cast<std::size_t
>(inactive_slot))
348 = last_used_index - Kokkos::atomic_fetch_add(&offset(), 1);
356 Kokkos::atomic_exchange(&position(inactive_slot),
357 position(replacement_index));
360 for (std::size_t i_properties = 0; i_properties < M::n_var;
363 model(inactive_slot, i_properties)
364 = model(replacement_index, i_properties);
367 for (std::size_t i_c = 0; i_c < M::n_c; ++i_c)
369 contribs(inactive_slot, i_c) = contribs(replacement_index, i_c);
372 ages(inactive_slot, 0) = ages(replacement_index, 0);
373 ages(inactive_slot, 1) = ages(replacement_index, 1);
378 M::SelfParticle model;
379 M::SelfContribs contribs;
382 Kokkos::View<std::size_t, ComputeSpace> offset;
383 std::size_t to_remove;
384 std::size_t last_used_index;
403 template <ModelType M>
struct InsertFunctor
405 InsertFunctor(std::size_t _original_size,
406 M::SelfParticle _model,
409 M::SelfParticle _buffer_model,
411 : original_size(_original_size), model(std::move(_model)),
412 ages(std::move(_ages)), position(std::move(_position)),
413 buffer_model(std::move(_buffer_model)),
414 buffer_position(std::move(_buffer_position))
417 KOKKOS_INLINE_FUNCTION
419 operator()(
const TeamMember& team)
const
421 auto range = M::n_var;
422 const int i = team.league_rank();
424 Kokkos::parallel_for(
425 Kokkos::TeamVectorRange(team, range),
427 { model(original_size + i, j) = buffer_model(i, j); });
428 position(original_size + i) = buffer_position(i);
433 ages(original_size + i, 0) = 0;
434 ages(original_size + i, 1) = 0;
437 std::size_t original_size;
438 M::SelfParticle model;
441 M::SelfParticle buffer_model;
447 template <ModelType Model>
448 [[nodiscard]] KOKKOS_INLINE_FUNCTION std::size_t
454 template <ModelType Model>
455 [[nodiscard]] std::size_t
461 template <ModelType Model>
469 template <ModelType Model>
470 [[maybe_unused]] [[nodiscard]]
auto
477 template <ModelType Model>
484 template <ModelType Model>
485 [[nodiscard]] std::size_t
491 template <ModelType Model>
492 template <
class Archive>
502 deserialize_view(ar,
status);
503 deserialize_view(ar,
model);
504 deserialize_view(ar,
ages);
506 if (Model::n_var !=
model.extent(1))
508 throw std::runtime_error(
509 "Error when deserialze, model number of property mismatch");
517 Kokkos::printf(
"ParticlesContainer::load: Check if load_tuning_constant "
518 "works with different value");
524 template <ModelType Model>
525 template <
class Archive>
534 serialize_view(ar,
status);
535 serialize_view(ar,
model);
536 serialize_view(ar,
ages);
539 template <ModelType Model>
542 const std::size_t dead)
548 const auto _threshold = std::max(
551 *
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);
575 template <ModelType Model>
579 PROFILE_SECTION(
"ParticlesContainer::merge_buffer")
586 _resize(original_size + n_add_item);
587 Kokkos::parallel_for(
"insert_merge",
588 TeamPolicy(n_add_item, Kokkos::AUTO, Model::n_var),
589 InsertFunctor<Model>(original_size,
601 template <ModelType Model>
605 PROFILE_SECTION(
"ParticlesContainer::_resize")
614 const auto new_allocated_size =
static_cast<std::size_t
>(std::ceil(
615 static_cast<double>(new_size) *
rt_params.allocation_factor));
622 Kokkos::resize(
model,
669 template <ModelType Model>
673 PROFILE_SECTION(
"ParticlesContainer::__allocate_buffer__")
675 const auto required_buffer_size =
static_cast<std::size_t
>(
682 Kokkos::realloc(
buffer_model, required_buffer_size, Model::n_var);
688#define alloc_without_init(name) \
689 Kokkos::view_alloc(Kokkos::WithoutInitializing, name)
692 template <ModelType M>
695 std::size_t n_particle,
696 [[maybe_unused]] std::size_t _n_samples)
697 :
model(alloc_without_init(
"particle_model"), 0, 0),
699 contribs(alloc_without_init(
"particle_contribs"), 0),
700 position(alloc_without_init(
"particle_position"), 0),
701 status(alloc_without_init(
"particle_status"), 0),
702 weights(alloc_without_init(
"particle_weigth"), 0),
703 ages(alloc_without_init(
"particle_age"), 0),
729 template <ModelType M>
735 template <ModelType M>
740 PROFILE_SECTION(
"ParticlesContainer::remove_inactive_particles")
754 throw std::runtime_error(
755 "remove_inactive_particles: Error in kernel cannot remove more "
756 "element than existing");
764 Kokkos::parallel_scan(
767 CompactParticlesFunctor<M>(
status,
782 const bool do_shrink = n_used_elements <= static_cast<std::size_t>(
798 template <ModelType M>
799 [[nodiscard]] KOKKOS_INLINE_FUNCTION M::FloatType
812 template <ModelType M>
819 template <ModelType M>
827 const int bin_size = 2048;
832 using ExecSpace = Kokkos::DefaultExecutionSpace;
833 using view_type =
decltype(
position);
834 auto binop = Kokkos::BinOp1D<view_type>(bin_size, 0, n_c);
836 auto sorter = Kokkos::BinSort<view_type, decltype(binop)>(
837 ExecSpace(), sn, binop,
true);
RuntimeParameters rt_params
Definition particles_container.hpp:231
void update_and_remove_inactive(std::size_t out, std::size_t dead)
Definition particles_container.hpp:541
double get_allocation_factor() const noexcept
Get the allocation factor.
Definition particles_container.hpp:479
Model::SelfContribs contribs
Definition particles_container.hpp:83
~ParticlesContainer()=default
Default destructor.
void force_remove_dead()
Clean all the non-idle particle even if number smaller than threshold.
Definition particles_container.hpp:463
Model::SelfParticle model
Definition particles_container.hpp:82
KOKKOS_INLINE_FUNCTION std::size_t n_particles() const
Definition particles_container.hpp:449
KOKKOS_INLINE_FUNCTION bool handle_division(const MC::pool_type &random_pool, std::size_t idx1) const
Get the contribution if particle at index idx.
Definition particles_container.hpp:561
Kokkos::View< uint64_t, Kokkos::SharedSpace > buffer_index
Definition particles_container.hpp:224
void _resize(std::size_t new_size, bool force=false)
Definition particles_container.hpp:603
void remove_inactive_particles(std::size_t to_remove)
Definition particles_container.hpp:737
void __allocate_buffer__()
Definition particles_container.hpp:671
ParticlesContainer()
Definition particles_container.hpp:730
ParticlesContainer & operator=(ParticlesContainer &&)=default
std::size_t get_inactive() const noexcept
Definition particles_container.hpp:456
std::size_t capacity() const noexcept
Definition particles_container.hpp:486
uint64_t n_used_elements
Definition particles_container.hpp:226
auto get_buffer_index() const
Definition particles_container.hpp:471
KOKKOS_INLINE_FUNCTION Model::FloatType get_weight(std::size_t idx) const
Return the particle weight.
Definition particles_container.hpp:800
void merge_buffer()
Insert particle buffer into the main container.
Definition particles_container.hpp:577
Model::SelfParticle buffer_model
Definition particles_container.hpp:222
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:693
ParticlePositions buffer_position
Definition particles_container.hpp:223
MC::ParticlePositions position
Definition particles_container.hpp:85
void save(Archive &ar) const
Save data into ar for serialization.
Definition particles_container.hpp:527
std::size_t inactive_counter
Definition particles_container.hpp:227
ParticleAges ages
Definition particles_container.hpp:88
Model UsedModel
Definition particles_container.hpp:59
ParticleWeigths< typename Model::FloatType > weights
Definition particles_container.hpp:87
ParticlesContainer & operator=(const ParticlesContainer &)=default
void _sort(size_t n_c)
Definition particles_container.hpp:821
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:494
ParticlesContainer(ParticlesContainer &&)=default
std::size_t n_allocated_elements
Definition particles_container.hpp:225
void change_runtime(RuntimeParameters &¶meters) noexcept
Definition particles_container.hpp:814
MC::ParticleStatus status
Definition particles_container.hpp:86
Concept to check if a model type has uniform_weight
Definition traits.hpp:189
Namespace that contains classes and structures related to Monte Carlo (MC) simulations.
Definition alias.hpp:16
Kokkos::View< Status *, ComputeSpace > ParticleStatus
Definition alias.hpp:141
Kokkos::View< uint64_t *, ComputeSpace > ParticlePositions
Definition alias.hpp:140
@ Idle
Definition alias.hpp:126
ParticleAgesBase< ComputeSpace > ParticleAges
Definition alias.hpp:144
gen_pool_type< Kokkos::DefaultExecutionSpace > pool_type
Definition alias.hpp:100
Kokkos::View< ftype *, ComputeSpace > ParticleWeigths
Definition alias.hpp:143
Definition particles_container.hpp:26
void serialize(Archive &ar)
Definition particles_container.hpp:35
double buffer_ratio
Definition particles_container.hpp:28
double dead_particle_ratio_threshold
Definition particles_container.hpp:31
uint64_t minimum_dead_particle_removal
Definition particles_container.hpp:27
double shrink_ratio
Definition particles_container.hpp:30
double allocation_factor
Definition particles_container.hpp:29