123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319 |
- /****************************************************************************
- Copyright (c) 2014-2017 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.
- ****************************************************************************/
- #ifndef CC_ALLOCATOR_STRATEGY_GLOBAL_SMALL_BLOCK_H
- #define CC_ALLOCATOR_STRATEGY_GLOBAL_SMALL_BLOCK_H
- /// @cond DO_NOT_SHOW
- /****************************************************************************
- 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/CCAllocatorMacros.h"
- #include "base/allocator/CCAllocatorBase.h"
- #include "base/allocator/CCAllocatorGlobal.h"
- #include "base/allocator/CCAllocatorStrategyFixedBlock.h"
- NS_CC_BEGIN
- NS_CC_ALLOCATOR_BEGIN
- #if CC_ENABLE_ALLOCATOR_DIAGNOSTICS
- #define TRACK(slot, size, op) _smallBlockAllocations[slot] op size
- #else
- #define TRACK(...)
- #endif
- // @brief
- class AllocatorStrategyGlobalSmallBlock
- : public AllocatorBase
- {
- public:
-
- // default number of block to allocate per page.
- static const size_t kDefaultSmallBlockCount = 100;
-
- // default max small block size pool.
- static const size_t kMaxSmallBlockPower = 13; // 2^13 8kb
-
- // @brief define for allocator strategy, cannot be typedef because we want to eval at use
- #define SType(size) AllocatorStrategyFixedBlock<size>
-
- void _lazy_init()
- {
- // this gets called before static constructors
- // so make sure we only get called once.
- static bool once = true;
- if (once)
- {
- once = false;
-
- // call our own constructor. Global new can be called before the constructors are called.
- // Make sure it gets called by having it done lazily in the call to allocate.
- new (this) AllocatorStrategyGlobalSmallBlock();
- }
- }
-
- AllocatorStrategyGlobalSmallBlock()
- {
- // this gets called before static constructors
- // so make sure we only get called once.
- static bool once = true;
- if (once)
- {
- once = false;
-
- _maxBlockSize = 1 << kMaxSmallBlockPower;
-
- #if CC_ENABLE_ALLOCATOR_DIAGNOSTICS
- AllocatorDiagnostics::instance()->trackAllocator(this);
- AllocatorBase::setTag("GlobalSmallBlock");
- #endif
-
- memset(_smallBlockAllocators, 0, sizeof(_smallBlockAllocators));
- #if CC_ENABLE_ALLOCATOR_DIAGNOSTICS
- memset(_smallBlockAllocations, 0, sizeof(_smallBlockAllocations));
- #endif
- // cannot call new on the allocator here because it will recurse
- // so instead we allocate from the global allocator and construct in place.
- #define SBA(n, size) \
- if (size <= _maxBlockSize) \
- { \
- auto v = ccAllocatorGlobal.allocate(sizeof(SType(size))); \
- _smallBlockAllocators[n] = (AllocatorBase*)(new (v) SType(size)("GlobalSmallBlock::"#size)); \
- }
-
- SBA(2, 4)
- SBA(3, 8);
- SBA(4, 16);
- SBA(5, 32);
- SBA(6, 64);
- SBA(7, 128);
- SBA(8, 256);
- SBA(9, 512);
- SBA(10, 1024);
- SBA(11, 2048);
- SBA(12, 4096);
- SBA(13, 8192);
-
- #undef SBA
- }
- }
-
- virtual ~AllocatorStrategyGlobalSmallBlock()
- {
- for (int i = 0; i <= kMaxSmallBlockPower; ++i)
- if (_smallBlockAllocators[i])
- ccAllocatorGlobal.deallocate(_smallBlockAllocators[i]);
-
- #if CC_ENABLE_ALLOCATOR_DIAGNOSTICS
- AllocatorDiagnostics::instance()->untrackAllocator(this);
- #endif
- }
-
- // @brief Allocate a block of some size. If the block is <= 8192 it is allocated out of an array
- // of fixed size block allocators. If larger, then we default back to the global allocator.
- // @param size Size of block to allocate. This will be rounded to the next power of two.
- CC_ALLOCATOR_INLINE void* allocate(size_t size)
- {
- _lazy_init();
-
- if (size < sizeof(intptr_t)) // always allocate at least enough space to store a pointer. this is
- size = sizeof(intptr_t); // so we can link the empty blocks together in the block allocator.
-
- // if the size is greater than what we determine to be a small block
- // size then fall through to calling the global allocator instead.
- if (size > _maxBlockSize)
- return ccAllocatorGlobal.allocate(size);
-
- // make sure the size fits into one of the
- // fixed sized block allocators we have above.
- size_t adjusted_size = AllocatorBase::nextPow2BlockSize(size);
-
- #define ALLOCATE(slot, size) \
- case size: \
- { \
- void* v = _smallBlockAllocators[slot]; \
- CC_ASSERT(nullptr != v); \
- auto a = (SType(size)*)v; \
- address = a->allocate(adjusted_size); \
- TRACK(slot, size, +=); \
- } \
- break;
-
- void* address = nullptr;
-
- switch (adjusted_size)
- {
- ALLOCATE(2, 4);
- ALLOCATE(3, 8);
- ALLOCATE(4, 16);
- ALLOCATE(5, 32);
- ALLOCATE(6, 64);
- ALLOCATE(7, 128);
- ALLOCATE(8, 256);
- ALLOCATE(9, 512);
- ALLOCATE(10, 1024);
- ALLOCATE(11, 2048);
- ALLOCATE(12, 4096);
- ALLOCATE(13, 8192);
- default:
- CC_ASSERT(false);
- break;
- }
-
- #undef ALLOCATE
-
- CC_ASSERT(adjusted_size < AllocatorBase::kDefaultAlignment || 0 == ((intptr_t)address & (AllocatorBase::kDefaultAlignment - 1)));
- CC_ASSERT(nullptr != address);
-
- return address;
- }
-
- // @brief Deallocate a block by choosing one of the fixed size block allocators
- // or defaulting to the global allocator if we do not own this block.
- CC_ALLOCATOR_INLINE void deallocate(void* address, size_t size = 0)
- {
- // if we didn't get a size, then we need to find the allocator
- // by asking each if they own the block. For allocators that
- // have few large pages, this is extremely fast.
- if (0 == size)
- {
- #define OWNS(slot, S, address) \
- case S: \
- { \
- void* v = _smallBlockAllocators[slot]; \
- if (v) \
- { \
- auto a = (SType(S)*)v; \
- if (a->owns(address)) \
- { \
- size = SType(S)::block_size; \
- break; \
- } \
- } \
- }
-
- // falls through until found
- switch (sizeof(uint32_t))
- {
- OWNS(2, 4, address);
- OWNS(3, 8, address);
- OWNS(4, 16, address);
- OWNS(5, 32, address);
- OWNS(6, 64, address);
- OWNS(7, 128, address);
- OWNS(8, 256, address);
- OWNS(9, 512, address);
- OWNS(10, 1024, address);
- OWNS(11, 2048, address);
- OWNS(12, 4096, address);
- OWNS(13, 8192, address);
- }
- }
-
- // if the size is greater than what we determine to be a small block
- // size then default to calling the global allocator instead.
- if (0 == size || size > _maxBlockSize)
- return ccAllocatorGlobal.deallocate(address, size);
-
- if (size < sizeof(intptr_t)) // always allocate at least enough space to store a pointer. this is
- size = sizeof(intptr_t); // so we can link the empty blocks together in the block allocator.
-
- // make sure the size fits into one of the
- // fixed sized block allocators we have above.
- size_t adjusted_size = AllocatorBase::nextPow2BlockSize(size);
-
- #define DEALLOCATE(slot, size, address) \
- case size: \
- { \
- void* v = _smallBlockAllocators[slot]; \
- CC_ASSERT(nullptr != v); \
- auto a = (SType(size)*)v; \
- a->deallocate(address, size); \
- TRACK(slot, size, -=); \
- } \
- break;
-
- switch (adjusted_size)
- {
- DEALLOCATE(2, 4, address);
- DEALLOCATE(3, 8, address);
- DEALLOCATE(4, 16, address);
- DEALLOCATE(5, 32, address);
- DEALLOCATE(6, 64, address);
- DEALLOCATE(7, 128, address);
- DEALLOCATE(8, 256, address);
- DEALLOCATE(9, 512, address);
- DEALLOCATE(10, 1024, address);
- DEALLOCATE(11, 2048, address);
- DEALLOCATE(12, 4096, address);
- DEALLOCATE(13, 8192, address);
- default:
- CC_ASSERT(false);
- }
-
- #undef DEALLOCATE
- }
-
- #if CC_ENABLE_ALLOCATOR_DIAGNOSTICS
- std::string diagnostics() const
- {
- std::stringstream s;
- size_t total = 0;
- for (auto i = 2; i <= kMaxSmallBlockPower; ++i)
- {
- auto a = _smallBlockAllocators[i];
- if (a)
- {
- total += _smallBlockAllocations[i];
- s << a->tag() << " allocated:" << _smallBlockAllocations[i] << "\n";
- }
- }
- s << "Total:" << total << "\n";
- return s.str();
- }
- size_t _highestCount;
- #endif
-
- protected:
-
- // @brief the max size of a block this allocator will pool before using global allocator
- size_t _maxBlockSize;
-
- // @brief array of small block allocators from 2^0 -> 2^kMaxSmallBlockPower
- AllocatorBase* _smallBlockAllocators[kMaxSmallBlockPower + 1];
-
- #if CC_ENABLE_ALLOCATOR_DIAGNOSTICS
- size_t _smallBlockAllocations[kMaxSmallBlockPower + 1];
- #endif
- };
- NS_CC_ALLOCATOR_END
- NS_CC_END
- /// @endcond
- #endif//CC_ALLOCATOR_STRATEGY_GLOBAL_SMALL_BLOCK_H
|