// Copyright (C) 2017 Jérôme Leclercq // This file is part of the "Nazara Engine - Core module" // For conditions of distribution and use, see copyright notice in Config.hpp // I'm not proud of those five following lines but ti's hard to do with another way now #ifdef NAZARA_DEBUG_NEWREDEFINITION_DISABLE_REDEFINITION #define NAZARA_DEBUG_NEWREDEFINITION_DISABLE_REDEFINITION_DEFINED #else #define NAZARA_DEBUG_NEWREDEFINITION_DISABLE_REDEFINITION #endif #include #include #include #include #include #include #include namespace Nz { /*! * \ingroup core * \fn Nz::MemoryHelper * \brief Core functions that helps the handle of memory in the engine */ /*! * \brief Calls the operator delete on the pointer * * \remark Uses MemoryManager with NAZARA_CORE_MANAGE_MEMORY defined else operator delete */ inline void OperatorDelete(void* ptr) { #if NAZARA_CORE_MANAGE_MEMORY MemoryManager::Free(ptr); #else operator delete(ptr); #endif } /*! * \brief Calls the operator new on the pointer * * \remark Uses MemoryManager with NAZARA_CORE_MANAGE_MEMORY defined else operator new */ inline void* OperatorNew(std::size_t size) { #if NAZARA_CORE_MANAGE_MEMORY return MemoryManager::Allocate(size); #else return operator new(size); #endif } /*! * \brief Constructs the object inplace * \return Pointer to the constructed object * * \param ptr Pointer to raw memory allocated * \param args Arguments for the constructor */ template T* PlacementNew(T* ptr, Args&&... args) { return new (ptr) T(std::forward(args)...); } /*! * \brief Calls the object destructor explicitly * * \param ptr Pointer to a previously constructed pointer on raw memory * * \remark This does not deallocate memory, and is a no-op on a null pointer */ template void PlacementDestroy(T* ptr) { if (ptr) ptr->~T(); } /*! * \ingroup core * \class Nz::StackArray * \brief Core class that represents a stack-allocated (if alloca is present) array */ template StackArray::StackArray(T* stackMemory, std::size_t size) : m_size(size), m_ptr(stackMemory) { for (std::size_t i = 0; i < m_size; ++i) PlacementNew(&m_ptr[i]); } template StackArray::StackArray(T* stackMemory, std::size_t size, NoInitTag) : m_size(size), m_ptr(stackMemory) { } template StackArray::~StackArray() { for (std::size_t i = 0; i < m_size; ++i) m_ptr[i].~T(); #ifndef NAZARA_ALLOCA_SUPPORT OperatorDelete(m_ptr); #endif } template typename StackArray::reference StackArray::back() { assert(m_size != 0); return m_ptr[m_size - 1]; } template typename StackArray::const_reference StackArray::back() const { assert(m_size != 0); return m_ptr[m_size - 1]; } template typename StackArray::iterator StackArray::begin() noexcept { return iterator(&m_ptr[0]); } template typename StackArray::const_iterator StackArray::begin() const noexcept { return const_iterator(&m_ptr[0]); } template typename StackArray::const_iterator StackArray::cbegin() const noexcept { return const_iterator(&m_ptr[0]); } template typename StackArray::const_iterator StackArray::cend() const noexcept { return const_iterator(&m_ptr[m_size]); } template typename StackArray::const_reverse_iterator StackArray::crbegin() const noexcept { return const_reverse_iterator(&m_ptr[m_size]); } template typename StackArray::const_reverse_iterator StackArray::crend() const noexcept { return const_reverse_iterator(&m_ptr[0]); } template T* StackArray::data() noexcept { return m_ptr; } template const T* StackArray::data() const noexcept { return m_ptr; } template bool StackArray::empty() const noexcept { return m_size == 0; } template typename StackArray::iterator StackArray::end() noexcept { return iterator(&m_ptr[m_size]); } template typename StackArray::const_iterator StackArray::end() const noexcept { return const_iterator(&m_ptr[m_size]); } template void StackArray::fill(const T& value) { std::fill(begin(), end(), value); } template typename StackArray::reference StackArray::front() noexcept { return m_ptr[0]; } template typename StackArray::const_reference StackArray::front() const noexcept { return m_ptr[0]; } template typename StackArray::size_type StackArray::max_size() const noexcept { return size(); } template typename StackArray::reverse_iterator StackArray::rbegin() noexcept { return reverse_iterator(&m_ptr[m_size]); } template typename StackArray::const_reverse_iterator StackArray::rbegin() const noexcept { return reverse_iterator(&m_ptr[m_size]); } template typename StackArray::reverse_iterator StackArray::rend() noexcept { return reverse_iterator(&m_ptr[0]); } template typename StackArray::const_reverse_iterator StackArray::rend() const noexcept { return reverse_iterator(&m_ptr[0]); } template typename StackArray::size_type StackArray::size() const noexcept { return m_size; } template typename StackArray::reference StackArray::operator[](size_type pos) { assert(pos < m_size); return m_ptr[pos]; } template typename StackArray::const_reference StackArray::operator[](size_type pos) const { assert(pos < m_size); return m_ptr[pos]; } } #include // If we have defined the constant, then we have to undefine it (to avoid bloating in the engine) #ifndef NAZARA_DEBUG_NEWREDEFINITION_DISABLE_REDEFINITION_DEFINED #undef NAZARA_DEBUG_NEWREDEFINITION_DISABLE_REDEFINITION #endif