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