Skip to content

Commit

Permalink
Convert from fill with filter empty to atomic grid create
Browse files Browse the repository at this point in the history
  • Loading branch information
Samuel Reeve authored and streeve committed Sep 1, 2023
1 parent d101d22 commit a3763e5
Show file tree
Hide file tree
Showing 2 changed files with 38 additions and 100 deletions.
129 changes: 34 additions & 95 deletions cajita/src/Cajita_ParticleInit.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -40,63 +40,6 @@
namespace Cajita
{

//---------------------------------------------------------------------------//
/*!
\brief Filter out empty particles that weren't created.
\param exec_space Kokkos execution space.
\param local_num_create The number of particles created.
\param aosoa The particle AoSoA.
\param particle_created Whether to remove unused allocated space.
\param shrink_to_fit Whether to remove unused allocated space.
*/
template <class CreationView, class ParticleAoSoA, class ExecutionSpace>
void filterEmpties( const ExecutionSpace& exec_space,
const int local_num_create,
const CreationView& particle_created, ParticleAoSoA& aosoa,
const bool shrink_to_fit )
{
using memory_space = typename CreationView::memory_space;

// Determine the empty particle positions in the compaction zone.
int num_particles = aosoa.size();
Kokkos::View<int*, memory_space> empties(
Kokkos::ViewAllocateWithoutInitializing( "empties" ),
std::min( num_particles - local_num_create, local_num_create ) );
Kokkos::parallel_scan(
"Cabana::ParticleInit::FindEmpty",
Kokkos::RangePolicy<ExecutionSpace>( exec_space, 0, local_num_create ),
KOKKOS_LAMBDA( const int i, int& count, const bool final_pass ) {
if ( !particle_created( i ) )
{
if ( final_pass )
{
empties( count ) = i;
}
++count;
}
} );

// Compact the list so the it only has real particles.
Kokkos::parallel_scan(
"Cabana::ParticleInit::RemoveEmpty",
Kokkos::RangePolicy<ExecutionSpace>( exec_space, local_num_create,
num_particles ),
KOKKOS_LAMBDA( const int i, int& count, const bool final_pass ) {
if ( particle_created( i ) )
{
if ( final_pass )
{
aosoa.setTuple( empties( count ), aosoa.getTuple( i ) );
}
++count;
}
} );
aosoa.resize( local_num_create );
if ( shrink_to_fit )
aosoa.shrinkToFit();
}

//---------------------------------------------------------------------------//
/*!
\brief Initialize a random number of particles in each cell given an
Expand Down Expand Up @@ -154,17 +97,13 @@ int createParticles(
int num_particles = particles_per_cell * owned_cells.size();
aosoa.resize( num_particles );

// Creation status.
auto particle_created = Kokkos::View<bool*, memory_space>(
Kokkos::ViewAllocateWithoutInitializing( "particle_created" ),
num_particles );
// Creation count.
auto count = Kokkos::View<int*, memory_space>( "particle_count", 1 );

// Initialize particles.
int local_num_create = 0;
Cajita::grid_parallel_reduce(
Cajita::grid_parallel_for(
"Cajita::ParticleInit::Random", exec_space, owned_cells,
KOKKOS_LAMBDA( const int i, const int j, const int k,
int& create_count ) {
KOKKOS_LAMBDA( const int i, const int j, const int k ) {
// Compute the owned local cell id.
int i_own = i - owned_cells.min( Dim::I );
int j_own = j - owned_cells.min( Dim::J );
Expand Down Expand Up @@ -209,23 +148,24 @@ int createParticles(

// Create a new particle with the given logical coordinates.
auto particle = particle_list.getParticle( pid );
particle_created( pid ) =
create_functor( pid, px, pv, particle );
bool created = create_functor( pid, px, pv, particle );

// If we created a new particle insert it into the list.
if ( particle_created( pid ) )
if ( created )
{
particle_list.setParticle( particle, pid );
++create_count;
auto c = Kokkos::atomic_fetch_add( &count( 0 ), 1 );
particle_list.setParticle( particle, c );
}
}
},
local_num_create );
} );
Kokkos::fence();

// Filter empties.
filterEmpties( exec_space, local_num_create, particle_created, aosoa,
shrink_to_fit );
return local_num_create;
auto count_host =
Kokkos::create_mirror_view_and_copy( Kokkos::HostSpace(), count );
aosoa.resize( count_host( 0 ) );
if ( shrink_to_fit )
aosoa.shrinkToFit();
return count_host( 0 );
}

/*!
Expand Down Expand Up @@ -417,17 +357,13 @@ int createParticles(
int num_particles = particles_per_cell * owned_cells.size();
aosoa.resize( num_particles );

// Creation status.
auto particle_created = Kokkos::View<bool*, memory_space>(
Kokkos::ViewAllocateWithoutInitializing( "particle_created" ),
num_particles );
// Creation count.
auto count = Kokkos::View<int*, memory_space>( "particle_count", 1 );

// Initialize particles.
int local_num_create = 0;
Cajita::grid_parallel_reduce(
Cajita::grid_parallel_for(
"Cajita::ParticleInit::Uniform", exec_space, owned_cells,
KOKKOS_LAMBDA( const int i, const int j, const int k,
int& create_count ) {
KOKKOS_LAMBDA( const int i, const int j, const int k ) {
// Compute the owned local cell id.
int i_own = i - owned_cells.min( Dim::I );
int j_own = j - owned_cells.min( Dim::J );
Expand Down Expand Up @@ -482,23 +418,26 @@ int createParticles(
// Create a new particle with the given logical
// coordinates.
auto particle = particle_list.getParticle( pid );
particle_created( pid ) =
create_functor( pid, px, pv, particle );
bool created = create_functor( pid, px, pv, particle );

// If we created a new particle insert it into the list.
if ( particle_created( pid ) )
if ( created )
{
particle_list.setParticle( particle, pid );
++create_count;
// TODO: replace atomic with fill (use pid directly)
// and filter of empty list entries.
auto c = Kokkos::atomic_fetch_add( &count( 0 ), 1 );
particle_list.setParticle( particle, c );
}
}
},
local_num_create );
} );
Kokkos::fence();

// Filter empties.
filterEmpties( exec_space, local_num_create, particle_created, aosoa,
shrink_to_fit );
return local_num_create;
auto count_host =
Kokkos::create_mirror_view_and_copy( Kokkos::HostSpace(), count );
aosoa.resize( count_host( 0 ) );
if ( shrink_to_fit )
aosoa.shrinkToFit();
return count_host( 0 );
}

/*!
Expand Down
9 changes: 4 additions & 5 deletions cajita/unit_test/tstParticleInit.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -106,15 +106,14 @@ void initParticleListTest( InitType init_type, int ppc )
};

// Initialize particles.
Cajita::createParticles( init_type, TEST_EXECSPACE(), init_func, particles,
ppc, *local_grid );
int num_p = Cajita::createParticles( init_type, TEST_EXECSPACE(), init_func,
particles, ppc, *local_grid );

// Check that we made particles.
int num_p = particles.size();
EXPECT_TRUE( num_p > 0 );

// Compute the global number of particles.
int global_num_particle = num_p;
int global_num_particle = particles.size();
MPI_Allreduce( MPI_IN_PLACE, &global_num_particle, 1, MPI_INT, MPI_SUM,
MPI_COMM_WORLD );
int expect_num_particle =
Expand Down Expand Up @@ -144,7 +143,7 @@ void initParticleListTest( InitType init_type, int ppc )
EXPECT_TRUE( px < box[2 * d + 1] );
}
auto pv = get( particle, Bar() );
EXPECT_EQ( pv, volume );
EXPECT_DOUBLE_EQ( pv, volume );
}
}

Expand Down

0 comments on commit a3763e5

Please sign in to comment.