2006-02-24 04:48:15 +00:00
|
|
|
/*
|
|
|
|
** tarray.h
|
|
|
|
** Templated, automatically resizing array
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
** Copyright 1998-2005 Randy Heit
|
|
|
|
** All rights reserved.
|
|
|
|
**
|
|
|
|
** Redistribution and use in source and binary forms, with or without
|
|
|
|
** modification, are permitted provided that the following conditions
|
|
|
|
** are met:
|
|
|
|
**
|
|
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
|
|
** documentation and/or other materials provided with the distribution.
|
|
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
** derived from this software without specific prior written permission.
|
|
|
|
**
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __TARRAY_H__
|
|
|
|
#define __TARRAY_H__
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
2006-05-04 03:49:46 +00:00
|
|
|
#include <assert.h>
|
|
|
|
#include <new>
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
#include "m_alloc.h"
|
|
|
|
|
|
|
|
|
|
|
|
// This function is called once for each entry in the TArray after it grows.
|
|
|
|
// The old entries will be immediately freed afterwards, so if they need to
|
|
|
|
// be destroyed, it needs to happen in this function.
|
|
|
|
template<class T>
|
|
|
|
void CopyForTArray (T &dst, T &src)
|
|
|
|
{
|
2006-05-04 03:49:46 +00:00
|
|
|
::new((void*)&dst) T(src);
|
|
|
|
src.~T();
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This function is called by Push to copy the the element to an unconstructed
|
|
|
|
// area of memory. Basically, if NeedsDestructor is overloaded to return true,
|
|
|
|
// then this should be overloaded to use placement new.
|
|
|
|
template<class T>
|
|
|
|
void ConstructInTArray (T *dst, const T &src)
|
|
|
|
{
|
2006-05-04 03:49:46 +00:00
|
|
|
::new((void*)dst) T(src);
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This function is much like the above function, except it is called when
|
|
|
|
// the array is explicitly enlarged without using Push.
|
|
|
|
template<class T>
|
|
|
|
void ConstructEmptyInTArray (T *dst)
|
|
|
|
{
|
2006-05-04 03:49:46 +00:00
|
|
|
::new((void*)dst) T;
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
class TArray
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
TArray ()
|
|
|
|
{
|
|
|
|
Most = 0;
|
|
|
|
Count = 0;
|
|
|
|
Array = NULL;
|
|
|
|
}
|
|
|
|
TArray (int max)
|
|
|
|
{
|
|
|
|
Most = max;
|
|
|
|
Count = 0;
|
2006-05-04 03:49:46 +00:00
|
|
|
Array = (T *)M_Malloc (sizeof(T)*max);
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
TArray (const TArray<T> &other)
|
|
|
|
{
|
|
|
|
DoCopy (other);
|
|
|
|
}
|
|
|
|
TArray<T> &operator= (const TArray<T> &other)
|
|
|
|
{
|
|
|
|
if (&other != this)
|
|
|
|
{
|
|
|
|
if (Array != NULL)
|
|
|
|
{
|
2006-05-04 03:49:46 +00:00
|
|
|
if (Count > 0)
|
|
|
|
{
|
|
|
|
DoDelete (0, Count-1);
|
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
free (Array);
|
|
|
|
}
|
|
|
|
DoCopy (other);
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
~TArray ()
|
|
|
|
{
|
|
|
|
if (Array)
|
|
|
|
{
|
2006-05-04 03:49:46 +00:00
|
|
|
if (Count > 0)
|
|
|
|
{
|
|
|
|
DoDelete (0, Count-1);
|
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
free (Array);
|
2006-05-04 03:49:46 +00:00
|
|
|
Array = NULL;
|
|
|
|
Count = 0;
|
|
|
|
Most = 0;
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
T &operator[] (unsigned int index) const
|
|
|
|
{
|
|
|
|
return Array[index];
|
|
|
|
}
|
|
|
|
unsigned int Push (const T &item)
|
|
|
|
{
|
2006-05-04 06:14:52 +00:00
|
|
|
Grow (1);
|
2006-02-24 04:48:15 +00:00
|
|
|
ConstructInTArray (&Array[Count], item);
|
|
|
|
return Count++;
|
|
|
|
}
|
|
|
|
bool Pop (T &item)
|
|
|
|
{
|
|
|
|
if (Count > 0)
|
|
|
|
{
|
|
|
|
item = Array[--Count];
|
|
|
|
DoDelete (Count, Count);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
void Delete (unsigned int index)
|
|
|
|
{
|
|
|
|
if (index < Count)
|
2006-05-04 03:49:46 +00:00
|
|
|
{
|
|
|
|
for (unsigned int i = index; i < Count - 1; ++i)
|
|
|
|
{
|
|
|
|
Array[i] = Array[i+1];
|
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
Count--;
|
2006-05-04 03:49:46 +00:00
|
|
|
DoDelete (Count, Count);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Inserts an item into the array, shifting elements as needed
|
|
|
|
void Insert (unsigned int index, const T &item)
|
|
|
|
{
|
|
|
|
if (index >= Count)
|
|
|
|
{
|
|
|
|
// Inserting somewhere past the end of the array, so we can
|
|
|
|
// just add it without moving things.
|
|
|
|
Resize (index + 1);
|
|
|
|
ConstructInTArray (&Array[index], item);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Inserting somewhere in the middle of the array, so make
|
|
|
|
// room for it and shift old entries out of the way.
|
|
|
|
|
|
|
|
Resize (Count + 1);
|
|
|
|
|
|
|
|
// Now copy items from the index and onward
|
|
|
|
for (unsigned int i = Count - 1; i-- > index; )
|
|
|
|
{
|
|
|
|
Array[i+1] = Array[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now put the new element in
|
|
|
|
DoDelete (index, index);
|
|
|
|
ConstructInTArray (&Array[index], item);
|
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
void ShrinkToFit ()
|
|
|
|
{
|
|
|
|
if (Most > Count)
|
|
|
|
{
|
|
|
|
Most = Count;
|
|
|
|
if (Most == 0)
|
|
|
|
{
|
|
|
|
if (Array != NULL)
|
|
|
|
{
|
|
|
|
free (Array);
|
|
|
|
Array = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DoResize ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Grow Array to be large enough to hold amount more entries without
|
|
|
|
// further growing.
|
|
|
|
void Grow (unsigned int amount)
|
|
|
|
{
|
|
|
|
if (Count + amount > Most)
|
|
|
|
{
|
|
|
|
const unsigned int choicea = Count + amount;
|
2006-05-04 06:14:52 +00:00
|
|
|
const unsigned int choiceb = Most = (Most >= 16) ? Most + Most / 2 : 16;
|
2006-02-24 04:48:15 +00:00
|
|
|
Most = (choicea > choiceb ? choicea : choiceb);
|
|
|
|
DoResize ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Resize Array so that it has exactly amount entries in use.
|
|
|
|
void Resize (unsigned int amount)
|
|
|
|
{
|
|
|
|
if (Count < amount)
|
|
|
|
{
|
2006-05-04 06:14:52 +00:00
|
|
|
// Adding new entries
|
2006-02-24 04:48:15 +00:00
|
|
|
Grow (amount - Count);
|
2006-05-04 06:14:52 +00:00
|
|
|
for (unsigned int i = Count; i < amount; ++i)
|
|
|
|
{
|
|
|
|
ConstructEmptyInTArray (&Array[i]);
|
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
2006-05-04 06:14:52 +00:00
|
|
|
else if (Count != amount)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-05-04 06:14:52 +00:00
|
|
|
// Deleting old entries
|
|
|
|
DoDelete (amount, Count - 1);
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
Count = amount;
|
|
|
|
}
|
|
|
|
// Reserves amount entries at the end of the array, but does nothing
|
|
|
|
// with them.
|
|
|
|
unsigned int Reserve (unsigned int amount)
|
|
|
|
{
|
2006-05-04 06:14:52 +00:00
|
|
|
Grow (amount);
|
2006-02-24 04:48:15 +00:00
|
|
|
unsigned int place = Count;
|
|
|
|
Count += amount;
|
|
|
|
return place;
|
|
|
|
}
|
|
|
|
unsigned int Size () const
|
|
|
|
{
|
|
|
|
return Count;
|
|
|
|
}
|
|
|
|
unsigned int Max () const
|
|
|
|
{
|
|
|
|
return Most;
|
|
|
|
}
|
|
|
|
void Clear ()
|
|
|
|
{
|
2006-05-04 03:49:46 +00:00
|
|
|
if (Count > 0)
|
|
|
|
{
|
|
|
|
DoDelete (0, Count-1);
|
|
|
|
Count = 0;
|
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
private:
|
|
|
|
T *Array;
|
|
|
|
unsigned int Most;
|
|
|
|
unsigned int Count;
|
|
|
|
|
|
|
|
void DoCopy (const TArray<T> &other)
|
|
|
|
{
|
|
|
|
Most = Count = other.Count;
|
|
|
|
if (Count != 0)
|
|
|
|
{
|
2006-05-04 03:49:46 +00:00
|
|
|
Array = (T *)M_Malloc (sizeof(T)*Most);
|
2006-02-24 04:48:15 +00:00
|
|
|
for (unsigned int i = 0; i < Count; ++i)
|
|
|
|
{
|
|
|
|
Array[i] = other.Array[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Array = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DoResize ()
|
|
|
|
{
|
2006-05-04 03:49:46 +00:00
|
|
|
T *newarray = (T *)M_Malloc (sizeof(T)*Most);
|
2006-02-24 04:48:15 +00:00
|
|
|
for (unsigned int i = 0; i < Count; ++i)
|
|
|
|
{
|
|
|
|
CopyForTArray (newarray[i], Array[i]);
|
|
|
|
}
|
|
|
|
free (Array);
|
|
|
|
Array = newarray;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DoDelete (unsigned int first, unsigned int last)
|
|
|
|
{
|
2006-05-04 03:49:46 +00:00
|
|
|
assert (last != ~0u);
|
|
|
|
for (unsigned int i = first; i <= last; ++i)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-05-04 03:49:46 +00:00
|
|
|
Array[i].~T();
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// An array with accessors that automatically grow the
|
|
|
|
// array as needed. But can still be used as a normal
|
|
|
|
// TArray if needed. Used by ACS world and global arrays.
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
class TAutoGrowArray : public TArray<T>
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
T GetVal (unsigned int index)
|
|
|
|
{
|
|
|
|
if (index >= this->Size())
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return (*this)[index];
|
|
|
|
}
|
|
|
|
void SetVal (unsigned int index, T val)
|
|
|
|
{
|
|
|
|
if (index >= this->Size())
|
|
|
|
{
|
|
|
|
this->Resize (index + 1);
|
|
|
|
}
|
|
|
|
(*this)[index] = val;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif //__TARRAY_H__
|