#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 #include #include 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 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