|
|
|
/*==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/>.
|
|
|
|
|
|
|
|
Additional permissions under GNU GPL version 3 section 7
|
|
|
|
|
|
|
|
If you modify this Program, or any covered work, by linking or
|
|
|
|
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
|
|
|
|
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
|
|
|
|
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
|
|
|
|
(or a modified version of those libraries),
|
|
|
|
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
|
|
|
|
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
|
|
|
|
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
|
|
|
|
licensors of this Program grant you additional
|
|
|
|
permission to convey the resulting work. Corresponding Source for a
|
|
|
|
non-source form of such a combination shall include the source code for
|
|
|
|
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
|
|
|
|
work.
|
|
|
|
|
|
|
|
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_t fCurrBlock;
|
|
|
|
uint32_t fNumBlockAlloc;
|
|
|
|
|
|
|
|
uint32_t fCurrElem;
|
|
|
|
uint32_t fNumElemAlloc;
|
|
|
|
|
|
|
|
uint32_t fGrowBy; // def = 0, to double
|
|
|
|
uint32_t fMinSize; // def = 1
|
|
|
|
|
|
|
|
hsTempPointer<T>& operator=(const hsTempPointer<T>&);
|
|
|
|
|
|
|
|
void IConsolidate();
|
|
|
|
void IGrow();
|
|
|
|
|
|
|
|
public:
|
|
|
|
hsTempPointer(uint32_t minSize = 1, uint32_t 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_t minSize, uint32_t 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_t numUsed = fCurrBlock * fNumElemAlloc + fCurrElem;
|
|
|
|
|
|
|
|
uint32_t 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_t)n > (uint32_t)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
|