1 #include "gtest/gtest.h"
2
3 #include "chre/util/memory_pool.h"
4
5 #include <random>
6 #include <vector>
7
8 using chre::MemoryPool;
9
TEST(MemoryPool,ExhaustPool)10 TEST(MemoryPool, ExhaustPool) {
11 MemoryPool<int, 3> memoryPool;
12 EXPECT_EQ(memoryPool.getFreeBlockCount(), 3);
13 EXPECT_NE(memoryPool.allocate(), nullptr);
14 EXPECT_EQ(memoryPool.getFreeBlockCount(), 2);
15 EXPECT_NE(memoryPool.allocate(), nullptr);
16 EXPECT_EQ(memoryPool.getFreeBlockCount(), 1);
17 EXPECT_NE(memoryPool.allocate(), nullptr);
18 EXPECT_EQ(memoryPool.getFreeBlockCount(), 0);
19 EXPECT_EQ(memoryPool.allocate(), nullptr);
20 EXPECT_EQ(memoryPool.getFreeBlockCount(), 0);
21 }
22
TEST(MemoryPool,ExhaustPoolThenDeallocateOneAndAllocateOne)23 TEST(MemoryPool, ExhaustPoolThenDeallocateOneAndAllocateOne) {
24 MemoryPool<int, 3> memoryPool;
25
26 // Exhaust the pool.
27 int *element1 = memoryPool.allocate();
28 int *element2 = memoryPool.allocate();
29 int *element3 = memoryPool.allocate();
30
31 // Perform some simple assignments. There is a chance we crash here if things
32 // are not implemented correctly.
33 *element1 = 0xcafe;
34 *element2 = 0xbeef;
35 *element3 = 0xface;
36
37 // Free one element and then allocate another.
38 memoryPool.deallocate(element1);
39 EXPECT_EQ(memoryPool.getFreeBlockCount(), 1);
40 element1 = memoryPool.allocate();
41 EXPECT_NE(element1, nullptr);
42
43 // Ensure that the pool remains exhausted.
44 EXPECT_EQ(memoryPool.allocate(), nullptr);
45
46 // Perform another simple assignment. There is a hope that this can crash if
47 // the pointer returned is very bad (like nullptr).
48 *element1 = 0xfade;
49
50 // Verify that the values stored were not corrupted by the deallocate
51 // allocate cycle.
52 EXPECT_EQ(*element1, 0xfade);
53 EXPECT_EQ(*element2, 0xbeef);
54 EXPECT_EQ(*element3, 0xface);
55 }
56
57 /*
58 * Pair an allocated pointer with the expected value that should be stored in
59 * that location.
60 */
61 struct AllocationExpectedValuePair {
62 size_t *allocation;
63 size_t expectedValue;
64 };
65
TEST(MemoryPool,ExhaustPoolThenRandomDeallocate)66 TEST(MemoryPool, ExhaustPoolThenRandomDeallocate) {
67 // The number of times to allocate and deallocate in random order.
68 const size_t kStressTestCount = 64;
69
70 // Construct a memory pool and a vector to maintain a list of all allocations.
71 const size_t kMemoryPoolSize = 64;
72 MemoryPool<size_t, kMemoryPoolSize> memoryPool;
73 std::vector<AllocationExpectedValuePair> allocations;
74
75 for (size_t i = 0; i < kStressTestCount; i++) {
76 // Exhaust the memory pool.
77 for (size_t j = 0; j < kMemoryPoolSize; j++) {
78 AllocationExpectedValuePair allocation = {
79 .allocation = memoryPool.allocate(),
80 .expectedValue = j,
81 };
82
83 *allocation.allocation = j;
84 allocations.push_back(allocation);
85 }
86
87 // Seed a random number generator with the loop iteration so that order is
88 // preserved across test runs.
89 std::mt19937 randomGenerator(i);
90
91 while (!allocations.empty()) {
92 // Generate a number with a uniform distribution between zero and the
93 // number of allocations remaining.
94 std::uniform_int_distribution<> distribution(0, allocations.size() - 1);
95 size_t deallocateIndex = distribution(randomGenerator);
96
97 // Verify the expected value and free the allocation.
98 EXPECT_EQ(*allocations[deallocateIndex].allocation,
99 allocations[deallocateIndex].expectedValue);
100 memoryPool.deallocate(allocations[deallocateIndex].allocation);
101
102 // Remove the freed allocation from the allocation list.
103 allocations.erase(allocations.begin() + deallocateIndex);
104 }
105 }
106 }
107