-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathlinear_allocator.cpp
139 lines (135 loc) · 5.24 KB
/
linear_allocator.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
#include "acl/allocators/linear_allocator.hpp"
#include "acl/allocators/linear_arena_allocator.hpp"
#include "acl/allocators/linear_stack_allocator.hpp"
#include "catch2/catch_all.hpp"
// NOLINTBEGIN
TEST_CASE("Validate linear_allocator", "[linear_allocator]")
{
using allocator_t = acl::linear_allocator<>;
struct record
{
void* data;
std::uint32_t size;
};
constexpr std::uint32_t k_arena_size = 1000;
allocator_t allocator(k_arena_size);
auto start = acl::allocate<std::uint8_t>(allocator, 40);
auto off100 = acl::allocate<std::uint8_t>(allocator, 100);
CHECK(start + 40 == off100);
allocator.deallocate(off100, 100);
off100 = acl::allocate<std::uint8_t>(allocator, 100);
CHECK(start + 40 == off100);
}
TEST_CASE("Validate linear_arena_allocator without alignment", "[linear_arena_allocator]")
{
using namespace acl;
using allocator_t = acl::linear_arena_allocator<
acl::config<acl::cfg::underlying_allocator<default_allocator<acl::config<acl::cfg::min_alignment<8>>>>>>;
struct record
{
void* data;
std::uint32_t size;
};
constexpr std::uint32_t k_arena_size = 1000;
allocator_t allocator(k_arena_size);
auto start = acl::allocate<std::uint8_t>(allocator, 40);
auto first = start;
auto off100 = acl::allocate<std::uint8_t>(allocator, 100);
CHECK(start + 40 == off100);
allocator.deallocate(off100, 100);
off100 = acl::allocate<std::uint8_t>(allocator, 100);
CHECK(start + 40 == off100);
auto new_arena = acl::allocate<std::uint8_t>(allocator, 1000);
CHECK(2 == allocator.get_arena_count());
auto from_old = acl::allocate<std::uint8_t>(allocator, 40);
CHECK(off100 + 100 == from_old);
allocator.deallocate(new_arena, 1000);
new_arena = acl::allocate<std::uint8_t>(allocator, 1000);
CHECK(2 == allocator.get_arena_count());
allocator.rewind();
start = acl::allocate<std::uint8_t>(allocator, 40);
CHECK(start == first);
CHECK(2 == allocator.get_arena_count());
allocator.smart_rewind();
start = acl::allocate<std::uint8_t>(allocator, 40);
CHECK(start == first);
CHECK(1 == allocator.get_arena_count());
}
TEST_CASE("Validate linear_arena_allocator with alignment", "[linear_arena_allocator]")
{
using namespace acl;
using allocator_t = acl::linear_arena_allocator<
acl::config<acl::cfg::underlying_allocator<default_allocator<acl::config<acl::cfg::min_alignment<128>>>>>>;
struct record
{
void* data;
std::uint32_t size;
};
constexpr std::uint32_t k_arena_size = 1152;
allocator_t allocator(k_arena_size);
auto start = acl::allocate<std::uint8_t>(allocator, 256, 128);
auto first = start;
auto off100 = acl::allocate<std::uint8_t>(allocator, 512, 128);
CHECK(start + 256 == off100);
allocator.deallocate(off100, 512);
off100 = acl::allocate<std::uint8_t>(allocator, 512, 128);
CHECK(start + 256 == off100);
auto new_arena = acl::allocate<std::uint8_t>(allocator, 1024, 128);
CHECK(2 == allocator.get_arena_count());
auto from_old = acl::allocate<std::uint8_t>(allocator, 256);
CHECK(off100 + 512 == from_old);
allocator.deallocate(new_arena, 1024);
new_arena = acl::allocate<std::uint8_t>(allocator, 1024, 128);
CHECK(2 == allocator.get_arena_count());
allocator.rewind();
start = acl::allocate<std::uint8_t>(allocator, 64, 128);
CHECK(start == first);
CHECK(2 == allocator.get_arena_count());
allocator.smart_rewind();
start = acl::allocate<std::uint8_t>(allocator, 64, 128);
CHECK(start == first);
CHECK(1 == allocator.get_arena_count());
}
TEST_CASE("Validate linear_stack_allocator with alignment", "[linear_stack_allocator]")
{
using namespace acl;
using allocator_t = linear_stack_allocator<>;
struct record
{
void* data;
std::uint32_t size;
};
constexpr std::uint32_t k_arena_size = 1;
allocator_t allocator(64);
std::uint8_t* first = nullptr;
{
auto ar = allocator.get_auto_rewind_point();
auto r1 = allocator.get_rewind_point();
auto a1 = acl::allocate<std::uint8_t>(allocator, 32, 0);
first = a1;
allocator.rewind(r1);
auto a2 = acl::allocate<std::uint8_t>(allocator, 32, 0);
CHECK(a1 == a2);
allocator.rewind(r1);
a1 = acl::allocate<std::uint8_t>(allocator, 32, 0);
a2 = acl::allocate<std::uint8_t>(allocator, 32, 0);
auto a3 = acl::allocate<std::uint8_t>(allocator, 16, 0);
auto r2 = allocator.get_rewind_point();
auto a4 = acl::allocate<std::uint8_t>(allocator, 32, 0);
auto a6 = acl::allocate<std::uint8_t>(allocator, 32, 0);
allocator.rewind(r2);
auto a7 = acl::allocate<std::uint8_t>(allocator, 32, 0);
CHECK(a4 == a7);
auto r3 = allocator.get_rewind_point();
a7 = acl::allocate<std::uint8_t>(allocator, 2, 0);
auto a8 = acl::allocate<std::uint8_t>(allocator, 128, 0);
auto a9 = acl::allocate<std::uint8_t>(allocator, 32, 0);
auto a10 = acl::allocate<std::uint8_t>(allocator, 64, 0);
allocator.rewind(r3);
auto a11 = acl::allocate<std::uint8_t>(allocator, 16, 0);
CHECK(a7 == a11);
}
auto a1 = acl::allocate<std::uint8_t>(allocator, 32, 0);
CHECK(a1 == first);
}
// NOLINTEND