Skip to content

Commit

Permalink
fix memory leak
Browse files Browse the repository at this point in the history
With ComputationalRadiationPhysics#44 we tried to fix the memory leak which can happen if more than
one thread is allocating a new chunk.
The PR ComputationalRadiationPhysics#44 was not fixing the issue and introduces a double free issue.
  • Loading branch information
psychocoderHPC committed Dec 11, 2023
1 parent 7f2c3e4 commit e325faa
Showing 1 changed file with 22 additions and 24 deletions.
46 changes: 22 additions & 24 deletions redGrapes/util/chunked_list.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -79,7 +79,7 @@ template <
>
struct ChunkedList
{
using chunk_offset_t = int16_t;
using chunk_offset_t = uint16_t;
using refcount_t = uint16_t;

struct Item
Expand Down Expand Up @@ -163,14 +163,11 @@ struct ChunkedList

struct Chunk
{
/* counts the number of alive elements in this chunk.
* Whenever `item_count` reaches zero, the chunk will be deleted.
* `item_count` starts with 1 to keep the chunk at least until
* its full capacity is used. This initial offset is
* compensated by not increasing the count when the last
* element is inserted.
/* counts the number of freed elements in this chunk.
* Whenever `freed_chunks` reaches the last chunk, the chunk
* will be deleted.
*/
std::atomic< chunk_offset_t > item_count{ 0 };
std::atomic< chunk_offset_t > freed_chunks{ 0 };

/* lowest index with free slot that can
* be used to add a new element
Expand Down Expand Up @@ -440,7 +437,7 @@ struct ChunkedList
*/
void release_chunk( typename memory::AtomicList< Chunk, Allocator >::MutBackwardIterator chunk )
{
if( chunk->item_count.fetch_sub(1) == 0 )
if( chunk->item_count.fetch_add(1) == chunk_size - 1u )
chunks.erase( chunk );
}

Expand All @@ -453,25 +450,26 @@ struct ChunkedList
auto chunk = chunks.rbegin();
if( chunk != chunks.rend() )
{
if( chunk->item_count.fetch_add(1) < chunk_size )
{
unsigned chunk_off = chunk->next_idx.fetch_add(1);

if( chunk_off < chunk_size )
{

unsigned chunk_off = chunk->next_idx.fetch_add(1);

if( chunk_off < chunk_size )
{
chunk->items()[ chunk_off ] = item;
chunk->last_idx ++;
if( chunk_off == chunk_size - 1)
{
// the thread who took the last chunk must allocate
// the next batch of chunks
chunks.allocate_item();
}
return MutBackwardIterator( chunk_size, chunk, chunk_off );
}
}

release_chunk(chunk);
}
else
{
// avoid that the counter can overflow
chunk->next_idx--;
}
}

auto prev_chunk = chunks.allocate_item();
if( prev_chunk != chunks.rend() )
release_chunk( prev_chunk );
}
}

Expand Down

0 comments on commit e325faa

Please sign in to comment.