| 
									
										
										
										
											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-08-07 09:50:34 +02:00
										 |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2019-09-13 21:12:26 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup bli | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * Some of the functions below have very similar alternatives in the standard library. However, it | 
					
						
							|  |  |  |  * is rather annoying to use those when debugging. Therefore, some more specialized and easier to | 
					
						
							|  |  |  |  * debug functions are provided here. | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2020-06-11 15:13:19 +02:00
										 |  |  | #include <new>
 | 
					
						
							| 
									
										
										
										
											2020-07-08 22:27:25 +02:00
										 |  |  | #include <type_traits>
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 16:45:06 +01:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * Call the destructor on n consecutive values. For trivially destructible types, this does | 
					
						
							|  |  |  |  * nothing. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Exception Safety: Destructors shouldn't throw exceptions. | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Before: | 
					
						
							|  |  |  |  *  ptr: initialized | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * After: | 
					
						
							|  |  |  |  *  ptr: uninitialized | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename T> void destruct_n(T *ptr, int64_t n) | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   BLI_assert(n >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |   static_assert(std::is_nothrow_destructible_v<T>, | 
					
						
							|  |  |  |                 "This should be true for all types. Destructors are noexcept by default."); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /* This is not strictly necessary, because the loop below will be optimized away anyway. It is
 | 
					
						
							| 
									
										
										
										
											2020-06-13 12:50:07 +10:00
										 |  |  |    * nice to make behavior this explicitly, though. */ | 
					
						
							| 
									
										
										
										
											2020-07-08 20:39:12 +02:00
										 |  |  |   if (std::is_trivially_destructible_v<T>) { | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   for (int64_t i = 0; i < n; i++) { | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |     ptr[i].~T(); | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * Call the default constructor on n consecutive elements. For trivially constructible types, this | 
					
						
							|  |  |  |  * does nothing. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Exception Safety: Strong. | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Before: | 
					
						
							|  |  |  |  *  ptr: uninitialized | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * After: | 
					
						
							|  |  |  |  *  ptr: initialized | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename T> void default_construct_n(T *ptr, int64_t n) | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   BLI_assert(n >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   /* This is not strictly necessary, because the loop below will be optimized away anyway. It is
 | 
					
						
							| 
									
										
										
										
											2020-06-13 12:50:07 +10:00
										 |  |  |    * nice to make behavior this explicitly, though. */ | 
					
						
							| 
									
										
										
										
											2020-07-08 20:39:12 +02:00
										 |  |  |   if (std::is_trivially_constructible_v<T>) { | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   int64_t current = 0; | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |   try { | 
					
						
							|  |  |  |     for (; current < n; current++) { | 
					
						
							|  |  |  |       new ((void *)(ptr + current)) T; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   catch (...) { | 
					
						
							|  |  |  |     destruct_n(ptr, current); | 
					
						
							|  |  |  |     throw; | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Copy n values from src to dst. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * Exception Safety: Basic. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * Before: | 
					
						
							|  |  |  |  *  src: initialized | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  * After: | 
					
						
							|  |  |  |  *  src: initialized | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename T> void initialized_copy_n(const T *src, int64_t n, T *dst) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   BLI_assert(n >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (int64_t i = 0; i < n; i++) { | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |     dst[i] = src[i]; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Copy n values from src to dst. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * Exception Safety: Strong. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * Before: | 
					
						
							|  |  |  |  *  src: initialized | 
					
						
							|  |  |  |  *  dst: uninitialized | 
					
						
							|  |  |  |  * After: | 
					
						
							|  |  |  |  *  src: initialized | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename T> void uninitialized_copy_n(const T *src, int64_t n, T *dst) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   BLI_assert(n >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int64_t current = 0; | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |   try { | 
					
						
							|  |  |  |     for (; current < n; current++) { | 
					
						
							|  |  |  |       new ((void *)(dst + current)) T(src[current]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   catch (...) { | 
					
						
							|  |  |  |     destruct_n(dst, current); | 
					
						
							|  |  |  |     throw; | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-08 22:27:25 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Convert n values from type `From` to type `To`. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Exception Safety: Strong. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Before: | 
					
						
							|  |  |  |  *  src: initialized | 
					
						
							|  |  |  |  *  dst: uninitialized | 
					
						
							|  |  |  |  * After: | 
					
						
							|  |  |  |  *  src: initialized | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename From, typename To> | 
					
						
							|  |  |  | void uninitialized_convert_n(const From *src, int64_t n, To *dst) | 
					
						
							| 
									
										
										
										
											2020-07-08 22:27:25 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   BLI_assert(n >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int64_t current = 0; | 
					
						
							| 
									
										
										
										
											2020-07-08 22:27:25 +02:00
										 |  |  |   try { | 
					
						
							|  |  |  |     for (; current < n; current++) { | 
					
						
							|  |  |  |       new ((void *)(dst + current)) To((To)src[current]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   catch (...) { | 
					
						
							|  |  |  |     destruct_n(dst, current); | 
					
						
							|  |  |  |     throw; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Move n values from src to dst. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * Exception Safety: Basic. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * Before: | 
					
						
							|  |  |  |  *  src: initialized | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  * After: | 
					
						
							|  |  |  |  *  src: initialized, moved-from | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename T> void initialized_move_n(T *src, int64_t n, T *dst) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   BLI_assert(n >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (int64_t i = 0; i < n; i++) { | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |     dst[i] = std::move(src[i]); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Move n values from src to dst. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * Exception Safety: Basic. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * Before: | 
					
						
							|  |  |  |  *  src: initialized | 
					
						
							|  |  |  |  *  dst: uninitialized | 
					
						
							|  |  |  |  * After: | 
					
						
							|  |  |  |  *  src: initialized, moved-from | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename T> void uninitialized_move_n(T *src, int64_t n, T *dst) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   BLI_assert(n >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int64_t current = 0; | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |   try { | 
					
						
							|  |  |  |     for (; current < n; current++) { | 
					
						
							|  |  |  |       new ((void *)(dst + current)) T(std::move(src[current])); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   catch (...) { | 
					
						
							|  |  |  |     destruct_n(dst, current); | 
					
						
							|  |  |  |     throw; | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Relocate n values from src to dst. Relocation is a move followed by destruction of the src | 
					
						
							|  |  |  |  * value. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * Exception Safety: Basic. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * Before: | 
					
						
							|  |  |  |  *  src: initialized | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  * After: | 
					
						
							|  |  |  |  *  src: uninitialized | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename T> void initialized_relocate_n(T *src, int64_t n, T *dst) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   BLI_assert(n >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   initialized_move_n(src, n, dst); | 
					
						
							|  |  |  |   destruct_n(src, n); | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Relocate n values from src to dst. Relocation is a move followed by destruction of the src | 
					
						
							|  |  |  |  * value. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * Exception Safety: Basic. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * Before: | 
					
						
							|  |  |  |  *  src: initialized | 
					
						
							| 
									
										
										
										
											2020-06-13 12:50:07 +10:00
										 |  |  |  *  dst: uninitialized | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * After: | 
					
						
							|  |  |  |  *  src: uninitialized | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename T> void uninitialized_relocate_n(T *src, int64_t n, T *dst) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   BLI_assert(n >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  |   uninitialized_move_n(src, n, dst); | 
					
						
							|  |  |  |   destruct_n(src, n); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Copy the value to n consecutive elements. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  * Exception Safety: Basic. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * Before: | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  * After: | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename T> void initialized_fill_n(T *dst, int64_t n, const T &value) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   BLI_assert(n >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (int64_t i = 0; i < n; i++) { | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |     dst[i] = value; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Copy the value to n consecutive elements. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |  *  Exception Safety: Strong. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * Before: | 
					
						
							|  |  |  |  *  dst: uninitialized | 
					
						
							|  |  |  |  * After: | 
					
						
							|  |  |  |  *  dst: initialized | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename T> void uninitialized_fill_n(T *dst, int64_t n, const T &value) | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   BLI_assert(n >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int64_t current = 0; | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |   try { | 
					
						
							|  |  |  |     for (; current < n; current++) { | 
					
						
							|  |  |  |       new ((void *)(dst + current)) T(value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   catch (...) { | 
					
						
							|  |  |  |     destruct_n(dst, current); | 
					
						
							|  |  |  |     throw; | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-09-12 14:23:21 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  | template<typename T> struct DestructValueAtAddress { | 
					
						
							|  |  |  |   void operator()(T *ptr) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     ptr->~T(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * A destruct_ptr is like unique_ptr, but it will only call the destructor and will not free the | 
					
						
							|  |  |  |  * memory. This is useful when using custom allocators. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  | template<typename T> using destruct_ptr = std::unique_ptr<T, DestructValueAtAddress<T>>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |  * An `AlignedBuffer` is a byte array with at least the given size and alignment. The buffer will | 
					
						
							| 
									
										
										
										
											2020-06-09 10:10:56 +02:00
										 |  |  |  * not be initialized by the default constructor. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<size_t Size, size_t Alignment> class alignas(Alignment) AlignedBuffer { | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |  private: | 
					
						
							|  |  |  |   /* Don't create an empty array. This causes problems with some compilers. */ | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |   char buffer_[(Size > 0) ? Size : 1]; | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   operator void *() | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return (void *)buffer_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   operator const void *() const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return (void *)buffer_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   void *ptr() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return (void *)buffer_; | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const void *ptr() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-03 14:15:05 +02:00
										 |  |  |     return (const void *)buffer_; | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * This can be used to reserve memory for C++ objects whose lifetime is different from the | 
					
						
							|  |  |  |  * lifetime of the object they are embedded in. It's used by containers with small buffer | 
					
						
							|  |  |  |  * optimization and hash table implementations. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  | template<typename T, int64_t Size = 1> class TypedBuffer { | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |  private: | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   AlignedBuffer<sizeof(T) * (size_t)Size, alignof(T)> buffer_; | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							|  |  |  |   operator T *() | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return (T *)&buffer_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   operator const T *() const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return (const T *)&buffer_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   T &operator*() | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return *(T *)&buffer_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const T &operator*() const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return *(const T *)&buffer_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   T *ptr() | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return (T *)&buffer_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const T *ptr() const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return (const T *)&buffer_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   T &ref() | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return *(T *)&buffer_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const T &ref() const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return *(const T *)&buffer_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 15:58:14 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * This can be used by container constructors. A parameter of this type should be used to indicate | 
					
						
							|  |  |  |  * that the constructor does not construct the elements. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class NoInitialization { | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-08 22:27:25 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Helper variable that checks if a pointer type can be converted into another pointer type without | 
					
						
							|  |  |  |  * issues. Possible issues are casting away const and casting a pointer to a child class. | 
					
						
							|  |  |  |  * Adding const or casting to a parent class is fine. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<typename From, typename To> | 
					
						
							|  |  |  | inline constexpr bool is_convertible_pointer_v = | 
					
						
							|  |  |  |     std::is_convertible_v<From, To> &&std::is_pointer_v<From> &&std::is_pointer_v<To>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 16:00:20 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Inline buffers for small-object-optimization should be disable by default. Otherwise we might | 
					
						
							|  |  |  |  * get large unexpected allocations on the stack. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | inline constexpr int64_t default_inline_buffer_capacity(size_t element_size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return ((int64_t)element_size < 100) ? 4 : 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  | }  // namespace blender
 |