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; | 20 struct LifeTimeCheck; |
21 | 21 |
22 struct MethodBindTester { | 22 struct MethodBindTester { |
23 void NullaryVoid() { ++call_count; } | 23 void NullaryVoid() { ++call_count; } |
24 int NullaryInt() { ++call_count; return 1; } | 24 int NullaryInt() { ++call_count; return 1; } |
25 int NullaryConst() const { ++call_count; return 2; } | 25 int NullaryConst() const { ++call_count; return 2; } |
26 void UnaryVoid(int dummy) { ++call_count; } | 26 void UnaryVoid(int dummy) { ++call_count; } |
27 template <class T> T Identity(T value) { ++call_count; return value; } | 27 template <class T> T Identity(T value) { ++call_count; return value; } |
28 int UnaryByRef(int& value) const { ++call_count; return ++value; } // NOLINT | 28 int UnaryByPointer(int* value) const { |
29 ++call_count; return ++(*value); | |
30 } | |
31 int UnaryByRef(const int& value) const { | |
32 ++call_count; | |
33 return ++const_cast<int&>(value); | |
34 } | |
29 int Multiply(int a, int b) const { ++call_count; return a * b; } | 35 int Multiply(int a, int b) const { ++call_count; return a * b; } |
30 void RefArgument(const scoped_refptr<LifeTimeCheck>& object) { | 36 void RefArgument(const scoped_refptr<LifeTimeCheck>& object) { |
31 EXPECT_TRUE(object.get() != nullptr); | 37 EXPECT_TRUE(object.get() != nullptr); |
32 } | 38 } |
33 | 39 |
34 mutable int call_count; | 40 mutable int call_count; |
35 }; | 41 }; |
36 | 42 |
37 struct A { int dummy; }; | 43 struct A { int dummy; }; |
38 struct B: public RefCountInterface { int dummy; }; | 44 struct B: public RefCountInterface { int dummy; }; |
(...skipping 18 matching lines...) Expand all Loading... | |
57 }; | 63 }; |
58 | 64 |
59 int Return42() { return 42; } | 65 int Return42() { return 42; } |
60 int Negate(int a) { return -a; } | 66 int Negate(int a) { return -a; } |
61 int Multiply(int a, int b) { return a * b; } | 67 int Multiply(int a, int b) { return a * b; } |
62 | 68 |
63 } // namespace | 69 } // namespace |
64 | 70 |
65 // Try to catch any problem with scoped_refptr type deduction in rtc::Bind at | 71 // Try to catch any problem with scoped_refptr type deduction in rtc::Bind at |
66 // compile time. | 72 // compile time. |
67 static_assert(is_same<detail::RemoveScopedPtrRef< | 73 static_assert(is_same<detail::RemoveAllRef< |
68 const scoped_refptr<RefCountInterface>&>::type, | 74 const scoped_refptr<RefCountInterface>&>::type, |
69 scoped_refptr<RefCountInterface>>::value, | 75 const scoped_refptr<RefCountInterface>>::value, |
70 "const scoped_refptr& should be captured by value"); | 76 "const scoped_refptr& should be captured by value"); |
71 | 77 |
72 static_assert(is_same<detail::RemoveScopedPtrRef<const scoped_refptr<F>&>::type, | 78 static_assert(is_same<detail::RemoveAllRef<const scoped_refptr<F>&>::type, |
73 scoped_refptr<F>>::value, | 79 const scoped_refptr<F>>::value, |
74 "const scoped_refptr& should be captured by value"); | 80 "const scoped_refptr& should be captured by value"); |
75 | 81 |
76 static_assert( | 82 static_assert( |
77 is_same<detail::RemoveScopedPtrRef<const int&>::type, const int&>::value, | 83 is_same<detail::RemoveAllRef<const int&>::type, const int>::value, |
78 "const int& should be captured as const int&"); | 84 "const int& should be captured as const int"); |
79 | 85 |
80 static_assert( | 86 static_assert( |
81 is_same<detail::RemoveScopedPtrRef<const F&>::type, const F&>::value, | 87 is_same<detail::RemoveAllRef<const F&>::type, const F>::value, |
82 "const F& should be captured as const F&"); | 88 "const F& should be captured as const F"); |
83 | 89 |
84 static_assert( | 90 static_assert( |
85 is_same<detail::RemoveScopedPtrRef<F&>::type, F&>::value, | 91 is_same<detail::RemoveAllRef<F&>::type, F>::value, |
86 "F& should be captured as F&"); | 92 "F& should be captured as F"); |
87 | 93 |
88 #define EXPECT_IS_CAPTURED_AS_PTR(T) \ | 94 #define EXPECT_IS_CAPTURED_AS_PTR(T) \ |
89 static_assert(is_same<detail::PointerType<T>::type, T*>::value, \ | 95 static_assert(is_same<detail::PointerType<T>::type, T*>::value, \ |
90 "PointerType") | 96 "PointerType") |
91 #define EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(T) \ | 97 #define EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(T) \ |
92 static_assert( \ | 98 static_assert( \ |
93 is_same<detail::PointerType<T>::type, scoped_refptr<T>>::value, \ | 99 is_same<detail::PointerType<T>::type, scoped_refptr<T>>::value, \ |
94 "PointerType") | 100 "PointerType") |
95 | 101 |
96 EXPECT_IS_CAPTURED_AS_PTR(void); | 102 EXPECT_IS_CAPTURED_AS_PTR(void); |
(...skipping 25 matching lines...) Expand all Loading... | |
122 EXPECT_EQ(3, object.call_count); | 128 EXPECT_EQ(3, object.call_count); |
123 Bind(&MethodBindTester::UnaryVoid, &object, 5)(); | 129 Bind(&MethodBindTester::UnaryVoid, &object, 5)(); |
124 EXPECT_EQ(4, object.call_count); | 130 EXPECT_EQ(4, object.call_count); |
125 EXPECT_EQ(100, Bind(&MethodBindTester::Identity<int>, &object, 100)()); | 131 EXPECT_EQ(100, Bind(&MethodBindTester::Identity<int>, &object, 100)()); |
126 EXPECT_EQ(5, object.call_count); | 132 EXPECT_EQ(5, object.call_count); |
127 const std::string string_value("test string"); | 133 const std::string string_value("test string"); |
128 EXPECT_EQ(string_value, Bind(&MethodBindTester::Identity<std::string>, | 134 EXPECT_EQ(string_value, Bind(&MethodBindTester::Identity<std::string>, |
129 &object, string_value)()); | 135 &object, string_value)()); |
130 EXPECT_EQ(6, object.call_count); | 136 EXPECT_EQ(6, object.call_count); |
131 int value = 11; | 137 int value = 11; |
132 EXPECT_EQ(12, Bind(&MethodBindTester::UnaryByRef, &object, value)()); | 138 // Bind binds by value, even if the method signature is by reference, so |
139 // "reference" binds require pointers. | |
140 EXPECT_EQ(12, Bind(&MethodBindTester::UnaryByPointer, &object, &value)()); | |
133 EXPECT_EQ(12, value); | 141 EXPECT_EQ(12, value); |
134 EXPECT_EQ(7, object.call_count); | 142 EXPECT_EQ(7, object.call_count); |
143 // It's possible to bind to a function that takes a const reference, though | |
144 // the capture will be a copy. See UnaryByRef hackery above where it removes | |
145 // the const to make sure the underlying storage is, in fact, a copy. | |
146 EXPECT_EQ(13, Bind(&MethodBindTester::UnaryByRef, &object, value)()); | |
147 // But the original value is unmodified. | |
148 EXPECT_EQ(12, value); | |
149 EXPECT_EQ(8, object.call_count); | |
135 EXPECT_EQ(56, Bind(&MethodBindTester::Multiply, &object, 7, 8)()); | 150 EXPECT_EQ(56, Bind(&MethodBindTester::Multiply, &object, 7, 8)()); |
136 EXPECT_EQ(8, object.call_count); | 151 EXPECT_EQ(9, object.call_count); |
137 } | 152 } |
138 | 153 |
139 TEST(BindTest, BindToFunction) { | 154 TEST(BindTest, BindToFunction) { |
140 EXPECT_EQ(42, Bind(&Return42)()); | 155 EXPECT_EQ(42, Bind(&Return42)()); |
141 EXPECT_EQ(3, Bind(&Negate, -3)()); | 156 EXPECT_EQ(3, Bind(&Negate, -3)()); |
142 EXPECT_EQ(56, Bind(&Multiply, 8, 7)()); | 157 EXPECT_EQ(56, Bind(&Multiply, 8, 7)()); |
143 } | 158 } |
144 | 159 |
145 // 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 |
146 // pointer. | 161 // pointer. |
(...skipping 56 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
203 scoped_object = nullptr; | 218 scoped_object = nullptr; |
204 EXPECT_EQ(object.ref_count_, 0); | 219 EXPECT_EQ(object.ref_count_, 0); |
205 } | 220 } |
206 | 221 |
207 namespace { | 222 namespace { |
208 | 223 |
209 const int* Ref(const int& a) { return &a; } | 224 const int* Ref(const int& a) { return &a; } |
210 | 225 |
211 } // anonymous namespace | 226 } // anonymous namespace |
212 | 227 |
213 // Test Bind with non-scoped_refptr<> reference argument. | 228 // Test Bind with non-scoped_refptr<> reference argument, which should be |
229 // modified to a non-reference capture. | |
214 TEST(BindTest, RefArgument) { | 230 TEST(BindTest, RefArgument) { |
215 const int x = 42; | 231 const int x = 42; |
216 EXPECT_TRUE(Ref(x) == &x); | 232 EXPECT_EQ(&x, Ref(x)); |
217 // Bind() should not make a copy of |x|, i.e. the pointers should be the same. | 233 // Bind() should make a copy of |x|, i.e. the pointers should be the same. |
magjed_webrtc
2015/10/23 05:17:01
s/i.e. the pointers should be the same/i.e. the po
noahric
2015/10/23 06:02:16
Done.
| |
218 auto functor = Bind(&Ref, x); | 234 auto functor = Bind(&Ref, x); |
219 EXPECT_TRUE(functor() == &x); | 235 EXPECT_NE(&x, functor()); |
220 } | 236 } |
221 | 237 |
222 } // namespace rtc | 238 } // namespace rtc |
OLD | NEW |