2006-02-24 04:48:15 +00:00
|
|
|
/*
|
|
|
|
** tarray.h
|
|
|
|
** Templated, automatically resizing array
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
2006-06-11 01:37:00 +00:00
|
|
|
** Copyright 1998-2006 Randy Heit
|
2006-02-24 04:48:15 +00:00
|
|
|
** 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"
|
|
|
|
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
class TArray
|
|
|
|
{
|
|
|
|
public:
|
2006-05-06 23:43:44 +00:00
|
|
|
////////
|
|
|
|
// This is a dummy constructor that does nothing. The purpose of this
|
|
|
|
// is so you can create a global TArray in the data segment that gets
|
|
|
|
// used by code before startup without worrying about the constructor
|
|
|
|
// resetting it after it's already been used. You MUST NOT use it for
|
|
|
|
// heap- or stack-allocated TArrays.
|
|
|
|
enum ENoInit
|
|
|
|
{
|
|
|
|
NoInit
|
|
|
|
};
|
|
|
|
TArray (ENoInit dummy)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
////////
|
2006-02-24 04:48:15 +00:00
|
|
|
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-05-16 02:50:18 +00:00
|
|
|
::new((void*)&Array[Count]) T(item);
|
2006-02-24 04:48:15 +00:00
|
|
|
return Count++;
|
|
|
|
}
|
|
|
|
bool Pop (T &item)
|
|
|
|
{
|
|
|
|
if (Count > 0)
|
|
|
|
{
|
|
|
|
item = Array[--Count];
|
2006-05-16 02:50:18 +00:00
|
|
|
Array[Count].~T();
|
2006-02-24 04:48:15 +00:00
|
|
|
return true;
|
2006-12-29 03:38:37 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2006-02-24 04:48:15 +00:00
|
|
|
void Delete (unsigned int index)
|
|
|
|
{
|
|
|
|
if (index < Count)
|
2006-05-04 03:49:46 +00:00
|
|
|
{
|
2006-05-16 02:50:18 +00:00
|
|
|
Array[index].~T();
|
2006-05-16 17:53:15 +00:00
|
|
|
memmove (&Array[index], &Array[index+1], sizeof(T)*(Count - index - 1));
|
2006-02-24 04:48:15 +00:00
|
|
|
Count--;
|
2006-05-04 03:49:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// 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);
|
2006-05-16 02:50:18 +00:00
|
|
|
::new ((void *)&Array[index]) T(item);
|
2006-05-04 03:49:46 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2006-05-16 02:50:18 +00:00
|
|
|
// Inserting somewhere in the middle of the array,
|
|
|
|
// so make room for it
|
2006-05-04 03:49:46 +00:00
|
|
|
Resize (Count + 1);
|
|
|
|
|
2006-05-16 02:50:18 +00:00
|
|
|
// Now move items from the index and onward out of the way
|
|
|
|
memmove (&Array[index+1], &Array[index], sizeof(T)*(Count - index - 1));
|
2006-05-04 03:49:46 +00:00
|
|
|
|
2006-05-16 02:50:18 +00:00
|
|
|
// And put the new element in
|
|
|
|
::new ((void *)&Array[index]) T(item);
|
2006-05-04 03:49:46 +00:00
|
|
|
}
|
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)
|
|
|
|
{
|
2006-05-16 02:50:18 +00:00
|
|
|
::new((void *)&Array[i]) T;
|
2006-05-04 06:14:52 +00:00
|
|
|
}
|
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-06 23:43:44 +00:00
|
|
|
size_t allocsize = sizeof(T)*Most;
|
2006-05-16 02:50:18 +00:00
|
|
|
Array = (T *)M_Realloc (Array, allocsize);
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
{
|
2006-07-31 10:22:53 +00:00
|
|
|
if ((int)index < 0) return; // These always result in an out of memory condition.
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
if (index >= this->Size())
|
|
|
|
{
|
|
|
|
this->Resize (index + 1);
|
|
|
|
}
|
|
|
|
(*this)[index] = val;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif //__TARRAY_H__
|