diff --git a/src/containers/Set.h b/src/containers/Set.h index 71dd1fc1..93cba1ca 100755 --- a/src/containers/Set.h +++ b/src/containers/Set.h @@ -542,6 +542,20 @@ class Set { return end(); } + /// Return a list with all the values of the set + List toList(MemoryAllocator& listAllocator) { + + List list(listAllocator); + + for (int i=0; i < mCapacity; i++) { + if (mEntries[i].value != nullptr) { + list.add(*(mEntries[i].value)); + } + } + + return list; + } + /// Clear the set void clear(bool releaseMemory = false) { diff --git a/src/containers/Stack.h b/src/containers/Stack.h index 4fb6e518..29b81576 100644 --- a/src/containers/Stack.h +++ b/src/containers/Stack.h @@ -27,6 +27,7 @@ #define REACTPHYSICS3D_STACK_H // Libraries +#include #include "configuration.h" #include "memory/MemoryAllocator.h" @@ -34,10 +35,9 @@ namespace reactphysics3d { // Class Stack /** - * This class represents a simple generic stack with an initial capacity. If the number - * of elements exceeds the capacity, the heap will be used to allocated more memory. + * This class represents a simple generic stack. */ -template +template class Stack { private: @@ -47,84 +47,134 @@ class Stack { /// Reference to the memory allocator MemoryAllocator& mAllocator; - /// Initial array that contains the elements of the stack - T mInitArray[capacity]; - - /// Pointer to the first element of the stack - T* mElements; + /// Array that contains the elements of the stack + T* mArray; /// Number of elements in the stack uint mNbElements; /// Number of allocated elements in the stack - uint mNbAllocatedElements; + uint mCapacity; + + // -------------------- Methods -------------------- // + + /// Allocate more memory + void allocate(size_t capacity) { + + T* newArray = static_cast(mAllocator.allocate(capacity * sizeof(T))); + assert(newArray != nullptr); + + // If there + if (mCapacity > 0) { + + if (mNbElements > 0) { + + // Copy the previous items in the new array + std::uninitialized_copy(mArray, mArray + mNbElements, newArray); + } + + // Release memory of the previous array + mAllocator.release(mArray, mCapacity * sizeof(T)); + } + + mArray = newArray; + + mCapacity = capacity; + } public: // -------------------- Methods -------------------- // /// Constructor - Stack(MemoryAllocator& allocator) - :mAllocator(allocator), mElements(mInitArray), mNbElements(0), mNbAllocatedElements(capacity) { + Stack(MemoryAllocator& allocator, size_t capacity = 0) + :mAllocator(allocator), mArray(nullptr), mNbElements(0), mCapacity(0) { + if (capacity > 0) { + allocate(capacity); + } + } + + /// Copy constructor + Stack(const Stack& stack) + :mAllocator(stack.mAllocator), mArray(nullptr), + mNbElements(stack.mNbElements), mCapacity(stack.mCapacity) { + + if (mCapacity > 0) { + + // Allocate memory for the buckets + mArray = static_cast(mAllocator.allocate(mCapacity * sizeof(T))); + assert(mArray != nullptr); + + if (mNbElements > 0) { + + // Copy the items + std::uninitialized_copy(stack.mArray, stack.mArray + mNbElements, mArray); + } + } } /// Destructor ~Stack() { - // If elements have been allocated on the heap - if (mInitArray != mElements) { + clear(); - // Release the memory allocated on the heap - mAllocator.release(mElements, mNbAllocatedElements * sizeof(T)); + // Release the memory allocated on the heap + mAllocator.release(mArray, mCapacity * sizeof(T)); + } + + /// Remove all the items from the stack + void clear() { + + // Destruct the items + for (size_t i = 0; i < mNbElements; i++) { + mArray[i].~T(); } + + mNbElements = 0; } /// Push an element into the stack - void push(const T& element); + void push(const T& element) { + + // If we need to allocate more elements + if (mNbElements == mCapacity) { + + allocate(mCapacity > 0 ? mCapacity * 2 : 1); + } + + // Copy the item into the array + new (mArray + mNbElements) T(element); + + mNbElements++; + } /// Pop an element from the stack (remove it from the stack and return it) - T pop(); + T pop() { - /// Return the number of elments in the stack - uint getNbElements() const; + assert(mNbElements > 0); -}; + mNbElements--; -// Push an element into the stack -template -inline void Stack::push(const T& element) { + // Copy the item + T item = mArray[mNbElements]; - // If we need to allocate more elements - if (mNbElements == mNbAllocatedElements) { - T* oldElements = mElements; - uint oldNbAllocatedElements = mNbAllocatedElements; - mNbAllocatedElements *= 2; - mElements = static_cast(mAllocator.allocate(mNbAllocatedElements * sizeof(T))); - assert(mElements); - memcpy(mElements, oldElements, mNbElements * sizeof(T)); - if (oldElements != mInitArray) { - mAllocator.release(oldElements, oldNbAllocatedElements * sizeof(T)); + // Call the destructor + mArray[mNbElements].~T(); + + return item; } - } - mElements[mNbElements] = element; - mNbElements++; -} + /// Return the number of items in the stack + size_t size() const { + return mNbElements; + } -// Pop an element from the stack (remove it from the stack and return it) -template -inline T Stack::pop() { - assert(mNbElements > 0); - mNbElements--; - return mElements[mNbElements]; -} - -// Return the number of elments in the stack -template -inline uint Stack::getNbElements() const { - return mNbElements; -} + /// Return the capacity of the stack + size_t capacity() const { + return mCapacity; + } +}; } diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index 15f42bc6..69e5c45e 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/TestStack.h" "tests/containers/TestDeque.h" "tests/mathematics/TestMathematicsFunctions.h" "tests/mathematics/TestMatrix2x2.h" diff --git a/test/main.cpp b/test/main.cpp index 95b73296..6a965039 100644 --- a/test/main.cpp +++ b/test/main.cpp @@ -43,6 +43,7 @@ #include "tests/containers/TestMap.h" #include "tests/containers/TestSet.h" #include "tests/containers/TestDeque.h" +#include "tests/containers/TestStack.h" using namespace reactphysics3d; @@ -56,6 +57,7 @@ int main() { testSuite.addTest(new TestMap("Map")); testSuite.addTest(new TestSet("Set")); testSuite.addTest(new TestDeque("Deque")); + testSuite.addTest(new TestStack("Stack")); // ---------- Mathematics tests ---------- // diff --git a/test/tests/containers/TestSet.h b/test/tests/containers/TestSet.h index b6cf1b54..5b75cd71 100644 --- a/test/tests/containers/TestSet.h +++ b/test/tests/containers/TestSet.h @@ -90,6 +90,7 @@ class TestSet : public Test { testEquality(); testAssignment(); testIterators(); + testConverters(); } void testConstructors() { @@ -419,6 +420,31 @@ class TestSet : public Test { } rp3d_test(set1.size() == size); } + + void testConverters() { + + Set set1(mAllocator); + + rp3d_test(set1.begin() == set1.end()); + + set1.add(1); + set1.add(2); + set1.add(3); + set1.add(4); + + List list1 = set1.toList(mAllocator); + rp3d_test(list1.size() == 4); + rp3d_test(list1.find(1) != list1.end()); + rp3d_test(list1.find(2) != list1.end()); + rp3d_test(list1.find(3) != list1.end()); + rp3d_test(list1.find(4) != list1.end()); + rp3d_test(list1.find(5) == list1.end()); + rp3d_test(list1.find(6) == list1.end()); + + Set set2(mAllocator); + List list2 = set2.toList(mAllocator); + rp3d_test(list2.size() == 0); + } }; } diff --git a/test/tests/containers/TestStack.h b/test/tests/containers/TestStack.h new file mode 100644 index 00000000..e014fc4b --- /dev/null +++ b/test/tests/containers/TestStack.h @@ -0,0 +1,133 @@ +/******************************************************************************** +* 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_STACK_H +#define TEST_STACK_H + +// Libraries +#include "Test.h" +#include "containers/Stack.h" +#include "memory/DefaultAllocator.h" + +/// Reactphysics3D namespace +namespace reactphysics3d { + +// Class TestStack +/** + * Unit test for the Stack class + */ +class TestStack : public Test { + + private : + + // ---------- Atributes ---------- // + + DefaultAllocator mAllocator; + + public : + + // ---------- Methods ---------- // + + /// Constructor + TestStack(const std::string& name) : Test(name) { + + } + + /// Run the tests + void run() { + + testConstructor(); + testPushPop(); + } + + void testConstructor() { + + // ----- Constructors ----- // + + Stack stack1(mAllocator); + rp3d_test(stack1.capacity() == 0); + rp3d_test(stack1.size() == 0); + + Stack stack2(mAllocator, 100); + rp3d_test(stack2.capacity() >= 100); + rp3d_test(stack2.size() == 0); + + // ----- Copy Constructors ----- // + Stack stack3(stack2); + rp3d_test(stack3.capacity() == stack2.capacity()); + rp3d_test(stack3.size() == stack2.size()); + + Stack stack4(mAllocator); + stack4.push(10); + stack4.push(20); + stack4.push(30); + rp3d_test(stack4.capacity() >= 3); + rp3d_test(stack4.size() == 3); + stack4.push(40); + rp3d_test(stack4.size() == 4); + + Stack set5(stack4); + rp3d_test(set5.capacity() == stack4.capacity()); + rp3d_test(set5.size() == stack4.size()); + rp3d_test(set5.pop() == 40); + rp3d_test(set5.size() == 3); + rp3d_test(set5.pop() == 30); + rp3d_test(set5.pop() == 20); + rp3d_test(set5.pop() == 10); + rp3d_test(set5.size() == 0); + } + + void testPushPop() { + + Stack stack1(mAllocator); + stack1.push(10); + stack1.push(80); + stack1.push(130); + rp3d_test(stack1.size() == 3); + rp3d_test(stack1.pop() == 130); + rp3d_test(stack1.pop() == 80); + rp3d_test(stack1.pop() == 10); + rp3d_test(stack1.size() == 0); + stack1.push(10); + rp3d_test(stack1.pop() == 10); + stack1.push(10); + stack1.push(80); + stack1.pop(); + rp3d_test(stack1.pop() == 10); + rp3d_test(stack1.size() == 0); + stack1.push(10); + stack1.push(80); + stack1.push(130); + stack1.push(56); + stack1.push(89); + stack1.push(131); + stack1.clear(); + rp3d_test(stack1.size() == 0); + } + }; + +} + +#endif