axmol/cocos/base/allocator/CCAllocatorStrategyFixedBlo...

294 lines
9.2 KiB
C++

#ifndef CC_ALLOCATOR_STRATEGY_FIXED_BLOCK_H
#define CC_ALLOCATOR_STRATEGY_FIXED_BLOCK_H
/****************************************************************************
Copyright (c) 2014 Chukong Technologies Inc.
Author: Justin Graham (https://github.com/mannewalis)
http://www.cocos2d-x.org
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
****************************************************************************/
/****************************************************************************
WARNING!
Do not use Console::log or any other methods that use NEW inside of this
allocator. Failure to do so will result in recursive memory allocation.
****************************************************************************/
#include "base/allocator/CCAllocatorBase.h"
#include "base/allocator/CCAllocatorMacros.h"
#include "base/allocator/CCAllocatorGlobal.h"
#include "base/allocator/CCAllocatorMutex.h"
#include "base/allocator/CCAllocatorDiagnostics.h"
#include <vector>
#include <typeinfo>
#include <sstream>
NS_CC_BEGIN
NS_CC_ALLOCATOR_BEGIN
// @brief define this to cause this allocator to fallback to the global allocator
// this is just for testing purposes to see if this allocator is broken.
//#define FALLBACK_TO_GLOBAL
// @brief
// Fixed sized block allocator strategy for allocating blocks
// of memory that are the same size.
// Optionally takes a page size which determines how many blocks
// are added when the allocator needs more storage.
// @param _block_size the size of the fixed block allocated by this allocator.
// @param _page_size the number of blocks to allocate when growing the free list.
// @param _alignment the alignment size in bytes of each block.
template <size_t _block_size, size_t _alignment = 16>
class AllocatorStrategyFixedBlock
: public AllocatorBase
{
public:
static constexpr size_t block_size = _block_size;
static constexpr size_t alignment = _alignment;
AllocatorStrategyFixedBlock(const char* tag = nullptr, size_t pageSize = 100)
: _list(nullptr)
, _pages(nullptr)
, _allocated(0)
, _pageSize(pageSize)
{
#if CC_ENABLE_ALLOCATOR_DIAGNOSTICS
_highestCount = 0;
AllocatorDiagnostics::instance()->trackAllocator(this);
AllocatorBase::setTag(tag ? tag : typeid(AllocatorStrategyFixedBlock).name());
#endif
}
virtual ~AllocatorStrategyFixedBlock()
{
#if CC_ENABLE_ALLOCATOR_DIAGNOSTICS
AllocatorDiagnostics::instance()->untrackAllocator(this);
#endif
do
{
intptr_t* page = (intptr_t*)_pages;
intptr_t* next = (intptr_t*)*page;
ccAllocatorGlobal.deallocate(page);
_pages = (void*)next;
}
while (_pages);
}
// @brief
// allocate a block of memory by returning the first item in the list or if empty
// then allocate a new page of blocks, and return the first element and store the rest.
// if _block_size does not match the requested size, then we assert.
CC_ALLOCATOR_INLINE void* allocate(size_t size)
{
CC_ASSERT(block_size == size);
#ifdef FALLBACK_TO_GLOBAL
return ccAllocatorGlobal.allocate(size);
#else
return pop_front();
#endif
}
// @brief Deallocate a block by pushing it on the head of a linked list of free blocks.
CC_ALLOCATOR_INLINE void deallocate(void* address, size_t size = 0)
{
CC_ASSERT(0 == size || block_size == size);
#ifdef FALLBACK_TO_GLOBAL
ccAllocatorGlobal.deallocate(address);
#else
push_front(address);
#endif
}
// @brief Checks allocated pages to determine whether or not a block
// is owned by this allocator. This should be reasonably fast
// for properly configured allocators with few large pages.
CC_ALLOCATOR_INLINE bool owns(const void* const address)
{
#ifdef FALLBACK_TO_GLOBAL
return true; // since everything uses the global allocator, we can just lie and say we own this address.
#else
LOCK(_mutex);
const uint8_t* const a = (const uint8_t* const)address;
const uint8_t* p = (uint8_t*)_pages;
const size_t pSize = pageSize();
while (p)
{
if (a >= p && a < (p + pSize))
{
UNLOCK(_mutex);
return true;
}
p = (uint8_t*)(*(uintptr_t*)p);
}
UNLOCK(_mutex);
return false;
#endif
}
#if CC_ENABLE_ALLOCATOR_DIAGNOSTICS
std::string diagnostics() const
{
std::stringstream s;
s << AllocatorBase::tag() << " initial:" << _pageSize << " count:" << _allocated << " highest:" << _highestCount << "\n";
return s.str();
}
size_t _highestCount;
#endif
protected:
#if COCOS2D_DEBUG
#define VALIDATE \
if (nullptr != _list) \
{ \
CC_ASSERT(nullptr != _pages); \
}
#else
#define VALIDATE
#endif
// @brief Method to push an allocated block onto the free list.
// No check is made that the block hasn't been already added to this allocator.
CC_ALLOCATOR_INLINE void push_front(void* block)
{
CC_ASSERT(block);
CC_ASSERT(block_size < AllocatorBase::kDefaultAlignment || 0 == ((intptr_t)block & (AllocatorBase::kDefaultAlignment - 1)));
#if COCOS2D_DEBUG
// additional check that we own this block
CC_ASSERT(true == owns(block));
#endif
LOCK(_mutex);
VALIDATE
if (nullptr == _list)
{
_list = block;
*(uintptr_t*)block = 0;
}
else
{
uintptr_t* p = (uintptr_t*)(block);
*p = (uintptr_t)_list;
_list = block;
}
CC_ASSERT(_allocated > 0);
--_allocated;
UNLOCK(_mutex);
}
// @brief Method to pop a block off the free list.
// If no blocks are available, then the list is grown by _page_size
// Tuning of the page size is critical to getting good performance.
// Ideally you would use a page size that is around the high water mark
// for the number of blocks of this size being allocated.
CC_ALLOCATOR_INLINE void* pop_front()
{
LOCK(_mutex);
VALIDATE
if (nullptr == _list)
{
allocatePage();
}
auto next = (void*)*(uintptr_t*)_list;
auto block = _list;
_list = next;
++_allocated;
#if CC_ENABLE_ALLOCATOR_DIAGNOSTICS
if (_allocated > _highestCount)
_highestCount = _allocated;
#endif
UNLOCK(_mutex);
CC_ASSERT(block_size < AllocatorBase::kDefaultAlignment || 0 == ((intptr_t)block & (AllocatorBase::kDefaultAlignment - 1)));
return block;
}
protected:
// @brief Returns the size of a page in bytes + overhead.
constexpr size_t pageSize() const
{
return AllocatorBase::kDefaultAlignment + AllocatorBase::nextPow2BlockSize(block_size) * _pageSize;
}
// @brief Allocates a new page from the global allocator,
// and adds all the blocks to the free list.
CC_ALLOCATOR_INLINE void allocatePage()
{
uint8_t* p = (uint8_t*)AllocatorBase::aligned(ccAllocatorGlobal.allocate(pageSize()));
intptr_t* page = (intptr_t*)p;
if (nullptr == _pages)
{
_pages = page;
*page = 0;
}
else
{
*page = (intptr_t)_pages;
_pages = page;
}
p += AllocatorBase::kDefaultAlignment; // step past the linked list node
_allocated += _pageSize;
size_t aligned_size = AllocatorBase::nextPow2BlockSize(block_size);
uint8_t* block = (uint8_t*)p;
for (int i = 0; i < _pageSize; ++i, block += aligned_size)
{
push_front(block);
}
}
protected:
// @brief Linked list of free blocks.
void* _list;
// @brief Linked list of allocated pages.
void* _pages;
// @brief number of blocks per page.
size_t _pageSize;
// @brief Number of blocks that are currently allocated.
size_t _allocated;
// @brief mutex for thread safety.
AllocatorMutex _mutex;
};
NS_CC_ALLOCATOR_END
NS_CC_END
#endif//CC_ALLOCATOR_STRATEGY_FIXED_BLOCK_H