blob: f77a93e3119d572f059dbf500cc6bc10613bf3cb [file] [log] [blame]
// Copyright 2023 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/fallback_allocator.h"
#include "pw_allocator/allocator_testing.h"
#include "pw_status/status.h"
#include "pw_unit_test/framework.h"
namespace pw::allocator {
namespace {
// Test fixtures.
class FallbackAllocatorTest : public ::testing::Test {
protected:
void SetUp() override { allocator_.Init(*primary_, *secondary_); }
test::AllocatorForTest<128> primary_;
test::AllocatorForTest<128> secondary_;
FallbackAllocatorImpl<internal::Metrics> allocator_;
};
// Unit tests.
TEST_F(FallbackAllocatorTest, ExplicitlyInitialized) {
FallbackAllocatorImpl<internal::Metrics> fallback;
metric::Group& group = fallback.metric_group();
EXPECT_EQ(group.metrics().size(), 0U);
EXPECT_EQ(group.children().size(), 0U);
fallback.Init(*primary_, *secondary_);
EXPECT_EQ(group.metrics().size(), 3U);
EXPECT_EQ(group.children().size(), 0U);
}
TEST_F(FallbackAllocatorTest, AutomaticallyInitialized) {
FallbackAllocatorImpl<internal::Metrics> fallback(*primary_, *secondary_);
metric::Group& group = fallback.metric_group();
EXPECT_EQ(group.metrics().size(), 3U);
EXPECT_EQ(group.children().size(), 0U);
}
TEST_F(FallbackAllocatorTest, QueryValidPrimary) {
Layout layout = Layout::Of<uint32_t>();
void* ptr = primary_->Allocate(layout);
EXPECT_TRUE(primary_->Query(ptr, layout).ok());
EXPECT_EQ(secondary_->Query(ptr, layout), Status::OutOfRange());
EXPECT_TRUE(allocator_.Query(ptr, layout).ok());
}
TEST_F(FallbackAllocatorTest, QueryValidSecondary) {
Layout layout = Layout::Of<uint32_t>();
void* ptr = secondary_->Allocate(layout);
EXPECT_FALSE(primary_->Query(ptr, layout).ok());
EXPECT_TRUE(secondary_->Query(ptr, layout).ok());
EXPECT_TRUE(allocator_.Query(ptr, layout).ok());
}
TEST_F(FallbackAllocatorTest, QueryInvalidPtr) {
test::AllocatorForTest<128> other;
Layout layout = Layout::Of<uint32_t>();
void* ptr = other->Allocate(layout);
EXPECT_FALSE(primary_->Query(ptr, layout).ok());
EXPECT_FALSE(secondary_->Query(ptr, layout).ok());
EXPECT_FALSE(allocator_.Query(ptr, layout).ok());
}
TEST_F(FallbackAllocatorTest, AllocateFromPrimary) {
Layout layout = Layout::Of<uint32_t>();
void* ptr = allocator_.Allocate(layout);
EXPECT_NE(ptr, nullptr);
EXPECT_EQ(primary_->allocate_size(), layout.size());
EXPECT_EQ(secondary_->allocate_size(), 0U);
}
TEST_F(FallbackAllocatorTest, AllocateFromSecondary) {
primary_->Exhaust();
Layout layout = Layout::Of<uint32_t>();
void* ptr = allocator_.Allocate(layout);
EXPECT_NE(ptr, nullptr);
EXPECT_EQ(primary_->allocate_size(), layout.size());
EXPECT_EQ(secondary_->allocate_size(), layout.size());
}
TEST_F(FallbackAllocatorTest, AllocateFailure) {
Layout layout = Layout::Of<uint32_t[0x10000]>();
void* ptr = allocator_.Allocate(layout);
EXPECT_EQ(ptr, nullptr);
EXPECT_EQ(primary_->allocate_size(), layout.size());
EXPECT_EQ(secondary_->allocate_size(), layout.size());
}
TEST_F(FallbackAllocatorTest, DeallocateUsingPrimary) {
Layout layout = Layout::Of<uint32_t>();
void* ptr = allocator_.Allocate(layout);
ASSERT_NE(ptr, nullptr);
allocator_.Deallocate(ptr, layout);
EXPECT_EQ(primary_->deallocate_ptr(), ptr);
EXPECT_EQ(primary_->deallocate_size(), layout.size());
EXPECT_EQ(secondary_->deallocate_ptr(), nullptr);
EXPECT_EQ(secondary_->deallocate_size(), 0U);
}
TEST_F(FallbackAllocatorTest, DeallocateUsingSecondary) {
primary_->Exhaust();
Layout layout = Layout::Of<uint32_t>();
void* ptr = allocator_.Allocate(layout);
ASSERT_NE(ptr, nullptr);
allocator_.Deallocate(ptr, layout);
EXPECT_EQ(primary_->deallocate_ptr(), nullptr);
EXPECT_EQ(primary_->deallocate_size(), 0U);
EXPECT_EQ(secondary_->deallocate_ptr(), ptr);
EXPECT_EQ(secondary_->deallocate_size(), layout.size());
}
TEST_F(FallbackAllocatorTest, ResizePrimary) {
Layout old_layout = Layout::Of<uint32_t>();
void* ptr = allocator_.Allocate(old_layout);
ASSERT_NE(ptr, nullptr);
size_t new_size = sizeof(uint32_t[3]);
EXPECT_TRUE(allocator_.Resize(ptr, old_layout, new_size));
EXPECT_EQ(primary_->resize_ptr(), ptr);
EXPECT_EQ(primary_->resize_old_size(), old_layout.size());
EXPECT_EQ(primary_->resize_new_size(), new_size);
// Secondary should not be touched.
EXPECT_EQ(secondary_->resize_ptr(), nullptr);
EXPECT_EQ(secondary_->resize_old_size(), 0U);
EXPECT_EQ(secondary_->resize_new_size(), 0U);
}
TEST_F(FallbackAllocatorTest, ResizePrimaryFailure) {
Layout old_layout = Layout::Of<uint32_t>();
void* ptr = allocator_.Allocate(old_layout);
ASSERT_NE(ptr, nullptr);
primary_->Exhaust();
size_t new_size = sizeof(uint32_t[3]);
EXPECT_FALSE(allocator_.Resize(ptr, old_layout, new_size));
EXPECT_EQ(primary_->resize_ptr(), ptr);
EXPECT_EQ(primary_->resize_old_size(), old_layout.size());
EXPECT_EQ(primary_->resize_new_size(), new_size);
// Secondary should not be touched.
EXPECT_EQ(secondary_->resize_ptr(), nullptr);
EXPECT_EQ(secondary_->resize_old_size(), 0U);
EXPECT_EQ(secondary_->resize_new_size(), 0U);
}
TEST_F(FallbackAllocatorTest, ResizeSecondary) {
primary_->Exhaust();
Layout old_layout = Layout::Of<uint32_t>();
void* ptr = allocator_.Allocate(old_layout);
ASSERT_NE(ptr, nullptr);
size_t new_size = sizeof(uint32_t[3]);
EXPECT_TRUE(allocator_.Resize(ptr, old_layout, new_size));
EXPECT_EQ(secondary_->resize_ptr(), ptr);
EXPECT_EQ(secondary_->resize_old_size(), old_layout.size());
EXPECT_EQ(secondary_->resize_new_size(), new_size);
// Primary should not be touched.
EXPECT_EQ(primary_->resize_ptr(), nullptr);
EXPECT_EQ(primary_->resize_old_size(), 0U);
EXPECT_EQ(primary_->resize_new_size(), 0U);
}
TEST_F(FallbackAllocatorTest, ResizeSecondaryFailure) {
primary_->Exhaust();
Layout old_layout = Layout::Of<uint32_t>();
void* ptr = allocator_.Allocate(old_layout);
ASSERT_NE(ptr, nullptr);
secondary_->Exhaust();
size_t new_size = sizeof(uint32_t[3]);
EXPECT_FALSE(allocator_.Resize(ptr, old_layout, new_size));
EXPECT_EQ(secondary_->resize_ptr(), ptr);
EXPECT_EQ(secondary_->resize_old_size(), old_layout.size());
EXPECT_EQ(secondary_->resize_new_size(), new_size);
// Primary should not be touched.
EXPECT_EQ(primary_->resize_ptr(), nullptr);
EXPECT_EQ(primary_->resize_old_size(), 0U);
EXPECT_EQ(primary_->resize_new_size(), 0U);
}
TEST_F(FallbackAllocatorTest, ReallocateSameAllocator) {
Layout old_layout = Layout::Of<uint32_t>();
void* ptr1 = allocator_.Allocate(old_layout);
ASSERT_NE(ptr1, nullptr);
// Claim subsequent memeory to force reallocation.
void* ptr2 = allocator_.Allocate(old_layout);
ASSERT_NE(ptr2, nullptr);
size_t new_size = sizeof(uint32_t[3]);
void* new_ptr = allocator_.Reallocate(ptr1, old_layout, new_size);
EXPECT_NE(new_ptr, nullptr);
EXPECT_EQ(primary_->deallocate_ptr(), ptr1);
EXPECT_EQ(primary_->deallocate_size(), old_layout.size());
EXPECT_EQ(primary_->allocate_size(), new_size);
}
TEST_F(FallbackAllocatorTest, ReallocateDifferentAllocator) {
Layout old_layout = Layout::Of<uint32_t>();
void* ptr = allocator_.Allocate(old_layout);
primary_->Exhaust();
size_t new_size = sizeof(uint32_t[3]);
void* new_ptr = allocator_.Reallocate(ptr, old_layout, new_size);
EXPECT_NE(new_ptr, nullptr);
EXPECT_EQ(primary_->deallocate_ptr(), ptr);
EXPECT_EQ(primary_->deallocate_size(), old_layout.size());
EXPECT_EQ(secondary_->allocate_size(), new_size);
}
TEST_F(FallbackAllocatorTest, GetMetrics) {
primary_->Exhaust();
Layout layout = Layout::Of<uint32_t>();
allocator_.Allocate(layout);
EXPECT_NE(primary_->used(), 0U);
EXPECT_EQ(primary_->peak(), primary_->used());
EXPECT_EQ(primary_->count(), 1U);
EXPECT_EQ(secondary_->used(), layout.size());
EXPECT_EQ(secondary_->peak(), layout.size());
EXPECT_EQ(secondary_->count(), 1U);
EXPECT_EQ(allocator_.used(), secondary_->used());
EXPECT_EQ(allocator_.peak(), secondary_->peak());
EXPECT_EQ(allocator_.count(), secondary_->count());
}
} // namespace
} // namespace pw::allocator