aboutsummaryrefslogtreecommitdiff
path: root/test/source/TestTupleVector.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'test/source/TestTupleVector.cpp')
-rw-r--r--test/source/TestTupleVector.cpp1540
1 files changed, 1540 insertions, 0 deletions
diff --git a/test/source/TestTupleVector.cpp b/test/source/TestTupleVector.cpp
new file mode 100644
index 0000000..8a83803
--- /dev/null
+++ b/test/source/TestTupleVector.cpp
@@ -0,0 +1,1540 @@
+/////////////////////////////////////////////////////////////////////////////
+// TestTupleVector.cpp
+//
+// Copyright (c) 2018, Electronic Arts Inc. All rights reserved.
+/////////////////////////////////////////////////////////////////////////////
+
+#include "EASTLTest.h"
+
+#include <EASTL/bonus/tuple_vector.h>
+
+#include <EASTL/sort.h>
+
+using namespace eastl;
+
+int TestTupleVector()
+{
+ int nErrorCount = 0;
+
+ // Test push-backs and accessors
+ {
+ tuple_vector<int> singleElementVec;
+ EATEST_VERIFY(singleElementVec.size() == 0);
+ EATEST_VERIFY(singleElementVec.capacity() == 0);
+ EATEST_VERIFY(singleElementVec.empty() == true);
+ EATEST_VERIFY(singleElementVec.validate());
+ singleElementVec.push_back_uninitialized();
+ singleElementVec.push_back(5);
+ EATEST_VERIFY(singleElementVec.size() == 2);
+ EATEST_VERIFY(singleElementVec.capacity() > 0);
+ EATEST_VERIFY(singleElementVec.get<0>()[1] == 5);
+ EATEST_VERIFY(singleElementVec.get<int>()[1] == 5);
+ EATEST_VERIFY(singleElementVec.empty() == false);
+ EATEST_VERIFY(singleElementVec.validate());
+
+ tuple_vector<int, float, bool> complexVec;
+ complexVec.reserve(5);
+ {
+ // need to call an overload of push_back that specifically grabs lvalue candidates - providing constants tend to prefer rvalue path
+ int intArg = 3;
+ float floatArg = 2.0f;
+ bool boolArg = true;
+ complexVec.push_back(intArg, floatArg, boolArg);
+ }
+ complexVec.push_back(1, 4.0f, false);
+ complexVec.push_back(2, 1.0f, true);
+ {
+ tuple<int, float, bool> complexTup(4, 3.0f, false);
+ complexVec.push_back(complexTup);
+ }
+ complexVec.push_back();
+ EATEST_VERIFY(complexVec.capacity() == 5);
+ EATEST_VERIFY(*(complexVec.get<0>()) == 3);
+ EATEST_VERIFY(complexVec.get<float>()[1] == 4.0f);
+ EATEST_VERIFY(complexVec.get<2>()[2] == complexVec.get<bool>()[2]);
+ EATEST_VERIFY(complexVec.validate());
+
+ tuple<int, float, bool> defaultComplexTup;
+ EATEST_VERIFY(complexVec.at(4) == defaultComplexTup);
+
+ tuple<int*, float*, bool*> complexPtrTuple = complexVec.data();
+ EATEST_VERIFY(get<0>(complexPtrTuple) != nullptr);
+ EATEST_VERIFY(get<2>(complexPtrTuple)[2] == complexVec.get<2>()[2]);
+
+ tuple<int&, float&, bool&> complexRefTuple = complexVec.at(2);
+ tuple<int&, float&, bool&> complexRefTupleBracket = complexVec[2];
+ tuple<int&, float&, bool&> complexRefTupleFront = complexVec.front();
+ tuple<int&, float&, bool&> complexRefTupleBack = complexVec.back();
+ EATEST_VERIFY(get<2>(complexRefTuple) == complexVec.get<2>()[2]);
+ EATEST_VERIFY(get<1>(complexRefTupleBracket) == 1.0f);
+ EATEST_VERIFY(get<1>(complexRefTupleFront) == 2.0f);
+ EATEST_VERIFY(get<1>(complexRefTupleBack) == 0.0f);
+
+ // verify the equivalent accessors for the const container exist/compile
+ {
+ const tuple_vector<int, float, bool>& constVec = complexVec;
+
+ EATEST_VERIFY(constVec.size() == 5);
+ EATEST_VERIFY(constVec.capacity() >= constVec.size());
+ EATEST_VERIFY(constVec.empty() == false);
+ EATEST_VERIFY(constVec.get<1>() == constVec.get<float>());
+
+ tuple<const int*, const float*, const bool*> constPtrTuple = constVec.data();
+ EATEST_VERIFY(get<0>(constPtrTuple) != nullptr);
+ EATEST_VERIFY(get<2>(constPtrTuple)[2] == constVec.get<2>()[2]);
+
+ tuple<const int&, const float&, const bool&> constRefTuple = constVec.at(2);
+ tuple<const int&, const float&, const bool&> constRefTupleBracket = constVec[2];
+ tuple<const int&, const float&, const bool&> constRefTupleFront = constVec.front();
+ tuple<const int&, const float&, const bool&> constRefTupleBack = constVec.back();
+ EATEST_VERIFY(get<2>(constRefTuple) == constVec.get<2>()[2]);
+ EATEST_VERIFY(get<1>(constRefTupleBracket) == 1.0f);
+ EATEST_VERIFY(get<1>(constRefTupleFront) == 2.0f);
+ EATEST_VERIFY(get<1>(constRefTupleBack) == 0.0f);
+
+ // check that return types of const-version of begin and cbegin (etc) match
+ static_assert(eastl::is_same<decltype(constVec.begin()), decltype(constVec.cbegin())>::value, "error");
+ static_assert(eastl::is_same<decltype(constVec.end()), decltype(constVec.cend())>::value, "error");
+ static_assert(eastl::is_same<decltype(constVec.rbegin()), decltype(constVec.crbegin())>::value, "error");
+ static_assert(eastl::is_same<decltype(constVec.rend()), decltype(constVec.crend())>::value, "error");
+
+ // check that return type of non-const version of begin and cbegin (etc) do _not_ match
+ static_assert(!eastl::is_same<decltype(complexVec.begin()), decltype(complexVec.cbegin())>::value, "error");
+ static_assert(!eastl::is_same<decltype(complexVec.end()), decltype(complexVec.cend())>::value, "error");
+ static_assert(!eastl::is_same<decltype(complexVec.rbegin()), decltype(complexVec.crbegin())>::value, "error");
+ static_assert(!eastl::is_same<decltype(complexVec.rend()), decltype(complexVec.crend())>::value, "error");
+
+ }
+ }
+
+ // test the memory layouts work for aligned structures
+ {
+ struct EA_ALIGN(16) AlignTestVec4
+ {
+ float a[4];
+ AlignTestVec4() : a{1.0f, 2.0f, 3.0f, 4.0f} {}
+ };
+
+ struct AlignTestByte3
+ {
+ char a[3];
+ AlignTestByte3() : a{1, 2, 3} {}
+ };
+
+ struct EA_ALIGN(8) AlignTestFourByte
+ {
+ int a[5];
+ AlignTestFourByte() : a{-1, -2, -3, -4, -5} {}
+ };
+
+ tuple_vector<bool, AlignTestVec4, AlignTestByte3, AlignTestFourByte> alignElementVec;
+ alignElementVec.push_back();
+ alignElementVec.push_back();
+ alignElementVec.push_back();
+ alignElementVec.push_back();
+ alignElementVec.push_back();
+
+ EATEST_VERIFY((uintptr_t)alignElementVec.get<AlignTestVec4>() % 16 == 0);
+ EATEST_VERIFY((uintptr_t)alignElementVec.get<AlignTestFourByte>() % 8 == 0);
+ }
+
+ // Test resize and various modifications
+ {
+ TestObject::Reset();
+
+ tuple_vector<bool, TestObject, float> testVec;
+ typedef tuple_vector<bool, TestObject, float>::size_type tuple_vector_size_type;
+ testVec.reserve(10);
+ for (int i = 0; i < 10; ++i)
+ {
+ testVec.push_back(i % 3 == 0, TestObject(i), (float)i);
+ }
+ testVec.pop_back();
+ EATEST_VERIFY(testVec.size() == 9);
+
+ // test resize that does destruction of objects
+ testVec.resize(5);
+ EATEST_VERIFY(testVec.size() == 5);
+ EATEST_VERIFY(TestObject::sTOCount == 5);
+ EATEST_VERIFY(testVec.capacity() == 10);
+
+ // test resize that does default construction of objects
+ testVec.resize(10);
+ EATEST_VERIFY(testVec.size() == 10);
+ EATEST_VERIFY(TestObject::sTOCount == 10);
+ EATEST_VERIFY(testVec.capacity() == 10);
+
+ // test resize that does default construction of objects and grows the vector
+ testVec.resize(15);
+ EATEST_VERIFY(testVec.size() == 15);
+ EATEST_VERIFY(TestObject::sTOCount == 15);
+ EATEST_VERIFY(testVec.capacity() > 10);
+ EATEST_VERIFY(testVec.validate());
+
+ // test resize with args that does destruction of objects
+ auto testVecCapacity = testVec.capacity();
+ testVec.resize(5, true, TestObject(5), 5.0f);
+ EATEST_VERIFY(testVec.size() == 5);
+ EATEST_VERIFY(TestObject::sTOCount == 5);
+ EATEST_VERIFY(testVec.capacity() == testVecCapacity);
+
+ // test resize with args that does construction of objects
+ testVec.resize(15, true, TestObject(5), 5.0f);
+ EATEST_VERIFY(testVec.size() == 15);
+ EATEST_VERIFY(TestObject::sTOCount == 15);
+ EATEST_VERIFY(testVec.capacity() == testVecCapacity);
+
+ // test resize with args that does construction of objects and grows the vector
+ auto newTestVecSize = testVecCapacity + 5;
+ testVec.resize(newTestVecSize, true, TestObject(5), 5.0f);
+ EATEST_VERIFY(testVec.size() == newTestVecSize);
+ EATEST_VERIFY(static_cast<tuple_vector_size_type>(TestObject::sTOCount) == newTestVecSize);
+ EATEST_VERIFY(testVec.capacity() > newTestVecSize);
+ EATEST_VERIFY(testVec.validate());
+ for (unsigned int i = 5; i < newTestVecSize; ++i)
+ {
+ EATEST_VERIFY(testVec.get<0>()[i] == true);
+ EATEST_VERIFY(testVec.get<1>()[i] == TestObject(5));
+ EATEST_VERIFY(testVec.get<2>()[i] == 5.0f);
+ }
+
+ {
+ tuple<bool, TestObject, float> resizeTup(true, TestObject(10), 10.0f);
+ typedef tuple_vector<bool, TestObject, float>::size_type tuple_vector_size_type;
+
+ // test resize with tuple that does destruction of objects
+ testVecCapacity = testVec.capacity();
+ EATEST_VERIFY(testVecCapacity >= 15); // check for next two resizes to make sure we don't grow vec
+
+ testVec.resize(20, resizeTup);
+ EATEST_VERIFY(testVec.size() == 20);
+ EATEST_VERIFY(TestObject::sTOCount == 20 + 1);
+ EATEST_VERIFY(testVec.capacity() == testVecCapacity);
+
+ // test resize with tuple that does construction of objects
+ testVec.resize(25, resizeTup);
+ EATEST_VERIFY(testVec.size() == 25);
+ EATEST_VERIFY(TestObject::sTOCount == 25 + 1);
+ EATEST_VERIFY(testVec.capacity() == testVecCapacity);
+
+ // test resize with tuple that does construction of objects and grows the vector
+ newTestVecSize = testVecCapacity + 5;
+ testVec.resize(newTestVecSize, resizeTup);
+ EATEST_VERIFY(testVec.size() == newTestVecSize);
+ EATEST_VERIFY(static_cast<tuple_vector_size_type>(TestObject::sTOCount) == newTestVecSize + 1);
+ EATEST_VERIFY(testVec.capacity() > newTestVecSize);
+ EATEST_VERIFY(testVec.validate());
+ for (unsigned int i = 5; i < 20; ++i)
+ {
+ EATEST_VERIFY(testVec.get<0>()[i] == true);
+ EATEST_VERIFY(testVec.get<1>()[i] == TestObject(5));
+ EATEST_VERIFY(testVec.get<2>()[i] == 5.0f);
+ }
+ for (unsigned int i = 20; i < testVecCapacity; ++i)
+ {
+ EATEST_VERIFY(testVec.get<0>()[i] == get<0>(resizeTup));
+ EATEST_VERIFY(testVec.get<1>()[i] == get<1>(resizeTup));
+ EATEST_VERIFY(testVec.get<2>()[i] == get<2>(resizeTup));
+ }
+ }
+
+ // test other modifiers
+ testVec.pop_back();
+ EATEST_VERIFY(testVec.size() == newTestVecSize - 1);
+ EATEST_VERIFY(static_cast<decltype(testVec)::size_type>(TestObject::sTOCount) == newTestVecSize - 1); // down 2 from last sTOCount check - resizeTup dtor and pop_back
+
+ EATEST_VERIFY(testVec.capacity() > newTestVecSize);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == testVec.size());
+ EATEST_VERIFY(testVec.validate());
+
+ testVec.clear();
+ EATEST_VERIFY(testVec.empty());
+ EATEST_VERIFY(testVec.validate());
+ EATEST_VERIFY(TestObject::IsClear());
+
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 0);
+ EATEST_VERIFY(testVec.validate());
+ TestObject::Reset();
+ }
+
+ // Test insert
+ {
+ TestObject::Reset();
+
+ // test insert with n values and lvalue args
+ {
+ tuple_vector<bool, TestObject, float> testVec;
+ bool boolArg = true;
+ TestObject toArg = TestObject(0);
+ float floatArg = 0.0f;
+ testVec.reserve(10);
+
+ // test insert on empty vector that doesn't cause growth
+ toArg = TestObject(3);
+ floatArg = 3.0f;
+ auto insertIter = testVec.insert(testVec.begin(), 3, boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 3);
+ EATEST_VERIFY(insertIter == testVec.begin());
+
+ // test insert to end of vector that doesn't cause growth
+ toArg = TestObject(5);
+ floatArg = 5.0f;
+ insertIter = testVec.insert(testVec.end(), 3, boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 6);
+ EATEST_VERIFY(insertIter == testVec.begin() + 3);
+
+ // test insert to middle of vector that doesn't cause growth
+ toArg = TestObject(4);
+ floatArg = 4.0f;
+ testVec.insert(testVec.begin() + 3, 3, boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 9);
+ EATEST_VERIFY(testVec.capacity() == 10);
+
+ // test insert to end of vector that causes growth
+ toArg = TestObject(6);
+ floatArg = 6.0f;
+ testVec.insert(testVec.end(), 3, boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 12);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 12);
+
+ // test insert to beginning of vector that causes growth
+ toArg = TestObject(1);
+ floatArg = 1.0f;
+ testVec.insert(testVec.begin(), 3, boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 15);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 15);
+
+ // test insert to middle of vector that causes growth
+ toArg = TestObject(2);
+ floatArg = 2.0f;
+ testVec.insert(testVec.begin() + 3, 3, boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 18);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 18);
+
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec.get<1>()[i] == TestObject(i / 3 + 1));
+ }
+ EATEST_VERIFY(testVec.validate());
+ }
+
+ // test insert with lvalue args
+ {
+ tuple_vector<bool, TestObject, float> testVec;
+ bool boolArg = true;
+ TestObject toArg = TestObject(0);
+ float floatArg = 0.0f;
+ testVec.reserve(3);
+
+ // test insert on empty vector that doesn't cause growth
+ toArg = TestObject(3);
+ floatArg = 3.0f;
+ testVec.insert(testVec.begin(), boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 1);
+
+ // test insert to end of vector that doesn't cause growth
+ toArg = TestObject(5);
+ floatArg = 5.0f;
+ testVec.insert(testVec.end(), boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 2);
+
+ // test insert to middle of vector that doesn't cause growth
+ toArg = TestObject(4);
+ floatArg = 4.0f;
+ testVec.insert(testVec.begin() + 1, boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 3);
+ EATEST_VERIFY(testVec.capacity() == 3);
+
+ // test insert to end of vector that causes growth
+ toArg = TestObject(6);
+ floatArg = 6.0f;
+ testVec.insert(testVec.end(), boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 4);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 4);
+
+ // test insert to beginning of vector that causes growth
+ toArg = TestObject(1);
+ floatArg = 1.0f;
+ testVec.insert(testVec.begin(), boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 5);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 5);
+
+ // test insert to middle of vector that causes growth
+ toArg = TestObject(2);
+ floatArg = 2.0f;
+ testVec.insert(testVec.begin() + 1, boolArg, toArg, floatArg);
+ EATEST_VERIFY(testVec.size() == 6);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 6);
+
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec.get<1>()[i] == TestObject(i + 1));
+ }
+ EATEST_VERIFY(testVec.validate());
+ }
+
+ // test insert with n and tuple
+ {
+ tuple_vector<bool, TestObject, float> testVec;
+ tuple<bool, TestObject, float> testTup;
+ testVec.reserve(10);
+
+ // test insert on empty vector that doesn't cause growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(3), 3.0f);
+ testVec.insert(testVec.begin(), 3, testTup);
+ EATEST_VERIFY(testVec.size() == 3);
+
+ // test insert to end of vector that doesn't cause growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(5), 5.0f);
+ testVec.insert(testVec.end(), 3, testTup);
+ EATEST_VERIFY(testVec.size() == 6);
+
+ // test insert to middle of vector that doesn't cause growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(4), 4.0f);
+ testVec.insert(testVec.begin() + 3, 3, testTup);
+ EATEST_VERIFY(testVec.size() == 9);
+ EATEST_VERIFY(testVec.capacity() == 10);
+
+ // test insert to end of vector that causes growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(6), 6.0f);
+ testVec.insert(testVec.end(), 3, testTup);
+ EATEST_VERIFY(testVec.size() == 12);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 12);
+
+ // test insert to beginning of vector that causes growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(1), 1.0f);
+ testVec.insert(testVec.begin(), 3, testTup);
+ EATEST_VERIFY(testVec.size() == 15);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 15);
+
+ // test insert to middle of vector that causes growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(2), 2.0f);
+ testVec.insert(testVec.begin() + 3, 3, testTup);
+ EATEST_VERIFY(testVec.size() == 18);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 18);
+
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec.get<1>()[i] == TestObject(i / 3 + 1));
+ }
+ EATEST_VERIFY(testVec.validate());
+ }
+
+ // test insert with tuple
+ {
+ tuple_vector<bool, TestObject, float> testVec;
+ tuple<bool, TestObject, float> testTup;
+ testVec.reserve(3);
+
+ // test insert on empty vector that doesn't cause growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(3), 3.0f);
+ testVec.insert(testVec.begin(), testTup);
+ EATEST_VERIFY(testVec.size() == 1);
+
+ // test insert to end of vector that doesn't cause growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(5), 5.0f);
+ testVec.insert(testVec.end(), testTup);
+ EATEST_VERIFY(testVec.size() == 2);
+
+ // test insert to middle of vector that doesn't cause growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(4), 4.0f);
+ testVec.insert(testVec.begin() + 1, testTup);
+ EATEST_VERIFY(testVec.size() == 3);
+ EATEST_VERIFY(testVec.capacity() == 3);
+
+ // test insert to end of vector that causes growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(6), 6.0f);
+ testVec.insert(testVec.end(), 1, testTup);
+ EATEST_VERIFY(testVec.size() == 4);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 4);
+
+ // test insert to beginning of vector that causes growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(1), 1.0f);
+ testVec.insert(testVec.begin(), 1, testTup);
+ EATEST_VERIFY(testVec.size() == 5);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 5);
+
+ // test insert to middle of vector that causes growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(2), 2.0f);
+ testVec.insert(testVec.begin() + 1, 1, testTup);
+ EATEST_VERIFY(testVec.size() == 6);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 6);
+
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec.get<1>()[i] == TestObject(i + 1));
+ }
+ EATEST_VERIFY(testVec.validate());
+ }
+
+ // test insert with initList
+ {
+ tuple_vector<bool, TestObject, float> testVec;
+ tuple<bool, TestObject, float> testTup;
+ testVec.reserve(10);
+
+ // test insert on empty vector that doesn't cause growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(3), 3.0f);
+ testVec.insert(testVec.begin(), {
+ {true, TestObject(3), 3.0f},
+ testTup,
+ {true, TestObject(3), 3.0f}
+ });
+ EATEST_VERIFY(testVec.size() == 3);
+
+ // test insert to end of vector that doesn't cause growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(5), 5.0f);
+ testVec.insert(testVec.end(), {
+ {true, TestObject(5), 5.0f},
+ testTup,
+ {true, TestObject(5), 5.0f}
+ });
+ EATEST_VERIFY(testVec.size() == 6);
+
+ // test insert to middle of vector that doesn't cause growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(4), 4.0f);
+ testVec.insert(testVec.begin() + 3, {
+ {true, TestObject(4), 4.0f},
+ testTup,
+ {true, TestObject(4), 4.0f}
+ });
+ EATEST_VERIFY(testVec.size() == 9);
+ EATEST_VERIFY(testVec.capacity() == 10);
+
+ // test insert to end of vector that causes growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(6), 6.0f);
+ testVec.insert(testVec.end(), {
+ {true, TestObject(6), 6.0f},
+ testTup,
+ {true, TestObject(6), 6.0f}
+ });
+ EATEST_VERIFY(testVec.size() == 12);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 12);
+
+ // test insert to beginning of vector that causes growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(1), 1.0f);
+ testVec.insert(testVec.begin(), {
+ {true, TestObject(1), 1.0f},
+ testTup,
+ {true, TestObject(1), 1.0f}
+ });
+ EATEST_VERIFY(testVec.size() == 15);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 15);
+
+ // test insert to middle of vector that causes growth
+ testTup = tuple<bool, TestObject, float>(true, TestObject(2), 2.0f);
+ testVec.insert(testVec.begin() + 3, {
+ {true, TestObject(2), 2.0f},
+ testTup,
+ {true, TestObject(2), 2.0f
+ } });
+ EATEST_VERIFY(testVec.size() == 18);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 18);
+
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec.get<1>()[i] == TestObject(i / 3 + 1));
+ }
+ EATEST_VERIFY(testVec.validate());
+ }
+
+ // test insert with rvalue args
+ {
+ tuple_vector<int, MoveOnlyType, TestObject> testVec;
+ testVec.reserve(3);
+
+ // test insert on empty vector that doesn't cause growth
+ testVec.insert(testVec.begin(), 3, MoveOnlyType(3), TestObject(3));
+ EATEST_VERIFY(testVec.size() == 1);
+
+ // test insert to end of vector that doesn't cause growth
+ testVec.insert(testVec.end(), 5, MoveOnlyType(5), TestObject(5));
+ EATEST_VERIFY(testVec.size() == 2);
+
+ // test insert to middle of vector that doesn't cause growth
+ testVec.insert(testVec.begin() + 1, 4, MoveOnlyType(4), TestObject(4));
+ EATEST_VERIFY(testVec.size() == 3);
+ EATEST_VERIFY(testVec.capacity() == 3);
+
+ // test insert to end of vector that causes growth
+ testVec.insert(testVec.end(), 6, MoveOnlyType(6), TestObject(6));
+ EATEST_VERIFY(testVec.size() == 4);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 4);
+
+ // test insert to beginning of vector that causes growth
+ testVec.insert(testVec.begin(), 1, MoveOnlyType(1), TestObject(1));
+ EATEST_VERIFY(testVec.size() == 5);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 5);
+
+ // test insert to middle of vector that causes growth
+ testVec.insert(testVec.begin() + 1, 2, MoveOnlyType(2), TestObject(2));
+ EATEST_VERIFY(testVec.size() == 6);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 6);
+
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec.get<2>()[i] == TestObject(i + 1));
+ }
+ EATEST_VERIFY(testVec.validate());
+ }
+
+ // test insert with rvalue tuple
+ {
+ tuple_vector<int, MoveOnlyType, TestObject> testVec;
+ testVec.reserve(3);
+
+ // test insert on empty vector that doesn't cause growth
+ testVec.insert(testVec.begin(), forward_as_tuple(3, MoveOnlyType(3), TestObject(3)));
+ EATEST_VERIFY(testVec.size() == 1);
+
+ // test insert to end of vector that doesn't cause growth
+ testVec.insert(testVec.end(), forward_as_tuple(5, MoveOnlyType(5), TestObject(5)));
+ EATEST_VERIFY(testVec.size() == 2);
+
+ // test insert to middle of vector that doesn't cause growth
+ testVec.insert(testVec.begin() + 1, forward_as_tuple(4, MoveOnlyType(4), TestObject(4)));
+ EATEST_VERIFY(testVec.size() == 3);
+ EATEST_VERIFY(testVec.capacity() == 3);
+
+ // test insert to end of vector that causes growth
+ testVec.insert(testVec.end(), forward_as_tuple(6, MoveOnlyType(6), TestObject(6)));
+ EATEST_VERIFY(testVec.size() == 4);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 4);
+
+ // test insert to beginning of vector that causes growth
+ testVec.insert(testVec.begin(), forward_as_tuple(1, MoveOnlyType(1), TestObject(1)));
+ EATEST_VERIFY(testVec.size() == 5);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 5);
+
+ // test insert to middle of vector that causes growth
+ testVec.insert(testVec.begin() + 1, forward_as_tuple(2, MoveOnlyType(2), TestObject(2)));
+ EATEST_VERIFY(testVec.size() == 6);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 6);
+
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec.get<2>()[i] == TestObject(i + 1));
+ }
+ EATEST_VERIFY(testVec.validate());
+ }
+
+ // test insert with iterator range
+ {
+ tuple_vector<bool, TestObject, float> srcVec;
+ for (unsigned int i = 0; i < 20; ++i)
+ {
+ srcVec.push_back(true, TestObject(i), (float)i);
+ }
+
+ tuple_vector<bool, TestObject, float> testVec;
+ testVec.reserve(10);
+
+ // test insert on empty vector that doesn't cause growth
+ testVec.insert(testVec.begin(), srcVec.begin() + 6, srcVec.begin() + 9);
+ EATEST_VERIFY(testVec.size() == 3);
+
+ // test insert to end of vector that doesn't cause growth
+ testVec.insert(testVec.end(), srcVec.begin() + 12, srcVec.begin() + 15);
+ EATEST_VERIFY(testVec.size() == 6);
+
+ // test insert to middle of vector that doesn't cause growth
+ testVec.insert(testVec.begin() + 3, srcVec.begin() + 9, srcVec.begin() + 12);
+ EATEST_VERIFY(testVec.size() == 9);
+ EATEST_VERIFY(testVec.capacity() == 10);
+
+ // test insert to end of vector that causes growth
+ testVec.insert(testVec.end(), srcVec.begin() + 15, srcVec.begin() + 18);
+ EATEST_VERIFY(testVec.size() == 12);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 12);
+
+ // test insert to beginning of vector that causes growth
+ testVec.insert(testVec.begin(), srcVec.begin(), srcVec.begin() + 3);
+ EATEST_VERIFY(testVec.size() == 15);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 15);
+
+ // test insert to middle of vector that causes growth
+ testVec.insert(testVec.begin() + 3, srcVec.begin() + 3, srcVec.begin() + 6);
+ EATEST_VERIFY(testVec.size() == 18);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 18);
+
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i), (float)i));
+ }
+ EATEST_VERIFY(testVec.validate());
+ }
+ EATEST_VERIFY(TestObject::IsClear());
+ TestObject::Reset();
+ }
+
+ // Test assign
+ {
+ {
+ tuple_vector<bool, TestObject, float> testVec;
+
+ // test assign that grows the capacity
+ testVec.assign(20, true, TestObject(1), 1.0f);
+ EATEST_VERIFY(testVec.size() == 20);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(1), 1.0f));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 20);
+
+ // test assign that shrinks the vector
+ testVec.assign(10, true, TestObject(2), 2.0f);
+ EATEST_VERIFY(testVec.size() == 10);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(2), 2.0f));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 10);
+
+ // test assign for when there's enough capacity
+ testVec.assign(15, true, TestObject(3), 3.0f);
+ EATEST_VERIFY(testVec.size() == 15);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(3), 3.0f));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 15);
+ }
+
+ {
+ tuple<bool, TestObject, float> srcTup;
+ tuple_vector<bool, TestObject, float> testVec;
+
+ // test assign from tuple that grows the capacity
+ srcTup = make_tuple(true, TestObject(1), 1.0f);
+ testVec.assign(20, srcTup);
+ EATEST_VERIFY(testVec.size() == 20);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == srcTup);
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 20 + 1);
+
+ // test assign from tuple that shrinks the vector
+ srcTup = make_tuple(true, TestObject(2), 2.0f);
+ testVec.assign(10, srcTup);
+ EATEST_VERIFY(testVec.size() == 10);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == srcTup);
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 10 + 1);
+
+ // test assign from tuple for when there's enough capacity
+ srcTup = make_tuple(true, TestObject(3), 3.0f);
+ testVec.assign(15, srcTup);
+ EATEST_VERIFY(testVec.size() == 15);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == srcTup);
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 15 + 1);
+ }
+
+ {
+ tuple_vector<bool, TestObject, float> srcVec;
+ for (unsigned int i = 0; i < 20; ++i)
+ {
+ srcVec.push_back(true, TestObject(i), (float)i);
+ }
+ tuple_vector<bool, TestObject, float> testVec;
+
+ // test assign from iter range that grows the capacity
+ testVec.assign(srcVec.begin() + 5, srcVec.begin() + 15);
+ EATEST_VERIFY(testVec.size() == 10);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == srcVec[i+5]);
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 10 + 20);
+
+ // test assign from iter range that shrinks the vector
+ testVec.assign(srcVec.begin() + 2, srcVec.begin() + 7);
+ EATEST_VERIFY(testVec.size() == 5);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == srcVec[i + 2]);
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 5 + 20);
+
+ // test assign from iter range for when there's enough capacity
+ testVec.assign(srcVec.begin() + 5, srcVec.begin() + 15);
+ EATEST_VERIFY(testVec.size() == 10);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == srcVec[i + 5]);
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 10 + 20);
+ }
+
+ {
+ tuple_vector<bool, TestObject, float> testVec;
+
+ // test assign from initList that grows the capacity
+ testVec.assign({
+ { true, TestObject(1), 1.0f },
+ { true, TestObject(2), 2.0f },
+ { true, TestObject(3), 3.0f }
+ });
+ EATEST_VERIFY(testVec.size() == 3);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i + 1), (float)i + 1.0f));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 3);
+
+ // test assign from initList that shrinks the vector
+ testVec.assign({
+ { true, TestObject(4), 4.0f }
+ });
+ EATEST_VERIFY(testVec.size() == 1);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i + 4), (float)i + 4.0f));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 1);
+
+ // test assign from initList for when there's enough capacity
+ testVec.assign({
+ { true, TestObject(5), 5.0f },
+ { true, TestObject(6), 6.0f }
+ });
+ EATEST_VERIFY(testVec.size() == 2);
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i + 5), (float)i + 5.0f));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 2);
+ }
+
+ EATEST_VERIFY(TestObject::IsClear());
+ TestObject::Reset();
+ }
+
+ // Test erase functions
+ {
+ {
+ tuple_vector<bool, TestObject, float> srcVec;
+ for (unsigned int i = 0; i < 20; ++i)
+ {
+ srcVec.push_back(true, TestObject(i), (float)i);
+ }
+ tuple_vector<bool, TestObject, float> testVec;
+
+ // test erase on an iter range
+ testVec.assign(srcVec.begin(), srcVec.end());
+ auto eraseIter = testVec.erase(testVec.begin() + 5, testVec.begin() + 10);
+ EATEST_VERIFY(eraseIter == testVec.begin() + 5);
+ EATEST_VERIFY(testVec.size() == 15);
+ EATEST_VERIFY(testVec.validate());
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ if (i < 5)
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i), (float)i));
+ else
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i + 5), (float)(i + 5)));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 15 + 20);
+
+ // test erase on one position
+ testVec.assign(srcVec.begin(), srcVec.end());
+ eraseIter = testVec.erase(testVec.begin() + 5);
+ EATEST_VERIFY(eraseIter == testVec.begin() + 5);
+ EATEST_VERIFY(testVec.size() == 19);
+ EATEST_VERIFY(testVec.validate());
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ if (i < 5)
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i), (float)i));
+ else
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i + 1), (float)(i + 1)));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 19 + 20);
+
+ // test erase_unsorted
+ testVec.assign(srcVec.begin(), srcVec.end());
+ eraseIter = testVec.erase_unsorted(testVec.begin() + 5);
+ EATEST_VERIFY(eraseIter == testVec.begin() + 5);
+ EATEST_VERIFY(testVec.size() == 19);
+ EATEST_VERIFY(testVec.validate());
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ if (i != 5)
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i), (float)i));
+ else
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(19), (float)(19)));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 19 + 20);
+ }
+
+ // test erase again but with reverse iterators everywhere
+ {
+ tuple_vector<bool, TestObject, float> srcVec;
+ for (unsigned int i = 0; i < 20; ++i)
+ {
+ srcVec.push_back(true, TestObject(i), (float)i);
+ }
+ tuple_vector<bool, TestObject, float> testVec;
+
+ // test erase on an iter range
+ testVec.assign(srcVec.begin(), srcVec.end());
+ auto eraseIter = testVec.erase(testVec.rbegin() + 5, testVec.rbegin() + 10);
+ EATEST_VERIFY(eraseIter == testVec.rbegin() + 5);
+ EATEST_VERIFY(testVec.size() == 15);
+ EATEST_VERIFY(testVec.validate());
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ if (i < 10)
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i), (float)i));
+ else
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i + 5), (float)(i + 5)));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 15 + 20);
+
+ // test erase on one position
+ testVec.assign(srcVec.begin(), srcVec.end());
+ eraseIter = testVec.erase(testVec.rbegin() + 5);
+ EATEST_VERIFY(eraseIter == testVec.rbegin() + 5);
+ EATEST_VERIFY(testVec.size() == 19);
+ EATEST_VERIFY(testVec.validate());
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ if (i < 14)
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i), (float)i));
+ else
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i + 1), (float)(i + 1)));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 19 + 20);
+
+ // test erase_unsorted
+ testVec.assign(srcVec.begin(), srcVec.end());
+ eraseIter = testVec.erase_unsorted(testVec.rbegin() + 5);
+ EATEST_VERIFY(eraseIter == testVec.rbegin() + 5);
+ EATEST_VERIFY(testVec.size() == 19);
+ EATEST_VERIFY(testVec.validate());
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ if (i != 14)
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(i), (float)i));
+ else
+ EATEST_VERIFY(testVec[i] == make_tuple(true, TestObject(19), (float)(19)));
+ }
+ EATEST_VERIFY(TestObject::sTOCount == 19 + 20);
+ }
+ EATEST_VERIFY(TestObject::IsClear());
+ TestObject::Reset();
+ }
+
+ // Test multitude of constructors
+ {
+ MallocAllocator ma;
+ TestObject::Reset();
+
+ // test ctor via initlist to prime srcVec
+ tuple_vector<bool, TestObject, float> srcVec({
+ { true, TestObject(0), 0.0f},
+ { false, TestObject(1), 1.0f},
+ { false, TestObject(2), 2.0f},
+ { true, TestObject(3), 3.0f},
+ { false, TestObject(4), 4.0f},
+ { false, TestObject(5), 5.0f},
+ { true, TestObject(6), 6.0f},
+ { false, TestObject(7), 7.0f},
+ { false, TestObject(8), 8.0f},
+ { true, TestObject(9), 9.0f}
+ });
+
+ // copy entire tuple_vector in ctor
+ {
+ tuple_vector<bool, TestObject, float> ctorFromConstRef(srcVec);
+ EATEST_VERIFY(ctorFromConstRef.size() == 10);
+ EATEST_VERIFY(ctorFromConstRef.validate());
+ for (int i = 0; i < 10; ++i)
+ {
+ EATEST_VERIFY(ctorFromConstRef.get<0>()[i] == (i % 3 == 0));
+ EATEST_VERIFY(ctorFromConstRef.get<1>()[i] == TestObject(i));
+ EATEST_VERIFY(ctorFromConstRef.get<2>()[i] == (float)i);
+ }
+ }
+
+ // copy entire tuple_vector via assignment
+ {
+ tuple_vector<bool, TestObject, float> ctorFromAssignment;
+ ctorFromAssignment = srcVec;
+ EATEST_VERIFY(ctorFromAssignment.size() == 10);
+ EATEST_VERIFY(ctorFromAssignment.validate());
+ for (int i = 0; i < 10; ++i)
+ {
+ EATEST_VERIFY(ctorFromAssignment.get<0>()[i] == (i % 3 == 0));
+ EATEST_VERIFY(ctorFromAssignment.get<1>()[i] == TestObject(i));
+ EATEST_VERIFY(ctorFromAssignment.get<2>()[i] == (float)i);
+ }
+ }
+
+ // copy entire tuple_vector via assignment of init-list
+ {
+ tuple_vector<bool, TestObject, float> ctorFromAssignment;
+ ctorFromAssignment = {
+ { true, TestObject(0), 0.0f},
+ { false, TestObject(1), 1.0f},
+ { false, TestObject(2), 2.0f},
+ { true, TestObject(3), 3.0f},
+ { false, TestObject(4), 4.0f},
+ { false, TestObject(5), 5.0f},
+ { true, TestObject(6), 6.0f},
+ { false, TestObject(7), 7.0f},
+ { false, TestObject(8), 8.0f},
+ { true, TestObject(9), 9.0f}
+ };
+ EATEST_VERIFY(ctorFromAssignment.size() == 10);
+ EATEST_VERIFY(ctorFromAssignment.validate());
+ for (int i = 0; i < 10; ++i)
+ {
+ EATEST_VERIFY(ctorFromAssignment.get<0>()[i] == (i % 3 == 0));
+ EATEST_VERIFY(ctorFromAssignment.get<1>()[i] == TestObject(i));
+ EATEST_VERIFY(ctorFromAssignment.get<2>()[i] == (float)i);
+ }
+ }
+
+ // ctor tuple_vector with iterator range
+ {
+ tuple_vector<bool, TestObject, float> ctorFromIters(srcVec.begin() + 2, srcVec.begin() + 7);
+ EATEST_VERIFY(ctorFromIters.size() == 5);
+ EATEST_VERIFY(ctorFromIters.validate());
+ for (int i = 2; i < 7; ++i)
+ {
+ EATEST_VERIFY(ctorFromIters.get<0>()[i - 2] == (i % 3 == 0));
+ EATEST_VERIFY(ctorFromIters.get<1>()[i - 2] == TestObject(i));
+ EATEST_VERIFY(ctorFromIters.get<2>()[i - 2] == (float)i);
+ }
+ }
+
+ // ctor tuple_vector with initial size
+ {
+ tuple_vector<bool, TestObject, float> ctorFromFill(10);
+ EATEST_VERIFY(ctorFromFill.size() == 10);
+ EATEST_VERIFY(ctorFromFill.validate());
+ for (int i = 0; i < 10; ++i)
+ {
+ EATEST_VERIFY(ctorFromFill.get<0>()[i] == false);
+ EATEST_VERIFY(ctorFromFill.get<1>()[i] == TestObject());
+ EATEST_VERIFY(ctorFromFill.get<2>()[i] == 0.0f);
+ }
+ }
+
+ // ctor tuple_vector with initial size and args
+ {
+ tuple_vector<bool, TestObject, float> ctorFromFillArgs(10, true, TestObject(5), 5.0f);
+ EATEST_VERIFY(ctorFromFillArgs.size() == 10);
+ EATEST_VERIFY(ctorFromFillArgs.validate());
+ for (int i = 0; i < 10; ++i)
+ {
+ EATEST_VERIFY(ctorFromFillArgs.get<0>()[i] == true);
+ EATEST_VERIFY(ctorFromFillArgs.get<1>()[i] == TestObject(5));
+ EATEST_VERIFY(ctorFromFillArgs.get<2>()[i] == 5.0f);
+ }
+ }
+
+ // ctor tuple_vector with initial size and tuple
+ {
+ tuple<bool, TestObject, float> tup(true, TestObject(5), 5.0f);
+ tuple_vector<bool, TestObject, float> ctorFromFillTup(10, tup);
+ EATEST_VERIFY(ctorFromFillTup.size() == 10);
+ EATEST_VERIFY(ctorFromFillTup.validate());
+ for (int i = 0; i < 10; ++i)
+ {
+ EATEST_VERIFY(ctorFromFillTup.get<0>()[i] == true);
+ EATEST_VERIFY(ctorFromFillTup.get<1>()[i] == TestObject(5));
+ EATEST_VERIFY(ctorFromFillTup.get<2>()[i] == 5.0f);
+ }
+ }
+
+ // ctor tuple_Vector with custom mallocator
+ {
+ tuple_vector_alloc<MallocAllocator, bool, TestObject, float> ctorWithAlloc(ma);
+ tuple_vector<bool, TestObject, float> ctorDefault;
+
+ ctorWithAlloc.push_back();
+ ctorDefault.push_back();
+
+ EATEST_VERIFY(ctorWithAlloc == ctorDefault);
+ EATEST_VERIFY(ctorWithAlloc.validate());
+ }
+
+ // ctor tuple_vector_alloc with copy (from diff. allocator)
+ {
+ tuple_vector_alloc<MallocAllocator, bool, TestObject, float> ctorFromConstRef(srcVec, ma);
+ EATEST_VERIFY(ctorFromConstRef.size() == 10);
+ EATEST_VERIFY(ctorFromConstRef.validate());
+ for (int i = 0; i < 10; ++i)
+ {
+ EATEST_VERIFY(ctorFromConstRef.get<0>()[i] == (i % 3 == 0));
+ EATEST_VERIFY(ctorFromConstRef.get<1>()[i] == TestObject(i));
+ EATEST_VERIFY(ctorFromConstRef.get<2>()[i] == (float)i);
+ }
+ EATEST_VERIFY(ctorFromConstRef.validate());
+ }
+
+ // ctor tuple_vector with initial size and args
+ {
+ tuple_vector_alloc<MallocAllocator, bool, TestObject, float> ctorFromFillArgs(10, true, TestObject(5), 5.0f, ma);
+ EATEST_VERIFY(ctorFromFillArgs.size() == 10);
+ EATEST_VERIFY(ctorFromFillArgs.validate());
+ for (int i = 0; i < 10; ++i)
+ {
+ EATEST_VERIFY(ctorFromFillArgs.get<0>()[i] == true);
+ EATEST_VERIFY(ctorFromFillArgs.get<1>()[i] == TestObject(5));
+ EATEST_VERIFY(ctorFromFillArgs.get<2>()[i] == 5.0f);
+ }
+ }
+
+ // ctor tuple_vector via move
+ {
+ tuple_vector<int, MoveOnlyType, TestObject> srcMoveVec;
+ for (int i = 0; i < 10; ++i)
+ {
+ srcMoveVec.emplace_back(move(i), MoveOnlyType(i), TestObject(i));
+ }
+
+ tuple_vector<int, MoveOnlyType, TestObject> ctorFromMove(move(srcMoveVec));
+
+ EATEST_VERIFY(ctorFromMove.size() == 10);
+ EATEST_VERIFY(ctorFromMove.validate());
+ for (int i = 0; i < 10; ++i)
+ {
+ EATEST_VERIFY(ctorFromMove.get<0>()[i] == i);
+ EATEST_VERIFY(ctorFromMove.get<1>()[i] == MoveOnlyType(i));
+ EATEST_VERIFY(ctorFromMove.get<2>()[i] == TestObject(i));
+ }
+ EATEST_VERIFY(srcMoveVec.size() == 0);
+ EATEST_VERIFY(srcMoveVec.validate());
+ }
+
+ // ctor tuple_vector via move (from diff. allocator)
+ {
+ tuple_vector_alloc<MallocAllocator, int, MoveOnlyType, TestObject> srcMoveVec;
+ for (int i = 0; i < 10; ++i)
+ {
+ srcMoveVec.emplace_back(move(i), MoveOnlyType(i), TestObject(i));
+ }
+
+ MallocAllocator otherMa;
+ tuple_vector_alloc<MallocAllocator, int, MoveOnlyType, TestObject> ctorFromMove(move(srcMoveVec), otherMa);
+
+ EATEST_VERIFY(ctorFromMove.size() == 10);
+ EATEST_VERIFY(ctorFromMove.validate());
+ for (int i = 0; i < 10; ++i)
+ {
+ EATEST_VERIFY(ctorFromMove.get<0>()[i] == i);
+ EATEST_VERIFY(ctorFromMove.get<1>()[i] == MoveOnlyType(i));
+ EATEST_VERIFY(ctorFromMove.get<2>()[i] == TestObject(i));
+ }
+ EATEST_VERIFY(srcMoveVec.size() == 0);
+ EATEST_VERIFY(srcMoveVec.validate());
+
+ // bonus test for specifying a custom allocator, but using the same one as above
+ tuple_vector_alloc<MallocAllocator, int, MoveOnlyType, TestObject> ctorFromMoveSameAlloc(move(ctorFromMove), otherMa);
+ EATEST_VERIFY(ctorFromMoveSameAlloc.size() == 10);
+ EATEST_VERIFY(ctorFromMoveSameAlloc.validate());
+ for (int i = 0; i < 10; ++i)
+ {
+ EATEST_VERIFY(ctorFromMoveSameAlloc.get<0>()[i] == i);
+ EATEST_VERIFY(ctorFromMoveSameAlloc.get<1>()[i] == MoveOnlyType(i));
+ EATEST_VERIFY(ctorFromMoveSameAlloc.get<2>()[i] == TestObject(i));
+ }
+ EATEST_VERIFY(ctorFromMove.size() == 0);
+ EATEST_VERIFY(ctorFromMove.validate());
+ }
+
+ // ctor tuple_vector via move-iters
+ {
+ tuple_vector<int, MoveOnlyType, TestObject> srcMoveVec;
+ for (int i = 0; i < 10; ++i)
+ {
+ srcMoveVec.emplace_back(move(i), MoveOnlyType(i), TestObject(i));
+ }
+
+ tuple_vector<int, MoveOnlyType, TestObject> ctorFromMove(make_move_iterator(srcMoveVec.begin() + 2), make_move_iterator(srcMoveVec.begin() + 7));
+
+ EATEST_VERIFY(ctorFromMove.size() == 5);
+ EATEST_VERIFY(ctorFromMove.validate());
+ for (int i = 2; i < 7; ++i)
+ {
+ EATEST_VERIFY(ctorFromMove.get<0>()[i-2] == i);
+ EATEST_VERIFY(ctorFromMove.get<1>()[i-2] == MoveOnlyType(i));
+ EATEST_VERIFY(ctorFromMove.get<2>()[i-2] == TestObject(i));
+ }
+ EATEST_VERIFY(srcMoveVec.size() == 10);
+ EATEST_VERIFY(srcMoveVec.validate());
+ for (int i = 0; i < 2; ++i)
+ {
+ EATEST_VERIFY(srcMoveVec.get<0>()[i] == i);
+ EATEST_VERIFY(srcMoveVec.get<1>()[i] == MoveOnlyType(i));
+ EATEST_VERIFY(srcMoveVec.get<2>()[i] == TestObject(i));
+ }
+ for (int i = 2; i < 7; ++i)
+ {
+ EATEST_VERIFY(srcMoveVec.get<0>()[i] == i); // int's just get copied because they're POD
+ EATEST_VERIFY(srcMoveVec.get<1>()[i] == MoveOnlyType(0));
+ EATEST_VERIFY(srcMoveVec.get<2>()[i] == TestObject(0));
+ }
+ for (int i = 7; i < 10; ++i)
+ {
+ EATEST_VERIFY(srcMoveVec.get<0>()[i] == i);
+ EATEST_VERIFY(srcMoveVec.get<1>()[i] == MoveOnlyType(i));
+ EATEST_VERIFY(srcMoveVec.get<2>()[i] == TestObject(i));
+ }
+ }
+
+ srcVec.clear();
+ EATEST_VERIFY(TestObject::IsClear());
+
+ TestObject::Reset();
+ }
+
+ // Test swap
+ {
+ tuple_vector<int, float, bool> complexVec;
+ complexVec.push_back(3, 2.0f, true);
+ complexVec.push_back(1, 4.0f, false);
+ complexVec.push_back(2, 1.0f, true);
+ complexVec.push_back(4, 3.0f, false);
+
+ tuple_vector<int, float, bool> otherComplexVec;
+ complexVec.swap(otherComplexVec);
+
+ EATEST_VERIFY(complexVec.size() == 0);
+ EATEST_VERIFY(complexVec.validate());
+ EATEST_VERIFY(otherComplexVec.validate());
+ EATEST_VERIFY(otherComplexVec.get<0>()[0] == 3);
+ EATEST_VERIFY(otherComplexVec.get<float>()[1] == 4.0f);
+
+ complexVec.push_back(10, 10.0f, true);
+ swap(complexVec, otherComplexVec);
+
+ EATEST_VERIFY(complexVec.validate());
+ EATEST_VERIFY(*(complexVec.get<0>()) == 3);
+ EATEST_VERIFY(complexVec.get<float>()[1] == 4.0f);
+
+ EATEST_VERIFY(otherComplexVec.validate());
+ EATEST_VERIFY(otherComplexVec.get<float>()[0] == 10.0f);
+ EATEST_VERIFY(otherComplexVec.size() == 1);
+
+ }
+
+
+ // Test tuple_Vector in a ranged for, and other large-scale iterator testing
+ {
+ tuple_vector<int, float, int> tripleElementVec;
+ tripleElementVec.push_back(1, 2.0f, 6);
+ tripleElementVec.push_back(2, 3.0f, 7);
+ tripleElementVec.push_back(3, 4.0f, 8);
+ tripleElementVec.push_back(4, 5.0f, 9);
+ tripleElementVec.push_back(5, 6.0f, 10);
+
+
+ // test copyConstructible, copyAssignable, swappable, prefix inc, !=, reference convertible to value_type (InputIterator!)
+ {
+ tuple_vector<int, float, int>::iterator iter = tripleElementVec.begin();
+ ++iter;
+ auto copiedIter(iter);
+ EATEST_VERIFY(get<2>(*copiedIter) == 7);
+ EATEST_VERIFY(copiedIter == iter);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(iter) != isf_none);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(copiedIter) != isf_none);
+
+ ++iter;
+ copiedIter = iter;
+ EATEST_VERIFY(get<2>(*copiedIter) == 8);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(iter) != isf_none);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(copiedIter) != isf_none);
+
+ ++iter;
+ swap(iter, copiedIter);
+ EATEST_VERIFY(get<2>(*iter) == 8);
+ EATEST_VERIFY(get<2>(*copiedIter) == 9);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(iter) != isf_none);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(copiedIter) != isf_none);
+
+ EATEST_VERIFY(copiedIter != iter);
+
+ tuple<const int&, const float&, const int&> ref(*iter);
+ tuple<int, float, int> value(*iter);
+ EATEST_VERIFY(get<2>(ref) == get<2>(value));
+ }
+
+ // test postfix increment, default constructible (ForwardIterator)
+ {
+ tuple_vector<int, float, int>::iterator iter = tripleElementVec.begin();
+ auto prefixIter = ++iter;
+
+ tuple_vector<int, float, int>::iterator postfixIter;
+ postfixIter = iter++;
+ EATEST_VERIFY(prefixIter == postfixIter);
+ EATEST_VERIFY(get<2>(*prefixIter) == 7);
+ EATEST_VERIFY(get<2>(*iter) == 8);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(iter) != isf_none);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(prefixIter) != isf_none);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(postfixIter) != isf_none);
+ }
+
+ // test prefix decrement and postfix decrement (BidirectionalIterator)
+ {
+ tuple_vector<int, float, int>::iterator iter = tripleElementVec.end();
+ auto prefixIter = --iter;
+
+ tuple_vector<int, float, int>::iterator postfixIter;
+ postfixIter = iter--;
+ EATEST_VERIFY(prefixIter == postfixIter);
+ EATEST_VERIFY(get<2>(*prefixIter) == 10);
+ EATEST_VERIFY(get<2>(*iter) == 9);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(iter) != isf_none);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(prefixIter) != isf_none);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(postfixIter) != isf_none);
+ }
+
+ // test many arithmetic operations (RandomAccessIterator)
+ {
+ tuple_vector<int, float, int>::iterator iter = tripleElementVec.begin();
+ auto symmetryOne = iter + 2;
+ auto symmetryTwo = 2 + iter;
+ iter += 2;
+ EATEST_VERIFY(symmetryOne == symmetryTwo);
+ EATEST_VERIFY(symmetryOne == iter);
+
+ symmetryOne = iter - 2;
+ symmetryTwo = 2 - iter;
+ iter -= 2;
+ EATEST_VERIFY(symmetryOne == symmetryTwo);
+ EATEST_VERIFY(symmetryOne == iter);
+
+ iter += 2;
+ EATEST_VERIFY(iter - symmetryOne == 2);
+
+ tuple<int&, float&, int&> symmetryRef = symmetryOne[2];
+ EATEST_VERIFY(get<2>(symmetryRef) == get<2>(*iter));
+
+ EATEST_VERIFY(symmetryOne < iter);
+ EATEST_VERIFY(iter > symmetryOne);
+ EATEST_VERIFY(symmetryOne >= symmetryTwo && iter >= symmetryOne);
+ EATEST_VERIFY(symmetryOne <= symmetryTwo && symmetryOne <= iter);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(iter) != isf_none);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(symmetryOne) != isf_none);
+ EATEST_VERIFY(tripleElementVec.validate_iterator(symmetryTwo) != isf_none);
+ }
+
+ // test simple iteration, and reverse iteration
+ {
+ float i = 0;
+ int j = 0;
+ EATEST_VERIFY(&get<0>(*tripleElementVec.begin()) == tripleElementVec.get<0>());
+ EATEST_VERIFY(&get<1>(*tripleElementVec.begin()) == tripleElementVec.get<1>());
+ for (auto iter : tripleElementVec)
+ {
+ i += get<1>(iter);
+ j += get<2>(iter);
+ }
+ EATEST_VERIFY(i == 20.0f);
+ EATEST_VERIFY(j == 40);
+
+ float reverse_i = 0;
+ int reverse_j = 0;
+
+ eastl::for_each(tripleElementVec.rbegin(), tripleElementVec.rend(),
+ [&](const tuple<int, float, int> tup)
+ {
+ reverse_i += get<1>(tup);
+ reverse_j += get<2>(tup);
+ });
+ EATEST_VERIFY(i == reverse_i);
+ EATEST_VERIFY(j == reverse_j);
+ EATEST_VERIFY(get<0>(*tripleElementVec.rbegin()) == 5);
+ }
+ }
+
+ // Test move operations
+ {
+ TestObject::Reset();
+
+ // test emplace
+ {
+ tuple_vector<int, MoveOnlyType, TestObject> testVec;
+ testVec.reserve(3);
+
+ // test emplace on empty vector that doesn't cause growth
+ testVec.emplace(testVec.begin(), 3, MoveOnlyType(3), TestObject(3));
+ EATEST_VERIFY(testVec.size() == 1);
+
+ // test emplace to end of vector that doesn't cause growth
+ testVec.emplace(testVec.end(), 5, MoveOnlyType(5), TestObject(5));
+ EATEST_VERIFY(testVec.size() == 2);
+
+ // test emplace to middle of vector that doesn't cause growth
+ testVec.emplace(testVec.begin() + 1, 4, MoveOnlyType(4), TestObject(4));
+ EATEST_VERIFY(testVec.size() == 3);
+ EATEST_VERIFY(testVec.capacity() == 3);
+
+ // test emplace to end of vector that causes growth
+ testVec.emplace(testVec.end(), 6, MoveOnlyType(6), TestObject(6));
+ EATEST_VERIFY(testVec.size() == 4);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 4);
+
+ // test emplace to beginning of vector that causes growth
+ testVec.emplace(testVec.begin(), 1, MoveOnlyType(1), TestObject(1));
+ EATEST_VERIFY(testVec.size() == 5);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 5);
+
+ // test emplace to middle of vector that causes growth
+ testVec.emplace(testVec.begin() + 1, 2, MoveOnlyType(2), TestObject(2));
+ EATEST_VERIFY(testVec.size() == 6);
+ testVec.shrink_to_fit();
+ EATEST_VERIFY(testVec.capacity() == 6);
+
+ for (unsigned int i = 0; i < testVec.size(); ++i)
+ {
+ EATEST_VERIFY(testVec.get<2>()[i] == TestObject(i + 1));
+ }
+ EATEST_VERIFY(testVec.validate());
+ }
+
+ // test some other miscellania around rvalues, including...
+ // push_back with rvalue args, push_back with rvalue tuple,
+ // emplace_back with args, and emplace_back with tup
+ {
+ tuple_vector<int, MoveOnlyType, TestObject> v1;
+ tuple_vector<int, MoveOnlyType, TestObject> v2;
+ // add some data in the vector so we can move it to the other vector.
+ v1.reserve(5);
+ auto emplacedTup = v1.emplace_back(1, MoveOnlyType(1), TestObject(1));
+ EATEST_VERIFY(emplacedTup == v1.back());
+ v1.push_back(3, MoveOnlyType(3), TestObject(3));
+ v1.emplace_back(forward_as_tuple(5, MoveOnlyType(5), TestObject(5)));
+ v1.push_back(forward_as_tuple(6, MoveOnlyType(6), TestObject(6)));
+ v1.emplace(v1.begin() + 1, 2, MoveOnlyType(2), TestObject(2));
+ v1.emplace(v1.begin() + 3, make_tuple(4, MoveOnlyType(4), TestObject(4)));
+
+ tuple<int&, MoveOnlyType&, TestObject&> movedTup = v1.at(0);
+ EATEST_VERIFY(v1.validate());
+ EATEST_VERIFY(get<0>(movedTup) == 1);
+ EATEST_VERIFY(get<0>(*v1.begin()) == 1);
+
+ for (int i = 0; i < static_cast<int>(v1.size()); ++i)
+ {
+ EATEST_VERIFY(v1.get<0>()[i] == i + 1);
+ }
+ EATEST_VERIFY(!v1.empty() && v2.empty());
+ v2 = eastl::move(v1);
+ EATEST_VERIFY(v2.validate());
+ EATEST_VERIFY(v1.empty() && !v2.empty());
+ v1.swap(v2);
+ EATEST_VERIFY(v1.validate());
+ EATEST_VERIFY(v2.validate());
+ EATEST_VERIFY(!v1.empty() && v2.empty());
+ }
+ EATEST_VERIFY(TestObject::IsClear());
+ TestObject::Reset();
+ }
+
+ // Test comparisons
+ {
+ MallocAllocator ma;
+ tuple_vector<bool, TestObject, float> equalsVec1, equalsVec2;
+ for (int i = 0; i < 10; ++i)
+ {
+ equalsVec1.push_back(i % 3 == 0, TestObject(i), (float)i);
+ equalsVec2.push_back(i % 3 == 0, TestObject(i), (float)i);
+ }
+ EATEST_VERIFY(equalsVec1 == equalsVec2);
+
+ tuple_vector<bool, TestObject, float> smallSizeVec(5);
+ tuple_vector<bool, TestObject, float> lessThanVec(10);
+ tuple_vector_alloc<MallocAllocator, bool, TestObject, float> greaterThanVec(10, ma);
+ for (int i = 0; i < 10; ++i)
+ {
+ lessThanVec.push_back(i % 3 == 0, TestObject(i), (float)i);
+ greaterThanVec.push_back(i % 3 == 0, TestObject(i * 2), (float)i * 2);
+ }
+ EATEST_VERIFY(equalsVec1 != smallSizeVec);
+ EATEST_VERIFY(equalsVec1 != lessThanVec);
+ EATEST_VERIFY(equalsVec1 != greaterThanVec);
+ EATEST_VERIFY(lessThanVec < greaterThanVec);
+ EATEST_VERIFY(greaterThanVec > lessThanVec);
+ EATEST_VERIFY(lessThanVec <= greaterThanVec);
+ EATEST_VERIFY(equalsVec1 <= equalsVec2);
+ EATEST_VERIFY(equalsVec1 >= equalsVec2);
+ }
+
+ // Test partition
+ {
+ {
+ tuple_vector<bool, TestObject, float, MoveOnlyType> vec;
+ for (int i = 0; i < 10; ++i)
+ {
+ vec.push_back(i % 3 == 0, TestObject(i), (float)i, MoveOnlyType(i));
+ }
+
+ eastl::partition(vec.begin(), vec.end(), [](tuple<bool&, TestObject&, float&, MoveOnlyType&> a)
+ { return get<0>(a) == true; });
+
+ // partition will split the array into 4 elements where the bool property is true, and 6 where it's false
+ for (int i = 0; i < 4; ++i)
+ EATEST_VERIFY(vec.get<0>()[i] == true);
+ for (int i = 4; i < 10; ++i)
+ EATEST_VERIFY(vec.get<0>()[i] == false);
+
+ EATEST_VERIFY(vec.validate());
+ EATEST_VERIFY(TestObject::sTOCount == 10);
+ }
+ EATEST_VERIFY(TestObject::IsClear());
+ TestObject::Reset();
+ }
+
+ // Test allocator manipulation
+ {
+ InstanceAllocator ia0((uint8_t)0), ia1((uint8_t)1);
+ tuple_vector_alloc<InstanceAllocator, int> vec(ia0);
+
+ // private vector allocator was copied from ia0 and should have matching id
+ EATEST_VERIFY(vec.get_allocator() == ia0);
+
+ // Assigning allocator
+ vec.set_allocator(ia1);
+ EATEST_VERIFY(vec.get_allocator() == ia1);
+ }
+
+ return nErrorCount;
+}
+
+