| 
									
										
										
										
											2022-02-11 09:07:11 +11:00
										 |  |  | /* SPDX-License-Identifier: GPL-2.0-or-later */ | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup bli | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * A linear allocator is the simplest form of an allocator. It never reuses any memory, and | 
					
						
							|  |  |  |  * therefore does not need a deallocation method. It simply hands out consecutive buffers of | 
					
						
							|  |  |  |  * memory. When the current buffer is full, it reallocates a new larger buffer and continues. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-07 09:50:34 +02:00
										 |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_string_ref.hh"
 | 
					
						
							|  |  |  | #include "BLI_utility_mixins.hh"
 | 
					
						
							|  |  |  | #include "BLI_vector.hh"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  | namespace blender { | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | template<typename Allocator = GuardedAllocator> class LinearAllocator : NonCopyable, NonMovable { | 
					
						
							|  |  |  |  private: | 
					
						
							| 
									
										
										
										
											2022-05-25 16:28:07 +02:00
										 |  |  |   BLI_NO_UNIQUE_ADDRESS Allocator allocator_; | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |   Vector<void *> owned_buffers_; | 
					
						
							|  |  |  |   Vector<Span<char>> unused_borrowed_buffers_; | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |   uintptr_t current_begin_; | 
					
						
							|  |  |  |   uintptr_t current_end_; | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   int64_t debug_allocated_amount_ = 0; | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-07 14:15:20 +01:00
										 |  |  |   /* Buffers larger than that are not packed together with smaller allocations to avoid wasting
 | 
					
						
							|  |  |  |    * memory. */ | 
					
						
							|  |  |  |   constexpr static inline int64_t large_buffer_threshold = 4096; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |  public: | 
					
						
							|  |  |  |   LinearAllocator() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     current_begin_ = 0; | 
					
						
							|  |  |  |     current_end_ = 0; | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ~LinearAllocator() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     for (void *ptr : owned_buffers_) { | 
					
						
							|  |  |  |       allocator_.deallocate(ptr); | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Get a pointer to a memory buffer with the given size an alignment. The memory buffer will be | 
					
						
							|  |  |  |    * freed when this LinearAllocator is destructed. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * The alignment has to be a power of 2. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   void *allocate(const int64_t size, const int64_t alignment) | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     BLI_assert(size >= 0); | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |     BLI_assert(alignment >= 1); | 
					
						
							|  |  |  |     BLI_assert(is_power_of_2_i(alignment)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:52:51 +02:00
										 |  |  |     const uintptr_t alignment_mask = alignment - 1; | 
					
						
							|  |  |  |     const uintptr_t potential_allocation_begin = (current_begin_ + alignment_mask) & | 
					
						
							|  |  |  |                                                  ~alignment_mask; | 
					
						
							|  |  |  |     const uintptr_t potential_allocation_end = potential_allocation_begin + size; | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     if (potential_allocation_end <= current_end_) { | 
					
						
							| 
									
										
										
										
											2021-03-07 14:15:20 +01:00
										 |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  |       debug_allocated_amount_ += size; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |       current_begin_ = potential_allocation_end; | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |       return reinterpret_cast<void *>(potential_allocation_begin); | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-07 14:15:20 +01:00
										 |  |  |     if (size <= large_buffer_threshold) { | 
					
						
							|  |  |  |       this->allocate_new_buffer(size + alignment, alignment); | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |       return this->allocate(size, alignment); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-07 14:15:20 +01:00
										 |  |  |     return this->allocator_large_buffer(size, alignment); | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Allocate a memory buffer that can hold an instance of T. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * This method only allocates memory and does not construct the instance. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   template<typename T> T *allocate() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return static_cast<T *>(this->allocate(sizeof(T), alignof(T))); | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Allocate a memory buffer that can hold T array with the given size. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * This method only allocates memory and does not construct the instance. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   template<typename T> MutableSpan<T> allocate_array(int64_t size) | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     T *array = static_cast<T *>(this->allocate(sizeof(T) * size, alignof(T))); | 
					
						
							|  |  |  |     return MutableSpan<T>(array, size); | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Construct an instance of T in memory provided by this allocator. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Arguments passed to this method will be forwarded to the constructor of T. | 
					
						
							|  |  |  |    * | 
					
						
							| 
									
										
										
										
											2021-03-07 14:24:52 +01:00
										 |  |  |    * You must not call `delete` on the returned value. | 
					
						
							|  |  |  |    * Instead, only the destructor has to be called. | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2021-07-30 16:19:19 +10:00
										 |  |  |   template<typename T, typename... Args> destruct_ptr<T> construct(Args &&...args) | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     void *buffer = this->allocate(sizeof(T), alignof(T)); | 
					
						
							|  |  |  |     T *value = new (buffer) T(std::forward<Args>(args)...); | 
					
						
							| 
									
										
										
										
											2021-03-07 14:24:52 +01:00
										 |  |  |     return destruct_ptr<T>(value); | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-13 12:58:02 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Construct multiple instances of a type in an array. The constructor of is called with the | 
					
						
							|  |  |  |    * given arguments. The caller is responsible for calling the destructor (and not `delete`) on | 
					
						
							|  |  |  |    * the constructed elements. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   template<typename T, typename... Args> | 
					
						
							| 
									
										
										
										
											2021-07-30 16:19:19 +10:00
										 |  |  |   MutableSpan<T> construct_array(int64_t size, Args &&...args) | 
					
						
							| 
									
										
										
										
											2021-05-13 12:58:02 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     MutableSpan<T> array = this->allocate_array<T>(size); | 
					
						
							|  |  |  |     for (const int64_t i : IndexRange(size)) { | 
					
						
							|  |  |  |       new (&array[i]) T(std::forward<Args>(args)...); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return array; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Copy the given array into a memory buffer provided by this allocator. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   template<typename T> MutableSpan<T> construct_array_copy(Span<T> src) | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-04-01 12:25:30 +02:00
										 |  |  |     if (src.is_empty()) { | 
					
						
							|  |  |  |       return {}; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |     MutableSpan<T> dst = this->allocate_array<T>(src.size()); | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |     uninitialized_copy_n(src.data(), src.size(), dst.data()); | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |     return dst; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Copy the given string into a memory buffer provided by this allocator. The returned string is | 
					
						
							|  |  |  |    * always null terminated. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   StringRefNull copy_string(StringRef str) | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     const int64_t alloc_size = str.size() + 1; | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     char *buffer = static_cast<char *>(this->allocate(alloc_size, 1)); | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |     str.copy(buffer, alloc_size); | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return StringRefNull(static_cast<const char *>(buffer)); | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   MutableSpan<void *> allocate_elements_and_pointer_array(int64_t element_amount, | 
					
						
							|  |  |  |                                                           int64_t element_size, | 
					
						
							|  |  |  |                                                           int64_t element_alignment) | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |     void *pointer_buffer = this->allocate(element_amount * sizeof(void *), alignof(void *)); | 
					
						
							|  |  |  |     void *elements_buffer = this->allocate(element_amount * element_size, element_alignment); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |     MutableSpan<void *> pointers((void **)pointer_buffer, element_amount); | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |     void *next_element_buffer = elements_buffer; | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     for (int64_t i : IndexRange(element_amount)) { | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |       pointers[i] = next_element_buffer; | 
					
						
							|  |  |  |       next_element_buffer = POINTER_OFFSET(next_element_buffer, element_size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return pointers; | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   template<typename T, typename... Args> | 
					
						
							| 
									
										
										
										
											2021-07-30 16:19:19 +10:00
										 |  |  |   Span<T *> construct_elements_and_pointer_array(int64_t n, Args &&...args) | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |     MutableSpan<void *> void_pointers = this->allocate_elements_and_pointer_array( | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |         n, sizeof(T), alignof(T)); | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |     MutableSpan<T *> pointers = void_pointers.cast<T *>(); | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     for (int64_t i : IndexRange(n)) { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |       new (static_cast<void *>(pointers[i])) T(std::forward<Args>(args)...); | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return pointers; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Tell the allocator to use up the given memory buffer, before allocating new memory from the | 
					
						
							|  |  |  |    * system. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   void provide_buffer(void *buffer, uint size) | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     unused_borrowed_buffers_.append(Span<char>(static_cast<char *>(buffer), size)); | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   template<size_t Size, size_t Alignment> | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   void provide_buffer(AlignedBuffer<Size, Alignment> &aligned_buffer) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     this->provide_buffer(aligned_buffer.ptr(), Size); | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  private: | 
					
						
							| 
									
										
										
										
											2021-03-07 14:15:20 +01:00
										 |  |  |   void allocate_new_buffer(int64_t min_allocation_size, int64_t min_alignment) | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     for (int64_t i : unused_borrowed_buffers_.index_range()) { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |       Span<char> buffer = unused_borrowed_buffers_[i]; | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |       if (buffer.size() >= min_allocation_size) { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |         unused_borrowed_buffers_.remove_and_reorder(i); | 
					
						
							|  |  |  |         current_begin_ = (uintptr_t)buffer.begin(); | 
					
						
							|  |  |  |         current_end_ = (uintptr_t)buffer.end(); | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-07 14:15:20 +01:00
										 |  |  |     /* Possibly allocate more bytes than necessary for the current allocation. This way more small
 | 
					
						
							|  |  |  |      * allocations can be packed together. Large buffers are allocated exactly to avoid wasting too | 
					
						
							|  |  |  |      * much memory. */ | 
					
						
							|  |  |  |     int64_t size_in_bytes = min_allocation_size; | 
					
						
							|  |  |  |     if (size_in_bytes <= large_buffer_threshold) { | 
					
						
							|  |  |  |       /* Gradually grow buffer size with each allocation, up to a maximum. */ | 
					
						
							| 
									
										
										
										
											2021-03-07 14:27:08 +01:00
										 |  |  |       const int grow_size = 1 << std::min<int>(owned_buffers_.size() + 6, 20); | 
					
						
							|  |  |  |       size_in_bytes = std::min(large_buffer_threshold, | 
					
						
							|  |  |  |                                std::max<int64_t>(size_in_bytes, grow_size)); | 
					
						
							| 
									
										
										
										
											2021-03-07 14:15:20 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-07 14:15:20 +01:00
										 |  |  |     void *buffer = allocator_.allocate(size_in_bytes, min_alignment, __func__); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     owned_buffers_.append(buffer); | 
					
						
							|  |  |  |     current_begin_ = (uintptr_t)buffer; | 
					
						
							|  |  |  |     current_end_ = current_begin_ + size_in_bytes; | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-03-07 14:15:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   void *allocator_large_buffer(const int64_t size, const int64_t alignment) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     void *buffer = allocator_.allocate(size, alignment, __func__); | 
					
						
							|  |  |  |     owned_buffers_.append(buffer); | 
					
						
							|  |  |  |     return buffer; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-04-24 23:52:55 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  | }  // namespace blender
 |