| //Has to be first for StackAllocator swap overload to be taken |
| //into account (at least using GCC 4.0.1) |
| #include "stack_allocator.h" |
| |
| #include <vector> |
| #include <algorithm> |
| #if !defined (STLPORT) || defined (_STLP_USE_EXCEPTIONS) |
| # include <stdexcept> |
| #endif |
| |
| #include "cppunit/cppunit_proxy.h" |
| |
| #if !defined (STLPORT) || defined(_STLP_USE_NAMESPACES) |
| using namespace std; |
| #endif |
| |
| // |
| // TestCase class |
| // |
| class VectorTest : public CPPUNIT_NS::TestCase |
| { |
| CPPUNIT_TEST_SUITE(VectorTest); |
| CPPUNIT_TEST(vec_test_1); |
| CPPUNIT_TEST(vec_test_2); |
| CPPUNIT_TEST(vec_test_3); |
| CPPUNIT_TEST(vec_test_4); |
| CPPUNIT_TEST(vec_test_5); |
| CPPUNIT_TEST(vec_test_6); |
| CPPUNIT_TEST(vec_test_7); |
| CPPUNIT_TEST(capacity); |
| CPPUNIT_TEST(at); |
| CPPUNIT_TEST(pointer); |
| CPPUNIT_TEST(auto_ref); |
| CPPUNIT_TEST(allocator_with_state); |
| CPPUNIT_TEST(iterators); |
| #if defined (STLPORT) && defined (_STLP_NO_MEMBER_TEMPLATES) |
| CPPUNIT_IGNORE; |
| #endif |
| CPPUNIT_TEST(optimizations_check); |
| CPPUNIT_TEST(assign_check); |
| CPPUNIT_STOP_IGNORE; |
| CPPUNIT_TEST(ebo); |
| CPPUNIT_TEST_SUITE_END(); |
| |
| protected: |
| void vec_test_1(); |
| void vec_test_2(); |
| void vec_test_3(); |
| void vec_test_4(); |
| void vec_test_5(); |
| void vec_test_6(); |
| void vec_test_7(); |
| void capacity(); |
| void at(); |
| void pointer(); |
| void auto_ref(); |
| void allocator_with_state(); |
| void iterators(); |
| void optimizations_check(); |
| void assign_check(); |
| void ebo(); |
| }; |
| |
| CPPUNIT_TEST_SUITE_REGISTRATION(VectorTest); |
| |
| // |
| // tests implementation |
| // |
| void VectorTest::vec_test_1() |
| { |
| vector<int> v1; // Empty vector of integers. |
| |
| CPPUNIT_ASSERT( v1.empty() == true ); |
| CPPUNIT_ASSERT( v1.size() == 0 ); |
| |
| // CPPUNIT_ASSERT( v1.max_size() == INT_MAX / sizeof(int) ); |
| // cout << "max_size = " << v1.max_size() << endl; |
| v1.push_back(42); // Add an integer to the vector. |
| |
| CPPUNIT_ASSERT( v1.size() == 1 ); |
| |
| CPPUNIT_ASSERT( v1[0] == 42 ); |
| |
| { |
| vector<vector<int> > vect(10); |
| vector<vector<int> >::iterator it(vect.begin()), end(vect.end()); |
| for (; it != end; ++it) { |
| CPPUNIT_ASSERT( (*it).empty() ); |
| CPPUNIT_ASSERT( (*it).size() == 0 ); |
| CPPUNIT_ASSERT( (*it).capacity() == 0 ); |
| CPPUNIT_ASSERT( (*it).begin() == (*it).end() ); |
| } |
| } |
| } |
| |
| void VectorTest::vec_test_2() |
| { |
| vector<double> v1; // Empty vector of doubles. |
| v1.push_back(32.1); |
| v1.push_back(40.5); |
| vector<double> v2; // Another empty vector of doubles. |
| v2.push_back(3.56); |
| |
| CPPUNIT_ASSERT( v1.size() == 2 ); |
| CPPUNIT_ASSERT( v1[0] == 32.1 ); |
| CPPUNIT_ASSERT( v1[1] == 40.5 ); |
| |
| CPPUNIT_ASSERT( v2.size() == 1 ); |
| CPPUNIT_ASSERT( v2[0] == 3.56 ); |
| size_t v1Cap = v1.capacity(); |
| size_t v2Cap = v2.capacity(); |
| |
| v1.swap(v2); // Swap the vector's contents. |
| |
| CPPUNIT_ASSERT( v1.size() == 1 ); |
| CPPUNIT_ASSERT( v1.capacity() == v2Cap ); |
| CPPUNIT_ASSERT( v1[0] == 3.56 ); |
| |
| CPPUNIT_ASSERT( v2.size() == 2 ); |
| CPPUNIT_ASSERT( v2.capacity() == v1Cap ); |
| CPPUNIT_ASSERT( v2[0] == 32.1 ); |
| CPPUNIT_ASSERT( v2[1] == 40.5 ); |
| |
| v2 = v1; // Assign one vector to another. |
| |
| CPPUNIT_ASSERT( v2.size() == 1 ); |
| CPPUNIT_ASSERT( v2[0] == 3.56 ); |
| } |
| |
| void VectorTest::vec_test_3() |
| { |
| typedef vector<char> vec_type; |
| |
| vec_type v1; // Empty vector of characters. |
| v1.push_back('h'); |
| v1.push_back('i'); |
| |
| CPPUNIT_ASSERT( v1.size() == 2 ); |
| CPPUNIT_ASSERT( v1[0] == 'h' ); |
| CPPUNIT_ASSERT( v1[1] == 'i' ); |
| |
| vec_type v2(v1.begin(), v1.end()); |
| v2[1] = 'o'; // Replace second character. |
| |
| CPPUNIT_ASSERT( v2.size() == 2 ); |
| CPPUNIT_ASSERT( v2[0] == 'h' ); |
| CPPUNIT_ASSERT( v2[1] == 'o' ); |
| |
| CPPUNIT_ASSERT( (v1 == v2) == false ); |
| |
| CPPUNIT_ASSERT( (v1 < v2) == true ); |
| } |
| |
| void VectorTest::vec_test_4() |
| { |
| vector<int> v(4); |
| |
| v[0] = 1; |
| v[1] = 4; |
| v[2] = 9; |
| v[3] = 16; |
| |
| CPPUNIT_ASSERT( v.front() == 1 ); |
| CPPUNIT_ASSERT( v.back() == 16 ); |
| |
| v.push_back(25); |
| |
| CPPUNIT_ASSERT( v.back() == 25 ); |
| CPPUNIT_ASSERT( v.size() == 5 ); |
| |
| v.pop_back(); |
| |
| CPPUNIT_ASSERT( v.back() == 16 ); |
| CPPUNIT_ASSERT( v.size() == 4 ); |
| } |
| |
| void VectorTest::vec_test_5() |
| { |
| int array [] = { 1, 4, 9, 16 }; |
| |
| vector<int> v(array, array + 4); |
| |
| CPPUNIT_ASSERT( v.size() == 4 ); |
| |
| CPPUNIT_ASSERT( v[0] == 1 ); |
| CPPUNIT_ASSERT( v[1] == 4 ); |
| CPPUNIT_ASSERT( v[2] == 9 ); |
| CPPUNIT_ASSERT( v[3] == 16 ); |
| } |
| |
| void VectorTest::vec_test_6() |
| { |
| int array [] = { 1, 4, 9, 16, 25, 36 }; |
| |
| vector<int> v(array, array + 6); |
| vector<int>::iterator vit; |
| |
| CPPUNIT_ASSERT( v.size() == 6 ); |
| CPPUNIT_ASSERT( v[0] == 1 ); |
| CPPUNIT_ASSERT( v[1] == 4 ); |
| CPPUNIT_ASSERT( v[2] == 9 ); |
| CPPUNIT_ASSERT( v[3] == 16 ); |
| CPPUNIT_ASSERT( v[4] == 25 ); |
| CPPUNIT_ASSERT( v[5] == 36 ); |
| |
| vit = v.erase( v.begin() ); // Erase first element. |
| CPPUNIT_ASSERT( *vit == 4 ); |
| |
| CPPUNIT_ASSERT( v.size() == 5 ); |
| CPPUNIT_ASSERT( v[0] == 4 ); |
| CPPUNIT_ASSERT( v[1] == 9 ); |
| CPPUNIT_ASSERT( v[2] == 16 ); |
| CPPUNIT_ASSERT( v[3] == 25 ); |
| CPPUNIT_ASSERT( v[4] == 36 ); |
| |
| vit = v.erase(v.end() - 1); // Erase last element. |
| CPPUNIT_ASSERT( vit == v.end() ); |
| |
| CPPUNIT_ASSERT( v.size() == 4 ); |
| CPPUNIT_ASSERT( v[0] == 4 ); |
| CPPUNIT_ASSERT( v[1] == 9 ); |
| CPPUNIT_ASSERT( v[2] == 16 ); |
| CPPUNIT_ASSERT( v[3] == 25 ); |
| |
| |
| v.erase(v.begin() + 1, v.end() - 1); // Erase all but first and last. |
| |
| CPPUNIT_ASSERT( v.size() == 2 ); |
| CPPUNIT_ASSERT( v[0] == 4 ); |
| CPPUNIT_ASSERT( v[1] == 25 ); |
| |
| } |
| |
| void VectorTest::vec_test_7() |
| { |
| int array1 [] = { 1, 4, 25 }; |
| int array2 [] = { 9, 16 }; |
| |
| vector<int> v(array1, array1 + 3); |
| vector<int>::iterator vit; |
| vit = v.insert(v.begin(), 0); // Insert before first element. |
| CPPUNIT_ASSERT( *vit == 0 ); |
| |
| vit = v.insert(v.end(), 36); // Insert after last element. |
| CPPUNIT_ASSERT( *vit == 36 ); |
| |
| CPPUNIT_ASSERT( v.size() == 5 ); |
| CPPUNIT_ASSERT( v[0] == 0 ); |
| CPPUNIT_ASSERT( v[1] == 1 ); |
| CPPUNIT_ASSERT( v[2] == 4 ); |
| CPPUNIT_ASSERT( v[3] == 25 ); |
| CPPUNIT_ASSERT( v[4] == 36 ); |
| |
| // Insert contents of array2 before fourth element. |
| v.insert(v.begin() + 3, array2, array2 + 2); |
| |
| CPPUNIT_ASSERT( v.size() == 7 ); |
| |
| CPPUNIT_ASSERT( v[0] == 0 ); |
| CPPUNIT_ASSERT( v[1] == 1 ); |
| CPPUNIT_ASSERT( v[2] == 4 ); |
| CPPUNIT_ASSERT( v[3] == 9 ); |
| CPPUNIT_ASSERT( v[4] == 16 ); |
| CPPUNIT_ASSERT( v[5] == 25 ); |
| CPPUNIT_ASSERT( v[6] == 36 ); |
| |
| v.clear(); |
| CPPUNIT_ASSERT( v.empty() ); |
| |
| v.insert(v.begin(), 5, 10); |
| CPPUNIT_ASSERT( v.size() == 5 ); |
| CPPUNIT_ASSERT( v[0] == 10 ); |
| CPPUNIT_ASSERT( v[1] == 10 ); |
| CPPUNIT_ASSERT( v[2] == 10 ); |
| CPPUNIT_ASSERT( v[3] == 10 ); |
| CPPUNIT_ASSERT( v[4] == 10 ); |
| |
| /* |
| { |
| vector<float> vf(2.0f, 3.0f); |
| CPPUNIT_ASSERT( vf.size() == 2 ); |
| CPPUNIT_ASSERT( vf.front() == 3.0f ); |
| CPPUNIT_ASSERT( vf.back() == 3.0f ); |
| } |
| */ |
| } |
| |
| struct TestStruct |
| { |
| unsigned int a[3]; |
| }; |
| |
| void VectorTest::capacity() |
| { |
| { |
| vector<int> v; |
| |
| CPPUNIT_ASSERT( v.capacity() == 0 ); |
| v.push_back(42); |
| CPPUNIT_ASSERT( v.capacity() >= 1 ); |
| v.reserve(5000); |
| CPPUNIT_ASSERT( v.capacity() >= 5000 ); |
| } |
| |
| { |
| //Test that used to generate an assertion when using __debug_alloc. |
| vector<TestStruct> va; |
| va.reserve(1); |
| va.reserve(2); |
| } |
| } |
| |
| void VectorTest::at() { |
| vector<int> v; |
| vector<int> const& cv = v; |
| |
| v.push_back(10); |
| CPPUNIT_ASSERT( v.at(0) == 10 ); |
| v.at(0) = 20; |
| CPPUNIT_ASSERT( cv.at(0) == 20 ); |
| |
| #if !defined (STLPORT) || defined (_STLP_USE_EXCEPTIONS) |
| try { |
| v.at(1) = 20; |
| CPPUNIT_FAIL; |
| } |
| catch (out_of_range const&) { |
| } |
| catch (...) { |
| CPPUNIT_FAIL; |
| } |
| #endif |
| } |
| |
| void VectorTest::pointer() |
| { |
| vector<int *> v1; |
| vector<int *> v2 = v1; |
| vector<int *> v3; |
| |
| v3.insert( v3.end(), v1.begin(), v1.end() ); |
| } |
| |
| void VectorTest::auto_ref() |
| { |
| vector<int> ref; |
| for (int i = 0; i < 5; ++i) { |
| ref.push_back(i); |
| } |
| |
| vector<vector<int> > v_v_int(1, ref); |
| v_v_int.push_back(v_v_int[0]); |
| v_v_int.push_back(ref); |
| v_v_int.push_back(v_v_int[0]); |
| v_v_int.push_back(v_v_int[0]); |
| v_v_int.push_back(ref); |
| |
| vector<vector<int> >::iterator vvit(v_v_int.begin()), vvitEnd(v_v_int.end()); |
| for (; vvit != vvitEnd; ++vvit) { |
| CPPUNIT_ASSERT( *vvit == ref ); |
| } |
| |
| /* |
| * Forbidden by the Standard: |
| v_v_int.insert(v_v_int.end(), v_v_int.begin(), v_v_int.end()); |
| for (vvit = v_v_int.begin(), vvitEnd = v_v_int.end(); |
| vvit != vvitEnd; ++vvit) { |
| CPPUNIT_ASSERT( *vvit == ref ); |
| } |
| */ |
| } |
| |
| void VectorTest::allocator_with_state() |
| { |
| char buf1[1024]; |
| StackAllocator<int> stack1(buf1, buf1 + sizeof(buf1)); |
| |
| char buf2[1024]; |
| StackAllocator<int> stack2(buf2, buf2 + sizeof(buf2)); |
| |
| { |
| typedef vector<int, StackAllocator<int> > VectorInt; |
| VectorInt vint1(10, 0, stack1); |
| VectorInt vint1Cpy(vint1); |
| |
| VectorInt vint2(10, 1, stack2); |
| VectorInt vint2Cpy(vint2); |
| |
| vint1.swap(vint2); |
| |
| CPPUNIT_ASSERT( vint1.get_allocator().swaped() ); |
| CPPUNIT_ASSERT( vint2.get_allocator().swaped() ); |
| |
| CPPUNIT_ASSERT( vint1 == vint2Cpy ); |
| CPPUNIT_ASSERT( vint2 == vint1Cpy ); |
| CPPUNIT_ASSERT( vint1.get_allocator() == stack2 ); |
| CPPUNIT_ASSERT( vint2.get_allocator() == stack1 ); |
| } |
| CPPUNIT_ASSERT( stack1.ok() ); |
| CPPUNIT_ASSERT( stack2.ok() ); |
| } |
| |
| struct Point { |
| int x, y; |
| }; |
| |
| struct PointEx : public Point { |
| PointEx() : builtFromBase(false) {} |
| PointEx(const Point&) : builtFromBase(true) {} |
| |
| bool builtFromBase; |
| }; |
| |
| #if defined (STLPORT) |
| # if defined (_STLP_USE_NAMESPACES) |
| namespace std { |
| # endif |
| _STLP_TEMPLATE_NULL |
| struct __type_traits<PointEx> { |
| typedef __false_type has_trivial_default_constructor; |
| typedef __true_type has_trivial_copy_constructor; |
| typedef __true_type has_trivial_assignment_operator; |
| typedef __true_type has_trivial_destructor; |
| typedef __true_type is_POD_type; |
| }; |
| # if defined (_STLP_USE_NAMESPACES) |
| } |
| # endif |
| #endif |
| |
| //This test check that vector implementation do not over optimize |
| //operation as PointEx copy constructor is trivial |
| void VectorTest::optimizations_check() |
| { |
| #if !defined (STLPORT) || !defined (_STLP_NO_MEMBER_TEMPLATES) |
| vector<Point> v1(1); |
| CPPUNIT_ASSERT( v1.size() == 1 ); |
| |
| vector<PointEx> v2(v1.begin(), v1.end()); |
| CPPUNIT_ASSERT( v2.size() == 1 ); |
| CPPUNIT_ASSERT( v2[0].builtFromBase == true ); |
| #endif |
| } |
| |
| void VectorTest::assign_check() |
| { |
| #if !defined (STLPORT) || !defined (_STLP_NO_MEMBER_TEMPLATES) |
| vector<int> v(3,1); |
| int array[] = { 1, 2, 3, 4, 5 }; |
| |
| v.assign( array, array + 5 ); |
| CPPUNIT_CHECK( v[4] == 5 ); |
| CPPUNIT_CHECK( v[0] == 1 ); |
| CPPUNIT_CHECK( v[1] == 2 ); |
| #endif |
| } |
| |
| void VectorTest::iterators() |
| { |
| vector<int> vint(10, 0); |
| vector<int> const& crvint = vint; |
| |
| CPPUNIT_ASSERT( vint.begin() == vint.begin() ); |
| CPPUNIT_ASSERT( crvint.begin() == vint.begin() ); |
| CPPUNIT_ASSERT( vint.begin() == crvint.begin() ); |
| CPPUNIT_ASSERT( crvint.begin() == crvint.begin() ); |
| |
| CPPUNIT_ASSERT( vint.begin() != vint.end() ); |
| CPPUNIT_ASSERT( crvint.begin() != vint.end() ); |
| CPPUNIT_ASSERT( vint.begin() != crvint.end() ); |
| CPPUNIT_ASSERT( crvint.begin() != crvint.end() ); |
| |
| CPPUNIT_ASSERT( vint.rbegin() == vint.rbegin() ); |
| // Not Standard: |
| //CPPUNIT_ASSERT( vint.rbegin() == crvint.rbegin() ); |
| //CPPUNIT_ASSERT( crvint.rbegin() == vint.rbegin() ); |
| CPPUNIT_ASSERT( crvint.rbegin() == crvint.rbegin() ); |
| |
| CPPUNIT_ASSERT( vint.rbegin() != vint.rend() ); |
| // Not Standard: |
| //CPPUNIT_ASSERT( vint.rbegin() != crvint.rend() ); |
| //CPPUNIT_ASSERT( crvint.rbegin() != vint.rend() ); |
| CPPUNIT_ASSERT( crvint.rbegin() != crvint.rend() ); |
| } |
| |
| |
| #if !defined (STLPORT) || \ |
| !defined (_STLP_USE_PTR_SPECIALIZATIONS) || defined (_STLP_CLASS_PARTIAL_SPECIALIZATION) |
| /* Simple compilation test: Check that nested types like iterator |
| * can be access even if type used to instanciate container is not |
| * yet completely defined. |
| */ |
| class IncompleteClass |
| { |
| vector<IncompleteClass> instances; |
| typedef vector<IncompleteClass>::iterator it; |
| }; |
| #endif |
| |
| #if defined (STLPORT) |
| # define NOTHROW _STLP_NOTHROW |
| #else |
| # define NOTHROW throw() |
| #endif |
| |
| /* This allocator implementation purpose is simply to break some |
| * internal STLport mecanism specific to the STLport own allocator |
| * implementation. */ |
| template <class _Tp> |
| struct NotSTLportAllocator : public allocator<_Tp> { |
| #if !defined (STLPORT) || defined (_STLP_MEMBER_TEMPLATE_CLASSES) |
| template <class _Tp1> struct rebind { |
| typedef NotSTLportAllocator<_Tp1> other; |
| }; |
| #endif |
| NotSTLportAllocator() NOTHROW {} |
| #if !defined (STLPORT) || defined (_STLP_MEMBER_TEMPLATES) |
| template <class _Tp1> NotSTLportAllocator(const NotSTLportAllocator<_Tp1>&) NOTHROW {} |
| #endif |
| NotSTLportAllocator(const NotSTLportAllocator<_Tp>&) NOTHROW {} |
| ~NotSTLportAllocator() NOTHROW {} |
| }; |
| |
| /* This test check a potential issue with empty base class |
| * optimization. Some compilers (VC6) do not implement it |
| * correctly resulting ina wrong behavior. */ |
| void VectorTest::ebo() |
| { |
| // We use heap memory as test failure can corrupt vector internal |
| // representation making executable crash on vector destructor invocation. |
| // We prefer a simple memory leak, internal corruption should be reveal |
| // by size or capacity checks. |
| typedef vector<int, NotSTLportAllocator<int> > V; |
| V *pv1 = new V(1, 1); |
| V *pv2 = new V(10, 2); |
| |
| size_t v1Capacity = pv1->capacity(); |
| size_t v2Capacity = pv2->capacity(); |
| |
| pv1->swap(*pv2); |
| |
| CPPUNIT_ASSERT( pv1->size() == 10 ); |
| CPPUNIT_ASSERT( pv1->capacity() == v2Capacity ); |
| CPPUNIT_ASSERT( (*pv1)[5] == 2 ); |
| |
| CPPUNIT_ASSERT( pv2->size() == 1 ); |
| CPPUNIT_ASSERT( pv2->capacity() == v1Capacity ); |
| CPPUNIT_ASSERT( (*pv2)[0] == 1 ); |
| |
| delete pv2; |
| delete pv1; |
| } |
| |