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-04-21 17:31:56 +02:00
|
|
|
#ifndef __BLI_MEMORY_UTILS_HH__
|
|
|
|
#define __BLI_MEMORY_UTILS_HH__
|
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
|
2019-09-13 21:12:26 +10:00
|
|
|
|
2020-04-21 17:31:56 +02:00
|
|
|
#endif /* __BLI_MEMORY_UTILS_HH__ */
|