OLD | NEW |
---|---|
1 /* | 1 /* |
2 * Copyright 2004 The WebRTC Project Authors. All rights reserved. | 2 * Copyright 2004 The WebRTC Project Authors. All rights reserved. |
3 * | 3 * |
4 * Use of this source code is governed by a BSD-style license | 4 * Use of this source code is governed by a BSD-style license |
5 * that can be found in the LICENSE file in the root of the source | 5 * that can be found in the LICENSE file in the root of the source |
6 * tree. An additional intellectual property rights grant can be found | 6 * tree. An additional intellectual property rights grant can be found |
7 * in the file PATENTS. All contributing project authors may | 7 * in the file PATENTS. All contributing project authors may |
8 * be found in the AUTHORS file in the root of the source tree. | 8 * be found in the AUTHORS file in the root of the source tree. |
9 */ | 9 */ |
10 | 10 |
11 #include "webrtc/base/bind.h" | 11 #include "webrtc/base/bind.h" |
12 #include "webrtc/base/gunit.h" | 12 #include "webrtc/base/gunit.h" |
13 | 13 |
14 #include "webrtc/base/refcount.h" | 14 #include "webrtc/base/refcount.h" |
15 | 15 |
16 namespace rtc { | 16 namespace rtc { |
17 | 17 |
18 namespace { | 18 namespace { |
19 | 19 |
20 struct LifeTimeCheck; | |
21 | |
20 struct MethodBindTester { | 22 struct MethodBindTester { |
21 void NullaryVoid() { ++call_count; } | 23 void NullaryVoid() { ++call_count; } |
22 int NullaryInt() { ++call_count; return 1; } | 24 int NullaryInt() { ++call_count; return 1; } |
23 int NullaryConst() const { ++call_count; return 2; } | 25 int NullaryConst() const { ++call_count; return 2; } |
24 void UnaryVoid(int dummy) { ++call_count; } | 26 void UnaryVoid(int dummy) { ++call_count; } |
25 template <class T> T Identity(T value) { ++call_count; return value; } | 27 template <class T> T Identity(T value) { ++call_count; return value; } |
26 int UnaryByRef(int& value) const { ++call_count; return ++value; } // NOLINT | 28 int UnaryByRef(int& value) const { ++call_count; return ++value; } // NOLINT |
27 int Multiply(int a, int b) const { ++call_count; return a * b; } | 29 int Multiply(int a, int b) const { ++call_count; return a * b; } |
30 void RefArgument(const scoped_refptr<LifeTimeCheck>& object) { | |
31 EXPECT_TRUE(object.get() != nullptr); | |
32 } | |
33 | |
28 mutable int call_count; | 34 mutable int call_count; |
29 }; | 35 }; |
30 | 36 |
31 struct A { int dummy; }; | 37 struct A { int dummy; }; |
32 struct B: public RefCountInterface { int dummy; }; | 38 struct B: public RefCountInterface { int dummy; }; |
33 struct C: public A, B {}; | 39 struct C: public A, B {}; |
34 struct D { | 40 struct D { |
35 int AddRef(); | 41 int AddRef(); |
36 }; | 42 }; |
37 struct E: public D { | 43 struct E: public D { |
38 int Release(); | 44 int Release(); |
39 }; | 45 }; |
40 struct F { | 46 struct F { |
41 void AddRef(); | 47 void AddRef(); |
42 void Release(); | 48 void Release(); |
43 }; | 49 }; |
44 | 50 |
45 class LifeTimeCheck : public RefCountInterface { | 51 struct LifeTimeCheck { |
46 public: | 52 LifeTimeCheck() : ref_count_(0) {} |
47 LifeTimeCheck(bool* has_died) : has_died_(has_died), is_ok_to_die_(false) {} | 53 void AddRef() { ++ref_count_; } |
48 ~LifeTimeCheck() { | 54 void Release() { --ref_count_; } |
49 EXPECT_TRUE(is_ok_to_die_); | |
50 *has_died_ = true; | |
51 } | |
52 void PrepareToDie() { is_ok_to_die_ = true; } | |
53 void NullaryVoid() {} | 55 void NullaryVoid() {} |
54 | 56 int ref_count_; |
55 private: | |
56 bool* const has_died_; | |
57 bool is_ok_to_die_; | |
58 }; | 57 }; |
59 | 58 |
60 int Return42() { return 42; } | 59 int Return42() { return 42; } |
61 int Negate(int a) { return -a; } | 60 int Negate(int a) { return -a; } |
62 int Multiply(int a, int b) { return a * b; } | 61 int Multiply(int a, int b) { return a * b; } |
63 | 62 |
64 } // namespace | 63 } // namespace |
65 | 64 |
66 // Try to catch any problem with scoped_refptr type deduction in rtc::Bind at | 65 // Try to catch any problem with scoped_refptr type deduction in rtc::Bind at |
67 // compile time. | 66 // compile time. |
67 static_assert(is_same<detail::RemoveScopedPtrRef< | |
tommi
2015/08/24 15:43:31
these are nice
| |
68 const scoped_refptr<RefCountInterface>&>::type, | |
69 scoped_refptr<RefCountInterface>>::value, | |
70 "const scoped_refptr& should be captured by value"); | |
71 | |
72 static_assert(is_same<detail::RemoveScopedPtrRef<const scoped_refptr<F>&>::type, | |
73 scoped_refptr<F>>::value, | |
74 "const scoped_refptr& should be captured by value"); | |
75 | |
76 static_assert( | |
77 is_same<detail::RemoveScopedPtrRef<const int&>::type, const int&>::value, | |
78 "const int& should be captured as const int&"); | |
79 | |
80 static_assert( | |
81 is_same<detail::RemoveScopedPtrRef<const F&>::type, const F&>::value, | |
82 "const F& should be captured as const F&"); | |
83 | |
84 static_assert( | |
85 is_same<detail::RemoveScopedPtrRef<F&>::type, F&>::value, | |
86 "F& should be captured as F&"); | |
87 | |
68 #define EXPECT_IS_CAPTURED_AS_PTR(T) \ | 88 #define EXPECT_IS_CAPTURED_AS_PTR(T) \ |
69 static_assert(is_same<detail::PointerType<T>::type, T*>::value, \ | 89 static_assert(is_same<detail::PointerType<T>::type, T*>::value, \ |
70 "PointerType") | 90 "PointerType") |
71 #define EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(T) \ | 91 #define EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(T) \ |
72 static_assert( \ | 92 static_assert( \ |
73 is_same<detail::PointerType<T>::type, scoped_refptr<T>>::value, \ | 93 is_same<detail::PointerType<T>::type, scoped_refptr<T>>::value, \ |
74 "PointerType") | 94 "PointerType") |
75 | 95 |
76 EXPECT_IS_CAPTURED_AS_PTR(void); | 96 EXPECT_IS_CAPTURED_AS_PTR(void); |
77 EXPECT_IS_CAPTURED_AS_PTR(int); | 97 EXPECT_IS_CAPTURED_AS_PTR(int); |
(...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
117 | 137 |
118 TEST(BindTest, BindToFunction) { | 138 TEST(BindTest, BindToFunction) { |
119 EXPECT_EQ(42, Bind(&Return42)()); | 139 EXPECT_EQ(42, Bind(&Return42)()); |
120 EXPECT_EQ(3, Bind(&Negate, -3)()); | 140 EXPECT_EQ(3, Bind(&Negate, -3)()); |
121 EXPECT_EQ(56, Bind(&Multiply, 8, 7)()); | 141 EXPECT_EQ(56, Bind(&Multiply, 8, 7)()); |
122 } | 142 } |
123 | 143 |
124 // Test Bind where method object implements RefCountInterface and is passed as a | 144 // Test Bind where method object implements RefCountInterface and is passed as a |
125 // pointer. | 145 // pointer. |
126 TEST(BindTest, CapturePointerAsScopedRefPtr) { | 146 TEST(BindTest, CapturePointerAsScopedRefPtr) { |
127 bool object_has_died = false; | 147 LifeTimeCheck object; |
128 scoped_refptr<LifeTimeCheck> object = | 148 EXPECT_EQ(object.ref_count_, 0); |
129 new RefCountedObject<LifeTimeCheck>(&object_has_died); | 149 scoped_refptr<LifeTimeCheck> scoped_object(&object); |
150 EXPECT_EQ(object.ref_count_, 1); | |
130 { | 151 { |
131 auto functor = Bind(&LifeTimeCheck::PrepareToDie, object.get()); | 152 auto functor = Bind(&LifeTimeCheck::NullaryVoid, &object); |
132 object = nullptr; | 153 EXPECT_EQ(object.ref_count_, 2); |
133 EXPECT_FALSE(object_has_died); | 154 scoped_object = nullptr; |
134 // Run prepare to die via functor. | 155 EXPECT_EQ(object.ref_count_, 1); |
135 functor(); | |
136 } | 156 } |
137 EXPECT_TRUE(object_has_died); | 157 EXPECT_EQ(object.ref_count_, 0); |
138 } | 158 } |
139 | 159 |
140 // Test Bind where method object implements RefCountInterface and is passed as a | 160 // Test Bind where method object implements RefCountInterface and is passed as a |
141 // scoped_refptr<>. | 161 // scoped_refptr<>. |
142 TEST(BindTest, CaptureScopedRefPtrAsScopedRefPtr) { | 162 TEST(BindTest, CaptureScopedRefPtrAsScopedRefPtr) { |
143 bool object_has_died = false; | 163 LifeTimeCheck object; |
144 scoped_refptr<LifeTimeCheck> object = | 164 EXPECT_EQ(object.ref_count_, 0); |
145 new RefCountedObject<LifeTimeCheck>(&object_has_died); | 165 scoped_refptr<LifeTimeCheck> scoped_object(&object); |
166 EXPECT_EQ(object.ref_count_, 1); | |
146 { | 167 { |
147 auto functor = Bind(&LifeTimeCheck::PrepareToDie, object); | 168 auto functor = Bind(&LifeTimeCheck::NullaryVoid, scoped_object); |
148 object = nullptr; | 169 EXPECT_EQ(object.ref_count_, 2); |
149 EXPECT_FALSE(object_has_died); | 170 scoped_object = nullptr; |
150 // Run prepare to die via functor. | 171 EXPECT_EQ(object.ref_count_, 1); |
151 functor(); | |
152 } | 172 } |
153 EXPECT_TRUE(object_has_died); | 173 EXPECT_EQ(object.ref_count_, 0); |
154 } | 174 } |
155 | 175 |
156 // Test Bind where method object is captured as scoped_refptr<> and the functor | 176 // Test Bind where method object is captured as scoped_refptr<> and the functor |
157 // dies while there are references left. | 177 // dies while there are references left. |
158 TEST(BindTest, FunctorReleasesObjectOnDestruction) { | 178 TEST(BindTest, FunctorReleasesObjectOnDestruction) { |
159 bool object_has_died = false; | 179 LifeTimeCheck object; |
160 scoped_refptr<LifeTimeCheck> object = | 180 EXPECT_EQ(object.ref_count_, 0); |
161 new RefCountedObject<LifeTimeCheck>(&object_has_died); | 181 scoped_refptr<LifeTimeCheck> scoped_object(&object); |
162 Bind(&LifeTimeCheck::NullaryVoid, object.get())(); | 182 EXPECT_EQ(object.ref_count_, 1); |
163 EXPECT_FALSE(object_has_died); | 183 Bind(&LifeTimeCheck::NullaryVoid, &object)(); |
164 object->PrepareToDie(); | 184 EXPECT_EQ(object.ref_count_, 1); |
165 object = nullptr; | 185 scoped_object = nullptr; |
166 EXPECT_TRUE(object_has_died); | 186 EXPECT_EQ(object.ref_count_, 0); |
187 } | |
188 | |
189 // Test Bind with scoped_refptr<> argument. | |
190 TEST(BindTest, ScopedRefPointerArgument) { | |
191 LifeTimeCheck object; | |
192 EXPECT_EQ(object.ref_count_, 0); | |
193 scoped_refptr<LifeTimeCheck> scoped_object(&object); | |
194 EXPECT_EQ(object.ref_count_, 1); | |
195 { | |
196 MethodBindTester bind_tester; | |
197 auto functor = | |
198 Bind(&MethodBindTester::RefArgument, &bind_tester, scoped_object); | |
199 EXPECT_EQ(object.ref_count_, 2); | |
200 } | |
201 EXPECT_EQ(object.ref_count_, 1); | |
202 scoped_object = nullptr; | |
203 EXPECT_EQ(object.ref_count_, 0); | |
204 } | |
205 | |
206 namespace { | |
207 | |
208 const int* Ref(const int& a) { return &a; } | |
209 | |
210 } // anonymous namespace | |
211 | |
212 // Test Bind with non-scoped_refptr<> reference argument. | |
213 TEST(BindTest, RefArgument) { | |
214 const int x = 42; | |
215 EXPECT_TRUE(Ref(x) == &x); | |
216 // Bind() should not make a copy of |x|, i.e. the pointers should be the same. | |
217 auto functor = Bind(&Ref, x); | |
218 EXPECT_TRUE(functor() == &x); | |
167 } | 219 } |
168 | 220 |
169 } // namespace rtc | 221 } // namespace rtc |
OLD | NEW |