blob: ee316692a0d831d98d0c1220048ed25af3e84006 [file] [log] [blame]
/* Copyright 2019 Google LLC. All Rights Reserved.
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
http://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 "ruy/allocator.h"
#include "ruy/gtest_wrapper.h"
namespace ruy {
namespace {
TEST(AllocatorTest, ReturnsValidMemory) {
Allocator allocator;
int *p;
allocator.Allocate(1, &p);
ASSERT_NE(p, nullptr);
// If this is bogus memory, ASan will cause this test to fail.
*p = 42;
allocator.FreeAll();
}
TEST(AllocatorTest, NoLeak) {
Allocator allocator;
// Allocate and free some ridiculously large total amount of memory, so
// that a leak will hopefully cause some sort of resource exhaustion.
//
// Despite the large number of allocations, this test is actually quite
// fast, since our fast-path allocation logic is very fast.
constexpr int kNumAllocations = 100 * 1024;
constexpr int kAllocationSize = 1024 * 1024;
for (int i = 0; i < kNumAllocations; i++) {
char *p;
allocator.Allocate(kAllocationSize, &p);
allocator.FreeAll();
}
}
TEST(AllocatorTest, IncreasingSizes) {
Allocator allocator;
// Allocate sizes that increase by small amounts across FreeAll calls.
for (int i = 1; i < 100 * 1024; i++) {
char *p;
allocator.Allocate(i, &p);
allocator.FreeAll();
}
}
TEST(AllocatorTest, ManySmallAllocations) {
Allocator allocator;
// Allocate many small allocations between FreeAll calls.
for (int i = 0; i < 10 * 1024; i += 100) {
for (int j = 0; j < i; j++) {
char *p;
allocator.Allocate(1, &p);
}
allocator.FreeAll();
}
}
TEST(AllocatorTest, DestructorHandlesMainBumpPtr) {
// This is a white-box test.
Allocator allocator;
allocator.AllocateBytes(1);
allocator.FreeAll();
// After the call to FreeAll, the allocator will consolidate all of the memory
// into the main bump-ptr allocator's block, which we then expect to be freed
// in the destructor.
//
// We have no test assertions -- we primarily expect that this trigger a leak
// checker and cause the test to fail.
}
TEST(AllocatorTest, DestructorHandlesFallbackBlocks) {
// This is a white-box test.
Allocator allocator;
// Since we just created the allocator, this will allocate a fallback block,
// which we then expect to be freed in the destructor.
//
// We have no test assertions -- we primarily expect that this trigger a leak
// checker and cause the test to fail.
allocator.AllocateBytes(1);
}
TEST(AllocatorTest, AvoidAliasing) {
Allocator allocator;
// Run twice with a FreeAll in between, just in case some future
// change of internal logic makes that bug-prone.
for (int repeat = 0; repeat < 2; repeat++) {
for (int i = 1; i < 100; i++) {
const void *to_avoid =
reinterpret_cast<const void *>(0x1234567890123ull + 123 * i);
void *ptr = allocator.AllocateBytesAvoidingAliasingWith(i * 10, to_avoid);
auto unsigned_low_bits = [](const void *p) {
return static_cast<std::uint32_t>(reinterpret_cast<std::uintptr_t>(p));
};
static constexpr int kMinPeriod = 1024;
std::uint32_t unsigned_diff =
(unsigned_low_bits(ptr) - unsigned_low_bits(to_avoid)) % kMinPeriod;
std::uint32_t unsigned_diff_mod = unsigned_diff % kMinPeriod;
ASSERT_TRUE(unsigned_diff_mod >= (kMinPeriod / 4) &&
unsigned_diff_mod <= 3 * (kMinPeriod / 4));
}
allocator.FreeAll();
}
}
} // namespace
} // namespace ruy
int main(int argc, char **argv) {
::testing::InitGoogleTest(&argc, argv);
return RUN_ALL_TESTS();
}