/*---------------------------------------------------------------------------* Project: Horizon File: fnd_LinkedList.h Copyright (C)2009 Nintendo Co., Ltd. All rights reserved. These coded instructions, statements, and computer programs contain proprietary information of Nintendo of America Inc. and/or Nintendo Company Ltd., and are protected by Federal copyright law. They may not be disclosed to third parties or copied or duplicated in any form, in whole or in part, without the prior written consent of Nintendo. $Rev: 21468 $ *--------------------------------------------------------------------------- */ /*---------------------------------------------------------------------------* Project: NW4R - SystemLib - include - nw4r - ut File: LinkedList.h Copyright (C) 2005-2008 Nintendo. All rights reserved. These coded instructions, statements, and computer programs contain proprietary information of Nintendo of America Inc. and/or Nintendo Company Ltd., and are protected by Federal copyright law. They may not be disclosed to third parties or copied or duplicated in any form, in whole or in part, without the prior written consent of Nintendo. $Revision: 1.13 #$ *--------------------------------------------------------------------------- */ /* */ #ifndef NN_FND_FND_LINKEDLIST_H_ #define NN_FND_FND_LINKEDLIST_H_ #ifdef __cplusplus #include #include #include namespace nn { namespace fnd { // Usage examples // // class Hoge : public IntrusiveLinkedList::Item { ... }; // tyepdef IntrusiveLinkedList HogeList; // // Hoge a, b, c; // { // HogeList list; // list.PushFront(&a); // list.PushBack(&b); // list.PushFront(&a); // PANIC: Item cannot be inserted in list twice // if (Hoge* p = list.PopBack()) // { // // Processing using *p // } // list.Erase(&a); // Specify the element and delete // list.Clear(); // } // Automatic Clear() using destructor // TODO: Untested // NEED_DOCUMENT // NEED_TEST /* */ template class IntrusiveLinkedList : private nn::util::NonCopyable > { public: class Item; // /* */ IntrusiveLinkedList() : m_Head(0) {} /* */ ~IntrusiveLinkedList() { Clear(); } /* */ bool IsEmpty() const { return !m_Head; } /* */ void PushBack(T* p); /* */ void PushFront(T* p); /* */ T* GetFront() const; /* */ T* GetBack() const; /* */ T* PopFront(); /* */ T* PopBack(); /* */ T* GetNext(T* p) const; /* */ T* GetPrevious(T* p) const; /* */ void Insert(T* position, T* inserted); /* */ void Erase(T* p); /* */ void Clear(); private: Item* m_Head; // /* :private */ static void ClearLinks(Item* p); /* :private */ static void InsertBefore(Item* p, Item* q); }; // NEED_DOCUMENT // NEED_TEST /* :private */ template class IntrusiveLinkedList::Item : private nn::util::NonCopyable::Item> { friend class IntrusiveLinkedList; protected: Item() : m_PreviousLink(0), m_NextLink(0) {} ~Item() { NN_TASSERT_(!m_PreviousLink && !m_NextLink); } T* GetPrevious() const { return static_cast(m_PreviousLink); } T* GetNext() const { return static_cast(m_NextLink); } private: Item* m_PreviousLink; Item* m_NextLink; }; template inline void IntrusiveLinkedList::ClearLinks(Item* p) { p->m_PreviousLink = p->m_NextLink = 0; } template inline void IntrusiveLinkedList::InsertBefore(Item* p, Item* q) { q->m_NextLink = p; p->m_PreviousLink->m_NextLink = q; q->m_PreviousLink = p->m_PreviousLink; p->m_PreviousLink = q; } template inline void IntrusiveLinkedList::PushBack(T* p) { NN_ASSERT_WITH_RESULT(p, MakeResultInvalidAddress()); NN_TASSERT_(p); Item* pNode = static_cast(p); NN_ASSERT_WITH_RESULT(!pNode->m_PreviousLink, MakeResultAlreadyListed()); NN_TASSERT_(!pNode->m_PreviousLink); NN_TASSERT_(!pNode->m_NextLink); if (IsEmpty()) { p->m_PreviousLink = p->m_NextLink = p; this->m_Head = p; } else { InsertBefore(m_Head, pNode); } } template inline void IntrusiveLinkedList::PushFront(T* p) { NN_ASSERT_WITH_RESULT(p, MakeResultInvalidAddress()); NN_TASSERT_(p); Item* pNode = static_cast(p); NN_ASSERT_WITH_RESULT(!pNode->m_PreviousLink, MakeResultAlreadyListed()); NN_TASSERT_(!pNode->m_PreviousLink); NN_TASSERT_(!pNode->m_NextLink); if (IsEmpty()) { p->m_PreviousLink = p->m_NextLink = p; } else { InsertBefore(m_Head, pNode); } this->m_Head = p; } template inline T* IntrusiveLinkedList::GetFront() const { return static_cast(m_Head); } template inline T* IntrusiveLinkedList::GetBack() const { if (IsEmpty()) { return 0; } else { return static_cast(m_Head->m_PreviousLink); } } template inline T* IntrusiveLinkedList::PopFront() { if (T* ret = GetFront()) { this->Erase(ret); return ret; } else { return 0; } } template inline T* IntrusiveLinkedList::PopBack() { if (T* ret = GetBack()) { this->Erase(ret); return ret; } else { return 0; } } template inline T* IntrusiveLinkedList::GetPrevious(T* p) const { NN_ASSERT_WITH_RESULT(p, MakeResultInvalidAddress()); NN_TASSERT_(p); Item* pNode = static_cast(p); NN_ASSERT_WITH_RESULT(pNode->m_PreviousLink, MakeResultInvalidNode()); NN_TASSERT_(pNode->m_PreviousLink); NN_TASSERT_(pNode->m_NextLink); if (p == this->GetFront()) { return 0; } return static_cast(pNode->m_PreviousLink); } template inline T* IntrusiveLinkedList::GetNext(T* p) const { NN_ASSERT_WITH_RESULT(p, MakeResultInvalidAddress()); NN_TASSERT_(p); Item* pNode = static_cast(p); NN_ASSERT_WITH_RESULT(pNode->m_PreviousLink, MakeResultInvalidNode()); NN_TASSERT_(pNode->m_PreviousLink); NN_TASSERT_(pNode->m_NextLink); if (p == this->GetBack()) { return 0; } return static_cast(pNode->m_NextLink); } template inline void IntrusiveLinkedList::Insert(T* position, T* inserted) { NN_ASSERT_WITH_RESULT(inserted, MakeResultInvalidAddress()); NN_TASSERT_(inserted); Item* pNodeInserted = static_cast(inserted); Item* pNodePosition = static_cast(position); NN_ASSERT_WITH_RESULT(!pNodeInserted->m_PreviousLink, MakeResultAlreadyListed()); NN_TASSERT_(!pNodeInserted->m_PreviousLink); NN_TASSERT_(!pNodeInserted->m_NextLink); if (pNodePosition == m_Head) { PushFront(inserted); } else if (pNodePosition) { NN_ASSERT_WITH_RESULT(pNodePosition->m_PreviousLink, MakeResultInvalidNode()); NN_TASSERT_(pNodePosition->m_PreviousLink); NN_TASSERT_(pNodePosition->m_NextLink); InsertBefore(pNodePosition, pNodeInserted); } else { PushBack(inserted); } } template inline void IntrusiveLinkedList::Erase(T* p) { NN_ASSERT_WITH_RESULT(p, MakeResultInvalidAddress()); NN_TASSERT_(p); Item* pNode = static_cast(p); NN_ASSERT_WITH_RESULT(pNode->m_PreviousLink, MakeResultInvalidNode()); NN_TASSERT_(pNode->m_PreviousLink); NN_TASSERT_(pNode->m_NextLink); if (pNode == pNode->m_PreviousLink) { this->m_Head = 0; } else { if (m_Head == pNode) { this->m_Head = m_Head->m_NextLink; } pNode->m_NextLink->m_PreviousLink = pNode->m_PreviousLink; pNode->m_PreviousLink->m_NextLink = pNode->m_NextLink; } ClearLinks(pNode); } template inline void IntrusiveLinkedList::Clear() { if (m_Head) { Item* p = m_Head; while (p) { Item* q = p; p = p->m_NextLink; ClearLinks(q); } this->m_Head = 0; } } } // namespace util } // namespace nn #endif // __cplusplus #endif /* NN_UTIL_LINKEDLIST_H_ */