blob: 8fb710311052d1f06c6aae5cca60cdf8952469fc [file] [log] [blame]
// Copyright 2024 The Pigweed Authors
//
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
// use this file except in compliance with the License. You may obtain a copy of
// the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
// License for the specific language governing permissions and limitations under
// the License.
#include "pw_allocator/chunk_pool.h"
#include <array>
#include <cstddef>
#include <cstdint>
#include "pw_allocator/internal/counter.h"
#include "pw_allocator/testing.h"
#include "pw_unit_test/framework.h"
namespace {
// Test fixtures.
using ::pw::allocator::Layout;
using ::pw::allocator::test::Counter;
using ChunkPoolTest = pw::allocator::test::TestWithCounters;
struct U64 {
std::byte bytes[8];
};
// Unit tests.
TEST_F(ChunkPoolTest, Capabilities) {
std::array<std::byte, 256> buffer;
pw::allocator::ChunkPool pool(buffer, Layout::Of<U64>());
EXPECT_EQ(pool.capabilities(), pw::allocator::ChunkPool::kCapabilities);
}
TEST_F(ChunkPoolTest, AllocateDeallocate) {
std::array<std::byte, 256> buffer;
pw::allocator::ChunkPool pool(buffer, Layout::Of<U64>());
void* ptr = pool.Allocate();
ASSERT_NE(ptr, nullptr);
pool.Deallocate(ptr);
}
TEST_F(ChunkPoolTest, ExhaustTwice) {
constexpr size_t kNumU64s = 32;
constexpr size_t kBufferSize = sizeof(U64) * kNumU64s;
std::array<std::byte, kBufferSize> buffer;
pw::allocator::ChunkPool pool(buffer, Layout::Of<U64>());
// Allocate everything.
std::array<void*, kNumU64s> ptrs;
for (auto& ptr : ptrs) {
ptr = pool.Allocate();
ASSERT_NE(ptr, nullptr);
}
// At this point, the pool is empty.
EXPECT_EQ(pool.Allocate(), nullptr);
// Now refill the pool, and show it can be emptied again.
for (auto& ptr : ptrs) {
pool.Deallocate(ptr);
ptr = nullptr;
}
for (auto& ptr : ptrs) {
ptr = pool.Allocate();
ASSERT_NE(ptr, nullptr);
}
// Release everything.
for (auto& ptr : ptrs) {
pool.Deallocate(ptr);
ptr = nullptr;
}
}
TEST_F(ChunkPoolTest, NewDelete) {
std::array<std::byte, 256> buffer;
pw::allocator::ChunkPool pool(buffer, Layout::Of<Counter>());
auto* counter = pool.New<Counter>(867u);
ASSERT_NE(counter, nullptr);
EXPECT_EQ(counter->value(), 867u);
pool.Delete(counter);
EXPECT_EQ(Counter::TakeNumDtorCalls(), 1u);
}
TEST_F(ChunkPoolTest, NewDeleteBoundedArray) {
std::array<std::byte, 256> buffer;
Layout layout(sizeof(Counter) * 3, alignof(Counter));
pw::allocator::ChunkPool pool(buffer, layout);
auto* counters = pool.New<Counter[3]>();
ASSERT_NE(counters, nullptr);
for (size_t i = 0; i < 3; ++i) {
EXPECT_EQ(counters[i].value(), i);
}
pool.Delete<Counter[3]>(counters);
EXPECT_EQ(Counter::TakeNumDtorCalls(), 3u);
}
TEST_F(ChunkPoolTest, NewDeleteUnboundedArray) {
std::array<std::byte, 256> buffer;
Layout layout(sizeof(Counter) * 5, alignof(Counter));
pw::allocator::ChunkPool pool(buffer, layout);
auto* counters = pool.New<Counter[]>();
ASSERT_NE(counters, nullptr);
for (size_t i = 0; i < 5; ++i) {
EXPECT_EQ(counters[i].value(), i);
}
pool.Delete<Counter[]>(counters, 5);
EXPECT_EQ(Counter::TakeNumDtorCalls(), 5u);
}
TEST_F(ChunkPoolTest, NewDeleteArray) {
std::array<std::byte, 256> buffer;
Layout layout(sizeof(Counter) * 3, alignof(Counter));
pw::allocator::ChunkPool pool(buffer, layout);
auto* counters = pool.New<Counter[3]>();
ASSERT_NE(counters, nullptr);
for (size_t i = 0; i < 3; ++i) {
EXPECT_EQ(counters[i].value(), i);
}
pool.DeleteArray(counters, 3);
EXPECT_EQ(Counter::TakeNumDtorCalls(), 3u);
}
TEST_F(ChunkPoolTest, MakeUnique) {
std::array<std::byte, 256> buffer;
pw::allocator::ChunkPool pool(buffer, Layout::Of<Counter>());
{
auto counter = pool.MakeUnique<Counter>(5309u);
ASSERT_NE(counter, nullptr);
EXPECT_EQ(counter->value(), 5309u);
}
EXPECT_EQ(Counter::TakeNumDtorCalls(), 1u);
}
TEST_F(ChunkPoolTest, MakeUniqueBoundedArray) {
std::array<std::byte, 256> buffer;
Layout layout(sizeof(Counter) * 7, alignof(Counter));
pw::allocator::ChunkPool pool(buffer, layout);
{
auto counters = pool.MakeUnique<Counter[7]>();
ASSERT_NE(counters, nullptr);
for (size_t i = 0; i < 7; ++i) {
EXPECT_EQ(counters[i].value(), i);
}
}
EXPECT_EQ(Counter::TakeNumDtorCalls(), 7u);
}
TEST_F(ChunkPoolTest, MakeUniqueBoundedArrayDifferentType) {
std::array<std::byte, 256> buffer;
Layout layout(sizeof(Counter) * 7, alignof(Counter));
pw::allocator::ChunkPool pool(buffer, layout);
auto bytes = pool.MakeUnique<std::byte[sizeof(Counter) * 7]>();
ASSERT_NE(bytes, nullptr);
EXPECT_EQ(bytes.size(), layout.size());
}
TEST_F(ChunkPoolTest, MakeUniqueUnboundedArray) {
std::array<std::byte, 256> buffer;
Layout layout(sizeof(Counter) * 9, alignof(Counter));
pw::allocator::ChunkPool pool(buffer, layout);
{
auto counters = pool.MakeUnique<Counter[]>();
ASSERT_NE(counters, nullptr);
for (size_t i = 0; i < 9; ++i) {
EXPECT_EQ(counters[i].value(), i);
}
}
EXPECT_EQ(Counter::TakeNumDtorCalls(), 9u);
}
TEST_F(ChunkPoolTest, MakeUniqueUnboundedArrayDifferentType) {
std::array<std::byte, 256> buffer;
Layout layout(sizeof(Counter) * 9, alignof(Counter));
pw::allocator::ChunkPool pool(buffer, layout);
auto bytes = pool.MakeUnique<std::byte[]>();
ASSERT_NE(bytes, nullptr);
EXPECT_EQ(bytes.size(), layout.size());
}
} // namespace