-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpool.hh
122 lines (100 loc) · 2.87 KB
/
pool.hh
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
#ifndef UTIL_POOL_H
#define UTIL_POOL_H
#include <cassert>
#include <cstring>
#include <vector>
#include <stdint.h>
namespace util {
/* Very simple pool. It can only allocate memory. And all of the memory it
* allocates must be freed at the same time.
*/
class Pool {
public:
Pool();
~Pool();
void *Allocate(std::size_t size) {
void *ret = current_;
current_ += size;
if (current_ > current_end_) {
ret = More(size);
}
#ifdef DEBUG
base_check_ = ret;
#endif
return ret;
}
/** Extend (or contract) the most recent allocation.
* @param base The base pointer of the allocation. This must must have been
* returned by the MOST RECENT call to Allocate or Continue.
* @param additional Change in the size.
*
* In most cases, more memory from the same page is used, in which case
* base is unchanged and the function returns false.
* If the page runs out, a new page is created and the memory (from base)
* is copied. The function returns true.
*
* @return Whether the base had to be changed due to allocating a page.
*/
bool Continue(void *&base, std::ptrdiff_t additional) {
#ifdef DEBUG
assert(base == base_check_);
#endif
current_ += additional;
if (current_ > current_end_) {
std::size_t new_total = current_ - static_cast<uint8_t*>(base);
void *new_base = More(new_total);
std::memcpy(new_base, base, new_total - additional);
base = new_base;
#ifdef DEBUG
base_check_ = base;
#endif
return true;
}
return false;
}
void FreeAll();
private:
void *More(std::size_t size);
std::vector<void *> free_list_;
uint8_t *current_, *current_end_;
#ifdef DEBUG
// For debugging, check that Continue came from the most recent call.
void *base_check_;
#endif // DEBUG
// no copying
Pool(const Pool &);
Pool &operator=(const Pool &);
};
/**
* Pool designed to allow limited freeing.
* Keeps a linked list of free elements in the free spaces.
* Will not reduce in size until FreeAll is called.
*/
class FreePool {
public:
explicit FreePool(std::size_t element_size)
: free_list_(NULL),
element_size_(element_size),
padded_size_(std::max(element_size_, sizeof(void*))) {}
void *Allocate() {
if (free_list_) {
void *ret = free_list_;
free_list_ = *reinterpret_cast<void**>(free_list_);
return ret;
} else {
return backing_.Allocate(padded_size_);
}
}
void Free(void *ptr) {
*reinterpret_cast<void**>(ptr) = free_list_;
free_list_ = ptr;
}
std::size_t ElementSize() const { return element_size_; }
private:
void *free_list_;
Pool backing_;
const std::size_t element_size_;
const std::size_t padded_size_;
};
} // namespace util
#endif // UTIL_POOL_H