| 
									
										
										
										
											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-06-09 11:58:47 +02:00
										 |  |  | #ifndef __BLI_SPAN_HH__
 | 
					
						
							|  |  |  | #define __BLI_SPAN_HH__
 | 
					
						
							| 
									
										
										
										
											2019-09-13 21:12:26 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup bli | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |  * An `blender::Span<T>` references an array that is owned by someone else. It is just a | 
					
						
							|  |  |  |  * pointer and a size. Since the memory is not owned, Span should not be used to transfer | 
					
						
							|  |  |  |  * ownership. The array cannot be modified through the Span. However, if T is a non-const | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  |  * pointer, the pointed-to elements can be modified. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |  * There is also `blender::MutableSpan<T>`. It is mostly the same as Span, but allows the | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  |  * array to be modified. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |  * A (Mutable)Span can refer to data owned by many different data structures including | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  |  * blender::Vector, blender::Array, blender::VectorSet, std::vector, std::array, std::string, | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * std::initializer_list and c-style array. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |  * `blender::Span` is very similar to `std::span` (C++20). However, there are a few differences: | 
					
						
							|  |  |  |  * - `blender::Span` is const by default. This is to avoid making things mutable when they don't | 
					
						
							|  |  |  |  *   have to be. To get a non-const span, you need to use `blender::MutableSpan`. Below is a list | 
					
						
							|  |  |  |  *   of const-behavior-equivalent pairs of data structures: | 
					
						
							|  |  |  |  *   - std::span<int>                <==>  blender::MutableSpan<int> | 
					
						
							|  |  |  |  *   - std::span<const int>          <==>  blender::Span<int> | 
					
						
							|  |  |  |  *   - std::span<int *>              <==>  blender::MutableSpan<int *> | 
					
						
							|  |  |  |  *   - std::span<const int *>        <==>  blender::MutableSpan<const int *> | 
					
						
							|  |  |  |  *   - std::span<int * const>        <==>  blender::Span<int *> | 
					
						
							|  |  |  |  *   - std::span<const int * const>  <==>  blender::Span<const int *> | 
					
						
							|  |  |  |  * - `blender::Span` always has a dynamic extent, while `std::span` can have a size that is | 
					
						
							|  |  |  |  *   determined at compile time. I did not have a use case for that yet. If we need it, we can | 
					
						
							|  |  |  |  *   decide to add this functionality to `blender::Span` or introduce a new type like | 
					
						
							|  |  |  |  *   `blender::FixedSpan<T, N>`. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * `blender::Span<T>` should be your default choice when you have to pass a read-only array | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  |  * into a function. It is better than passing a `const Vector &`, because then the function only | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |  * works for vectors and not for e.g. arrays. Using Span as function parameter makes it usable | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  |  * in more contexts, better expresses the intent and does not sacrifice performance. It is also | 
					
						
							|  |  |  |  * better than passing a raw pointer and size separately, because it is more convenient and safe. | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |  * `blender::MutableSpan<T>` can be used when a function is supposed to return an array, the | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  |  * size of which is known before the function is called. One advantage of this approach is that the | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * caller is responsible for allocation and deallocation. Furthermore, the function can focus on | 
					
						
							|  |  |  |  * its task, without having to worry about memory allocation. Alternatively, a function could | 
					
						
							|  |  |  |  * return an Array or Vector. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |  * Note: When a function has a MutableSpan<T> output parameter and T is not a trivial type, | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * then the function has to specify whether the referenced array is expected to be initialized or | 
					
						
							|  |  |  |  * not. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |  * Since the arrays are only referenced, it is generally unsafe to store an Span. When you | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * store one, you should know who owns the memory. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |  * Instances of Span and MutableSpan are small and should be passed by value. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | #include <iostream>
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 17:31:56 +02:00
										 |  |  | #include "BLI_index_range.hh"
 | 
					
						
							|  |  |  | #include "BLI_memory_utils.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
										 |  |  |  * References an array of type T that is owned by someone else. The data in the array cannot be | 
					
						
							|  |  |  |  * modified. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  | template<typename T> class Span { | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  private: | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |   const T *start_ = nullptr; | 
					
						
							|  |  |  |   uint size_ = 0; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Create a reference to an empty array. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   Span() = default; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |   Span(const T *start, uint size) : start_(start), size_(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<U, T>> * = nullptr> | 
					
						
							|  |  |  |   Span(const U *start, uint size) : start_((const T *)start), size_(size) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Reference an initializer_list. Note that the data in the initializer_list is only valid until | 
					
						
							|  |  |  |    * the expression containing it is fully computed. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Do: | 
					
						
							|  |  |  |    *  call_function_with_array({1, 2, 3, 4}); | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Don't: | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    *  Span<int> span = {1, 2, 3, 4}; | 
					
						
							|  |  |  |    *  call_function_with_array(span); | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   Span(const std::initializer_list<T> &list) : Span(list.begin(), (uint)list.size()) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   Span(const std::vector<T> &vector) : Span(vector.data(), (uint)vector.size()) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   template<std::size_t N> Span(const std::array<T, N> &array) : Span(array.data(), N) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Support implicit conversions like the ones below: | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    *   Span<T *> -> Span<const T *> | 
					
						
							|  |  |  |    *   Span<Derived *> -> Span<Base *> | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-08 22:27:25 +02:00
										 |  |  |   template<typename U, typename std::enable_if_t<is_convertible_pointer_v<U, T>> * = nullptr> | 
					
						
							|  |  |  |   Span(Span<U> array) : start_((T *)array.data()), size_(array.size()) | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Returns a contiguous part of the array. This invokes undefined behavior when the slice does | 
					
						
							|  |  |  |    * not stay within the bounds of the array. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   Span slice(uint start, uint size) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(start + size <= this->size() || size == 0); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return Span(start_ + start, size); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   Span slice(IndexRange range) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     return this->slice(range.start(), range.size()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Returns a new Span with n elements removed from the beginning. This invokes undefined | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * behavior when the array is too small. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   Span drop_front(uint n) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n <= this->size()); | 
					
						
							|  |  |  |     return this->slice(n, this->size() - n); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Returns a new Span with n elements removed from the beginning. This invokes undefined | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * behavior when the array is too small. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   Span drop_back(uint n) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n <= this->size()); | 
					
						
							|  |  |  |     return this->slice(0, this->size() - n); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Returns a new Span that only contains the first n elements. This invokes undefined | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * behavior when the array is too small. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   Span take_front(uint n) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n <= this->size()); | 
					
						
							|  |  |  |     return this->slice(0, n); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Returns a new Span that only contains the last n elements. This invokes undefined | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * behavior when the array is too small. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   Span take_back(uint n) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n <= this->size()); | 
					
						
							|  |  |  |     return this->slice(this->size() - n, n); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Returns the pointer to the beginning of the referenced array. This may be nullptr when the | 
					
						
							|  |  |  |    * size is zero. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   const T *data() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const T *begin() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const T *end() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_ + size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Access an element in the array. This invokes undefined behavior when the index is out of | 
					
						
							|  |  |  |    * bounds. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							|  |  |  |   const T &operator[](uint index) const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     BLI_assert(index < size_); | 
					
						
							|  |  |  |     return start_[index]; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Returns the number of elements in the referenced array. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							|  |  |  |   uint size() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											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 if the size is zero. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   bool is_empty() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return size_ == 0; | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Returns the number of bytes referenced by this Span. | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    */ | 
					
						
							|  |  |  |   uint size_in_bytes() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return sizeof(T) * size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Does a linear search to see of the value is in the array. | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Returns true if it is, otherwise false. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							|  |  |  |   bool contains(const T &value) const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     for (const T &element : *this) { | 
					
						
							|  |  |  |       if (element == value) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Does a constant time check to see if the pointer points to a value in the referenced array. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    * Return true if it is, otherwise false. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   bool contains_ptr(const T *ptr) const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return (this->begin() <= ptr) && (ptr < this->end()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Does a linear search to count how often the value is in the array. | 
					
						
							| 
									
										
										
										
											2019-09-19 13:18:52 +10:00
										 |  |  |    * Returns the number of occurrences. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							|  |  |  |   uint count(const T &value) const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     uint counter = 0; | 
					
						
							|  |  |  |     for (const T &element : *this) { | 
					
						
							|  |  |  |       if (element == value) { | 
					
						
							|  |  |  |         counter++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return counter; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Return a reference to the first element in the array. This invokes undefined behavior when the | 
					
						
							|  |  |  |    * array is empty. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							|  |  |  |   const T &first() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     BLI_assert(size_ > 0); | 
					
						
							|  |  |  |     return start_[0]; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Returns a reference to the last element in the array. This invokes undefined behavior when the | 
					
						
							|  |  |  |    * array is empty. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							|  |  |  |   const T &last() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     BLI_assert(size_ > 0); | 
					
						
							|  |  |  |     return start_[size_ - 1]; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Returns the element at the given index. If the index is out of range, return the fallback | 
					
						
							|  |  |  |    * value. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							|  |  |  |   T get(uint index, const T &fallback) const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     if (index < size_) { | 
					
						
							|  |  |  |       return start_[index]; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     return fallback; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Check if the array contains duplicates. Does a linear search for every element. So the total | 
					
						
							|  |  |  |    * running time is O(n^2). Only use this for small arrays. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   bool has_duplicates__linear_search() const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     /* The size should really be smaller than that. If it is not, the calling code should be
 | 
					
						
							|  |  |  |      * changed. */ | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     BLI_assert(size_ < 1000); | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     for (uint i = 0; i < size_; i++) { | 
					
						
							|  |  |  |       const T &value = start_[i]; | 
					
						
							|  |  |  |       for (uint j = i + 1; j < size_; j++) { | 
					
						
							|  |  |  |         if (value == start_[j]) { | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |           return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns true when this and the other array have an element in common. This should only be | 
					
						
							|  |  |  |    * called on small arrays, because it has a running time of O(n*m) where n and m are the sizes of | 
					
						
							|  |  |  |    * the arrays. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   bool intersects__linear_search(Span other) const | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   { | 
					
						
							|  |  |  |     /* The size should really be smaller than that. If it is not, the calling code should be
 | 
					
						
							|  |  |  |      * changed. */ | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     BLI_assert(size_ < 1000); | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     for (uint i = 0; i < size_; i++) { | 
					
						
							|  |  |  |       const T &value = start_[i]; | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |       if (other.contains(value)) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns the index of the first occurrence of the given value. This invokes undefined behavior | 
					
						
							|  |  |  |    * when the value is not in the array. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   uint first_index(const T &search_value) const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:52:51 +02:00
										 |  |  |     const int index = this->first_index_try(search_value); | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |     BLI_assert(index >= 0); | 
					
						
							|  |  |  |     return (uint)index; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns the index of the first occurrence of the given value or -1 if it does not exist. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   int first_index_try(const T &search_value) const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     for (uint i = 0; i < size_; i++) { | 
					
						
							|  |  |  |       if (start_[i] == search_value) { | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |         return i; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Utility to make it more convenient to iterate over all indices that can be used with 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
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Returns a new Span to the same underlying memory buffer. No conversions are done. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   template<typename NewT> Span<NewT> cast() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     BLI_assert((size_ * sizeof(T)) % sizeof(NewT) == 0); | 
					
						
							|  |  |  |     uint new_size = size_ * sizeof(T) / sizeof(NewT); | 
					
						
							|  |  |  |     return Span<NewT>(reinterpret_cast<const NewT *>(start_), new_size); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * A debug utility to print the content of the Span. Every element will be printed on a | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    * separate line using the given callback. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   template<typename PrintLineF> void print_as_lines(std::string name, PrintLineF print_line) const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     std::cout << "Span: " << name << " \tSize:" << size_ << '\n'; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     for (const T &value : *this) { | 
					
						
							|  |  |  |       std::cout << "  "; | 
					
						
							|  |  |  |       print_line(value); | 
					
						
							|  |  |  |       std::cout << '\n'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * A debug utility to print the content of the span. Every element be printed on a separate | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * line. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   void print_as_lines(std::string name) const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     this->print_as_lines(name, [](const T &value) { std::cout << value; }); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |  * Mostly the same as Span, except that one can change the array elements through a | 
					
						
							|  |  |  |  * MutableSpan. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  | template<typename T> class MutableSpan { | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  private: | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |   T *start_; | 
					
						
							|  |  |  |   uint size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   MutableSpan() = default; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:52:51 +02:00
										 |  |  |   MutableSpan(T *start, const uint size) : start_(start), size_(size) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   MutableSpan(std::vector<T> &vector) : MutableSpan(vector.data(), vector.size()) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   template<std::size_t N> MutableSpan(std::array<T, N> &array) : MutableSpan(array.data(), N) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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>(start_, size_); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Returns the number of elements in the array. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							|  |  |  |   uint size() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Replace all elements in the referenced array with the given value. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   void fill(const T &value) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     initialized_fill_n(start_, size_, value); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Replace a subset of all elements with the given value. This invokes undefined behavior when | 
					
						
							|  |  |  |    * one of the indices is out of bounds. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   void fill_indices(Span<uint> indices, const T &value) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     for (uint i : indices) { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |       BLI_assert(i < size_); | 
					
						
							|  |  |  |       start_[i] = value; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Returns a pointer to the beginning of the referenced array. This may be nullptr, when the size | 
					
						
							|  |  |  |    * is zero. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   T *data() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   T *begin() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   T *end() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_ + size_; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:52:51 +02:00
										 |  |  |   T &operator[](const uint index) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(index < this->size()); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return start_[index]; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * Returns a contiguous part of the array. This invokes undefined behavior when the slice would | 
					
						
							|  |  |  |    * go out of bounds. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-03 14:52:51 +02:00
										 |  |  |   MutableSpan slice(const uint start, const uint length) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(start + length <= this->size()); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return MutableSpan(start_ + start, length); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Returns a new MutableSpan with n elements removed from the beginning. This invokes | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * undefined behavior when the array is too small. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-03 14:52:51 +02:00
										 |  |  |   MutableSpan drop_front(const uint n) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n <= this->size()); | 
					
						
							|  |  |  |     return this->slice(n, this->size() - n); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Returns a new MutableSpan with n elements removed from the end. This invokes undefined | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * behavior when the array is too small. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-03 14:52:51 +02:00
										 |  |  |   MutableSpan drop_back(const uint n) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n <= this->size()); | 
					
						
							|  |  |  |     return this->slice(0, this->size() - n); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Returns a new MutableSpan that only contains the first n elements. This invokes undefined | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * behavior when the array is too small. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-03 14:52:51 +02:00
										 |  |  |   MutableSpan take_front(const uint n) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n <= this->size()); | 
					
						
							|  |  |  |     return this->slice(0, n); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Return a new MutableSpan that only contains the last n elements. This invokes undefined | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * behavior when the array is too small. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-07-03 14:52:51 +02:00
										 |  |  |   MutableSpan take_back(const uint n) const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     BLI_assert(n <= this->size()); | 
					
						
							|  |  |  |     return this->slice(this->size() - n, n); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Returns an (immutable) Span that references the same array. This is usually not needed, | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |    * due to implicit conversions. However, sometimes automatic type deduction needs some help. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   Span<T> as_span() const | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return Span<T>(start_, size_); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-02-07 17:23:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Utility to make it more convenient to iterate over all indices that can be used with 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-02-10 13:54:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * Returns a reference to the last element. This invokes undefined behavior when the array is | 
					
						
							|  |  |  |    * empty. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 16:21:12 +02:00
										 |  |  |   T &last() const | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     BLI_assert(size_ > 0); | 
					
						
							|  |  |  |     return start_[size_ - 1]; | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |    * Returns a new span to the same underlying memory buffer. No conversions are done. | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  |   template<typename NewT> MutableSpan<NewT> cast() const | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     BLI_assert((size_ * sizeof(T)) % sizeof(NewT) == 0); | 
					
						
							|  |  |  |     uint new_size = size_ * sizeof(T) / sizeof(NewT); | 
					
						
							|  |  |  |     return MutableSpan<NewT>(reinterpret_cast<NewT *>(start_), new_size); | 
					
						
							| 
									
										
										
										
											2020-05-07 14:21:26 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Utilities to check that arrays have the same size in debug builds. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  | template<typename T1, typename T2> void assert_same_size(const T1 &v1, const T2 &v2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   UNUSED_VARS_NDEBUG(v1, v2); | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  |   uint size = v1.size(); | 
					
						
							|  |  |  |   BLI_assert(size == v1.size()); | 
					
						
							|  |  |  |   BLI_assert(size == v2.size()); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template<typename T1, typename T2, typename T3> | 
					
						
							|  |  |  | void assert_same_size(const T1 &v1, const T2 &v2, const T3 &v3) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   UNUSED_VARS_NDEBUG(v1, v2, v3); | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  |   uint size = v1.size(); | 
					
						
							|  |  |  |   BLI_assert(size == v1.size()); | 
					
						
							|  |  |  |   BLI_assert(size == v2.size()); | 
					
						
							|  |  |  |   BLI_assert(size == v3.size()); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  | } /* namespace blender */ | 
					
						
							| 
									
										
										
										
											2019-09-13 21:12:26 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 11:58:47 +02:00
										 |  |  | #endif /* __BLI_SPAN_HH__ */
 |