// NeL - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 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 Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #ifndef NL_BLOCK_MEMORY_H #define NL_BLOCK_MEMORY_H #include "types_nl.h" #include #include #include "debug.h" #include "mutex.h" namespace NLMISC { // *************************************************************************** /// See CBlockMemory::Purge extern bool NL3D_BlockMemoryAssertOnPurge; // =true. // *************************************************************************** /** * Block memory allocation * * This memory manager is a fast memory allocator, doing same thing as new/delete. It works by blocks. Blocks are always * allocated, never deleted. Allocation/free are in O(1). * * Elements with sizeof(T) class CBlockMemory { public: /// Constructor CBlockMemory(uint blockSize= 16) { nlassert(blockSize); _BlockSize= blockSize; _EltSize= std::max((uint)sizeof(T), (uint)sizeof(void*)); _NextFreeElt= NULL; _NAllocatedElts= 0; } // just copy setup from other blockMemory, don't copy data! CBlockMemory(const CBlockMemory &other) { _BlockSize= other._BlockSize; // if other block is rebinded, don't copy its rebinded size. _EltSize= (uint)std::max(sizeof(T), sizeof(void*)); // No elts allocated _NextFreeElt= NULL; _NAllocatedElts= 0; } /** purge() */ ~CBlockMemory() { purge(); } /// allocate an element. ctor is called. T* allocate() { _Mutex.enter(); // if not enough memory, aloc a block. if(!_NextFreeElt) { _Blocks.push_front(CBlock()); buildBlock(*_Blocks.begin()); // new free elt points to the beginning of this block. _NextFreeElt= (*_Blocks.begin()).Data; #ifdef NL_DEBUG // if debug, must decal for begin check. _NextFreeElt= (uint32*)_NextFreeElt + 1; #endif } // choose next free elt. nlassert(_NextFreeElt); T* ret= (T*)_NextFreeElt; // update _NextFreeElt, so it points to the next free element. _NextFreeElt= *(void**)_NextFreeElt; // construct the allocated element. if( __ctor_dtor__ ) new (ret) T; // some simple Check. #ifdef NL_DEBUG uint32 *checkStart= (uint32*)(void*)ret-1; uint32 *checkEnd = (uint32*)((uint8*)(void*)ret+_EltSize); nlassert( *checkStart == CheckDeletedIdent); nlassert( *checkEnd == CheckDeletedIdent); // if ok, mark this element as allocated. *checkStart= CheckAllocatedIdent; *checkEnd = CheckAllocatedIdent; #endif _NAllocatedElts++; _Mutex.leave(); return ret; } /// delete an element allocated with this manager. dtor is called. NULL is tested. void free(T* ptr) { if(!ptr) return; _Mutex.enter(); // some simple Check. nlassert(_NAllocatedElts>0); #ifdef NL_DEBUG uint32 *checkStart= (uint32*)(void*)ptr-1; uint32 *checkEnd = (uint32*)((uint8*)(void*)ptr+_EltSize); nlassert( *checkStart == CheckAllocatedIdent); nlassert( *checkEnd == CheckAllocatedIdent); // if ok, mark this element as deleted. *checkStart = *checkEnd = uint32(CheckDeletedIdent); #endif // destruct the element. if( __ctor_dtor__ ) ptr->~T(); // just append this freed element to the list. *(void**)ptr= _NextFreeElt; _NextFreeElt= (void*) ptr; _NAllocatedElts--; _Mutex.leave(); } /** delete all blocks, freeing all memory. It is an error to purge() or delete a CBlockMemory, while elements * still remains!! You must free your elements with free(). * NB: you can disable this assert if you set NL3D_BlockMemoryAssertOnPurge to false * (good to quit a program quickly without uninitialize). */ void purge () { _Mutex.enter(); if(NL3D_BlockMemoryAssertOnPurge) nlassert(_NAllocatedElts==0); while(_Blocks.begin()!=_Blocks.end()) { releaseBlock(*_Blocks.begin()); _Blocks.erase(_Blocks.begin()); } _NextFreeElt= NULL; _NAllocatedElts= 0; _Mutex.leave(); } // ******************** public: // This is to be used with CSTLBlockAllocator only!!! It changes the size of an element!! void __stl_alloc_changeEltSize(uint eltSize) { _Mutex.enter(); // must not be used with object ctor/dtor behavior. nlassert(__ctor_dtor__ == false); // format size. eltSize= std::max((uint)eltSize, (uint)sizeof(void*)); // if not the same size as before if(_EltSize!= eltSize) { // verify that rebind is made before any allocation!! nlassert(_Blocks.empty()); // change the size. _EltSize= eltSize; } _Mutex.leave(); }; // This is to be used with CSTLBlockAllocator only!!! uint __stl_alloc_getEltSize() const { return _EltSize; } private: /// size of a block. uint _BlockSize; /// size of an element in the block. uint _EltSize; /// number of elements allocated. sint _NAllocatedElts; /// next free element. void *_NextFreeElt; /// Must be ThreadSafe (eg: important for 3D PointLight and list of transform) CFastMutex _Mutex; /// a block. struct CBlock { /// The data allocated. void *Data; }; /// list of blocks. std::list _Blocks; /// For debug only, check ident. enum TCheckIdent { CheckAllocatedIdent= 0x01234567, CheckDeletedIdent= 0x89ABCDEF }; private: void buildBlock(CBlock &block) { uint i; uint32 nodeSize= _EltSize; #ifdef NL_DEBUG // must allocate more size for mem checks in debug. nodeSize+= 2*sizeof(uint32); #endif // allocate. block.Data = (void*)new uint8 [_BlockSize * nodeSize]; // by default, all elements are not allocated, build the list of free elements. void *ptr= block.Data; #ifdef NL_DEBUG // if debug, must decal for begin check. ptr= (uint32*)ptr + 1; #endif for(i=0; i<_BlockSize-1; i++) { // next elt. void *next= (uint8*)ptr + nodeSize; // points to the next element in this array. *(void**)ptr= next; // next. ptr= next; } // last element points to NULL. *(void**)ptr= NULL; // If debug, must init all check values to CheckDeletedIdent. #ifdef NL_DEBUG ptr= block.Data; // must decal for begin check. ptr= (uint32*)ptr + 1; // fill all nodes. for(i=0; i<_BlockSize; i++) { uint32 *checkStart= (uint32*)ptr-1; uint32 *checkEnd = (uint32*)((uint8*)ptr+_EltSize); // mark this element as deleted. *checkStart = *checkEnd = uint32(CheckDeletedIdent); // next elt. ptr= (uint8*)ptr + nodeSize; } #endif } void releaseBlock(CBlock &block) { delete [] ((uint8*)block.Data); } }; } // NLMISC #endif // NL_BLOCK_MEMORY_H /* End of block_memory.h */