Home | History | Annotate | Download | only in containers
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "base/containers/stack_container.h"
      6 
      7 #include <algorithm>
      8 
      9 #include "base/memory/aligned_memory.h"
     10 #include "base/memory/ref_counted.h"
     11 #include "testing/gtest/include/gtest/gtest.h"
     12 
     13 namespace base {
     14 
     15 namespace {
     16 
     17 class Dummy : public base::RefCounted<Dummy> {
     18  public:
     19   explicit Dummy(int* alive) : alive_(alive) {
     20     ++*alive_;
     21   }
     22 
     23  private:
     24   friend class base::RefCounted<Dummy>;
     25 
     26   ~Dummy() {
     27     --*alive_;
     28   }
     29 
     30   int* const alive_;
     31 };
     32 
     33 }  // namespace
     34 
     35 TEST(StackContainer, Vector) {
     36   const int stack_size = 3;
     37   StackVector<int, stack_size> vect;
     38   const int* stack_buffer = &vect.stack_data().stack_buffer()[0];
     39 
     40   // The initial |stack_size| elements should appear in the stack buffer.
     41   EXPECT_EQ(static_cast<size_t>(stack_size), vect.container().capacity());
     42   for (int i = 0; i < stack_size; i++) {
     43     vect.container().push_back(i);
     44     EXPECT_EQ(stack_buffer, &vect.container()[0]);
     45     EXPECT_TRUE(vect.stack_data().used_stack_buffer_);
     46   }
     47 
     48   // Adding more elements should push the array onto the heap.
     49   for (int i = 0; i < stack_size; i++) {
     50     vect.container().push_back(i + stack_size);
     51     EXPECT_NE(stack_buffer, &vect.container()[0]);
     52     EXPECT_FALSE(vect.stack_data().used_stack_buffer_);
     53   }
     54 
     55   // The array should still be in order.
     56   for (int i = 0; i < stack_size * 2; i++)
     57     EXPECT_EQ(i, vect.container()[i]);
     58 
     59   // Resize to smaller. Our STL implementation won't reallocate in this case,
     60   // otherwise it might use our stack buffer. We reserve right after the resize
     61   // to guarantee it isn't using the stack buffer, even though it doesn't have
     62   // much data.
     63   vect.container().resize(stack_size);
     64   vect.container().reserve(stack_size * 2);
     65   EXPECT_FALSE(vect.stack_data().used_stack_buffer_);
     66 
     67   // Copying the small vector to another should use the same allocator and use
     68   // the now-unused stack buffer. GENERALLY CALLERS SHOULD NOT DO THIS since
     69   // they have to get the template types just right and it can cause errors.
     70   std::vector<int, StackAllocator<int, stack_size> > other(vect.container());
     71   EXPECT_EQ(stack_buffer, &other.front());
     72   EXPECT_TRUE(vect.stack_data().used_stack_buffer_);
     73   for (int i = 0; i < stack_size; i++)
     74     EXPECT_EQ(i, other[i]);
     75 }
     76 
     77 TEST(StackContainer, VectorDoubleDelete) {
     78   // Regression testing for double-delete.
     79   typedef StackVector<scoped_refptr<Dummy>, 2> Vector;
     80   typedef Vector::ContainerType Container;
     81   Vector vect;
     82 
     83   int alive = 0;
     84   scoped_refptr<Dummy> dummy(new Dummy(&alive));
     85   EXPECT_EQ(alive, 1);
     86 
     87   vect->push_back(dummy);
     88   EXPECT_EQ(alive, 1);
     89 
     90   Dummy* dummy_unref = dummy.get();
     91   dummy = NULL;
     92   EXPECT_EQ(alive, 1);
     93 
     94   Container::iterator itr = std::find(vect->begin(), vect->end(), dummy_unref);
     95   EXPECT_EQ(itr->get(), dummy_unref);
     96   vect->erase(itr);
     97   EXPECT_EQ(alive, 0);
     98 
     99   // Shouldn't crash at exit.
    100 }
    101 
    102 namespace {
    103 
    104 template <size_t alignment>
    105 class AlignedData {
    106  public:
    107   AlignedData() { memset(data_.void_data(), 0, alignment); }
    108   ~AlignedData() {}
    109   base::AlignedMemory<alignment, alignment> data_;
    110 };
    111 
    112 }  // anonymous namespace
    113 
    114 #define EXPECT_ALIGNED(ptr, align) \
    115     EXPECT_EQ(0u, reinterpret_cast<uintptr_t>(ptr) & (align - 1))
    116 
    117 TEST(StackContainer, BufferAlignment) {
    118   StackVector<wchar_t, 16> text;
    119   text->push_back(L'A');
    120   EXPECT_ALIGNED(&text[0], ALIGNOF(wchar_t));
    121 
    122   StackVector<double, 1> doubles;
    123   doubles->push_back(0.0);
    124   EXPECT_ALIGNED(&doubles[0], ALIGNOF(double));
    125 
    126   StackVector<AlignedData<16>, 1> aligned16;
    127   aligned16->push_back(AlignedData<16>());
    128   EXPECT_ALIGNED(&aligned16[0], 16);
    129 
    130 #if !defined(__GNUC__) || defined(ARCH_CPU_X86_FAMILY)
    131   // It seems that non-X86 gcc doesn't respect greater than 16 byte alignment.
    132   // See http://gcc.gnu.org/bugzilla/show_bug.cgi?id=33721 for details.
    133   // TODO(sbc):re-enable this if GCC starts respecting higher alignments.
    134   StackVector<AlignedData<256>, 1> aligned256;
    135   aligned256->push_back(AlignedData<256>());
    136   EXPECT_ALIGNED(&aligned256[0], 256);
    137 #endif
    138 }
    139 
    140 template class StackVector<int, 2>;
    141 template class StackVector<scoped_refptr<Dummy>, 2>;
    142 
    143 }  // namespace base
    144