1 | //===----------------------------------------------------------------------===// |
2 | // |
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | // See https://llvm.org/LICENSE.txt for license information. |
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | // |
7 | //===----------------------------------------------------------------------===// |
8 | |
9 | #include <memory> |
10 | #include <memory_resource> |
11 | |
12 | #ifndef _LIBCPP_HAS_NO_ATOMIC_HEADER |
13 | # include <atomic> |
14 | #elif !defined(_LIBCPP_HAS_NO_THREADS) |
15 | # include <mutex> |
16 | # if defined(__ELF__) && defined(_LIBCPP_LINK_PTHREAD_LIB) |
17 | # pragma comment(lib, "pthread") |
18 | # endif |
19 | #endif |
20 | |
21 | _LIBCPP_BEGIN_NAMESPACE_STD |
22 | |
23 | namespace pmr { |
24 | |
25 | // memory_resource |
26 | |
27 | memory_resource::~memory_resource() = default; |
28 | |
29 | // new_delete_resource() |
30 | |
31 | #ifdef _LIBCPP_HAS_NO_ALIGNED_ALLOCATION |
32 | static bool is_aligned_to(void* ptr, size_t align) { |
33 | void* p2 = ptr; |
34 | size_t space = 1; |
35 | void* result = std::align(align, 1, p2, space); |
36 | return (result == ptr); |
37 | } |
38 | #endif |
39 | |
40 | class _LIBCPP_EXPORTED_FROM_ABI __new_delete_memory_resource_imp : public memory_resource { |
41 | void* do_allocate(size_t bytes, size_t align) override { |
42 | #ifndef _LIBCPP_HAS_NO_ALIGNED_ALLOCATION |
43 | return std::__libcpp_allocate(size: bytes, align: align); |
44 | #else |
45 | if (bytes == 0) |
46 | bytes = 1; |
47 | void* result = std::__libcpp_allocate(bytes, align); |
48 | if (!is_aligned_to(result, align)) { |
49 | std::__libcpp_deallocate(result, bytes, align); |
50 | __throw_bad_alloc(); |
51 | } |
52 | return result; |
53 | #endif |
54 | } |
55 | |
56 | void do_deallocate(void* p, size_t bytes, size_t align) override { std::__libcpp_deallocate(ptr: p, size: bytes, align: align); } |
57 | |
58 | bool do_is_equal(const memory_resource& other) const noexcept override { return &other == this; } |
59 | }; |
60 | |
61 | // null_memory_resource() |
62 | |
63 | class _LIBCPP_EXPORTED_FROM_ABI __null_memory_resource_imp : public memory_resource { |
64 | void* do_allocate(size_t, size_t) override { __throw_bad_alloc(); } |
65 | void do_deallocate(void*, size_t, size_t) override {} |
66 | bool do_is_equal(const memory_resource& other) const noexcept override { return &other == this; } |
67 | }; |
68 | |
69 | namespace { |
70 | |
71 | union ResourceInitHelper { |
72 | struct { |
73 | __new_delete_memory_resource_imp new_delete_res; |
74 | __null_memory_resource_imp null_res; |
75 | } resources; |
76 | char dummy; |
77 | constexpr ResourceInitHelper() : resources() {} |
78 | ~ResourceInitHelper() {} |
79 | }; |
80 | |
81 | // Pretend we're inside a system header so the compiler doesn't flag the use of the init_priority |
82 | // attribute with a value that's reserved for the implementation (we're the implementation). |
83 | #include "memory_resource_init_helper.h" |
84 | |
85 | } // end namespace |
86 | |
87 | memory_resource* new_delete_resource() noexcept { return &res_init.resources.new_delete_res; } |
88 | |
89 | memory_resource* null_memory_resource() noexcept { return &res_init.resources.null_res; } |
90 | |
91 | // default_memory_resource() |
92 | |
93 | static memory_resource* __default_memory_resource(bool set = false, memory_resource* new_res = nullptr) noexcept { |
94 | #ifndef _LIBCPP_HAS_NO_ATOMIC_HEADER |
95 | static constinit atomic<memory_resource*> __res{&res_init.resources.new_delete_res}; |
96 | if (set) { |
97 | new_res = new_res ? new_res : new_delete_resource(); |
98 | // TODO: Can a weaker ordering be used? |
99 | return std::atomic_exchange_explicit(o: &__res, d: new_res, m: memory_order_acq_rel); |
100 | } else { |
101 | return std::atomic_load_explicit(o: &__res, m: memory_order_acquire); |
102 | } |
103 | #elif !defined(_LIBCPP_HAS_NO_THREADS) |
104 | static constinit memory_resource* res = &res_init.resources.new_delete_res; |
105 | static mutex res_lock; |
106 | if (set) { |
107 | new_res = new_res ? new_res : new_delete_resource(); |
108 | lock_guard<mutex> guard(res_lock); |
109 | memory_resource* old_res = res; |
110 | res = new_res; |
111 | return old_res; |
112 | } else { |
113 | lock_guard<mutex> guard(res_lock); |
114 | return res; |
115 | } |
116 | #else |
117 | static constinit memory_resource* res = &res_init.resources.new_delete_res; |
118 | if (set) { |
119 | new_res = new_res ? new_res : new_delete_resource(); |
120 | memory_resource* old_res = res; |
121 | res = new_res; |
122 | return old_res; |
123 | } else { |
124 | return res; |
125 | } |
126 | #endif |
127 | } |
128 | |
129 | memory_resource* get_default_resource() noexcept { return __default_memory_resource(); } |
130 | |
131 | memory_resource* set_default_resource(memory_resource* __new_res) noexcept { |
132 | return __default_memory_resource(set: true, new_res: __new_res); |
133 | } |
134 | |
135 | // 23.12.5, mem.res.pool |
136 | |
137 | static size_t roundup(size_t count, size_t alignment) { |
138 | size_t mask = alignment - 1; |
139 | return (count + mask) & ~mask; |
140 | } |
141 | |
142 | struct unsynchronized_pool_resource::__adhoc_pool:: { |
143 | __chunk_footer* ; |
144 | char* ; |
145 | size_t ; |
146 | size_t () { return (reinterpret_cast<char*>(this) - __start_) + sizeof(*this); } |
147 | }; |
148 | |
149 | void unsynchronized_pool_resource::__adhoc_pool::__release_ptr(memory_resource* upstream) { |
150 | while (__first_ != nullptr) { |
151 | __chunk_footer* next = __first_->__next_; |
152 | upstream->deallocate(p: __first_->__start_, bytes: __first_->__allocation_size(), align: __first_->__align_); |
153 | __first_ = next; |
154 | } |
155 | } |
156 | |
157 | void* unsynchronized_pool_resource::__adhoc_pool::__do_allocate(memory_resource* upstream, size_t bytes, size_t align) { |
158 | const size_t = sizeof(__chunk_footer); |
159 | const size_t = alignof(__chunk_footer); |
160 | |
161 | if (align < footer_align) |
162 | align = footer_align; |
163 | |
164 | size_t aligned_capacity = roundup(count: bytes, alignment: footer_align) + footer_size; |
165 | |
166 | void* result = upstream->allocate(bytes: aligned_capacity, align: align); |
167 | |
168 | __chunk_footer* h = (__chunk_footer*)((char*)result + aligned_capacity - footer_size); |
169 | h->__next_ = __first_; |
170 | h->__start_ = (char*)result; |
171 | h->__align_ = align; |
172 | __first_ = h; |
173 | return result; |
174 | } |
175 | |
176 | void unsynchronized_pool_resource::__adhoc_pool::__do_deallocate( |
177 | memory_resource* upstream, void* p, size_t bytes, size_t align) { |
178 | _LIBCPP_ASSERT_NON_NULL(__first_ != nullptr, "deallocating a block that was not allocated with this allocator" ); |
179 | if (__first_->__start_ == p) { |
180 | __chunk_footer* next = __first_->__next_; |
181 | upstream->deallocate(p: p, bytes: __first_->__allocation_size(), align: __first_->__align_); |
182 | __first_ = next; |
183 | } else { |
184 | for (__chunk_footer* h = __first_; h->__next_ != nullptr; h = h->__next_) { |
185 | if (h->__next_->__start_ == p) { |
186 | __chunk_footer* next = h->__next_->__next_; |
187 | upstream->deallocate(p: p, bytes: h->__next_->__allocation_size(), align: h->__next_->__align_); |
188 | h->__next_ = next; |
189 | return; |
190 | } |
191 | } |
192 | // The request to deallocate memory ends up being a no-op, likely resulting in a memory leak. |
193 | _LIBCPP_ASSERT_VALID_DEALLOCATION(false, "deallocating a block that was not allocated with this allocator" ); |
194 | } |
195 | } |
196 | |
197 | class unsynchronized_pool_resource::__fixed_pool { |
198 | struct { |
199 | __chunk_footer* ; |
200 | char* ; |
201 | size_t ; |
202 | size_t () { return (reinterpret_cast<char*>(this) - __start_) + sizeof(*this); } |
203 | }; |
204 | |
205 | struct { |
206 | __vacancy_header* ; |
207 | }; |
208 | |
209 | __chunk_footer* __first_chunk_ = nullptr; |
210 | __vacancy_header* __first_vacancy_ = nullptr; |
211 | |
212 | public: |
213 | explicit __fixed_pool() = default; |
214 | |
215 | void __release_ptr(memory_resource* upstream) { |
216 | __first_vacancy_ = nullptr; |
217 | while (__first_chunk_ != nullptr) { |
218 | __chunk_footer* next = __first_chunk_->__next_; |
219 | upstream->deallocate(p: __first_chunk_->__start_, bytes: __first_chunk_->__allocation_size(), align: __first_chunk_->__align_); |
220 | __first_chunk_ = next; |
221 | } |
222 | } |
223 | |
224 | void* __try_allocate_from_vacancies() { |
225 | if (__first_vacancy_ != nullptr) { |
226 | void* result = __first_vacancy_; |
227 | __first_vacancy_ = __first_vacancy_->__next_vacancy_; |
228 | return result; |
229 | } |
230 | return nullptr; |
231 | } |
232 | |
233 | void* __allocate_in_new_chunk(memory_resource* upstream, size_t block_size, size_t chunk_size) { |
234 | _LIBCPP_ASSERT_INTERNAL(chunk_size % block_size == 0, "" ); |
235 | static_assert(__default_alignment >= alignof(std::max_align_t), "" ); |
236 | static_assert(__default_alignment >= alignof(__chunk_footer), "" ); |
237 | static_assert(__default_alignment >= alignof(__vacancy_header), "" ); |
238 | |
239 | const size_t = sizeof(__chunk_footer); |
240 | const size_t = alignof(__chunk_footer); |
241 | |
242 | size_t aligned_capacity = roundup(count: chunk_size, alignment: footer_align) + footer_size; |
243 | |
244 | void* result = upstream->allocate(bytes: aligned_capacity, align: __default_alignment); |
245 | |
246 | __chunk_footer* h = (__chunk_footer*)((char*)result + aligned_capacity - footer_size); |
247 | h->__next_ = __first_chunk_; |
248 | h->__start_ = (char*)result; |
249 | h->__align_ = __default_alignment; |
250 | __first_chunk_ = h; |
251 | |
252 | if (chunk_size > block_size) { |
253 | __vacancy_header* last_vh = this->__first_vacancy_; |
254 | for (size_t i = block_size; i != chunk_size; i += block_size) { |
255 | __vacancy_header* vh = (__vacancy_header*)((char*)result + i); |
256 | vh->__next_vacancy_ = last_vh; |
257 | last_vh = vh; |
258 | } |
259 | this->__first_vacancy_ = last_vh; |
260 | } |
261 | return result; |
262 | } |
263 | |
264 | void __evacuate(void* p) { |
265 | __vacancy_header* vh = (__vacancy_header*)(p); |
266 | vh->__next_vacancy_ = __first_vacancy_; |
267 | __first_vacancy_ = vh; |
268 | } |
269 | |
270 | size_t __previous_chunk_size_in_bytes() const { return __first_chunk_ ? __first_chunk_->__allocation_size() : 0; } |
271 | |
272 | static const size_t __default_alignment = alignof(max_align_t); |
273 | }; |
274 | |
275 | size_t unsynchronized_pool_resource::__pool_block_size(int i) const { return size_t(1) << __log2_pool_block_size(i: i); } |
276 | |
277 | int unsynchronized_pool_resource::__log2_pool_block_size(int i) const { return (i + __log2_smallest_block_size); } |
278 | |
279 | int unsynchronized_pool_resource::__pool_index(size_t bytes, size_t align) const { |
280 | if (align > alignof(std::max_align_t) || bytes > (size_t(1) << __num_fixed_pools_)) |
281 | return __num_fixed_pools_; |
282 | else { |
283 | int i = 0; |
284 | bytes = (bytes > align) ? bytes : align; |
285 | bytes -= 1; |
286 | bytes >>= __log2_smallest_block_size; |
287 | while (bytes != 0) { |
288 | bytes >>= 1; |
289 | i += 1; |
290 | } |
291 | return i; |
292 | } |
293 | } |
294 | |
295 | unsynchronized_pool_resource::unsynchronized_pool_resource(const pool_options& opts, memory_resource* upstream) |
296 | : __res_(upstream), __fixed_pools_(nullptr) { |
297 | size_t largest_block_size; |
298 | if (opts.largest_required_pool_block == 0) |
299 | largest_block_size = __default_largest_block_size; |
300 | else if (opts.largest_required_pool_block < __smallest_block_size) |
301 | largest_block_size = __smallest_block_size; |
302 | else if (opts.largest_required_pool_block > __max_largest_block_size) |
303 | largest_block_size = __max_largest_block_size; |
304 | else |
305 | largest_block_size = opts.largest_required_pool_block; |
306 | |
307 | if (opts.max_blocks_per_chunk == 0) |
308 | __options_max_blocks_per_chunk_ = __max_blocks_per_chunk; |
309 | else if (opts.max_blocks_per_chunk < __min_blocks_per_chunk) |
310 | __options_max_blocks_per_chunk_ = __min_blocks_per_chunk; |
311 | else if (opts.max_blocks_per_chunk > __max_blocks_per_chunk) |
312 | __options_max_blocks_per_chunk_ = __max_blocks_per_chunk; |
313 | else |
314 | __options_max_blocks_per_chunk_ = opts.max_blocks_per_chunk; |
315 | |
316 | __num_fixed_pools_ = 1; |
317 | size_t capacity = __smallest_block_size; |
318 | while (capacity < largest_block_size) { |
319 | capacity <<= 1; |
320 | __num_fixed_pools_ += 1; |
321 | } |
322 | } |
323 | |
324 | pool_options unsynchronized_pool_resource::options() const { |
325 | pool_options p; |
326 | p.max_blocks_per_chunk = __options_max_blocks_per_chunk_; |
327 | p.largest_required_pool_block = __pool_block_size(i: __num_fixed_pools_ - 1); |
328 | return p; |
329 | } |
330 | |
331 | void unsynchronized_pool_resource::release() { |
332 | __adhoc_pool_.__release_ptr(upstream: __res_); |
333 | if (__fixed_pools_ != nullptr) { |
334 | const int n = __num_fixed_pools_; |
335 | for (int i = 0; i < n; ++i) |
336 | __fixed_pools_[i].__release_ptr(upstream: __res_); |
337 | __res_->deallocate(p: __fixed_pools_, bytes: __num_fixed_pools_ * sizeof(__fixed_pool), align: alignof(__fixed_pool)); |
338 | __fixed_pools_ = nullptr; |
339 | } |
340 | } |
341 | |
342 | void* unsynchronized_pool_resource::do_allocate(size_t bytes, size_t align) { |
343 | // A pointer to allocated storage (6.6.4.4.1) with a size of at least bytes. |
344 | // The size and alignment of the allocated memory shall meet the requirements for |
345 | // a class derived from memory_resource (23.12). |
346 | // If the pool selected for a block of size bytes is unable to satisfy the memory request |
347 | // from its own internal data structures, it will call upstream_resource()->allocate() |
348 | // to obtain more memory. If bytes is larger than that which the largest pool can handle, |
349 | // then memory will be allocated using upstream_resource()->allocate(). |
350 | |
351 | int i = __pool_index(bytes, align); |
352 | if (i == __num_fixed_pools_) |
353 | return __adhoc_pool_.__do_allocate(upstream: __res_, bytes, align); |
354 | else { |
355 | if (__fixed_pools_ == nullptr) { |
356 | __fixed_pools_ = |
357 | (__fixed_pool*)__res_->allocate(bytes: __num_fixed_pools_ * sizeof(__fixed_pool), align: alignof(__fixed_pool)); |
358 | __fixed_pool* first = __fixed_pools_; |
359 | __fixed_pool* last = __fixed_pools_ + __num_fixed_pools_; |
360 | for (__fixed_pool* pool = first; pool != last; ++pool) |
361 | ::new ((void*)pool) __fixed_pool; |
362 | } |
363 | void* result = __fixed_pools_[i].__try_allocate_from_vacancies(); |
364 | if (result == nullptr) { |
365 | auto min = [](size_t a, size_t b) { return a < b ? a : b; }; |
366 | auto max = [](size_t a, size_t b) { return a < b ? b : a; }; |
367 | |
368 | size_t prev_chunk_size_in_bytes = __fixed_pools_[i].__previous_chunk_size_in_bytes(); |
369 | size_t prev_chunk_size_in_blocks = prev_chunk_size_in_bytes >> __log2_pool_block_size(i); |
370 | |
371 | size_t chunk_size_in_blocks; |
372 | |
373 | if (prev_chunk_size_in_blocks == 0) { |
374 | size_t min_blocks_per_chunk = max(__min_bytes_per_chunk >> __log2_pool_block_size(i), __min_blocks_per_chunk); |
375 | chunk_size_in_blocks = min_blocks_per_chunk; |
376 | } else { |
377 | static_assert(__max_bytes_per_chunk <= SIZE_MAX - (__max_bytes_per_chunk / 4), "unsigned overflow is possible" ); |
378 | chunk_size_in_blocks = prev_chunk_size_in_blocks + (prev_chunk_size_in_blocks / 4); |
379 | } |
380 | |
381 | size_t max_blocks_per_chunk = |
382 | min((__max_bytes_per_chunk >> __log2_pool_block_size(i)), |
383 | min(__max_blocks_per_chunk, __options_max_blocks_per_chunk_)); |
384 | if (chunk_size_in_blocks > max_blocks_per_chunk) |
385 | chunk_size_in_blocks = max_blocks_per_chunk; |
386 | |
387 | size_t block_size = __pool_block_size(i); |
388 | |
389 | size_t chunk_size_in_bytes = (chunk_size_in_blocks << __log2_pool_block_size(i)); |
390 | result = __fixed_pools_[i].__allocate_in_new_chunk(upstream: __res_, block_size, chunk_size: chunk_size_in_bytes); |
391 | } |
392 | return result; |
393 | } |
394 | } |
395 | |
396 | void unsynchronized_pool_resource::do_deallocate(void* p, size_t bytes, size_t align) { |
397 | // Returns the memory at p to the pool. It is unspecified if, |
398 | // or under what circumstances, this operation will result in |
399 | // a call to upstream_resource()->deallocate(). |
400 | |
401 | int i = __pool_index(bytes, align); |
402 | if (i == __num_fixed_pools_) |
403 | return __adhoc_pool_.__do_deallocate(upstream: __res_, p, bytes, align); |
404 | else { |
405 | _LIBCPP_ASSERT_NON_NULL( |
406 | __fixed_pools_ != nullptr, "deallocating a block that was not allocated with this allocator" ); |
407 | __fixed_pools_[i].__evacuate(p); |
408 | } |
409 | } |
410 | |
411 | bool synchronized_pool_resource::do_is_equal(const memory_resource& other) const noexcept { return &other == this; } |
412 | |
413 | // 23.12.6, mem.res.monotonic.buffer |
414 | |
415 | static void* align_down(size_t align, size_t size, void*& ptr, size_t& space) { |
416 | if (size > space) |
417 | return nullptr; |
418 | |
419 | char* p1 = static_cast<char*>(ptr); |
420 | char* new_ptr = reinterpret_cast<char*>(reinterpret_cast<uintptr_t>(p1 - size) & ~(align - 1)); |
421 | |
422 | if (new_ptr < (p1 - space)) |
423 | return nullptr; |
424 | |
425 | ptr = new_ptr; |
426 | space -= p1 - new_ptr; |
427 | |
428 | return ptr; |
429 | } |
430 | |
431 | void* monotonic_buffer_resource::__initial_descriptor::__try_allocate_from_chunk(size_t bytes, size_t align) { |
432 | if (!__cur_) |
433 | return nullptr; |
434 | void* new_ptr = static_cast<void*>(__cur_); |
435 | size_t new_capacity = (__cur_ - __start_); |
436 | void* aligned_ptr = align_down(align, size: bytes, ptr&: new_ptr, space&: new_capacity); |
437 | if (aligned_ptr != nullptr) |
438 | __cur_ = static_cast<char*>(new_ptr); |
439 | return aligned_ptr; |
440 | } |
441 | |
442 | void* monotonic_buffer_resource::__chunk_footer::(size_t bytes, size_t align) { |
443 | void* new_ptr = static_cast<void*>(__cur_); |
444 | size_t new_capacity = (__cur_ - __start_); |
445 | void* aligned_ptr = align_down(align, size: bytes, ptr&: new_ptr, space&: new_capacity); |
446 | if (aligned_ptr != nullptr) |
447 | __cur_ = static_cast<char*>(new_ptr); |
448 | return aligned_ptr; |
449 | } |
450 | |
451 | void* monotonic_buffer_resource::do_allocate(size_t bytes, size_t align) { |
452 | const size_t = sizeof(__chunk_footer); |
453 | const size_t = alignof(__chunk_footer); |
454 | |
455 | auto previous_allocation_size = [&]() { |
456 | if (__chunks_ != nullptr) |
457 | return __chunks_->__allocation_size(); |
458 | |
459 | size_t newsize = (__initial_.__start_ != nullptr) ? (__initial_.__end_ - __initial_.__start_) : __initial_.__size_; |
460 | |
461 | return roundup(count: newsize, alignment: footer_align) + footer_size; |
462 | }; |
463 | |
464 | if (void* result = __initial_.__try_allocate_from_chunk(bytes, align)) |
465 | return result; |
466 | if (__chunks_ != nullptr) { |
467 | if (void* result = __chunks_->__try_allocate_from_chunk(bytes, align)) |
468 | return result; |
469 | } |
470 | |
471 | // Allocate a brand-new chunk. |
472 | |
473 | if (align < footer_align) |
474 | align = footer_align; |
475 | |
476 | size_t aligned_capacity = roundup(count: bytes, alignment: footer_align) + footer_size; |
477 | size_t previous_capacity = previous_allocation_size(); |
478 | |
479 | if (aligned_capacity <= previous_capacity) { |
480 | size_t newsize = 2 * (previous_capacity - footer_size); |
481 | aligned_capacity = roundup(count: newsize, alignment: footer_align) + footer_size; |
482 | } |
483 | |
484 | char* start = (char*)__res_->allocate(bytes: aligned_capacity, align: align); |
485 | auto end = start + aligned_capacity - footer_size; |
486 | __chunk_footer* = (__chunk_footer*)(end); |
487 | footer->__next_ = __chunks_; |
488 | footer->__start_ = start; |
489 | footer->__cur_ = end; |
490 | footer->__align_ = align; |
491 | __chunks_ = footer; |
492 | |
493 | return __chunks_->__try_allocate_from_chunk(bytes, align); |
494 | } |
495 | |
496 | } // namespace pmr |
497 | |
498 | _LIBCPP_END_NAMESPACE_STD |
499 | |