/*==LICENSE==* CyanWorlds.com Engine - MMOG client, server and tools Copyright (C) 2011 Cyan Worlds, Inc. This program is free software: you can redistribute it and/or modify it under the terms of the GNU 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 General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>. You can contact Cyan Worlds, Inc. by email legal@cyan.com or by snail mail at: Cyan Worlds, Inc. 14617 N Newport Hwy Mead, WA 99021 *==LICENSE==*/ #ifndef hsTempPointer_inc #define hsTempPointer_inc #include "hsMemory.h" #include "hsExceptions.h" template <class T> class hsTempPointer { private: T** fArray; UInt32 fCurrBlock; UInt32 fNumBlockAlloc; UInt32 fCurrElem; UInt32 fNumElemAlloc; UInt32 fGrowBy; // def = 0, to double UInt32 fMinSize; // def = 1 hsTempPointer<T>& operator=(const hsTempPointer<T>&); void IConsolidate(); void IGrow(); public: hsTempPointer(UInt32 minSize = 1, UInt32 growBy = 0); ~hsTempPointer(); void Reset(); T* Next(); T* Array(int n); }; template <class T> hsTempPointer<T>::~hsTempPointer() { int i; for( i = 0; i <= fCurrBlock; i++ ) delete [] fArray[i]; delete [] fArray; } template <class T> hsTempPointer<T>::hsTempPointer(UInt32 minSize, UInt32 growBy) { fGrowBy = growBy; fMinSize = minSize; fArray = TRACKED_NEW T*[2]; fNumBlockAlloc = 2; fCurrBlock = 0; fArray[fCurrBlock] = TRACKED_NEW T[fMinSize]; fNumElemAlloc = minSize; fCurrElem = 0; } template <class T> void hsTempPointer<T>::IConsolidate() { hsAssert(fCurrBlock > 0, "Shouldn't consolidate when nothing to do"); UInt32 numUsed = fCurrBlock * fNumElemAlloc + fCurrElem; UInt32 newSize = fNumElemAlloc; if( !fGrowBy ) { while( newSize <= numUsed ) newSize <<= 1; } else { while( newSize <= numUsed ) newSize += fGrowBy; } int i; for( i = 0; i <= fCurrBlock; i++ ) delete [] fArray[i]; fArray[0] = TRACKED_NEW T[newSize]; fNumElemAlloc = newSize; fCurrElem = 0; fCurrBlock = 0; } template <class T> void hsTempPointer<T>::IGrow() { if( ++fCurrBlock >= fNumBlockAlloc ) { T** newBlockArray = TRACKED_NEW T*[fNumBlockAlloc <<= 1]; HSMemory::BlockMove(fArray, newBlockArray, fCurrBlock * sizeof(*fArray)); delete [] fArray; fArray = newBlockArray; } fArray[fCurrBlock] = TRACKED_NEW T[fNumElemAlloc]; fCurrElem = 0; } template <class T> T* hsTempPointer<T>::Next() { if( fCurrElem >= fNumElemAlloc ) IGrow(); return fArray[fCurrBlock] + fCurrElem++; } template <class T> T* hsTempPointer<T>::Array(int n) { // minSize (on constructor) should be greater than max n hsDebugCode(hsThrowIfBadParam((UInt32)n > (UInt32)fNumElemAlloc);) if( fCurrElem + n >= fNumElemAlloc ) IGrow(); int idx = fCurrElem; fCurrElem += n; return fArray[fCurrBlock] + idx; } template <class T> void hsTempPointer<T>::Reset() { if( fCurrBlock > 0 ) IConsolidate(); fCurrElem = 0; } #endif // hsTempPointer_inc