/*==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 .
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==*/
/*****************************************************************************
*
* $/Plasma20/Sources/Plasma/NucleusLib/pnUtils/Private/pnUtArray.h
*
***/
#ifndef PLASMA20_SOURCES_PLASMA_NUCLEUSLIB_PNUTILS_PRIVATE_PNUTARRAY_H
#define PLASMA20_SOURCES_PLASMA_NUCLEUSLIB_PNUTILS_PRIVATE_PNUTARRAY_H
#include "Pch.h"
#include "pnUtSort.h"
/****************************************************************************
*
* Macros
*
***/
#define ARRAY(type) TArray< type, TArrayCopyBits< type > >
#define ARRAYOBJ(type) TArray< type, TArrayCopyObject< type > >
#define FARRAY(type) TFArray< type, TArrayCopyBits< type > >
#define FARRAYOBJ(type) TFArray< type, TArrayCopyObject< type > >
/****************************************************************************
*
* CBuffer
*
***/
template
class TBuffer {
protected:
T * m_data;
public:
inline TBuffer ();
inline TBuffer (unsigned count);
inline TBuffer (const void * source, unsigned count);
inline TBuffer (const TBuffer & source);
inline ~TBuffer ();
inline TBuffer & operator= (const TBuffer & source);
inline bool operator== (const TBuffer & source) const;
inline T & operator[] (unsigned index);
inline T operator[] (unsigned index) const;
inline void Attach (T * source, unsigned count);
inline unsigned Bytes () const;
inline void Clear ();
inline unsigned Count () const;
inline T * Detach ();
inline void Fill (uint8_t value);
inline T * Ptr ();
inline const T * Ptr () const;
inline void Set (const T * source, unsigned count);
inline void SetBytes (unsigned bytes);
inline void SetCount (unsigned count);
inline void Zero ();
};
//===========================================================================
template
TBuffer::TBuffer () {
m_data = nil;
}
//===========================================================================
template
TBuffer::TBuffer (unsigned count) {
m_data = nil;
SetCount(count);
}
//===========================================================================
template
TBuffer::TBuffer (const void * source, unsigned count) {
m_data = nil;
SetCount(count);
memcpy(m_data, source, count * sizeof(T));
}
//===========================================================================
template
TBuffer::TBuffer (const TBuffer & source) {
m_data = nil;
unsigned bytes = source.Bytes();
SetBytes(bytes);
if (bytes)
memcpy(m_data, source.m_data, bytes);
}
//===========================================================================
template
TBuffer::~TBuffer () {
if (m_data)
free(m_data);
}
//===========================================================================
template
TBuffer & TBuffer::operator= (const TBuffer & source) {
unsigned newBytes = source.Bytes();
if (newBytes != Bytes())
SetBytes(newBytes);
if (&source != this)
memcpy(m_data, source.m_data, newBytes);
return *this;
}
//===========================================================================
template
bool TBuffer::operator== (const TBuffer & source) const {
unsigned size = _m_size(m_data);
return (size == _m_size(source.m_data)) && !memcmp(m_data, source.m_data, size);
}
//===========================================================================
template
T & TBuffer::operator[] (unsigned index) {
ASSERT(index < Count());
return m_data[index];
}
//===========================================================================
template
T TBuffer::operator[] (unsigned index) const {
ASSERT(index < Count());
return m_data[index];
}
//===========================================================================
template
void TBuffer::Attach (T * source, unsigned count) {
if (m_data)
free(m_data);
m_data = source;
ASSERT(_m_size(source) >= count * sizeof(T));
}
//===========================================================================
template
unsigned TBuffer::Bytes () const {
return m_data ? _m_size(m_data) : 0;
}
//===========================================================================
template
void TBuffer::Clear () {
if (m_data) {
free(m_data);
m_data = nil;
}
}
//===========================================================================
template
unsigned TBuffer::Count () const {
return m_data ? (_m_size(m_data) / sizeof(T)) : 0;
}
//===========================================================================
template
T * TBuffer::Detach () {
T * result = m_data;
m_data = nil;
return result;
}
//===========================================================================
template
void TBuffer::Fill (uint8_t value) {
if (m_data)
memset(m_data, value, Bytes());
}
//===========================================================================
template
T * TBuffer::Ptr () {
return m_data;
}
//===========================================================================
template
const T * TBuffer::Ptr () const {
return m_data;
}
//===========================================================================
template
void TBuffer::Set (const T * source, unsigned count) {
SetCount(count);
memcpy(m_data, source, count * sizeof(T));
}
//===========================================================================
template
void TBuffer::SetBytes (unsigned bytes) {
if (bytes)
m_data = (T *)realloc(m_data, bytes);
else if (m_data) {
free(m_data);
m_data = nil;
}
}
//===========================================================================
template
void TBuffer::SetCount (unsigned count) {
SetBytes(count * sizeof(T));
}
//===========================================================================
template
void TBuffer::Zero () {
if (m_data)
memset(m_data, 0, Bytes());
}
typedef TBuffer CBuffer;
/****************************************************************************
*
* CBaseArray
*
***/
class CBaseArray {
protected:
unsigned CalcAllocGrowth (unsigned newAlloc, unsigned oldAlloc, unsigned * chunkSize);
void * ReallocPtr (void * ptr, unsigned bytes);
};
/****************************************************************************
*
* TArrayCopyBits
*
***/
template
class TArrayCopyBits {
public:
inline static void Assign (T * dest, const T source[], unsigned count);
inline static void Construct (T * dest) { }
inline static void Construct (T * dest, unsigned count) { }
inline static void CopyConstruct (T * dest, const T & source);
inline static void CopyConstruct (T * dest, const T source[], unsigned count);
inline static void Destruct (T * dest) { }
inline static void Destruct (T * dest, unsigned count) { }
};
//===========================================================================
template
void TArrayCopyBits::Assign (T * dest, const T source[], unsigned count) {
memmove(dest, source, count * sizeof(T));
}
//===========================================================================
template
void TArrayCopyBits::CopyConstruct (T * dest, const T & source) {
*dest = source;
}
//===========================================================================
template
void TArrayCopyBits::CopyConstruct (T * dest, const T source[], unsigned count) {
ASSERT((dest + count <= source) || (source + count <= dest));
memcpy(dest, source, count * sizeof(T));
}
/****************************************************************************
*
* TArrayCopyObject
*
***/
template
class TArrayCopyObject {
public:
inline static void Assign (T * dest, const T source[], unsigned count);
inline static void Construct (T * dest);
inline static void Construct (T * dest, unsigned count);
inline static void CopyConstruct (T * dest, const T & source);
inline static void CopyConstruct (T * dest, const T source[], unsigned count);
inline static void Destruct (T * dest);
inline static void Destruct (T * dest, unsigned count);
};
//===========================================================================
template
void TArrayCopyObject::Assign (T * dest, const T source[], unsigned count) {
if (dest > source)
for (unsigned loop = count; loop--; )
dest[loop] = source[loop];
else if (dest < source)
for (unsigned loop = 0; loop < count; ++loop)
dest[loop] = source[loop];
}
//===========================================================================
template
void TArrayCopyObject::Construct (T * dest) {
new(dest) T;
}
//===========================================================================
template
void TArrayCopyObject::Construct (T * dest, unsigned count) {
for (unsigned loop = 0; loop < count; ++loop)
new(&dest[loop]) T;
}
//===========================================================================
template
void TArrayCopyObject::CopyConstruct (T * dest, const T & source) {
new(dest) T(source);
}
//===========================================================================
template
void TArrayCopyObject::CopyConstruct (T * dest, const T source[], unsigned count) {
ASSERT((dest + count <= source) || (source + count <= dest));
for (unsigned loop = 0; loop < count; ++loop)
new(&dest[loop]) T(source[loop]);
}
//===========================================================================
template
void TArrayCopyObject::Destruct (T * dest) {
dest->~T();
}
//===========================================================================
template
void TArrayCopyObject::Destruct (T * dest, unsigned count) {
for (unsigned loop = count; loop--; )
dest[loop].~T();
}
/****************************************************************************
*
* TFArray
*
***/
template
class TFArray : protected CBaseArray {
protected:
T * m_data;
unsigned m_alloc;
unsigned m_count;
inline void AdjustSize (unsigned newAlloc, unsigned newCount);
public:
inline TFArray ();
inline TFArray (unsigned count);
inline TFArray (const T * source, unsigned count);
inline TFArray (const TFArray & source);
inline ~TFArray ();
inline TFArray & operator= (const TFArray & source);
inline bool operator== (const TFArray & source) const;
inline T & operator[] (unsigned index);
inline const T & operator[] (unsigned index) const;
inline void Attach (T * source, unsigned count);
inline void AttachTemp (T * source, unsigned count);
inline unsigned Bytes () const;
inline void Clear ();
inline unsigned Count () const;
inline T * Detach ();
inline void Fill (uint8_t value);
inline T * Ptr ();
inline const T * Ptr () const;
inline void Set (const T * source, unsigned count);
inline void SetArray (const TFArray & source);
inline void SetCount (unsigned count);
inline T * Term ();
inline const T * Term () const;
inline T * Top ();
inline const T * Top () const;
inline void Zero ();
inline void ZeroCount ();
inline void ZeroRange (unsigned index, unsigned count);
};
//===========================================================================
template
TFArray::TFArray () {
m_alloc = 0;
m_count = 0;
m_data = nil;
}
//===========================================================================
template
TFArray::TFArray (unsigned count) {
m_alloc = m_count = count;
if (count) {
m_data = (T *)malloc(count * sizeof(T));
C::Construct(m_data, count);
}
else
m_data = nil;
}
//===========================================================================
template
TFArray::TFArray (const T * source, unsigned count) {
m_alloc = m_count = count;
if (count) {
m_data = (T *)malloc(count * sizeof(T));
C::CopyConstruct(m_data, source, count);
}
else
m_data = nil;
}
//===========================================================================
template
TFArray::TFArray (const TFArray & source) {
m_alloc = m_count = source.m_count;
if (m_count) {
m_data = (T *)malloc(m_count * sizeof(T));
C::CopyConstruct(m_data, source.m_data, m_count);
}
else
m_data = nil;
}
//===========================================================================
template
TFArray::~TFArray () {
Clear();
}
//===========================================================================
template
TFArray & TFArray::operator= (const TFArray & source) {
if (&source == this)
return *this;
AdjustSize(source.m_count, 0);
C::CopyConstruct(m_data, source.m_data, source.m_count);
m_count = source.m_count;
return *this;
}
//===========================================================================
template
inline bool TFArray::operator== (const TFArray & source) const {
if (m_count != source.m_count)
return false;
for (unsigned index = 0; index < m_count; ++index)
if (!((*this)[index] == source[index]))
return false;
return true;
}
//===========================================================================
template
T & TFArray::operator[] (unsigned index) {
ASSERT(index < m_count);
return m_data[index];
}
//===========================================================================
template
const T & TFArray::operator[] (unsigned index) const {
ASSERT(index < m_count);
return m_data[index];
}
//===========================================================================
template
void TFArray::AdjustSize (unsigned newAlloc, unsigned newCount) {
// Destruct elements if the array is shrinking
if (m_count > newCount) {
C::Destruct(m_data + newCount, m_count - newCount);
m_count = newCount;
}
// Change the memory allocation size if necessary
if (m_alloc != newAlloc) {
T * newData = (T *)ReallocPtr(m_data, newAlloc * sizeof(T));
if (newData != m_data) {
C::CopyConstruct(newData, m_data, m_count);
C::Destruct(m_data, m_count);
if (m_data)
free(m_data);
}
m_alloc = newAlloc;
m_data = newData;
}
// Construct elements if the array is growing
if (m_count < newCount) {
C::Construct(m_data + m_count, newCount - m_count);
m_count = newCount;
}
}
//===========================================================================
template
void TFArray::Attach (T * source, unsigned count) {
C::Destruct(m_data, m_count);
if (m_data)
free(m_data);
m_data = source;
m_alloc = _m_size(source) / sizeof(T);
m_count = count;
ASSERT(m_alloc >= m_count);
}
//===========================================================================
template
void TFArray::AttachTemp (T * source, unsigned count) {
C::Destruct(m_data, m_count);
if (m_data)
free(m_data);
m_data = source;
m_alloc = count;
m_count = count;
}
//===========================================================================
template
unsigned TFArray::Bytes () const {
return m_count * sizeof(T);
}
//===========================================================================
template
void TFArray::Clear () {
C::Destruct(m_data, m_count);
if (m_data)
free(m_data);
m_data = nil;
m_alloc = m_count = 0;
}
//===========================================================================
template
unsigned TFArray::Count () const {
return m_count;
}
//===========================================================================
template
T * TFArray::Detach () {
T * result = m_data;
m_data = nil;
m_alloc = 0;
m_count = 0;
return result;
}
//===========================================================================
template
void TFArray::Fill (uint8_t value) {
C::Destruct(m_data, m_count);
memset(m_data, value, m_count * sizeof(T));
C::Construct(m_data, m_count);
}
//===========================================================================
template
T * TFArray::Ptr () {
return m_data;
}
//===========================================================================
template
const T * TFArray::Ptr () const {
return m_data;
}
//===========================================================================
template
void TFArray::Set (const T * source, unsigned count) {
AdjustSize(count, 0);
C::CopyConstruct(m_data, source, count);
m_count = count;
}
//===========================================================================
template
void TFArray::SetArray (const TFArray & source) {
AdjustSize(source.m_count, 0);
C::CopyConstruct(m_data, source.m_data, source.m_count);
m_count = source.m_count;
}
//===========================================================================
template
void TFArray::SetCount (unsigned count) {
AdjustSize(count, count);
}
//===========================================================================
template
T * TFArray::Term () {
return m_data + m_count;
}
//===========================================================================
template
const T * TFArray::Term () const {
return m_data + m_count;
}
//===========================================================================
template
T * TFArray::Top () {
ASSERT(m_count);
return m_data + m_count - 1;
}
//===========================================================================
template
const T * TFArray::Top () const {
ASSERT(m_count);
return m_data + m_count - 1;
}
//===========================================================================
template
void TFArray::Zero () {
C::Destruct(m_data, m_count);
memset(m_data, 0, m_count * sizeof(T));
C::Construct(m_data, m_count);
}
//===========================================================================
template
void TFArray::ZeroCount () {
C::Destruct(m_data, m_count);
m_count = 0;
}
//===========================================================================
template
void TFArray::ZeroRange (unsigned index, unsigned count) {
ASSERT(index + count <= m_count);
C::Destruct(m_data + index, count);
memset(m_data + index, 0, count * sizeof(T));
C::Construct(m_data + index, count);
}
/****************************************************************************
*
* TArray
*
***/
template
class TArray : public TFArray {
private:
unsigned m_chunkSize;
inline void AdjustSizeChunked (unsigned newAlloc, unsigned newCount);
public:
inline TArray ();
inline TArray (const char file[], int line);
inline TArray (unsigned count);
inline TArray (const T * source, unsigned count);
inline TArray (const TArray & source);
inline TArray & operator= (const TArray & source);
inline unsigned Add (const T & source);
inline unsigned Add (const T * source, unsigned count);
inline unsigned AddArray (const TArray & source);
inline void Copy (unsigned destIndex, unsigned sourceIndex, unsigned count);
inline void DeleteOrdered (unsigned index);
inline void DeleteUnordered (unsigned index);
inline void GrowToCount (unsigned count, bool zero);
inline void GrowToFit (unsigned index, bool zero);
inline void ShrinkBy (unsigned count);
inline void Move (unsigned destIndex, unsigned sourceIndex, unsigned count);
inline T * New ();
inline T * New (unsigned count);
inline void Push (const T & source);
inline T Pop ();
inline void Reserve (unsigned additionalCount);
inline void Set (const T * source, unsigned count);
inline void SetChunkSize (unsigned chunkSize);
inline void SetCount (unsigned count);
inline void SetCountFewer (unsigned count);
inline void Trim ();
};
//===========================================================================
template
TArray::TArray () : TFArray() {
m_chunkSize = std::max(1u, 256 / sizeof(T));
}
//===========================================================================
template
TArray::TArray (const char file[], int line) : TFArray(file, line) {
m_chunkSize = std::max(1u, 256 / sizeof(T));
}
//===========================================================================
template
TArray::TArray (unsigned count) : TFArray(count) {
m_chunkSize = std::max(1u, 256 / sizeof(T));
}
//===========================================================================
template
TArray::TArray (const T * source, unsigned count) : TFArray(source, count) {
m_chunkSize = std::max(1u, 256 / sizeof(T));
}
//===========================================================================
template
TArray::TArray (const TArray & source) : TFArray(source) {
m_chunkSize = source.m_chunkSize;
}
//===========================================================================
template
TArray & TArray::operator= (const TArray & source) {
if (&source == this)
return *this;
m_chunkSize = source.m_chunkSize;
AdjustSize(max(this->m_alloc, source.m_count), 0);
C::CopyConstruct(this->m_data, source.m_data, source.m_count);
this->m_count = source.m_count;
return *this;
}
//===========================================================================
template
unsigned TArray::Add (const T & source) {
unsigned index = this->m_count;
Push(source);
return index;
}
//===========================================================================
template
unsigned TArray::Add (const T * source, unsigned count) {
unsigned index = this->m_count;
AdjustSizeChunked(this->m_count + count, this->m_count);
C::CopyConstruct(&this->m_data[this->m_count], source, count);
this->m_count += count;
return index;
}
//===========================================================================
template
unsigned TArray::AddArray (const TArray & source) {
unsigned index = this->m_count;
AdjustSizeChunked(this->m_count + source.m_count, this->m_count);
C::CopyConstruct(&this->m_data[this->m_count], source.m_data, source.m_count);
this->m_count += source.m_count;
return index;
}
//===========================================================================
template
void TArray::AdjustSizeChunked (unsigned newAlloc, unsigned newCount) {
// Disallow shrinking the allocation
if (newAlloc <= this->m_alloc)
newAlloc = this->m_alloc;
// Process growing the allocation
else
newAlloc = this->CalcAllocGrowth(newAlloc, this->m_alloc, &this->m_chunkSize);
// Perform the allocation
this->AdjustSize(newAlloc, newCount);
}
//===========================================================================
template
void TArray::Copy (unsigned destIndex, unsigned sourceIndex, unsigned count) {
// Copy the data to the destination
ASSERT(destIndex + count <= m_count);
ASSERT(sourceIndex + count <= m_count);
C::Assign(this->m_data + destIndex, this->m_data + sourceIndex, count);
}
//===========================================================================
template
void TArray::DeleteOrdered (unsigned index) {
ASSERT(index < this->m_count);
if (index + 1 < this->m_count)
C::Assign(&this->m_data[index], &this->m_data[index + 1], this->m_count - index - 1);
C::Destruct(&this->m_data[--this->m_count]);
}
//===========================================================================
template
void TArray::DeleteUnordered (unsigned index) {
ASSERT(index < this->m_count);
if (index + 1 < this->m_count)
C::Assign(&this->m_data[index], &this->m_data[this->m_count - 1], 1);
C::Destruct(&this->m_data[--this->m_count]);
}
//===========================================================================
template
void TArray::GrowToCount (unsigned count, bool zero) {
if (count <= this->m_count)
return;
AdjustSizeChunked(count, this->m_count);
if (zero)
memset(this->m_data + this->m_count, 0, (count - this->m_count) * sizeof(T));
C::Construct(this->m_data + this->m_count, count - this->m_count);
this->m_count = count;
}
//===========================================================================
template
void TArray::GrowToFit (unsigned index, bool zero) {
GrowToCount(index + 1, zero);
}
//===========================================================================
template
void TArray::ShrinkBy (unsigned count) {
ASSERT(count <= this->m_count);
C::Destruct(this->m_data + this->m_count - count, count);
this->m_count -= count;
}
//===========================================================================
template
void TArray::Move (unsigned destIndex, unsigned sourceIndex, unsigned count) {
// Copy the data to the destination
ASSERT(destIndex + count <= this->m_count);
ASSERT(sourceIndex + count <= this->m_count);
C::Assign(this->m_data + destIndex, this->m_data + sourceIndex, count);
// Remove it from the source
if (destIndex >= sourceIndex) {
C::Destruct(this->m_data + sourceIndex, std::min(count, destIndex - sourceIndex));
C::Construct(this->m_data + sourceIndex, std::min(count, destIndex - sourceIndex));
}
else {
unsigned overlap = (destIndex + count > sourceIndex) ? (destIndex + count - sourceIndex) : 0;
ASSERT(overlap <= count);
C::Destruct(this->m_data + sourceIndex + overlap, count - overlap);
C::Construct(this->m_data + sourceIndex + overlap, count - overlap);
}
}
//===========================================================================
template
T * TArray::New () {
AdjustSizeChunked(this->m_count + 1, this->m_count + 1);
return &this->m_data[this->m_count - 1];
}
//===========================================================================
template
T * TArray::New (unsigned count) {
AdjustSizeChunked(this->m_count + count, this->m_count + count);
return &this->m_data[this->m_count - count];
}
//===========================================================================
template
void TArray::Push (const T & source) {
AdjustSizeChunked(this->m_count + 1, this->m_count);
C::CopyConstruct(&this->m_data[this->m_count], source);
++this->m_count;
}
//===========================================================================
template
T TArray::Pop () {
ASSERT(this->m_count);
T result = this->m_data[--this->m_count];
C::Destruct(this->m_data + this->m_count);
return result;
}
//===========================================================================
template
void TArray::Reserve (unsigned additionalCount) {
AdjustSizeChunked(std::max(this->m_alloc, this->m_count + additionalCount), this->m_count);
}
//===========================================================================
template
void TArray::Set (const T * source, unsigned count) {
AdjustSizeChunked(count, 0);
C::CopyConstruct(this->m_data, source, count);
this->m_count = count;
}
//===========================================================================
template
void TArray::SetChunkSize (unsigned chunkSize) {
this->m_chunkSize = chunkSize;
}
//===========================================================================
template
void TArray::SetCount (unsigned count) {
AdjustSizeChunked(std::max(this->m_alloc, count), count);
}
//===========================================================================
template
void TArray::SetCountFewer (unsigned count) {
ASSERT(count <= this->m_count);
C::Destruct(this->m_data + count, this->m_count - count);
this->m_count = count;
}
//===========================================================================
template
void TArray::Trim () {
this->AdjustSize(this->m_count, this->m_count);
}
#endif