You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
335 lines
6.2 KiB
C++
335 lines
6.2 KiB
C++
//
|
|
// G A M E.O.N.E - LOW LEVEL LIB V1.0
|
|
// Copyright (C) 2001 LEVEL ONE ENTERTAINMENT,
|
|
// Licensed under the terms of LGPL.
|
|
//:---------------------------------------------------------------------------
|
|
//:Description
|
|
//
|
|
// LOW LEVEL LLQUEUE_DEQUEUE HEADERFILE
|
|
//
|
|
|
|
#ifndef _LLQUEUE_DEQUEUE_H
|
|
#define _LLQUEUE_DEQUEUE_H
|
|
|
|
|
|
#define LLQUEUE_DEQUEUE_DEFAULT_SIZE 32
|
|
#define EXPANDFAILED(s) DebugOutLn("Expand queue failed");
|
|
#define INDEXFAILED(s) DebugOutLn("Invalid Queue index");
|
|
|
|
|
|
/*
|
|
* deque
|
|
*/
|
|
template <class T>
|
|
class dequeue {
|
|
|
|
public:
|
|
dequeue();
|
|
virtual ~dequeue();
|
|
|
|
void Clear(void); // logically clear the queue
|
|
void Init(int max); // set size of queue (hint)
|
|
// Init destroys contents and resets all
|
|
void PutFront( T *element); // adds element on Front of queue
|
|
T *GetFront(void); // zero if queue empty
|
|
void PutBack( T *element); // adds element on Back of queue
|
|
T *GetBack(void); // zero if queue empty
|
|
void Put( T* element ) { PutBack(element);} // Put/Get Support
|
|
T* Get(void) { return GetBack();}
|
|
|
|
int InQueue(void); // returns number of elements queue
|
|
bool IsEmpty(void) const; // true if InQueue() returns zero
|
|
|
|
// iteration
|
|
void Set(int index); // sets iteration index
|
|
void Reset(void); // resets iteration index (same as Set(0) )
|
|
T *Next(void); // returns current element or 0 if no more left, Resets iterator
|
|
T *NextNoReset(void); // returns current element or 0 if no more left, Does not reset iterator
|
|
T* operator[](int i) const; // returns element at index i
|
|
T * SetAt(int i,T* newvalue); // direct access to element, replaces element
|
|
|
|
virtual void Extend(void); // called from Put*() if stacksize reaches max_elements
|
|
private:
|
|
int CalcIndex(int i) const;
|
|
void ExtendQueue(void);
|
|
T **m_array;
|
|
int m_iterator_index;
|
|
int m_num_elements;
|
|
int m_max_elements;
|
|
int m_HeadIndex;
|
|
int m_TailIndex;
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
* Clears logically all entries
|
|
* Caller must delete unused entries before calling Clear()
|
|
*/
|
|
template <class T>
|
|
inline void dequeue<T>::Clear(void)
|
|
{
|
|
m_num_elements=0;
|
|
m_iterator_index=0;
|
|
m_HeadIndex=0;
|
|
m_TailIndex=0;
|
|
}
|
|
|
|
|
|
/*
|
|
* resets iterator index
|
|
*/
|
|
template <class T>
|
|
inline void dequeue<T>::Reset(void)
|
|
{
|
|
m_iterator_index=0;
|
|
}
|
|
|
|
/*
|
|
* Calc Index
|
|
*/
|
|
template<class T>
|
|
inline int dequeue<T>::CalcIndex(int i) const
|
|
{
|
|
int Idx = m_HeadIndex + i;
|
|
if(Idx >= m_max_elements )
|
|
Idx -= m_max_elements;
|
|
return Idx;
|
|
}
|
|
|
|
/*
|
|
* set iterator index
|
|
*/
|
|
template <class T>
|
|
inline void dequeue<T>::Set(int index)
|
|
{
|
|
m_iterator_index=index<0?0:index;
|
|
}
|
|
|
|
/*
|
|
* returns current item
|
|
*/
|
|
template <class T>
|
|
inline T * dequeue<T>::Next(void)
|
|
{
|
|
if( m_iterator_index < m_num_elements )
|
|
{
|
|
int idx = CalcIndex(m_iterator_index++);
|
|
return m_array[idx];
|
|
}
|
|
Reset();
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* returns current item, does not Reset Iterator
|
|
* subsequent calls will return 0
|
|
*/
|
|
template <class T>
|
|
inline T * dequeue<T>::NextNoReset(void)
|
|
{
|
|
if( m_iterator_index < m_num_elements )
|
|
{
|
|
int idx = CalcIndex(m_iterator_index++);
|
|
return m_array[idx];
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* returns number of items of the normal queue
|
|
*/
|
|
template <class T>
|
|
inline int dequeue<T>::InQueue(void)
|
|
{
|
|
return m_num_elements;
|
|
}
|
|
|
|
|
|
/*
|
|
* get a element from the bottom of the stack
|
|
*/
|
|
template <class T>
|
|
inline T * dequeue<T>::GetBack(void)
|
|
{
|
|
if( m_num_elements > 0 )
|
|
{
|
|
m_num_elements--;
|
|
m_TailIndex--;
|
|
|
|
if(m_TailIndex < 0)
|
|
m_TailIndex += m_max_elements;
|
|
|
|
return m_array[m_TailIndex];
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* get a element from the bottom of the stack
|
|
*/
|
|
template <class T>
|
|
inline void dequeue<T>::PutBack(T* element)
|
|
{
|
|
if( m_num_elements < m_max_elements )
|
|
{
|
|
if( m_TailIndex == m_max_elements )
|
|
m_TailIndex = 0;
|
|
|
|
m_array[m_TailIndex] = element;
|
|
m_TailIndex++;
|
|
m_num_elements++;
|
|
}
|
|
else
|
|
{
|
|
Extend();
|
|
PutBack(element);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* get a element from the front of the stack
|
|
*/
|
|
template <class T>
|
|
inline T * dequeue<T>::GetFront(void)
|
|
{
|
|
if( m_num_elements > 0 )
|
|
{
|
|
if( m_HeadIndex == m_max_elements )
|
|
m_HeadIndex = 0;
|
|
|
|
T* element = m_array[m_HeadIndex];
|
|
|
|
m_HeadIndex++;
|
|
m_num_elements--;
|
|
|
|
return element;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* put a element on the stack
|
|
*/
|
|
template <class T>
|
|
inline void dequeue<T>::PutFront(T* element )
|
|
{
|
|
if( m_num_elements < m_max_elements )
|
|
{
|
|
m_num_elements++;
|
|
m_HeadIndex--;
|
|
if(m_HeadIndex < 0)
|
|
m_HeadIndex += m_max_elements;
|
|
|
|
m_array[m_HeadIndex] = element;
|
|
}
|
|
else
|
|
{
|
|
Extend();
|
|
PutFront(element);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Constructor
|
|
*/
|
|
template <class T>
|
|
dequeue<T>::dequeue()
|
|
{
|
|
m_array=0;
|
|
Clear();
|
|
m_max_elements=0;
|
|
}
|
|
|
|
template <class T>
|
|
dequeue<T>::~dequeue()
|
|
{
|
|
Clear();
|
|
if( m_array ) llmem_free(m_array);
|
|
m_array=0;
|
|
}
|
|
|
|
|
|
template <class T>
|
|
void dequeue<T>::Init( int imax )
|
|
{
|
|
Clear();
|
|
m_max_elements = imax>1?imax:LLQUEUE_DEQUEUE_DEFAULT_SIZE;
|
|
m_array = (T**)llmem_alloc(imax*sizeof(T*));
|
|
}
|
|
|
|
template <class T>
|
|
void dequeue<T>::Extend(void)
|
|
{
|
|
ExtendQueue();
|
|
}
|
|
|
|
template <class T>
|
|
void dequeue<T>::ExtendQueue(void)
|
|
{
|
|
int newsize = m_max_elements;
|
|
newsize += (newsize>>1);
|
|
if(newsize<LLQUEUE_DEQUEUE_DEFAULT_SIZE) newsize=LLQUEUE_DEQUEUE_DEFAULT_SIZE;
|
|
|
|
T** tmp=(T**)llmem_alloc(newsize*sizeof(T*));
|
|
if(!tmp) EXPANDFAILED(newsize*sizeof(T*));
|
|
|
|
if( m_num_elements )
|
|
{
|
|
if(m_TailIndex > m_HeadIndex)
|
|
memcpy(tmp, m_array + m_HeadIndex, m_num_elements*sizeof(T*));
|
|
else
|
|
{
|
|
int Rest = m_max_elements - m_HeadIndex;
|
|
memcpy(tmp, m_array + m_HeadIndex, Rest*sizeof(T*));
|
|
memcpy(tmp + Rest, m_array, m_TailIndex*sizeof(T*));
|
|
}
|
|
}
|
|
|
|
m_HeadIndex=0;
|
|
m_TailIndex=m_num_elements;
|
|
|
|
llmem_free(m_array);
|
|
m_array=tmp;
|
|
m_max_elements = newsize;
|
|
}
|
|
|
|
template<class T>
|
|
bool dequeue<T>::IsEmpty() const
|
|
{
|
|
return m_num_elements == 0 ? true : false;
|
|
}
|
|
|
|
template<class T>
|
|
T* dequeue<T>::operator[](int i) const
|
|
{
|
|
if( i >= 0 && i < m_num_elements )
|
|
{
|
|
return m_array[CalcIndex(i)];
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
template<class T>
|
|
T * dequeue<T>::SetAt(int i,T* newvalue)
|
|
{
|
|
T* oldvalue;
|
|
if( i < m_num_elements )
|
|
{
|
|
int index = CalcIndex(i);
|
|
oldvalue = m_array[index];
|
|
m_array[index] = newvalue;
|
|
return oldvalue;
|
|
}
|
|
INDEXFAILED(i);
|
|
return 0;
|
|
}
|
|
|
|
|
|
#endif
|