This repository has been archived on 2023-10-09. You can view files and clone it, but cannot push or open issues or pull requests.
Files
blender-archive/source/blender/blenlib/BLI_memory_utils.hh

312 lines
6.4 KiB
C++
Raw Normal View History

/*
* 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__
2019-09-13 21:12:26 +10:00
/** \file
* \ingroup bli
*/
#include <memory>
#include <new>
#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<typename T> void default_construct_n(T *ptr, uint n)
{
/* 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. */
if (std::is_trivially_constructible<T>::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<typename T> void destruct_n(T *ptr, uint n)
{
/* 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. */
if (std::is_trivially_destructible<T>::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<typename T> 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<typename T> 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<typename T> 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<typename T> 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<typename T> 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
2020-06-13 12:50:07 +10:00
* dst: uninitialized
* After:
* src: uninitialized
* dst: initialized
*/
template<typename T> 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<typename T> 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<typename T> 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<typename T> 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<typename T> using destruct_ptr = std::unique_ptr<T, DestructValueAtAddress<T>>;
/**
* An `AlignedBuffer` is a byte array with at least the given size and alignment. The buffer will
* not be initialized by the default constructor.
*/
template<size_t Size, size_t Alignment> class alignas(Alignment) AlignedBuffer {
private:
/* Don't create an empty array. This causes problems with some compilers. */
char buffer_[(Size > 0) ? Size : 1];
public:
operator void *()
{
return (void *)buffer_;
}
operator const void *() const
{
return (void *)buffer_;
}
void *ptr()
{
return (void *)buffer_;
}
const void *ptr() const
{
return (const void *)buffer_;
}
};
/**
* 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<typename T, size_t Size = 1> class TypedBuffer {
private:
AlignedBuffer<sizeof(T) * Size, alignof(T)> buffer_;
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_;
}
};
/**
* 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
2019-09-13 21:12:26 +10:00
#endif /* __BLI_MEMORY_UTILS_HH__ */