| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-07 09:50:34 +02:00
										 |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup bli | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  |  * A `blender::Array<T>` is a container for a fixed size array the size of which is NOT known at | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * compile time. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * If the size is known at compile time, `std::array<T, N>` should be used instead. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  |  * blender::Array should usually be used instead of blender::Vector whenever the number of elements | 
					
						
							|  |  |  |  * is known at construction time. Note however, that blender::Array will default construct all | 
					
						
							|  |  |  |  * elements when initialized with the size-constructor. For trivial types, this does nothing. In | 
					
						
							| 
									
										
										
										
											2020-06-30 15:58:14 +02:00
										 |  |  |  * all other cases, this adds overhead. | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * | 
					
						
							|  |  |  |  * A main benefit of using Array over Vector is that it expresses the intent of the developer | 
					
						
							|  |  |  |  * better. It indicates that the size of the data structure is not expected to change. Furthermore, | 
					
						
							|  |  |  |  * you can be more certain that an array does not overallocate. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  |  * blender::Array supports small object optimization to improve performance when the size turns out | 
					
						
							|  |  |  |  * to be small at run-time. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 17:31:56 +02:00
										 |  |  | #include "BLI_allocator.hh"
 | 
					
						
							|  |  |  | #include "BLI_index_range.hh"
 | 
					
						
							|  |  |  | #include "BLI_memory_utils.hh"
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  | #include "BLI_span.hh"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  | namespace blender { | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | template< | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * The type of the values stored in the array. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     typename T, | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * The number of values that can be stored in the array, without doing a heap allocation. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-07-20 16:00:20 +02:00
										 |  |  |     int64_t InlineBufferCapacity = default_inline_buffer_capacity(sizeof(T)), | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * The allocator used by this array. Should rarely be changed, except when you don't want that | 
					
						
							|  |  |  |      * MEM_* functions are used internally. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     typename Allocator = GuardedAllocator> | 
					
						
							| 
									
										
										
										
											2020-04-23 20:05:53 +02:00
										 |  |  | class Array { | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  private: | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /** The beginning of the array. It might point into the inline buffer. */ | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |   T *data_; | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Number of elements in the array. */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   int64_t size_; | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Used for allocations when the inline buffer is too small. */ | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |   Allocator allocator_; | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** A placeholder buffer that will remain uninitialized until it is used. */ | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   TypedBuffer<T, InlineBufferCapacity> inline_buffer_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * By default an empty array is created. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   Array() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |     data_ = inline_buffer_; | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     size_ = 0; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Create a new array that contains copies of all values. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-08 22:27:25 +02:00
										 |  |  |   template<typename U, typename std::enable_if_t<std::is_convertible_v<U, T>> * = nullptr> | 
					
						
							|  |  |  |   Array(Span<U> values, Allocator allocator = {}) : allocator_(allocator) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     size_ = values.size(); | 
					
						
							|  |  |  |     data_ = this->get_buffer_for_size(values.size()); | 
					
						
							| 
									
										
										
										
											2020-07-08 22:27:25 +02:00
										 |  |  |     uninitialized_convert_n<U, T>(values.data(), size_, data_); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Create a new array that contains copies of all values. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-08 22:27:25 +02:00
										 |  |  |   template<typename U, typename std::enable_if_t<std::is_convertible_v<U, T>> * = nullptr> | 
					
						
							|  |  |  |   Array(const std::initializer_list<U> &values) : Array(Span<U>(values)) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-13 10:51:46 +02:00
										 |  |  |   Array(const std::initializer_list<T> &values) : Array(Span<T>(values)) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Create a new array with the given size. All values will be default constructed. For trivial | 
					
						
							|  |  |  |    * types like int, default construction does nothing. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * We might want another version of this in the future, that does not do default construction | 
					
						
							|  |  |  |    * even for non-trivial types. This should not be the default though, because one can easily mess | 
					
						
							|  |  |  |    * up when dealing with uninitialized memory. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   explicit Array(int64_t size) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     size_ = size; | 
					
						
							|  |  |  |     data_ = this->get_buffer_for_size(size); | 
					
						
							|  |  |  |     default_construct_n(data_, size); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Create a new array with the given size. All values will be initialized by copying the given | 
					
						
							|  |  |  |    * default. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   Array(int64_t size, const T &value) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     BLI_assert(size >= 0); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     size_ = size; | 
					
						
							|  |  |  |     data_ = this->get_buffer_for_size(size); | 
					
						
							|  |  |  |     uninitialized_fill_n(data_, size_, value); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 15:58:14 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Create a new array with uninitialized elements. The caller is responsible for constructing the | 
					
						
							|  |  |  |    * elements. Moving, copying or destructing an Array with uninitialized elements invokes | 
					
						
							|  |  |  |    * undefined behavior. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * This should be used very rarely. Note, that the normal size-constructor also does not | 
					
						
							|  |  |  |    * initialize the elements when T is trivially constructible. Therefore, it only makes sense to | 
					
						
							|  |  |  |    * use this with non trivially constructible types. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Usage: | 
					
						
							|  |  |  |    *  Array<std::string> my_strings(10, NoInitialization()); | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   Array(int64_t size, NoInitialization) | 
					
						
							| 
									
										
										
										
											2020-06-30 15:58:14 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     BLI_assert(size >= 0); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     size_ = size; | 
					
						
							|  |  |  |     data_ = this->get_buffer_for_size(size); | 
					
						
							| 
									
										
										
										
											2020-06-30 15:58:14 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-07 20:32:40 +02:00
										 |  |  |   Array(const Array &other) : Array(other.as_span(), other.allocator_) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |   Array(Array &&other) noexcept : allocator_(other.allocator_) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     size_ = other.size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |     if (!other.uses_inline_buffer()) { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |       data_ = other.data_; | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |       data_ = this->get_buffer_for_size(size_); | 
					
						
							|  |  |  |       uninitialized_relocate_n(other.data_, size_, data_); | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |     other.data_ = other.inline_buffer_; | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     other.size_ = 0; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ~Array() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     destruct_n(data_, size_); | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |     if (!this->uses_inline_buffer()) { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |       allocator_.deallocate(static_cast<void *>(data_)); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Array &operator=(const Array &other) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (this == &other) { | 
					
						
							|  |  |  |       return *this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     this->~Array(); | 
					
						
							|  |  |  |     new (this) Array(other); | 
					
						
							|  |  |  |     return *this; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Array &operator=(Array &&other) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (this == &other) { | 
					
						
							|  |  |  |       return *this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     this->~Array(); | 
					
						
							|  |  |  |     new (this) Array(std::move(other)); | 
					
						
							|  |  |  |     return *this; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   T &operator[](int64_t index) | 
					
						
							| 
									
										
										
										
											2020-07-03 14:31:26 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     BLI_assert(index >= 0); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:31:26 +02:00
										 |  |  |     BLI_assert(index < size_); | 
					
						
							|  |  |  |     return data_[index]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   const T &operator[](int64_t index) const | 
					
						
							| 
									
										
										
										
											2020-07-03 14:31:26 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |     BLI_assert(index >= 0); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:31:26 +02:00
										 |  |  |     BLI_assert(index < size_); | 
					
						
							|  |  |  |     return data_[index]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   operator Span<T>() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return Span<T>(data_, size_); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   operator MutableSpan<T>() | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return MutableSpan<T>(data_, size_); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-08 22:27:25 +02:00
										 |  |  |   template<typename U, typename std::enable_if_t<is_convertible_pointer_v<T, U>> * = nullptr> | 
					
						
							|  |  |  |   operator Span<U>() const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return Span<U>(data_, size_); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   template<typename U, typename std::enable_if_t<is_convertible_pointer_v<T, U>> * = nullptr> | 
					
						
							|  |  |  |   operator MutableSpan<U>() | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return MutableSpan<U>(data_, size_); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   Span<T> as_span() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     return *this; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   MutableSpan<T> as_mutable_span() | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   { | 
					
						
							|  |  |  |     return *this; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns the number of elements in the array. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   int64_t size() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns true when the number of elements in the array is zero. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   bool is_empty() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return size_ == 0; | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 13:02:10 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Copies the given value to every element in the array. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   void fill(const T &value) const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     initialized_fill_n(data_, size_, value); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Get a pointer to the beginning of the array. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   const T *data() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return data_; | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   T *data() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return data_; | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   const T *begin() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return data_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   const T *end() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return data_ + size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   T *begin() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return data_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   T *end() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return data_ + size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-14 13:16:44 +02:00
										 |  |  |   std::reverse_iterator<T *> rbegin() | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return std::reverse_iterator<T *>(this->end()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   std::reverse_iterator<T *> rend() | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return std::reverse_iterator<T *>(this->begin()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   std::reverse_iterator<const T *> rbegin() const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return std::reverse_iterator<T *>(this->end()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   std::reverse_iterator<const T *> rend() const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return std::reverse_iterator<T *>(this->begin()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Get an index range containing all valid indices for this array. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-02-07 17:23:25 +01:00
										 |  |  |   IndexRange index_range() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return IndexRange(size_); | 
					
						
							| 
									
										
										
										
											2020-02-07 17:23:25 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Sets the size to zero. This should only be used when you have manually destructed all elements | 
					
						
							|  |  |  |    * in the array beforehand. Use with care. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   void clear_without_destruct() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     size_ = 0; | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Access the allocator used by this array. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-05-04 11:11:46 +02:00
										 |  |  |   Allocator &allocator() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return allocator_; | 
					
						
							| 
									
										
										
										
											2020-05-04 11:11:46 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Get the value of the InlineBufferCapacity template argument. This is the number of elements | 
					
						
							|  |  |  |    * that can be stored without doing an allocation. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   static int64_t inline_buffer_capacity() | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     return InlineBufferCapacity; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  private: | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   T *get_buffer_for_size(int64_t size) | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-04-23 20:05:53 +02:00
										 |  |  |     if (size <= InlineBufferCapacity) { | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |       return inline_buffer_; | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       return this->allocate(size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   T *allocate(int64_t size) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return static_cast<T *>( | 
					
						
							|  |  |  |         allocator_.allocate(static_cast<size_t>(size) * sizeof(T), alignof(T), AT)); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   bool uses_inline_buffer() const | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |     return data_ == inline_buffer_; | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 16:00:20 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Same as a normal Array, but does not use Blender's guarded allocator. This is useful when | 
					
						
							|  |  |  |  * allocating memory with static storage duration. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<typename T, int64_t InlineBufferCapacity = default_inline_buffer_capacity(sizeof(T))> | 
					
						
							|  |  |  | using RawArray = Array<T, InlineBufferCapacity, RawAllocator>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  | }  // namespace blender
 |