/* * 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. */ #ifndef __BLI_MEMORY_UTILS_HH__ #define __BLI_MEMORY_UTILS_HH__ /** \file * \ingroup bli */ #include #include #include "BLI_utildefines.h" namespace blender { /** * Call the default constructor on n consecutive elements. For trivially constructible types, this * does nothing. * * Before: * ptr: uninitialized * After: * ptr: initialized */ template void default_construct_n(T *ptr, uint n) { /* This is not strictly necessary, because the loop below will be optimized away anyway. It is * nice to make behavior this explicitly, though. */ if (std::is_trivially_constructible::value) { return; } for (uint i = 0; i < n; i++) { new ((void *)(ptr + i)) T; } } /** * Call the destructor on n consecutive values. For trivially destructible types, this does * nothing. * * Before: * ptr: initialized * After: * ptr: uninitialized */ template void destruct_n(T *ptr, uint n) { /* This is not strictly necessary, because the loop below will be optimized away anyway. It is * nice to make behavior this explicitly, though. */ if (std::is_trivially_destructible::value) { return; } for (uint i = 0; i < n; i++) { ptr[i].~T(); } } /** * Copy n values from src to dst. * * Before: * src: initialized * dst: initialized * After: * src: initialized * dst: initialized */ template void initialized_copy_n(const T *src, uint n, T *dst) { for (uint i = 0; i < n; i++) { dst[i] = src[i]; } } /** * Copy n values from src to dst. * * Before: * src: initialized * dst: uninitialized * After: * src: initialized * dst: initialized */ template void uninitialized_copy_n(const T *src, uint n, T *dst) { for (uint i = 0; i < n; i++) { new ((void *)(dst + i)) T(src[i]); } } /** * Move n values from src to dst. * * Before: * src: initialized * dst: initialized * After: * src: initialized, moved-from * dst: initialized */ template void initialized_move_n(T *src, uint n, T *dst) { for (uint i = 0; i < n; i++) { dst[i] = std::move(src[i]); } } /** * Move n values from src to dst. * * Before: * src: initialized * dst: uninitialized * After: * src: initialized, moved-from * dst: initialized */ template void uninitialized_move_n(T *src, uint n, T *dst) { for (uint i = 0; i < n; i++) { new ((void *)(dst + i)) T(std::move(src[i])); } } /** * Relocate n values from src to dst. Relocation is a move followed by destruction of the src * value. * * Before: * src: initialized * dst: initialized * After: * src: uninitialized * dst: initialized */ template void initialized_relocate_n(T *src, uint n, T *dst) { initialized_move_n(src, n, dst); destruct_n(src, n); } /** * Relocate n values from src to dst. Relocation is a move followed by destruction of the src * value. * * Before: * src: initialized * dst: uninitialized * After: * src: uninitialized * dst: initialized */ template void uninitialized_relocate_n(T *src, uint n, T *dst) { uninitialized_move_n(src, n, dst); destruct_n(src, n); } /** * Copy the value to n consecutive elements. * * Before: * dst: initialized * After: * dst: initialized */ template void initialized_fill_n(T *dst, uint n, const T &value) { for (uint i = 0; i < n; i++) { dst[i] = value; } } /** * Copy the value to n consecutive elements. * * Before: * dst: uninitialized * After: * dst: initialized */ template void uninitialized_fill_n(T *dst, uint n, const T &value) { for (uint i = 0; i < n; i++) { new ((void *)(dst + i)) T(value); } } template struct DestructValueAtAddress { void operator()(T *ptr) { ptr->~T(); } }; /** * 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. */ template using destruct_ptr = std::unique_ptr>; /** * An `AlignedBuffer` is simply a byte array with the given size and alignment. The buffer will * not be initialized by the default constructor. * * 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. */ template class alignas(Alignment) AlignedBuffer { private: /* Don't create an empty array. This causes problems with some compilers. */ char buffer_[(Size > 0) ? Size : 1]; public: void *ptr() { return (void *)buffer_; } const void *ptr() const { return (const void *)buffer_; } }; /** * 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 { }; } // namespace blender #endif /* __BLI_MEMORY_UTILS_HH__ */