/******************************************************************************* * This file is part of SWIFT. * Copyright (c) 2020 Josh Borrow (josh@joshborrow.com) * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published * by the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program. If not, see . * ******************************************************************************/ #ifndef SWIFT_PARTICLE_SPLITTING_H #define SWIFT_PARTICLE_SPLITTING_H #include "inline.h" #include "io_properties.h" #include "particle_splitting_struct.h" #include /** * @brief Initialise a particle_splitting_data struct * at the start of a run, given an initial * progenitor ID. * * @param splitting_data the uninitialised particle struct. * @param id the ID of the particle (used for progenitor_id) */ __attribute__((always_inline)) INLINE static void particle_splitting_mark_part_as_not_split( struct particle_splitting_data* restrict splitting_data, int id) { splitting_data->progenitor_id = id; splitting_data->split_tree = 0; splitting_data->split_count = 0; } /** * @brief Updates the binary trees of the particles that * have been split. pi should retain its original ID, * and hence gets the relevant part of the tree set * to zero. * * @param sdi first particle_splitting_data* resulting from * the splitting event. * @param sdj second particle_splitting_data* resulting from * the splitting event. */ __attribute__((always_inline)) INLINE static void particle_splitting_update_binary_tree( struct particle_splitting_data* restrict sdi, struct particle_splitting_data* restrict sdj) { /* Update the binary tree */ sdj->split_tree |= 1LL << sdj->split_count; /* Increase counters on both; sdi implicitly has a zero * in the relevant spot in its binary tree */ sdj->split_count++; sdi->split_count++; /* Print warnings if we have split these particles more * than the number of times the tree can accommodate. * Warning is only printed once for each particle */ if (sdi->split_count == 8 * sizeof(sdi->split_tree)) { message( "Warning: Particle with progenitor ID %lld with binary tree %lld has " "been split over the maximum %zu times, making its binary tree " "invalid.", sdi->progenitor_id, sdi->split_tree, sizeof(sdi->split_tree)); } } /** * @brief Specifies the particle-splitting related fields * to write to a dataset. * * @param parts The particle array. * @param xparts The extra particle array. * @param list The list of i/o properties to write. * @param with_cosmology Are we running with cosmology? * * @return Returns the number of fields to write. */ INLINE static int particle_splitting_write_particles(const struct part* parts, const struct xpart* xparts, struct io_props* list, const int with_cosmology) { list[0] = io_make_output_field( "ProgenitorParticleIDs", LONGLONG, 1, UNIT_CONV_NO_UNITS, 0.f, xparts, split_data.progenitor_id, "ID of the progenitor of this particle. If this particle is the result " "of one (or many) splitting events, this ID corresponds to the ID of the " "particle in the initial conditions that its lineage can be traced back " "to. If the particle was never split, this is the same as ParticleIDs."); list[1] = io_make_output_field( "SplitCounts", UINT8, 1, UNIT_CONV_NO_UNITS, 0.f, xparts, split_data.split_count, "Number of times this particle has been split. Note that both particles " "that take part in the splitting have counter incremented, so the " "number of splitting events in an entire simulation is half of the sum " "of all of these numbers."); list[2] = io_make_output_field( "SplitTrees", LONGLONG, 1, UNIT_CONV_NO_UNITS, 0.f, xparts, split_data.split_tree, "Binary tree describing splitting events. Particles that keep the " "original ID have a value of zero in a splitting event, whereas" "particles given a new ID have a value of one."); return 3; } /** * @brief Specifies which star particle fields to write to a dataset * * @param sparts The star particle array. * @param list The list of i/o properties to write. * * @return Returns the number of fields to write. */ INLINE static int particle_splitting_write_sparticles( const struct spart* sparts, struct io_props* list) { list[0] = io_make_output_field( "ProgenitorParticleIDs", LONGLONG, 1, UNIT_CONV_NO_UNITS, 0.f, sparts, split_data.progenitor_id, "Progenitor ID of the gas particle that became this star. If this " "particle is the result of one (or many) splitting events, this ID " "corresponds to the ID of the particle in the initial conditions that " "its lineage can be traced back to. If the particle was never split, " "this is the same as ParticleIDs."); list[1] = io_make_output_field( "SplitCounts", UINT8, 1, UNIT_CONV_NO_UNITS, 0.f, sparts, split_data.split_count, "Number of times the gas particle that turned into this star particle " "was split. Note that both particles that take part in the splitting " "have this counter incremented, so the number of splitting events in an " "entire simulation is half of the sum of all of these numbers."); list[2] = io_make_output_field( "SplitTrees", LONGLONG, 1, UNIT_CONV_NO_UNITS, 0.f, sparts, split_data.split_tree, "Binary tree describing splitting events. Particles that keep the " "original ID have a value of zero in a splitting event, whereas" "particles given a new ID have a value of one."); return 3; } /** * @brief Specifies which black hole particle fields to write to a dataset * * @param bparts The black hole particle array. * @param list The list of i/o properties to write. * * @return Returns the number of fields to write. */ INLINE static int particle_splitting_write_bparticles( const struct bpart* bparts, struct io_props* list) { list[0] = io_make_output_field( "ProgenitorParticleIDs", LONGLONG, 1, UNIT_CONV_NO_UNITS, 0.f, bparts, split_data.progenitor_id, "Progenitor ID of the gas particle that became the seed BH. If this " "particle is the result of one (or many) splitting events, this ID " "corresponds to the ID of the particle in the initial conditions that " "its lineage can be traced back to. If the particle was never split, " "this is the same as ParticleIDs."); list[1] = io_make_output_field( "SplitCounts", UINT8, 1, UNIT_CONV_NO_UNITS, 0.f, bparts, split_data.split_count, "Number of times the gas particle that became this BH seed " "was split. Note that both particles that take part in the splitting " "have this counter incremented, so the number of splitting events in an " "entire simulation is half of the sum of all of these numbers."); list[2] = io_make_output_field( "SplitTrees", LONGLONG, 1, UNIT_CONV_NO_UNITS, 0.f, bparts, split_data.split_tree, "Binary tree describing splitting events prior to BH seeding. Particles " "that keep the original ID have a value of zero in a splitting event, " "whereas particles given a new ID have a value of one."); return 3; } /** * @brief Write particle splitting data to the stdout for debugging purposes. * * @param p Particle data. * @param xp Extra particle data. */ __attribute__((always_inline)) INLINE static void particle_splitting_debug_particle(const struct part* p, const struct xpart* xp) { if (xp != NULL) { warning("[PID%lld] particle_splitting_data:", p->id); warning( "[PID%lld] progenitor_id = %lli, split_tree = %lli, " "split_count = %hhu", p->id, xp->split_data.progenitor_id, xp->split_data.split_tree, xp->split_data.split_count); } } #endif /* SWIFT_PARTICLE_SPLITTING_H */