mirror of
https://github.com/ZDoom/qzdoom-gpl.git
synced 2024-11-17 09:41:21 +00:00
295 lines
6.5 KiB
C
295 lines
6.5 KiB
C
|
/*
|
||
|
** 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>
|
||
|
#include "m_alloc.h"
|
||
|
|
||
|
// Specialize this function for any type that needs to have its destructor called.
|
||
|
template<class T>
|
||
|
bool NeedsDestructor ()
|
||
|
{
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// 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)
|
||
|
{
|
||
|
dst = src;
|
||
|
if (NeedsDestructor<T>())
|
||
|
{
|
||
|
src.~T();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// 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)
|
||
|
{
|
||
|
//new (dst) T(src);
|
||
|
*dst = src;
|
||
|
}
|
||
|
|
||
|
// 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)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
template <class T>
|
||
|
class TArray
|
||
|
{
|
||
|
public:
|
||
|
TArray ()
|
||
|
{
|
||
|
Most = 0;
|
||
|
Count = 0;
|
||
|
Array = NULL;
|
||
|
}
|
||
|
TArray (int max)
|
||
|
{
|
||
|
Most = max;
|
||
|
Count = 0;
|
||
|
Array = (T *)Malloc (sizeof(T)*max);
|
||
|
}
|
||
|
TArray (const TArray<T> &other)
|
||
|
{
|
||
|
DoCopy (other);
|
||
|
}
|
||
|
TArray<T> &operator= (const TArray<T> &other)
|
||
|
{
|
||
|
if (&other != this)
|
||
|
{
|
||
|
if (Array != NULL)
|
||
|
{
|
||
|
DoDelete (0, Count-1);
|
||
|
free (Array);
|
||
|
}
|
||
|
DoCopy (other);
|
||
|
}
|
||
|
return *this;
|
||
|
}
|
||
|
~TArray ()
|
||
|
{
|
||
|
if (Array)
|
||
|
{
|
||
|
DoDelete (0, Count-1);
|
||
|
free (Array);
|
||
|
}
|
||
|
}
|
||
|
T &operator[] (unsigned int index) const
|
||
|
{
|
||
|
return Array[index];
|
||
|
}
|
||
|
unsigned int Push (const T &item)
|
||
|
{
|
||
|
if (Count >= Most)
|
||
|
{
|
||
|
Most = (Most >= 16) ? Most + Most / 2 : 16;
|
||
|
DoResize ();
|
||
|
}
|
||
|
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)
|
||
|
{
|
||
|
DoDelete (index, index);
|
||
|
if (index < Count-1)
|
||
|
memmove (Array + index, Array + index + 1, (Count - index - 1) * sizeof(T));
|
||
|
if (index < Count)
|
||
|
Count--;
|
||
|
}
|
||
|
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;
|
||
|
const unsigned int choiceb = Most + Most/2;
|
||
|
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)
|
||
|
{
|
||
|
Grow (amount - Count);
|
||
|
}
|
||
|
if (NeedsDestructor<T>())
|
||
|
{
|
||
|
for (unsigned int i = Count; i < amount; ++i)
|
||
|
{
|
||
|
ConstructEmptyInTArray (&Array[i]);
|
||
|
}
|
||
|
}
|
||
|
Count = amount;
|
||
|
}
|
||
|
// Reserves amount entries at the end of the array, but does nothing
|
||
|
// with them.
|
||
|
unsigned int Reserve (unsigned int amount)
|
||
|
{
|
||
|
if (Count + amount > Most)
|
||
|
{
|
||
|
Grow (amount);
|
||
|
}
|
||
|
unsigned int place = Count;
|
||
|
Count += amount;
|
||
|
return place;
|
||
|
}
|
||
|
unsigned int Size () const
|
||
|
{
|
||
|
return Count;
|
||
|
}
|
||
|
unsigned int Max () const
|
||
|
{
|
||
|
return Most;
|
||
|
}
|
||
|
void Clear ()
|
||
|
{
|
||
|
DoDelete (0, Count-1);
|
||
|
Count = 0;
|
||
|
}
|
||
|
private:
|
||
|
T *Array;
|
||
|
unsigned int Most;
|
||
|
unsigned int Count;
|
||
|
|
||
|
void DoCopy (const TArray<T> &other)
|
||
|
{
|
||
|
Most = Count = other.Count;
|
||
|
if (Count != 0)
|
||
|
{
|
||
|
Array = (T *)Malloc (sizeof(T)*Most);
|
||
|
for (unsigned int i = 0; i < Count; ++i)
|
||
|
{
|
||
|
Array[i] = other.Array[i];
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
Array = NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void DoResize ()
|
||
|
{
|
||
|
T *newarray = (T *)Malloc (sizeof(T)*Most);
|
||
|
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)
|
||
|
{
|
||
|
if (NeedsDestructor<T>())
|
||
|
{
|
||
|
for (unsigned int i = first; i <= last; ++i)
|
||
|
{
|
||
|
Array[i].~T();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// 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__
|