diff --git a/CMakeLists.txt b/CMakeLists.txt index 9b5e9bb4..873f78e5 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -160,6 +160,7 @@ SET (REACTPHYSICS3D_HEADERS "src/containers/Map.h" "src/containers/Set.h" "src/containers/Pair.h" + "src/containers/Deque.h" "src/utils/Profiler.h" "src/utils/Logger.h" ) diff --git a/src/containers/Deque.h b/src/containers/Deque.h new file mode 100644 index 00000000..a17c50fd --- /dev/null +++ b/src/containers/Deque.h @@ -0,0 +1,673 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://www.reactphysics3d.com * +* Copyright (c) 2010-2018 Daniel Chappuis * +********************************************************************************* +* * +* This software is provided 'as-is', without any express or implied warranty. * +* In no event will the authors be held liable for any damages arising from the * +* use of this software. * +* * +* Permission is granted to anyone to use this software for any purpose, * +* including commercial applications, and to alter it and redistribute it * +* freely, subject to the following restrictions: * +* * +* 1. The origin of this software must not be misrepresented; you must not claim * +* that you wrote the original software. If you use this software in a * +* product, an acknowledgment in the product documentation would be * +* appreciated but is not required. * +* * +* 2. Altered source versions must be plainly marked as such, and must not be * +* misrepresented as being the original software. * +* * +* 3. This notice may not be removed or altered from any source distribution. * +* * +********************************************************************************/ + +#ifndef REACTPHYSICS3D_DEQUE_H +#define REACTPHYSICS3D_DEQUE_H + +// Libraries +#include "configuration.h" +#include "memory/MemoryAllocator.h" +#include +#include +#include +#include +#include + +namespace reactphysics3d { + +// Class Deque +/** + * This class represents a Deque. A Deque is a double ended queue. It is possible to + * push and pop items at both ends of the queue. Note that in the following code, the + * virtualIndex is the index of the items in the deque in range [0, mSize-1] where + * 0 is always the front item and mSize-1 the back item. There is a virtual index for + * each item that is in the deque. The allocatedIndex is the index of a allocated slot in the + * deque in range [0, (mNbChunks * CHUNK_NB_ITEMS)-1]. A given allocatedIndex corresponds to a slot + * in the deque that can be empty or contains an item. + */ +template +class Deque { + + private: + + // -------------------- Constants -------------------- // + + /// Number of items in a chunk + const uint CHUNK_NB_ITEMS = 17; + + /// First item index in a chunk + const uint CHUNK_FIRST_ITEM_INDEX = CHUNK_NB_ITEMS / 2; + + // -------------------- Attributes -------------------- // + + /// Array of chunks + T** mChunks; + + /// Number of current elements in the deque + size_t mSize; + + /// Number of chunks + size_t mNbChunks; + + /// Index of the chunk with the first element of the deque + size_t mFirstChunkIndex; + + /// Index of the chunk with the last element of the deque + size_t mLastChunkIndex; + + /// Index of the first element in the first chunk + uint8 mFirstItemIndex; + + /// Index of the last element in the last chunk + uint8 mLastItemIndex; + + /// Memory allocator + MemoryAllocator& mAllocator; + + // -------------------- Methods -------------------- // + + /// Return a reference to an item at the given virtual index in range [0; mSize-1] + T& getItem(size_t virtualIndex) const { + + // If the virtual index is valid + if (virtualIndex < mSize) { + + size_t chunkIndex = mFirstChunkIndex; + size_t itemIndex = mFirstItemIndex; + + const size_t nbItemsFirstChunk = CHUNK_NB_ITEMS - mFirstItemIndex; + if (virtualIndex < nbItemsFirstChunk) { + itemIndex += virtualIndex; + } + else { + + virtualIndex -= nbItemsFirstChunk; + chunkIndex++; + + chunkIndex += virtualIndex / CHUNK_NB_ITEMS; + itemIndex = virtualIndex % CHUNK_NB_ITEMS; + } + + return mChunks[chunkIndex][itemIndex]; + } + else { + assert(false); + } + } + + /// Add more chunks + void expandChunks(size_t atLeastNbChunks = 0) { + + // If it is not necessary to expand the chunks + if (atLeastNbChunks > 0 && atLeastNbChunks <= mNbChunks) { + return; + } + + size_t newNbChunks = mNbChunks == 0 ? 3 : 2 * mNbChunks - 1; + if (atLeastNbChunks > 0 && newNbChunks < atLeastNbChunks) { + newNbChunks = size_t(atLeastNbChunks / 2) * 2 + 1; + } + const size_t halfNbChunksToAdd = mNbChunks == 0 ? 1 : (mNbChunks - 1) / 2; + + // Allocate memory for the new array of pointers to chunk + void* newMemory = mAllocator.allocate(newNbChunks * sizeof(T*)); + assert(newMemory != nullptr); + T** newChunks = static_cast(newMemory); + + // If chunks have already been allocated + if (mNbChunks > 0) { + + // Copy the pointers to the previous chunks to the new allocated memory location + std::uninitialized_copy(mChunks, mChunks + mNbChunks, newChunks + halfNbChunksToAdd); + + // Release the previously allocated memory + mAllocator.release(mChunks, mNbChunks * sizeof(T*)); + } + + mChunks = newChunks; + + // If we need to allocate the first chunk (in the middle of the chunks array) + if (mNbChunks == 0) { + mChunks[newNbChunks / 2] = static_cast(mAllocator.allocate(sizeof(T) * CHUNK_NB_ITEMS)); + assert(mChunks[newNbChunks / 2] != nullptr); + } + + mNbChunks = newNbChunks; + + // Allocate memory for each new chunk + for (size_t i=0; i < halfNbChunksToAdd; i++) { + + // Allocate memory for the new chunk + mChunks[i] = static_cast(mAllocator.allocate(sizeof(T) * CHUNK_NB_ITEMS)); + assert(mChunks[i] != nullptr); + + mChunks[mNbChunks - 1 - i] = static_cast(mAllocator.allocate(sizeof(T) * CHUNK_NB_ITEMS)); + assert(mChunks[mNbChunks - 1 -i] != nullptr); + } + + // Update the first and last chunk index + mFirstChunkIndex += halfNbChunksToAdd; + mLastChunkIndex += halfNbChunksToAdd; + } + + /* + /// Get the next item in the deque (return false if we have reached the end) + bool getNextItem(size_t& chunkIndex, size_t& itemIndex) const { + + bool isValid = true; + + // If we have reached the last item + if (chunkIndex == mLastChunkIndex && itemIndex == mLastItemIndex) { + isValid = false; + } + + // If we were at the last item of the chunk + if (itemIndex == CHUNK_NB_ITEMS - 1) { + + // Move to the next chunk + chunkIndex++; + itemIndex = 0; + } + else { + + itemIndex++; + } + + return isValid; + } + + /// Get the previous item in the deque (return false if we have reached the front) + bool getPreviousItem(size_t& chunkIndex, size_t& itemIndex) const { + + // If we have reached the first item + if (chunkIndex == mFirstChunkIndex && itemIndex == mFirstItemIndex) { + return false; + } + + // If we were at the first item of the chunk + if (itemIndex == 0) { + + // Move to the previous chunk + chunkIndex--; + itemIndex = CHUNK_NB_ITEMS - 1; + } + else { + + itemIndex--; + } + + return true; + } + /// Return true if the two chunk/item indices are currently valid for the state of the deque + bool areValidIndices(size_t chunkIndex, size_t itemIndex) const { + + return chunkIndex >= mFirstChunkIndex && chunkIndex <= mLastChunkIndex && + itemIndex >= mFirstItemIndex && itemIndex <= mLastItemIndex; + } + */ + + public: + + /// Class Iterator + /** + * This class represents an iterator for the Deque + */ + class Iterator { + + private: + + size_t mVirtualIndex; + const Deque* mDeque; + + public: + + // Iterator traits + using value_type = T; + using difference_type = std::ptrdiff_t; + using pointer = T*; + using const_pointer = T const*; + using reference = T&; + using const_reference = const T&; + using iterator_category = std::random_access_iterator_tag; + + /// Constructor + Iterator() = default; + + /// Constructor + Iterator(const Deque* deque, size_t virtualIndex) : mVirtualIndex(virtualIndex), mDeque(deque) { + + } + + /// Copy constructor + Iterator(const Iterator& it) : mVirtualIndex(it.mVirtualIndex), mDeque(it.mDeque) { + + } + + /// Deferencable + reference operator*() { + assert(mVirtualIndex < mDeque->mSize); + return mDeque->getItem(mVirtualIndex); + } + + /// Const Deferencable + const_reference operator*() const { + assert(mVirtualIndex < mDeque->mSize); + return mDeque->getItem(mVirtualIndex); + } + + /// Deferencable + const_pointer operator->() const { + assert(mVirtualIndex < mDeque->mSize); + return &(mDeque->getItem(mVirtualIndex)); + } + + /// Post increment (it++) + Iterator& operator++() { + assert(mVirtualIndex < mDeque->mSize); + mVirtualIndex++; + return *this; + } + + /// Pre increment (++it) + Iterator operator++(int number) { + assert(mVirtualIndex < mDeque->mSize); + Iterator tmp = *this; + mVirtualIndex++; + return tmp; + } + + /// Post decrement (it--) + Iterator& operator--() { + mVirtualIndex--; + return *this; + } + + /// Pre decrement (--it) + Iterator operator--(int number) { + Iterator tmp = *this; + mVirtualIndex--; + return tmp; + } + + /// Plus operator + Iterator operator+(const difference_type& n) { + return Iterator(mDeque, mVirtualIndex + n); + } + + /// Plus operator + Iterator& operator+=(const difference_type& n) { + mVirtualIndex += n; + return *this; + } + + /// Minus operator + Iterator operator-(const difference_type& n) { + return Iterator(mDeque, mVirtualIndex - n); + } + + /// Minus operator + Iterator& operator-=(const difference_type& n) { + mVirtualIndex -= n; + return *this; + } + + /// Difference operator + difference_type operator-(const Iterator& iterator) const { + return mVirtualIndex - iterator.mVirtualIndex; + } + + /// Comparison operator + bool operator<(const Iterator& other) const { + + return mVirtualIndex < other.mVirtualIndex; + } + + /// Comparison operator + bool operator>(const Iterator& other) const { + + return mVirtualIndex > other.mVirtualIndex; + } + + /// Comparison operator + bool operator<=(const Iterator& other) const { + + return mVirtualIndex <= other.mVirtualIndex; + } + + /// Comparison operator + bool operator>=(const Iterator& other) const { + + return mVirtualIndex >= other.mVirtualIndex; + } + + /// Equality operator (it == end()) + bool operator==(const Iterator& iterator) const { + + return mVirtualIndex == iterator.mVirtualIndex; + } + + /// Inequality operator (it != end()) + bool operator!=(const Iterator& iterator) const { + return !(*this == iterator); + } + + /// Frienship + friend class Deque; + + }; + + // -------------------- Methods -------------------- // + + /// Constructor + Deque(MemoryAllocator& allocator) + : mChunks(nullptr), mSize(0), mNbChunks(0), mFirstChunkIndex(1), + mLastChunkIndex(1), mFirstItemIndex(CHUNK_FIRST_ITEM_INDEX), + mLastItemIndex(CHUNK_FIRST_ITEM_INDEX), mAllocator(allocator) { + + // Allocate memory for the chunks array + expandChunks(); + } + + /// Copy constructor + Deque(const Deque& deque) + : mSize(0), mNbChunks(0), mFirstChunkIndex(1), + mLastChunkIndex(1), mFirstItemIndex(CHUNK_FIRST_ITEM_INDEX), + mLastItemIndex(CHUNK_FIRST_ITEM_INDEX), mAllocator(deque.mAllocator) { + + // Allocate memory for the array of chunks + expandChunks(deque.mNbChunks); + + if (deque.mSize > 0) { + + const size_t dequeHalfSize1 = std::ceil(deque.mSize / 2.0f); + const size_t dequeHalfSize2 = deque.mSize - dequeHalfSize1; + + // Add the items into the deque + for(size_t i=0; i < dequeHalfSize1; i++) { + addFront(deque[dequeHalfSize1 - 1 - i]); + } + for(size_t i=0; i < dequeHalfSize2; i++) { + addBack(deque[dequeHalfSize1 + i]); + } + } + } + + /// Destructor + ~Deque() { + + clear(); + + // Release each chunk + for (size_t i=0; i < mNbChunks; i++) { + + mAllocator.release(static_cast(mChunks[i]), sizeof(T) * CHUNK_NB_ITEMS); + } + + // Release the chunks array + mAllocator.release(static_cast(mChunks), sizeof(T*) * mNbChunks); + } + + /// Add an element at the end of the deque + void addBack(const T& element) { + + // If we need to add the item in a another chunk + if (mLastItemIndex == CHUNK_NB_ITEMS - 1) { + + // If we need to add more chunks + if (mLastChunkIndex == mNbChunks - 1) { + + // Add more chunks + expandChunks(); + } + + mLastItemIndex = 0; + mLastChunkIndex++; + } + else if (mSize != 0) { + mLastItemIndex++; + } + + // Construct the element at its location in the chunk + new (static_cast(&(mChunks[mLastChunkIndex][mLastItemIndex]))) T(element); + + mSize++; + + assert(mFirstChunkIndex >= 0 && mLastChunkIndex < mNbChunks); + assert(mFirstItemIndex >= 0 && mFirstItemIndex < CHUNK_NB_ITEMS); + assert(mLastItemIndex >= 0 && mLastItemIndex < CHUNK_NB_ITEMS); + assert(mFirstChunkIndex <= mLastChunkIndex); + } + + /// Add an element at the front of the deque + void addFront(const T& element) { + + // If we need to add the item in another chunk + if (mFirstItemIndex == 0) { + + // If we need to add more chunks + if (mFirstChunkIndex == 0) { + + // Add more chunks + expandChunks(); + } + + mFirstItemIndex = CHUNK_NB_ITEMS - 1; + mFirstChunkIndex--; + } + else if (mSize != 0) { + mFirstItemIndex--; + } + + // Construct the element at its location in the chunk + new (static_cast(&(mChunks[mFirstChunkIndex][mFirstItemIndex]))) T(element); + + mSize++; + + assert(mFirstChunkIndex >= 0 && mLastChunkIndex < mNbChunks); + assert(mFirstItemIndex >= 0 && mFirstItemIndex < CHUNK_NB_ITEMS); + assert(mLastItemIndex >= 0 && mLastItemIndex < CHUNK_NB_ITEMS); + assert(mFirstChunkIndex <= mLastChunkIndex); + } + + /// Remove the first element of the deque + void popFront() { + + if (mSize > 0) { + + // Call the destructor of the first element + mChunks[mFirstChunkIndex][mFirstItemIndex].~T(); + + mSize--; + + if (mSize == 0) { + mFirstChunkIndex = mNbChunks / 2; + mFirstItemIndex = CHUNK_FIRST_ITEM_INDEX; + mLastChunkIndex = mFirstChunkIndex; + mLastItemIndex = CHUNK_FIRST_ITEM_INDEX; + } + else if (mFirstItemIndex == CHUNK_NB_ITEMS - 1){ + mFirstChunkIndex++; + mFirstItemIndex = 0; + } + else { + mFirstItemIndex++; + } + + assert(mFirstChunkIndex >= 0 && mLastChunkIndex < mNbChunks); + assert(mFirstItemIndex >= 0 && mFirstItemIndex < CHUNK_NB_ITEMS); + assert(mLastItemIndex >= 0 && mLastItemIndex < CHUNK_NB_ITEMS); + assert(mFirstChunkIndex <= mLastChunkIndex); + } + } + + /// Remove the last element of the deque + void popBack() { + + if (mSize > 0) { + + // Call the destructor of the last element + mChunks[mLastChunkIndex][mLastItemIndex].~T(); + + mSize--; + + if (mSize == 0) { + mFirstChunkIndex = mNbChunks / 2; + mFirstItemIndex = CHUNK_FIRST_ITEM_INDEX; + mLastChunkIndex = mFirstChunkIndex; + mLastItemIndex = CHUNK_FIRST_ITEM_INDEX; + } + else if (mLastItemIndex == 0){ + mLastChunkIndex--; + mLastItemIndex = CHUNK_NB_ITEMS - 1; + } + else { + mLastItemIndex--; + } + + assert(mFirstChunkIndex >= 0 && mLastChunkIndex < mNbChunks); + assert(mFirstItemIndex >= 0 && mFirstItemIndex < CHUNK_NB_ITEMS); + assert(mLastItemIndex >= 0 && mLastItemIndex < CHUNK_NB_ITEMS); + assert(mFirstChunkIndex <= mLastChunkIndex); + } + } + + /// Return a reference to the first item of the deque + const T& getFront() const { + if (mSize > 0) { + return mChunks[mFirstChunkIndex][mFirstItemIndex]; + } + assert(false); + } + + /// Return a reference to the last item of the deque + const T& getBack() const { + if (mSize > 0) { + return mChunks[mLastChunkIndex][mLastItemIndex]; + } + assert(false); + } + + /// Clear the elements of the deque + void clear() { + + if (mSize > 0) { + + // Call the destructor of every items + for (size_t i=0; i < mSize; i++) { + getItem(i).~T(); + } + + mSize = 0; + + mFirstChunkIndex = mNbChunks / 2; + mLastChunkIndex = mFirstChunkIndex; + mFirstItemIndex = CHUNK_FIRST_ITEM_INDEX; + mLastItemIndex = CHUNK_FIRST_ITEM_INDEX; + } + } + + /// Return the number of elements in the deque + size_t size() const { + return mSize; + } + + /// Overloaded index operator + T& operator[](const uint index) { + assert(index < mSize); + return getItem(index); + } + + /// Overloaded const index operator + const T& operator[](const uint index) const { + assert(index < mSize); + return getItem(index); + } + + /// Overloaded equality operator + bool operator==(const Deque& deque) const { + + if (mSize != deque.mSize) return false; + + for (size_t i=0; i < mSize; i++) { + if (getItem(i) != deque.getItem(i)) { + return false; + } + } + + return true; + } + + /// Overloaded not equal operator + bool operator!=(const Deque& deque) const { + + return !((*this) == deque); + } + + /// Overloaded assignment operator + Deque& operator=(const Deque& deque) { + + if (this != &deque) { + + // Clear all the elements + clear(); + + if (deque.mSize > 0) { + + // Number of used chunks + const size_t nbUsedChunks = deque.mLastChunkIndex - deque.mFirstChunkIndex + 1; + + // Expand the chunk if necessary + expandChunks(nbUsedChunks); + + const size_t dequeHalfSize1 = std::ceil(deque.mSize / 2.0f); + const size_t dequeHalfSize2 = deque.mSize - dequeHalfSize1; + + // Add the items into the deque + for(size_t i=0; i < dequeHalfSize1; i++) { + addFront(deque[dequeHalfSize1 - 1 - i]); + } + for(size_t i=0; i < dequeHalfSize2; i++) { + addBack(deque[dequeHalfSize1 + i]); + } + } + } + + return *this; + } + + /// Return a begin iterator + Iterator begin() const { + return Iterator(this, 0); + } + + /// Return a end iterator + Iterator end() const { + return Iterator(this, mSize); + } +}; + +} + +#endif diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index 02a7ec9b..15f42bc6 100755 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -18,6 +18,7 @@ SET (RP3D_TESTS_HEADERS "tests/containers/TestList.h" "tests/containers/TestMap.h" "tests/containers/TestSet.h" + "tests/containers/TestDeque.h" "tests/mathematics/TestMathematicsFunctions.h" "tests/mathematics/TestMatrix2x2.h" "tests/mathematics/TestMatrix3x3.h" diff --git a/test/main.cpp b/test/main.cpp index 24b9f003..95b73296 100644 --- a/test/main.cpp +++ b/test/main.cpp @@ -42,6 +42,7 @@ #include "tests/containers/TestList.h" #include "tests/containers/TestMap.h" #include "tests/containers/TestSet.h" +#include "tests/containers/TestDeque.h" using namespace reactphysics3d; @@ -54,6 +55,7 @@ int main() { testSuite.addTest(new TestList("List")); testSuite.addTest(new TestMap("Map")); testSuite.addTest(new TestSet("Set")); + testSuite.addTest(new TestDeque("Deque")); // ---------- Mathematics tests ---------- // diff --git a/test/tests/containers/TestDeque.h b/test/tests/containers/TestDeque.h new file mode 100644 index 00000000..8f3c9709 --- /dev/null +++ b/test/tests/containers/TestDeque.h @@ -0,0 +1,338 @@ +/******************************************************************************** +* ReactPhysics3D physics library, http://www.reactphysics3d.com * +* Copyright (c) 2010-2016 Daniel Chappuis * +********************************************************************************* +* * +* This software is provided 'as-is', without any express or implied warranty. * +* In no event will the authors be held liable for any damages arising from the * +* use of this software. * +* * +* Permission is granted to anyone to use this software for any purpose, * +* including commercial applications, and to alter it and redistribute it * +* freely, subject to the following restrictions: * +* * +* 1. The origin of this software must not be misrepresented; you must not claim * +* that you wrote the original software. If you use this software in a * +* product, an acknowledgment in the product documentation would be * +* appreciated but is not required. * +* * +* 2. Altered source versions must be plainly marked as such, and must not be * +* misrepresented as being the original software. * +* * +* 3. This notice may not be removed or altered from any source distribution. * +* * +********************************************************************************/ + +#ifndef TEST_DEQUE_H +#define TEST_DEQUE_H + +// Libraries +#include "Test.h" +#include "containers/Deque.h" +#include "memory/DefaultAllocator.h" + +/// Reactphysics3D namespace +namespace reactphysics3d { + +// Class TestDeque +/** + * Unit test for the Deque class + */ +class TestDeque : public Test { + + private : + + // ---------- Atributes ---------- // + + DefaultAllocator mAllocator; + + public : + + // ---------- Methods ---------- // + + /// Constructor + TestDeque(const std::string& name) : Test(name) { + + } + + /// Run the tests + void run() { + + testConstructors(); + testAddPopClear(); + testAssignment(); + testEquality(); + testIterators(); + } + + void testConstructors() { + + // ----- Constructors ----- // + + Deque deque1(mAllocator); + rp3d_test(deque1.size() == 0); + + Deque deque2(mAllocator); + deque2.addBack(1); + deque2.addBack(2); + deque2.addBack(3); + rp3d_test(deque2.size() == 3); + + // ----- Copy Constructors ----- // + + Deque deque3(deque1); + rp3d_test(deque3.size() == 0); + + Deque deque4(deque2); + rp3d_test(deque4.size() == deque2.size()); + Deque::Iterator it3 = deque2.begin(); + Deque::Iterator it5 = deque4.begin(); + for (uint i=0; i deque5(mAllocator); + for (uint i=0; i<300; i++) { + deque5.addBack("test"); + } + + // ----- Test [] operator ----- // + Deque deque6(deque2); + deque6.addBack(1); + rp3d_test(deque6[0] == 1); + deque6.addBack(2); + rp3d_test(deque6[0] == 1); + rp3d_test(deque6[1] == 2); + deque6.addFront(5); + rp3d_test(deque6[0] == 5); + rp3d_test(deque6[1] == 1); + rp3d_test(deque6[2] == 2); + deque6.addFront(7); + rp3d_test(deque6[0] == 7); + rp3d_test(deque6[1] == 5); + rp3d_test(deque6[2] == 1); + rp3d_test(deque6[3] == 2); + deque6.popFront(); + rp3d_test(deque6[0] == 5); + rp3d_test(deque6[1] == 1); + rp3d_test(deque6[2] == 2); + } + + void testAddPopClear() { + + // ----- Test addBack() ----- // + + Deque deque1(mAllocator); + deque1.addBack(4); + rp3d_test(deque1.size() == 1); + rp3d_test(deque1.getBack() == 4); + rp3d_test(deque1.getFront() == 4); + deque1.addBack(9); + rp3d_test(deque1.size() == 2); + rp3d_test(deque1.getBack() == 9); + rp3d_test(deque1.getFront() == 4); + deque1.addBack(1); + rp3d_test(deque1.size() == 3); + rp3d_test(deque1.getBack() == 1); + rp3d_test(deque1.getFront() == 4); + + // ----- Test addFront() ----- // + + Deque deque2(mAllocator); + deque2.addFront(4); + rp3d_test(deque2.size() == 1); + rp3d_test(deque2.getBack() == 4); + rp3d_test(deque2.getFront() == 4); + deque2.addFront(9); + rp3d_test(deque2.size() == 2); + rp3d_test(deque2.getBack() == 4); + rp3d_test(deque2.getFront() == 9); + deque2.addFront(1); + rp3d_test(deque2.size() == 3); + rp3d_test(deque2.getBack() == 4); + rp3d_test(deque2.getFront() == 1); + + // ----- Test popFront() ----- // + + deque1.popFront(); + rp3d_test(deque1.size() == 2); + rp3d_test(deque1.getFront() == 9); + deque1.popFront(); + rp3d_test(deque1.size() == 1); + rp3d_test(deque1.getFront() == 1); + deque1.popFront(); + rp3d_test(deque1.size() == 0); + + // ----- Test popBack() ----- // + + deque2.popBack(); + rp3d_test(deque2.size() == 2); + rp3d_test(deque2.getBack() == 9); + deque2.popBack(); + rp3d_test(deque2.size() == 1); + rp3d_test(deque2.getBack() == 1); + deque2.popBack(); + rp3d_test(deque2.size() == 0); + + deque2.addBack(1); + deque2.addFront(2); + rp3d_test(deque2.getBack() == 1); + rp3d_test(deque2.getFront() == 2); + deque2.addBack(3); + deque2.addFront(4); + rp3d_test(deque2.getBack() == 3); + rp3d_test(deque2.getFront() == 4); + deque2.popBack(); + deque2.popFront(); + rp3d_test(deque2.getBack() == 1); + rp3d_test(deque2.getFront() == 2); + + // ----- Test clear() ----- // + + deque1.clear(); + deque2.clear(); + rp3d_test(deque1.size() == 0); + rp3d_test(deque2.size() == 0); + + deque1.addBack(1); + deque1.addFront(2); + deque1.addBack(3); + deque1.addFront(4); + rp3d_test(deque1.getBack() == 3); + rp3d_test(deque1.getFront() == 4); + deque1.clear(); + rp3d_test(deque1.size() == 0); + + } + + void testAssignment() { + + Deque deque1(mAllocator); + deque1.addBack(1); + deque1.addBack(2); + deque1.addBack(3); + + Deque deque2(mAllocator); + deque2.addFront(5); + deque2.addBack(6); + + Deque deque3(mAllocator); + Deque deque4(mAllocator); + deque4.addBack(7); + deque4.addBack(9); + + deque3 = deque2; + rp3d_test(deque2.size() == deque3.size()); + rp3d_test(deque2.size() == 2); + rp3d_test(deque2.getBack() == deque3.getBack()); + rp3d_test(deque2.getFront() == deque3.getFront()); + + deque4 = deque1; + rp3d_test(deque4.size() == deque1.size()) + rp3d_test(deque4.size() == 3) + rp3d_test(deque4.getFront() == 1); + rp3d_test(deque4.getBack() == 3); + } + + void testEquality() { + + Deque deque1(mAllocator); + deque1.addFront(1); + deque1.addBack(2); + deque1.addBack(3); + + Deque deque2(mAllocator); + deque2.addBack(1); + deque2.addBack(2); + + Deque deque3(mAllocator); + deque3.addBack(1); + deque3.addBack(2); + deque3.addBack(3); + + Deque deque4(mAllocator); + deque4.addFront(1); + deque4.addFront(5); + deque4.addFront(3); + + Deque deque5(mAllocator); + deque5.addFront(3); + deque5.addFront(2); + deque5.addFront(1); + + rp3d_test(deque1 == deque1); + rp3d_test(deque1 != deque2); + rp3d_test(deque1 == deque3); + rp3d_test(deque1 != deque4); + rp3d_test(deque2 != deque4); + rp3d_test(deque1 == deque5); + } + + void testIterators() { + + Deque deque1(mAllocator); + + rp3d_test(deque1.begin() == deque1.end()); + + deque1.addBack(5); + deque1.addBack(6); + deque1.addBack(8); + deque1.addBack(-1); + + Deque::Iterator itBegin = deque1.begin(); + Deque::Iterator itEnd = deque1.end(); + Deque::Iterator it = deque1.begin(); + + rp3d_test(itBegin < itEnd); + rp3d_test(itBegin <= itEnd); + rp3d_test(itEnd > itBegin); + rp3d_test(itEnd >= itBegin); + + rp3d_test(itBegin == it); + rp3d_test(*it == 5); + rp3d_test(*(it++) == 5); + rp3d_test(*it == 6); + rp3d_test(*(it--) == 6); + rp3d_test(*it == 5); + rp3d_test(*(++it) == 6); + rp3d_test(*it == 6); + rp3d_test(*(--it) == 5); + rp3d_test(*it == 5); + rp3d_test(it == itBegin); + + it = deque1.end(); + rp3d_test(it == itEnd); + it--; + rp3d_test(*it == -1); + it++; + rp3d_test(it == itEnd); + + Deque deque2(mAllocator); + for (auto it = deque1.begin(); it != deque1.end(); ++it) { + deque2.addBack(*it); + } + + rp3d_test(deque1 == deque2); + + it = itBegin; + rp3d_test(*(it + 2) == 8); + it += 2; + rp3d_test(*it == 8); + rp3d_test(*(it - 2) == 5); + it -= 2; + rp3d_test(*it == 5); + rp3d_test((itEnd - itBegin) == 4); + + it = itBegin; + *it = 19; + rp3d_test(*it == 19); + } + + }; + +} + +#endif