| 
									
										
										
										
											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++) { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |       new (static_cast<void *>(ptr + current)) T; | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   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++) { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |       new (static_cast<void *>(dst + current)) T(src[current]); | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   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++) { | 
					
						
							| 
									
										
										
										
											2020-08-19 16:44:53 +02:00
										 |  |  |       new (static_cast<void *>(dst + current)) To(static_cast<To>(src[current])); | 
					
						
							| 
									
										
										
										
											2020-07-08 22:27:25 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   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++) { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |       new (static_cast<void *>(dst + current)) T(std::move(src[current])); | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   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++) { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |       new (static_cast<void *>(dst + current)) T(value); | 
					
						
							| 
									
										
										
										
											2020-07-06 09:08:53 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   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 *() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return buffer_; | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   operator const void *() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return buffer_; | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   void *ptr() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return buffer_; | 
					
						
							| 
									
										
										
										
											2020-02-10 13:54:57 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const void *ptr() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return 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 *() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return static_cast<T *>(buffer_.ptr()); | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   operator const T *() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return static_cast<const T *>(buffer_.ptr()); | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   T &operator*() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return *static_cast<T *>(buffer_.ptr()); | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const T &operator*() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return *static_cast<const T *>(buffer_.ptr()); | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   T *ptr() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return static_cast<T *>(buffer_.ptr()); | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const T *ptr() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return static_cast<const T *>(buffer_.ptr()); | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   T &ref() | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return *static_cast<T *>(buffer_.ptr()); | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const T &ref() const | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |     return *static_cast<const T *>(buffer_.ptr()); | 
					
						
							| 
									
										
										
										
											2020-07-06 10:56:26 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-08-19 16:44:53 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * This can be used to mark a constructor of an object that does not throw exceptions. Other | 
					
						
							|  |  |  |  * constructors can delegate to this constructor to make sure that the object lifetime starts. | 
					
						
							|  |  |  |  * With this, the destructor of the object will be called, even when the remaining constructor | 
					
						
							|  |  |  |  * throws. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class NoExceptConstructor { | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-12-16 15:59:58 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Helper variable that checks if a Span<From> can be converted to Span<To> safely, whereby From | 
					
						
							|  |  |  |  * and To are pointers. Adding const and casting to a void pointer is allowed. | 
					
						
							|  |  |  |  * Casting up and down a class hierarchy generally is not allowed, because this might change the | 
					
						
							|  |  |  |  * pointer under some circumstances. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<typename From, typename To> | 
					
						
							|  |  |  | inline constexpr bool is_span_convertible_pointer_v = | 
					
						
							|  |  |  |     /* Make sure we are working with pointers. */ | 
					
						
							|  |  |  |     std::is_pointer_v<From> &&std::is_pointer_v<To> && | 
					
						
							|  |  |  |     (/* No casting is necessary when both types are the same. */ | 
					
						
							|  |  |  |      std::is_same_v<From, To> || | 
					
						
							|  |  |  |      /* Allow adding const to the underlying type. */ | 
					
						
							|  |  |  |      std::is_same_v<const std::remove_pointer_t<From>, std::remove_pointer_t<To>> || | 
					
						
							|  |  |  |      /* Allow casting non-const pointers to void pointers. */ | 
					
						
							|  |  |  |      (!std::is_const_v<std::remove_pointer_t<From>> && std::is_same_v<To, void *>) || | 
					
						
							|  |  |  |      /* Allow casting any pointer to const void pointers. */ | 
					
						
							|  |  |  |      std::is_same_v<To, const void *>); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-07 18:24:59 +02:00
										 |  |  |   return (static_cast<int64_t>(element_size) < 100) ? 4 : 0; | 
					
						
							| 
									
										
										
										
											2020-07-20 16:00:20 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-19 16:44:53 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * This can be used by containers to implement an exception-safe copy-assignment-operator. | 
					
						
							|  |  |  |  * It assumes that the container has an exception safe copy constructor and an exception-safe | 
					
						
							|  |  |  |  * move-assignment-operator. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<typename Container> Container ©_assign_container(Container &dst, const Container &src) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (&src == &dst) { | 
					
						
							|  |  |  |     return dst; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Container container_copy{src}; | 
					
						
							|  |  |  |   dst = std::move(container_copy); | 
					
						
							|  |  |  |   return dst; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * This can be used by containers to implement an exception-safe move-assignment-operator. | 
					
						
							|  |  |  |  * It assumes that the container has an exception-safe move-constructor and a noexcept constructor | 
					
						
							|  |  |  |  * tagged with the NoExceptConstructor tag. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<typename Container> | 
					
						
							|  |  |  | Container &move_assign_container(Container &dst, Container &&src) noexcept( | 
					
						
							|  |  |  |     std::is_nothrow_move_constructible_v<Container>) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (&dst == &src) { | 
					
						
							|  |  |  |     return dst; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dst.~Container(); | 
					
						
							|  |  |  |   if constexpr (std::is_nothrow_move_constructible_v<Container>) { | 
					
						
							|  |  |  |     new (&dst) Container(std::move(src)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       new (&dst) Container(std::move(src)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     catch (...) { | 
					
						
							|  |  |  |       new (&dst) Container(NoExceptConstructor()); | 
					
						
							|  |  |  |       throw; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return dst; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-09 10:27:24 +02:00
										 |  |  | }  // namespace blender
 |