blob: dbff007ada7ecc27045acdff7082fb448024dab1 [file] [log] [blame]
[email protected]44106182012-04-06 03:53:021// Copyright (c) 2012 The Chromium Authors. All rights reserved.
license.botbf09a502008-08-24 00:55:552// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
initial.commitd7cae122008-07-26 21:49:384
[email protected]4ac8f672008-08-11 14:02:065#include "base/at_exit.h"
[email protected]3b63f8f42011-03-28 01:54:156#include "base/memory/singleton.h"
[email protected]4ac8f672008-08-11 14:02:067#include "testing/gtest/include/gtest/gtest.h"
8
9namespace {
10
[email protected]4ac8f672008-08-11 14:02:0611COMPILE_ASSERT(DefaultSingletonTraits<int>::kRegisterAtExit == true, a);
12
[email protected]4ac8f672008-08-11 14:02:0613typedef void (*CallbackFunc)();
14
[email protected]625332e02010-12-14 07:48:4915class IntSingleton {
16 public:
17 static IntSingleton* GetInstance() {
18 return Singleton<IntSingleton>::get();
19 }
20
21 int value_;
22};
23
24class Init5Singleton {
25 public:
26 struct Trait;
27
28 static Init5Singleton* GetInstance() {
29 return Singleton<Init5Singleton, Trait>::get();
30 }
31
32 int value_;
33};
34
35struct Init5Singleton::Trait : public DefaultSingletonTraits<Init5Singleton> {
36 static Init5Singleton* New() {
37 Init5Singleton* instance = new Init5Singleton();
38 instance->value_ = 5;
39 return instance;
[email protected]5820d2f02010-12-11 10:23:3740 }
41};
[email protected]49ab5562010-12-11 09:13:5442
[email protected]625332e02010-12-14 07:48:4943int* SingletonInt() {
44 return &IntSingleton::GetInstance()->value_;
[email protected]4ac8f672008-08-11 14:02:0645}
46
47int* SingletonInt5() {
[email protected]625332e02010-12-14 07:48:4948 return &Init5Singleton::GetInstance()->value_;
[email protected]4ac8f672008-08-11 14:02:0649}
50
[email protected]625332e02010-12-14 07:48:4951template <typename Type>
52struct CallbackTrait : public DefaultSingletonTraits<Type> {
53 static void Delete(Type* instance) {
54 if (instance->callback_)
55 (instance->callback_)();
56 DefaultSingletonTraits<Type>::Delete(instance);
57 }
58};
59
60class CallbackSingleton {
61 public:
62 CallbackSingleton() : callback_(NULL) { }
63 CallbackFunc callback_;
64};
65
66class CallbackSingletonWithNoLeakTrait : public CallbackSingleton {
67 public:
68 struct Trait : public CallbackTrait<CallbackSingletonWithNoLeakTrait> { };
69
70 CallbackSingletonWithNoLeakTrait() : CallbackSingleton() { }
71
72 static CallbackSingletonWithNoLeakTrait* GetInstance() {
73 return Singleton<CallbackSingletonWithNoLeakTrait, Trait>::get();
74 }
75};
76
77class CallbackSingletonWithLeakTrait : public CallbackSingleton {
78 public:
79 struct Trait : public CallbackTrait<CallbackSingletonWithLeakTrait> {
80 static const bool kRegisterAtExit = false;
81 };
82
83 CallbackSingletonWithLeakTrait() : CallbackSingleton() { }
84
85 static CallbackSingletonWithLeakTrait* GetInstance() {
86 return Singleton<CallbackSingletonWithLeakTrait, Trait>::get();
87 }
88};
89
90class CallbackSingletonWithStaticTrait : public CallbackSingleton {
91 public:
92 struct Trait;
93
94 CallbackSingletonWithStaticTrait() : CallbackSingleton() { }
95
96 static CallbackSingletonWithStaticTrait* GetInstance() {
97 return Singleton<CallbackSingletonWithStaticTrait, Trait>::get();
98 }
99};
100
101struct CallbackSingletonWithStaticTrait::Trait
102 : public StaticMemorySingletonTraits<CallbackSingletonWithStaticTrait> {
103 static void Delete(CallbackSingletonWithStaticTrait* instance) {
104 if (instance->callback_)
105 (instance->callback_)();
106 StaticMemorySingletonTraits<CallbackSingletonWithStaticTrait>::Delete(
107 instance);
108 }
109};
110
[email protected]cd924d62012-02-23 17:52:20111template <class Type>
112class AlignedTestSingleton {
113 public:
114 AlignedTestSingleton() {}
115 ~AlignedTestSingleton() {}
116 static AlignedTestSingleton* GetInstance() {
117 return Singleton<AlignedTestSingleton,
118 StaticMemorySingletonTraits<AlignedTestSingleton> >::get();
119 }
120
121 Type type_;
122};
123
[email protected]625332e02010-12-14 07:48:49124
[email protected]4ac8f672008-08-11 14:02:06125void SingletonNoLeak(CallbackFunc CallOnQuit) {
[email protected]625332e02010-12-14 07:48:49126 CallbackSingletonWithNoLeakTrait::GetInstance()->callback_ = CallOnQuit;
[email protected]4ac8f672008-08-11 14:02:06127}
128
129void SingletonLeak(CallbackFunc CallOnQuit) {
[email protected]625332e02010-12-14 07:48:49130 CallbackSingletonWithLeakTrait::GetInstance()->callback_ = CallOnQuit;
[email protected]4ac8f672008-08-11 14:02:06131}
132
133CallbackFunc* GetLeakySingleton() {
[email protected]625332e02010-12-14 07:48:49134 return &CallbackSingletonWithLeakTrait::GetInstance()->callback_;
135}
136
137void DeleteLeakySingleton() {
138 DefaultSingletonTraits<CallbackSingletonWithLeakTrait>::Delete(
139 CallbackSingletonWithLeakTrait::GetInstance());
[email protected]4ac8f672008-08-11 14:02:06140}
141
[email protected]297d0e52010-05-07 15:24:49142void SingletonStatic(CallbackFunc CallOnQuit) {
[email protected]625332e02010-12-14 07:48:49143 CallbackSingletonWithStaticTrait::GetInstance()->callback_ = CallOnQuit;
[email protected]297d0e52010-05-07 15:24:49144}
145
146CallbackFunc* GetStaticSingleton() {
[email protected]625332e02010-12-14 07:48:49147 return &CallbackSingletonWithStaticTrait::GetInstance()->callback_;
148}
149
[email protected]4ac8f672008-08-11 14:02:06150} // namespace
initial.commitd7cae122008-07-26 21:49:38151
152class SingletonTest : public testing::Test {
153 public:
[email protected]44106182012-04-06 03:53:02154 SingletonTest() {}
initial.commitd7cae122008-07-26 21:49:38155
dcheng8aef37612014-12-23 02:56:47156 void SetUp() override {
initial.commitd7cae122008-07-26 21:49:38157 non_leak_called_ = false;
158 leaky_called_ = false;
[email protected]297d0e52010-05-07 15:24:49159 static_called_ = false;
initial.commitd7cae122008-07-26 21:49:38160 }
161
initial.commitd7cae122008-07-26 21:49:38162 protected:
initial.commitd7cae122008-07-26 21:49:38163 void VerifiesCallbacks() {
164 EXPECT_TRUE(non_leak_called_);
165 EXPECT_FALSE(leaky_called_);
[email protected]297d0e52010-05-07 15:24:49166 EXPECT_TRUE(static_called_);
initial.commitd7cae122008-07-26 21:49:38167 non_leak_called_ = false;
168 leaky_called_ = false;
[email protected]297d0e52010-05-07 15:24:49169 static_called_ = false;
initial.commitd7cae122008-07-26 21:49:38170 }
171
172 void VerifiesCallbacksNotCalled() {
173 EXPECT_FALSE(non_leak_called_);
174 EXPECT_FALSE(leaky_called_);
[email protected]297d0e52010-05-07 15:24:49175 EXPECT_FALSE(static_called_);
initial.commitd7cae122008-07-26 21:49:38176 non_leak_called_ = false;
177 leaky_called_ = false;
[email protected]297d0e52010-05-07 15:24:49178 static_called_ = false;
initial.commitd7cae122008-07-26 21:49:38179 }
180
[email protected]4ac8f672008-08-11 14:02:06181 static void CallbackNoLeak() {
initial.commitd7cae122008-07-26 21:49:38182 non_leak_called_ = true;
183 }
184
[email protected]4ac8f672008-08-11 14:02:06185 static void CallbackLeak() {
initial.commitd7cae122008-07-26 21:49:38186 leaky_called_ = true;
187 }
188
[email protected]297d0e52010-05-07 15:24:49189 static void CallbackStatic() {
190 static_called_ = true;
191 }
192
initial.commitd7cae122008-07-26 21:49:38193 private:
initial.commitd7cae122008-07-26 21:49:38194 static bool non_leak_called_;
195 static bool leaky_called_;
[email protected]297d0e52010-05-07 15:24:49196 static bool static_called_;
initial.commitd7cae122008-07-26 21:49:38197};
198
199bool SingletonTest::non_leak_called_ = false;
200bool SingletonTest::leaky_called_ = false;
[email protected]297d0e52010-05-07 15:24:49201bool SingletonTest::static_called_ = false;
initial.commitd7cae122008-07-26 21:49:38202
initial.commitd7cae122008-07-26 21:49:38203TEST_F(SingletonTest, Basic) {
[email protected]625332e02010-12-14 07:48:49204 int* singleton_int;
initial.commitd7cae122008-07-26 21:49:38205 int* singleton_int_5;
[email protected]4ac8f672008-08-11 14:02:06206 CallbackFunc* leaky_singleton;
[email protected]297d0e52010-05-07 15:24:49207 CallbackFunc* static_singleton;
initial.commitd7cae122008-07-26 21:49:38208
initial.commitd7cae122008-07-26 21:49:38209 {
[email protected]4ea927b2009-11-19 09:11:39210 base::ShadowingAtExitManager sem;
initial.commitd7cae122008-07-26 21:49:38211 {
[email protected]625332e02010-12-14 07:48:49212 singleton_int = SingletonInt();
initial.commitd7cae122008-07-26 21:49:38213 }
214 // Ensure POD type initialization.
[email protected]625332e02010-12-14 07:48:49215 EXPECT_EQ(*singleton_int, 0);
216 *singleton_int = 1;
initial.commitd7cae122008-07-26 21:49:38217
[email protected]625332e02010-12-14 07:48:49218 EXPECT_EQ(singleton_int, SingletonInt());
219 EXPECT_EQ(*singleton_int, 1);
initial.commitd7cae122008-07-26 21:49:38220
221 {
[email protected]4ac8f672008-08-11 14:02:06222 singleton_int_5 = SingletonInt5();
initial.commitd7cae122008-07-26 21:49:38223 }
224 // Is default initialized to 5.
225 EXPECT_EQ(*singleton_int_5, 5);
initial.commitd7cae122008-07-26 21:49:38226
[email protected]4ac8f672008-08-11 14:02:06227 SingletonNoLeak(&CallbackNoLeak);
228 SingletonLeak(&CallbackLeak);
[email protected]297d0e52010-05-07 15:24:49229 SingletonStatic(&CallbackStatic);
230 static_singleton = GetStaticSingleton();
[email protected]4ac8f672008-08-11 14:02:06231 leaky_singleton = GetLeakySingleton();
initial.commitd7cae122008-07-26 21:49:38232 EXPECT_TRUE(leaky_singleton);
233 }
initial.commitd7cae122008-07-26 21:49:38234
235 // Verify that only the expected callback has been called.
236 VerifiesCallbacks();
[email protected]64e95e12011-08-17 17:41:02237 // Delete the leaky singleton.
[email protected]625332e02010-12-14 07:48:49238 DeleteLeakySingleton();
initial.commitd7cae122008-07-26 21:49:38239
[email protected]297d0e52010-05-07 15:24:49240 // The static singleton can't be acquired post-atexit.
241 EXPECT_EQ(NULL, GetStaticSingleton());
242
initial.commitd7cae122008-07-26 21:49:38243 {
[email protected]4ea927b2009-11-19 09:11:39244 base::ShadowingAtExitManager sem;
initial.commitd7cae122008-07-26 21:49:38245 // Verifiy that the variables were reset.
246 {
[email protected]625332e02010-12-14 07:48:49247 singleton_int = SingletonInt();
248 EXPECT_EQ(*singleton_int, 0);
initial.commitd7cae122008-07-26 21:49:38249 }
250 {
[email protected]4ac8f672008-08-11 14:02:06251 singleton_int_5 = SingletonInt5();
initial.commitd7cae122008-07-26 21:49:38252 EXPECT_EQ(*singleton_int_5, 5);
253 }
[email protected]297d0e52010-05-07 15:24:49254 {
255 // Resurrect the static singleton, and assert that it
256 // still points to the same (static) memory.
[email protected]625332e02010-12-14 07:48:49257 CallbackSingletonWithStaticTrait::Trait::Resurrect();
[email protected]297d0e52010-05-07 15:24:49258 EXPECT_EQ(GetStaticSingleton(), static_singleton);
259 }
initial.commitd7cae122008-07-26 21:49:38260 }
261 // The leaky singleton shouldn't leak since SingletonLeak has not been called.
initial.commitd7cae122008-07-26 21:49:38262 VerifiesCallbacksNotCalled();
263}
[email protected]cd924d62012-02-23 17:52:20264
265#define EXPECT_ALIGNED(ptr, align) \
266 EXPECT_EQ(0u, reinterpret_cast<uintptr_t>(ptr) & (align - 1))
267
268TEST_F(SingletonTest, Alignment) {
269 using base::AlignedMemory;
270
271 // Create some static singletons with increasing sizes and alignment
272 // requirements. By ordering this way, the linker will need to do some work to
273 // ensure proper alignment of the static data.
274 AlignedTestSingleton<int32>* align4 =
275 AlignedTestSingleton<int32>::GetInstance();
276 AlignedTestSingleton<AlignedMemory<32, 32> >* align32 =
277 AlignedTestSingleton<AlignedMemory<32, 32> >::GetInstance();
278 AlignedTestSingleton<AlignedMemory<128, 128> >* align128 =
279 AlignedTestSingleton<AlignedMemory<128, 128> >::GetInstance();
280 AlignedTestSingleton<AlignedMemory<4096, 4096> >* align4096 =
281 AlignedTestSingleton<AlignedMemory<4096, 4096> >::GetInstance();
282
283 EXPECT_ALIGNED(align4, 4);
284 EXPECT_ALIGNED(align32, 32);
285 EXPECT_ALIGNED(align128, 128);
286 EXPECT_ALIGNED(align4096, 4096);
287}