630 lines
19 KiB
630 lines
19 KiB
/*==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 <http://www.gnu.org/licenses/>. |
|
|
|
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/pnUtList.h |
|
* |
|
***/ |
|
|
|
#ifdef PLASMA20_SOURCES_PLASMA_NUCLEUSLIB_PNUTILS_PRIVATE_PNUTLIST_H |
|
#error "Header $/Plasma20/Sources/Plasma/NucleusLib/pnUtils/Private/pnUtList.h included more than once" |
|
#endif |
|
#define PLASMA20_SOURCES_PLASMA_NUCLEUSLIB_PNUTILS_PRIVATE_PNUTLIST_H |
|
|
|
|
|
/**************************************************************************** |
|
* |
|
* Constants |
|
* |
|
***/ |
|
|
|
enum ELinkType { |
|
kListUnlinked, |
|
kListLinkAfter, |
|
kListLinkBefore, |
|
kListHead = kListLinkAfter, |
|
kListTail = kListLinkBefore |
|
}; |
|
|
|
|
|
/**************************************************************************** |
|
* |
|
* Macros |
|
* |
|
***/ |
|
|
|
#define LINK(class) TLink< class > |
|
#define LIST(class) TList< class > |
|
#define LISTDECL(class,field) TListDecl< class, offsetof(class,field) > |
|
#define LISTDYN(class) TListDyn< class > |
|
|
|
|
|
/**************************************************************************** |
|
* |
|
* Forward declarations |
|
* |
|
***/ |
|
|
|
template<class T> |
|
class TLink; |
|
|
|
template<class T> |
|
class TList; |
|
|
|
|
|
/**************************************************************************** |
|
* |
|
* CBaseLink |
|
* |
|
***/ |
|
|
|
class CBaseLink { |
|
friend class CBaseList; |
|
|
|
inline static bool TermCheck (byte * ptr); |
|
inline static byte * TermMark (byte * ptr); |
|
inline static byte * TermUnmarkAlways (byte * ptr); |
|
|
|
protected: |
|
CBaseLink * volatile m_prevLink; |
|
byte * volatile m_next; |
|
|
|
inline int CalcLinkOffset () const; |
|
inline void InitializeLinks (); |
|
inline void InitializeLinksWithOffset (int linkOffset); |
|
inline void InsertAfter (byte * node, CBaseLink * prevLink, int linkOffset); |
|
inline void InsertBefore (byte * node, CBaseLink * nextLink); |
|
inline byte * Next () const; |
|
inline byte * NextIgnoreTerm () const; |
|
inline byte * NextUnchecked () const; |
|
inline CBaseLink * NextLink () const; |
|
inline CBaseLink * NextLink (int linkOffset) const; |
|
inline byte * Prev () const; |
|
inline void UnlinkFromNeighbors (); |
|
|
|
public: |
|
inline CBaseLink (); |
|
inline CBaseLink (const CBaseLink & source); |
|
inline ~CBaseLink (); |
|
inline CBaseLink & operator= (const CBaseLink & source); |
|
inline bool IsLinked () const; |
|
inline void Unlink (); |
|
|
|
}; |
|
|
|
//=========================================================================== |
|
CBaseLink::CBaseLink () { |
|
InitializeLinks(); |
|
} |
|
|
|
//=========================================================================== |
|
CBaseLink::CBaseLink (const CBaseLink & source) { |
|
#ifdef HS_DEBUGGING |
|
if (source.IsLinked()) |
|
FATAL("No copy constructor"); |
|
#endif |
|
InitializeLinks(); |
|
} |
|
|
|
//=========================================================================== |
|
CBaseLink::~CBaseLink () { |
|
UnlinkFromNeighbors(); |
|
} |
|
|
|
//=========================================================================== |
|
CBaseLink & CBaseLink::operator= (const CBaseLink & source) { |
|
#ifdef HS_DEBUGGING |
|
FATAL("No assignment operator"); |
|
#endif |
|
return *this; |
|
} |
|
|
|
//=========================================================================== |
|
int CBaseLink::CalcLinkOffset () const { |
|
return (int)((byte *)this - m_prevLink->NextIgnoreTerm()); |
|
} |
|
|
|
//=========================================================================== |
|
void CBaseLink::InitializeLinks () { |
|
ASSERT(!((unsigned_ptr)this & 3)); |
|
m_prevLink = this; |
|
m_next = TermMark((byte *)this); |
|
} |
|
|
|
//=========================================================================== |
|
void CBaseLink::InitializeLinksWithOffset (int linkOffset) { |
|
m_prevLink = this; |
|
m_next = TermMark((byte *)this - linkOffset); |
|
} |
|
|
|
//=========================================================================== |
|
void CBaseLink::InsertAfter (byte * node, CBaseLink * prevLink, int linkOffset) { |
|
UnlinkFromNeighbors(); |
|
m_prevLink = prevLink; |
|
m_next = prevLink->m_next; |
|
prevLink->NextLink(linkOffset)->m_prevLink = this; |
|
prevLink->m_next = node; |
|
} |
|
|
|
//=========================================================================== |
|
void CBaseLink::InsertBefore (byte * node, CBaseLink * nextLink) { |
|
UnlinkFromNeighbors(); |
|
m_prevLink = nextLink->m_prevLink; |
|
m_next = m_prevLink->m_next; |
|
nextLink->m_prevLink->m_next = node; |
|
nextLink->m_prevLink = this; |
|
} |
|
|
|
//=========================================================================== |
|
bool CBaseLink::IsLinked () const { |
|
return (m_prevLink != this); |
|
} |
|
|
|
//=========================================================================== |
|
byte * CBaseLink::Next () const { |
|
return TermCheck(m_next) ? nil : m_next; |
|
} |
|
|
|
//=========================================================================== |
|
byte * CBaseLink::NextIgnoreTerm () const { |
|
return TermUnmarkAlways(m_next); |
|
} |
|
|
|
//=========================================================================== |
|
byte * CBaseLink::NextUnchecked () const { |
|
return m_next; |
|
} |
|
|
|
//=========================================================================== |
|
CBaseLink * CBaseLink::NextLink () const { |
|
return (CBaseLink *)(NextIgnoreTerm() + CalcLinkOffset()); |
|
} |
|
|
|
//=========================================================================== |
|
CBaseLink * CBaseLink::NextLink (int linkOffset) const { |
|
ASSERT(linkOffset == CalcLinkOffset()); |
|
return (CBaseLink *)(NextIgnoreTerm() + linkOffset); |
|
} |
|
|
|
//=========================================================================== |
|
byte * CBaseLink::Prev () const { |
|
return m_prevLink->m_prevLink->Next(); |
|
} |
|
|
|
//=========================================================================== |
|
bool CBaseLink::TermCheck (byte * ptr) { |
|
return (unsigned_ptr)ptr & 1; |
|
} |
|
|
|
//=========================================================================== |
|
byte * CBaseLink::TermMark (byte * ptr) { |
|
// Converts an unmarked pointer to a marked pointer |
|
ASSERT(!TermCheck(ptr)); |
|
return (byte *)((unsigned_ptr)ptr + 1); |
|
} |
|
|
|
//=========================================================================== |
|
byte * CBaseLink::TermUnmarkAlways (byte * ptr) { |
|
// Returns an unmarked pointer regardless of whether the source pointer |
|
// was marked on unmarked |
|
return (byte *)((unsigned_ptr)ptr & ~1); |
|
} |
|
|
|
//=========================================================================== |
|
void CBaseLink::Unlink () { |
|
UnlinkFromNeighbors(); |
|
InitializeLinks(); |
|
} |
|
|
|
//=========================================================================== |
|
void CBaseLink::UnlinkFromNeighbors () { |
|
NextLink()->m_prevLink = m_prevLink; |
|
m_prevLink->m_next = m_next; |
|
} |
|
|
|
|
|
/**************************************************************************** |
|
* |
|
* TLink |
|
* |
|
***/ |
|
|
|
template<class T> |
|
class TLink : public CBaseLink { |
|
|
|
public: |
|
inline T * Next (); |
|
inline const T * Next () const; |
|
inline T * NextUnchecked (); |
|
inline const T * NextUnchecked () const; |
|
inline T * Prev (); |
|
inline const T * Prev () const; |
|
|
|
}; |
|
|
|
//=========================================================================== |
|
template<class T> |
|
T * TLink<T>::Next () { |
|
return (T *)CBaseLink::Next(); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
const T * TLink<T>::Next () const { |
|
return (const T *)CBaseLink::Next(); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
T * TLink<T>::NextUnchecked () { |
|
return (T *)CBaseLink::NextUnchecked(); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
const T * TLink<T>::NextUnchecked () const { |
|
return (const T *)CBaseLink::NextUnchecked(); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
T * TLink<T>::Prev () { |
|
return (T *)CBaseLink::Prev(); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
const T * TLink<T>::Prev () const { |
|
return (const T *)CBaseLink::Prev(); |
|
} |
|
|
|
|
|
/**************************************************************************** |
|
* |
|
* CBaseList |
|
* |
|
***/ |
|
|
|
class CBaseList { |
|
|
|
private: |
|
enum { LINK_OFFSET_UNINIT = 0xDDDDDDDD }; |
|
|
|
int m_linkOffset; |
|
CBaseLink m_terminator; |
|
|
|
protected: |
|
inline CBaseLink * GetLink (const byte * node) const; |
|
inline byte * Head () const; |
|
inline bool IsLinked (const byte * node) const; |
|
inline void Link (byte * node, ELinkType linkType, byte * existingNode); |
|
void Link (CBaseList * list, byte * afterNode, byte * beforeNode, ELinkType linkType, byte * existingNode); |
|
inline byte * Next (const byte * node) const; |
|
inline byte * NextUnchecked (const byte * node) const; |
|
inline byte * Prev (byte * node) const; |
|
inline byte * Tail () const; |
|
inline void Unlink (byte * node); |
|
|
|
public: |
|
inline CBaseList (); |
|
inline CBaseList (const CBaseList & source); |
|
inline ~CBaseList (); |
|
inline CBaseList & operator= (const CBaseList & source); |
|
inline void SetLinkOffset (int linkOffset); |
|
void UnlinkAll (); |
|
|
|
}; |
|
|
|
//=========================================================================== |
|
CBaseList::CBaseList () { |
|
m_linkOffset = LINK_OFFSET_UNINIT; |
|
} |
|
|
|
//=========================================================================== |
|
CBaseList::CBaseList (const CBaseList & source) { |
|
m_linkOffset = LINK_OFFSET_UNINIT; |
|
} |
|
|
|
//=========================================================================== |
|
CBaseList::~CBaseList () { |
|
if (m_terminator.IsLinked()) |
|
UnlinkAll(); |
|
} |
|
|
|
//=========================================================================== |
|
CBaseList & CBaseList::operator= (const CBaseList & source) { |
|
return *this; |
|
} |
|
|
|
//=========================================================================== |
|
CBaseLink * CBaseList::GetLink (const byte * node) const { |
|
return (CBaseLink *)(node + m_linkOffset); |
|
} |
|
|
|
//=========================================================================== |
|
byte * CBaseList::Head () const { |
|
return m_terminator.Next(); |
|
} |
|
|
|
//=========================================================================== |
|
bool CBaseList::IsLinked (const byte * node) const { |
|
ASSERT(node); |
|
return GetLink(node)->IsLinked(); |
|
} |
|
|
|
//=========================================================================== |
|
void CBaseList::Link (byte * node, ELinkType linkType, byte * existingNode) { |
|
ASSERT(node != existingNode); |
|
ASSERT(m_linkOffset != LINK_OFFSET_UNINIT); |
|
ASSERT((linkType == kListLinkAfter) || (linkType == kListLinkBefore)); |
|
if (linkType == kListLinkAfter) |
|
GetLink(node)->InsertAfter(node, existingNode ? GetLink(existingNode) : &m_terminator, m_linkOffset); |
|
else |
|
GetLink(node)->InsertBefore(node, existingNode ? GetLink(existingNode) : &m_terminator); |
|
} |
|
|
|
//=========================================================================== |
|
byte * CBaseList::Next (const byte * node) const { |
|
ASSERT(node); |
|
return GetLink(node)->Next(); |
|
} |
|
|
|
//=========================================================================== |
|
byte * CBaseList::NextUnchecked (const byte * node) const { |
|
ASSERT(node); |
|
return GetLink(node)->NextUnchecked(); |
|
} |
|
|
|
//=========================================================================== |
|
byte * CBaseList::Prev (byte * node) const { |
|
ASSERT(node); |
|
return GetLink(node)->Prev(); |
|
} |
|
|
|
//=========================================================================== |
|
void CBaseList::SetLinkOffset (int linkOffset) { |
|
ASSERT(!Head()); |
|
|
|
m_linkOffset = linkOffset; |
|
m_terminator.InitializeLinksWithOffset(linkOffset); |
|
} |
|
|
|
//=========================================================================== |
|
byte * CBaseList::Tail () const { |
|
return m_terminator.Prev(); |
|
} |
|
|
|
//=========================================================================== |
|
void CBaseList::Unlink (byte * node) { |
|
ASSERT(node); |
|
GetLink(node)->Unlink(); |
|
} |
|
|
|
|
|
/**************************************************************************** |
|
* |
|
* TList |
|
* |
|
***/ |
|
|
|
template<class T> |
|
class TList : public CBaseList { |
|
|
|
private: |
|
inline T * NewFlags (unsigned flags, ELinkType linkType, T * existingNode, const char file[], int line); |
|
|
|
public: |
|
inline void Clear (); |
|
inline void Delete (T * node); |
|
inline T * Head (); |
|
inline const T * Head () const; |
|
inline bool IsLinked (const T * node) const; |
|
inline void Link (T * node, ELinkType linkType = kListTail, T * existingNode = nil); |
|
inline void Link (TList<T> * list, ELinkType linkType = kListTail, T * existingNode = nil); |
|
inline void Link (TList<T> * list, T * afterNode, T * beforeNode, ELinkType linkType = kListTail, T * existingNode = nil); |
|
inline T * New (ELinkType linkType = kListTail, T * existingNode = nil, const char file[] = nil, int line = 0); |
|
inline T * NewZero (ELinkType linkType = kListTail, T * existingNode = nil, const char file[] = nil, int line = 0); |
|
inline T * Next (const T * node); |
|
inline const T * Next (const T * node) const; |
|
inline T * NextUnchecked (const T * node); |
|
inline const T * NextUnchecked (const T * node) const; |
|
inline T * Prev (const T * node); |
|
inline const T * Prev (const T * node) const; |
|
inline T * Tail (); |
|
inline const T * Tail () const; |
|
inline void Unlink (T * node); |
|
|
|
}; |
|
|
|
//=========================================================================== |
|
template<class T> |
|
void TList<T>::Clear () { |
|
for (T * curr; (curr = Head()) != nil; Delete(curr)) |
|
; |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
void TList<T>::Delete (T * node) { |
|
DEL(node); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
T * TList<T>::Head () { |
|
return (T *)CBaseList::Head(); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
const T * TList<T>::Head () const { |
|
return (const T *)CBaseList::Head(); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
bool TList<T>::IsLinked (const T * node) const { |
|
return CBaseList::IsLinked((const byte *)node); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
void TList<T>::Link (T * node, ELinkType linkType, T * existingNode) { |
|
CBaseList::Link((byte *)node, linkType, (byte *)existingNode); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
void TList<T>::Link (TList<T> * list, ELinkType linkType, T * existingNode) { |
|
CBaseList::Link(list, nil, nil, linkType, (byte *)existingNode); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
void TList<T>::Link (TList<T> * list, T * afterNode, T * beforeNode, ELinkType linkType, T * existingNode) { |
|
CBaseList::Link(list, (byte *)afterNode, (byte *)beforeNode, linkType, (byte *)existingNode); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline T * TList<T>::Next (const T * node) { |
|
return (T *)CBaseList::Next((byte *)node); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline const T * TList<T>::Next (const T * node) const { |
|
return (const T *)CBaseList::Next((byte *)node); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline T * TList<T>::NextUnchecked (const T * node) { |
|
return (T *)CBaseList::NextUnchecked((byte *)node); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline const T * TList<T>::NextUnchecked (const T * node) const { |
|
return (const T *)CBaseList::NextUnchecked((byte *)node); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline T * TList<T>::New (ELinkType linkType, T * existingNode, const char file[], int line) { |
|
return NewFlags(0, linkType, existingNode, file, line); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline T * TList<T>::NewFlags (unsigned flags, ELinkType linkType, T * existingNode, const char file[], int line) { |
|
if (!file) { |
|
file = __FILE__; |
|
line = __LINE__; |
|
} |
|
T * node = new(MemAlloc(sizeof(T), flags, file, line)) T; |
|
if (linkType != kListUnlinked) |
|
Link(node, linkType, existingNode); |
|
return node; |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline T * TList<T>::NewZero (ELinkType linkType, T * existingNode, const char file[], int line) { |
|
return NewFlags(MEM_ZERO, linkType, existingNode, file, line); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline T * TList<T>::Prev (const T * node) { |
|
return (T *)CBaseList::Prev((byte *)node); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline const T * TList<T>::Prev (const T * node) const { |
|
return (const T *)CBaseList::Prev((byte *)node); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline T * TList<T>::Tail () { |
|
return (T *)CBaseList::Tail(); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline const T * TList<T>::Tail () const { |
|
return (const T *)CBaseList::Tail(); |
|
} |
|
|
|
//=========================================================================== |
|
template<class T> |
|
inline void TList<T>::Unlink (T * node) { |
|
CBaseList::Unlink((byte *)node); |
|
} |
|
|
|
|
|
/**************************************************************************** |
|
* |
|
* TListDecl |
|
* |
|
***/ |
|
|
|
template<class T, int linkOffset> |
|
class TListDecl : public TList<T> { |
|
|
|
public: |
|
inline TListDecl (); |
|
|
|
}; |
|
|
|
//=========================================================================== |
|
template<class T, int linkOffset> |
|
TListDecl<T,linkOffset>::TListDecl () { |
|
SetLinkOffset(linkOffset); |
|
} |
|
|
|
|
|
/**************************************************************************** |
|
* |
|
* TListDyn |
|
* |
|
***/ |
|
|
|
template<class T> |
|
class TListDyn : public TList<T> { |
|
|
|
public: |
|
void Initialize (int linkOffset); |
|
|
|
}; |
|
|
|
//=========================================================================== |
|
template<class T> |
|
void TListDyn<T>::Initialize (int linkOffset) { |
|
SetLinkOffset(linkOffset); |
|
}
|
|
|