Skip to content

Commit

Permalink
[Support] Recycler: Match dealloc size and enforce min size
Browse files Browse the repository at this point in the history
  • Loading branch information
optimisan committed Jan 8, 2025
1 parent 162814a commit e31e945
Show file tree
Hide file tree
Showing 2 changed files with 27 additions and 1 deletion.
4 changes: 3 additions & 1 deletion llvm/include/llvm/Support/Recycler.h
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,7 @@ class Recycler {
void clear(AllocatorType &Allocator) {
while (FreeList) {
T *t = reinterpret_cast<T *>(pop_val());
Allocator.Deallocate(t);
Allocator.Deallocate(t, Size, Align);
}
}

Expand All @@ -89,6 +89,8 @@ class Recycler {
"Recycler allocation alignment is less than object align!");
static_assert(sizeof(SubClass) <= Size,
"Recycler allocation size is less than object size!");
static_assert(Size >= sizeof(FreeNode) &&
"Recycler allocation size must be at least sizeof(FreeNode)");
return FreeList ? reinterpret_cast<SubClass *>(pop_val())
: static_cast<SubClass *>(Allocator.Allocate(Size, Align));
}
Expand Down
24 changes: 24 additions & 0 deletions llvm/unittests/Support/RecyclerTest.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,10 @@ using namespace llvm;

namespace {

struct Object1 {
char Data[1];
};

struct Object8 {
char Data[8];
};
Expand All @@ -22,12 +26,32 @@ class DecoratedMallocAllocator : public MallocAllocator {
public:
int DeallocCount = 0;

void Deallocate(const void *Ptr, size_t Size, size_t Alignment) {
DeallocCount++;
MallocAllocator::Deallocate(Ptr, Size, Alignment);
}

template <typename T> void Deallocate(T *Ptr) {
DeallocCount++;
MallocAllocator::Deallocate(Ptr);
}
};

TEST(RecyclerTest, RecycleAllocation) {
DecoratedMallocAllocator Allocator;
// Recycler needs size to be atleast 8 bytes.
Recycler<Object1, 8, 8> R;
Object1 *A1 = R.Allocate(Allocator);
Object1 *A2 = R.Allocate(Allocator);
R.Deallocate(Allocator, A2);
Object1 *A3 = R.Allocate(Allocator);
EXPECT_EQ(A2, A3); // reuse the deallocated object.
R.Deallocate(Allocator, A1);
R.Deallocate(Allocator, A3);
R.clear(Allocator); // Should deallocate A1 and A3.
EXPECT_EQ(Allocator.DeallocCount, 2);
}

TEST(RecyclerTest, MoveConstructor) {
DecoratedMallocAllocator Allocator;
Recycler<Object8> R;
Expand Down

0 comments on commit e31e945

Please sign in to comment.