|  | /* | 
|  | *  Copyright 2004 The WebRTC Project Authors. All rights reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS.  All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | */ | 
|  |  | 
|  | #include <type_traits> | 
|  |  | 
|  | #include "rtc_base/bind.h" | 
|  | #include "rtc_base/gunit.h" | 
|  |  | 
|  | #include "rtc_base/refcount.h" | 
|  | #include "rtc_base/refcountedobject.h" | 
|  |  | 
|  | namespace rtc { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | struct LifeTimeCheck; | 
|  |  | 
|  | struct MethodBindTester { | 
|  | void NullaryVoid() { ++call_count; } | 
|  | int NullaryInt() { | 
|  | ++call_count; | 
|  | return 1; | 
|  | } | 
|  | int NullaryConst() const { | 
|  | ++call_count; | 
|  | return 2; | 
|  | } | 
|  | void UnaryVoid(int dummy) { ++call_count; } | 
|  | template <class T> | 
|  | T Identity(T value) { | 
|  | ++call_count; | 
|  | return value; | 
|  | } | 
|  | int UnaryByPointer(int* value) const { | 
|  | ++call_count; | 
|  | return ++(*value); | 
|  | } | 
|  | int UnaryByRef(const int& value) const { | 
|  | ++call_count; | 
|  | return ++const_cast<int&>(value); | 
|  | } | 
|  | int Multiply(int a, int b) const { | 
|  | ++call_count; | 
|  | return a * b; | 
|  | } | 
|  | void RefArgument(const scoped_refptr<LifeTimeCheck>& object) { | 
|  | EXPECT_TRUE(object.get() != nullptr); | 
|  | } | 
|  |  | 
|  | mutable int call_count; | 
|  | }; | 
|  |  | 
|  | struct A { | 
|  | int dummy; | 
|  | }; | 
|  | struct B : public RefCountInterface { | 
|  | int dummy; | 
|  | }; | 
|  | struct C : public A, B {}; | 
|  | struct D { | 
|  | int AddRef(); | 
|  | }; | 
|  | struct E : public D { | 
|  | int Release(); | 
|  | }; | 
|  | struct F { | 
|  | void AddRef(); | 
|  | void Release(); | 
|  | }; | 
|  |  | 
|  | struct LifeTimeCheck { | 
|  | LifeTimeCheck() : ref_count_(0) {} | 
|  | void AddRef() { ++ref_count_; } | 
|  | void Release() { --ref_count_; } | 
|  | void NullaryVoid() {} | 
|  | int ref_count_; | 
|  | }; | 
|  |  | 
|  | int Return42() { | 
|  | return 42; | 
|  | } | 
|  | int Negate(int a) { | 
|  | return -a; | 
|  | } | 
|  | int Multiply(int a, int b) { | 
|  | return a * b; | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | // Try to catch any problem with scoped_refptr type deduction in rtc::Bind at | 
|  | // compile time. | 
|  | #define EXPECT_IS_CAPTURED_AS_PTR(T)                              \ | 
|  | static_assert(is_same<detail::PointerType<T>::type, T*>::value, \ | 
|  | "PointerTyp"                                      \ | 
|  | "e") | 
|  | #define EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(T)                        \ | 
|  | static_assert(                                                      \ | 
|  | is_same<detail::PointerType<T>::type, scoped_refptr<T>>::value, \ | 
|  | "PointerType") | 
|  |  | 
|  | EXPECT_IS_CAPTURED_AS_PTR(void); | 
|  | EXPECT_IS_CAPTURED_AS_PTR(int); | 
|  | EXPECT_IS_CAPTURED_AS_PTR(double); | 
|  | EXPECT_IS_CAPTURED_AS_PTR(A); | 
|  | EXPECT_IS_CAPTURED_AS_PTR(D); | 
|  | EXPECT_IS_CAPTURED_AS_PTR(RefCountInterface*); | 
|  | EXPECT_IS_CAPTURED_AS_PTR( | 
|  | decltype(Unretained<RefCountedObject<RefCountInterface>>)); | 
|  |  | 
|  | EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(RefCountInterface); | 
|  | EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(B); | 
|  | EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(C); | 
|  | EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(E); | 
|  | EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(F); | 
|  | EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(RefCountedObject<RefCountInterface>); | 
|  | EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(RefCountedObject<B>); | 
|  | EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(RefCountedObject<C>); | 
|  | EXPECT_IS_CAPTURED_AS_SCOPED_REFPTR(const RefCountedObject<RefCountInterface>); | 
|  |  | 
|  | TEST(BindTest, BindToMethod) { | 
|  | MethodBindTester object = {0}; | 
|  | EXPECT_EQ(0, object.call_count); | 
|  | Bind(&MethodBindTester::NullaryVoid, &object)(); | 
|  | EXPECT_EQ(1, object.call_count); | 
|  | EXPECT_EQ(1, Bind(&MethodBindTester::NullaryInt, &object)()); | 
|  | EXPECT_EQ(2, object.call_count); | 
|  | EXPECT_EQ(2, Bind(&MethodBindTester::NullaryConst, | 
|  | static_cast<const MethodBindTester*>(&object))()); | 
|  | EXPECT_EQ(3, object.call_count); | 
|  | Bind(&MethodBindTester::UnaryVoid, &object, 5)(); | 
|  | EXPECT_EQ(4, object.call_count); | 
|  | EXPECT_EQ(100, Bind(&MethodBindTester::Identity<int>, &object, 100)()); | 
|  | EXPECT_EQ(5, object.call_count); | 
|  | const std::string string_value("test string"); | 
|  | EXPECT_EQ(string_value, Bind(&MethodBindTester::Identity<std::string>, | 
|  | &object, string_value)()); | 
|  | EXPECT_EQ(6, object.call_count); | 
|  | int value = 11; | 
|  | // Bind binds by value, even if the method signature is by reference, so | 
|  | // "reference" binds require pointers. | 
|  | EXPECT_EQ(12, Bind(&MethodBindTester::UnaryByPointer, &object, &value)()); | 
|  | EXPECT_EQ(12, value); | 
|  | EXPECT_EQ(7, object.call_count); | 
|  | // It's possible to bind to a function that takes a const reference, though | 
|  | // the capture will be a copy. See UnaryByRef hackery above where it removes | 
|  | // the const to make sure the underlying storage is, in fact, a copy. | 
|  | EXPECT_EQ(13, Bind(&MethodBindTester::UnaryByRef, &object, value)()); | 
|  | // But the original value is unmodified. | 
|  | EXPECT_EQ(12, value); | 
|  | EXPECT_EQ(8, object.call_count); | 
|  | EXPECT_EQ(56, Bind(&MethodBindTester::Multiply, &object, 7, 8)()); | 
|  | EXPECT_EQ(9, object.call_count); | 
|  | } | 
|  |  | 
|  | TEST(BindTest, BindToFunction) { | 
|  | EXPECT_EQ(42, Bind(&Return42)()); | 
|  | EXPECT_EQ(3, Bind(&Negate, -3)()); | 
|  | EXPECT_EQ(56, Bind(&Multiply, 8, 7)()); | 
|  | } | 
|  |  | 
|  | // Test Bind where method object implements RefCountInterface and is passed as a | 
|  | // pointer. | 
|  | TEST(BindTest, CapturePointerAsScopedRefPtr) { | 
|  | LifeTimeCheck object; | 
|  | EXPECT_EQ(object.ref_count_, 0); | 
|  | scoped_refptr<LifeTimeCheck> scoped_object(&object); | 
|  | EXPECT_EQ(object.ref_count_, 1); | 
|  | { | 
|  | auto functor = Bind(&LifeTimeCheck::NullaryVoid, &object); | 
|  | EXPECT_EQ(object.ref_count_, 2); | 
|  | scoped_object = nullptr; | 
|  | EXPECT_EQ(object.ref_count_, 1); | 
|  | } | 
|  | EXPECT_EQ(object.ref_count_, 0); | 
|  | } | 
|  |  | 
|  | // Test Bind where method object implements RefCountInterface and is passed as a | 
|  | // scoped_refptr<>. | 
|  | TEST(BindTest, CaptureScopedRefPtrAsScopedRefPtr) { | 
|  | LifeTimeCheck object; | 
|  | EXPECT_EQ(object.ref_count_, 0); | 
|  | scoped_refptr<LifeTimeCheck> scoped_object(&object); | 
|  | EXPECT_EQ(object.ref_count_, 1); | 
|  | { | 
|  | auto functor = Bind(&LifeTimeCheck::NullaryVoid, scoped_object); | 
|  | EXPECT_EQ(object.ref_count_, 2); | 
|  | scoped_object = nullptr; | 
|  | EXPECT_EQ(object.ref_count_, 1); | 
|  | } | 
|  | EXPECT_EQ(object.ref_count_, 0); | 
|  | } | 
|  |  | 
|  | // Test Bind where method object is captured as scoped_refptr<> and the functor | 
|  | // dies while there are references left. | 
|  | TEST(BindTest, FunctorReleasesObjectOnDestruction) { | 
|  | LifeTimeCheck object; | 
|  | EXPECT_EQ(object.ref_count_, 0); | 
|  | scoped_refptr<LifeTimeCheck> scoped_object(&object); | 
|  | EXPECT_EQ(object.ref_count_, 1); | 
|  | Bind(&LifeTimeCheck::NullaryVoid, &object)(); | 
|  | EXPECT_EQ(object.ref_count_, 1); | 
|  | scoped_object = nullptr; | 
|  | EXPECT_EQ(object.ref_count_, 0); | 
|  | } | 
|  |  | 
|  | // Test Bind with scoped_refptr<> argument. | 
|  | TEST(BindTest, ScopedRefPointerArgument) { | 
|  | LifeTimeCheck object; | 
|  | EXPECT_EQ(object.ref_count_, 0); | 
|  | scoped_refptr<LifeTimeCheck> scoped_object(&object); | 
|  | EXPECT_EQ(object.ref_count_, 1); | 
|  | { | 
|  | MethodBindTester bind_tester; | 
|  | auto functor = | 
|  | Bind(&MethodBindTester::RefArgument, &bind_tester, scoped_object); | 
|  | EXPECT_EQ(object.ref_count_, 2); | 
|  | } | 
|  | EXPECT_EQ(object.ref_count_, 1); | 
|  | scoped_object = nullptr; | 
|  | EXPECT_EQ(object.ref_count_, 0); | 
|  | } | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | const int* Ref(const int& a) { | 
|  | return &a; | 
|  | } | 
|  |  | 
|  | }  // anonymous namespace | 
|  |  | 
|  | // Test Bind with non-scoped_refptr<> reference argument, which should be | 
|  | // modified to a non-reference capture. | 
|  | TEST(BindTest, RefArgument) { | 
|  | const int x = 42; | 
|  | EXPECT_EQ(&x, Ref(x)); | 
|  | // Bind() should make a copy of |x|, i.e. the pointers should be different. | 
|  | auto functor = Bind(&Ref, x); | 
|  | EXPECT_NE(&x, functor()); | 
|  | } | 
|  |  | 
|  | }  // namespace rtc |