239 lines
7.8 KiB
C++
239 lines
7.8 KiB
C++
|
/*
|
||
|
* 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 "webrtc/base/bind.h"
|
||
|
#include "webrtc/base/gunit.h"
|
||
|
|
||
|
#include "webrtc/base/refcount.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.
|
||
|
static_assert(
|
||
|
is_same<
|
||
|
rtc::remove_reference<const scoped_refptr<RefCountInterface>&>::type,
|
||
|
const scoped_refptr<RefCountInterface>>::value,
|
||
|
"const scoped_refptr& should be captured by value");
|
||
|
|
||
|
static_assert(is_same<rtc::remove_reference<const scoped_refptr<F>&>::type,
|
||
|
const scoped_refptr<F>>::value,
|
||
|
"const scoped_refptr& should be captured by value");
|
||
|
|
||
|
static_assert(
|
||
|
is_same<rtc::remove_reference<const int&>::type, const int>::value,
|
||
|
"const int& should be captured as const int");
|
||
|
|
||
|
static_assert(is_same<rtc::remove_reference<const F&>::type, const F>::value,
|
||
|
"const F& should be captured as const F");
|
||
|
|
||
|
static_assert(is_same<rtc::remove_reference<F&>::type, F>::value,
|
||
|
"F& should be captured as F");
|
||
|
|
||
|
#define EXPECT_IS_CAPTURED_AS_PTR(T) \
|
||
|
static_assert(is_same<detail::PointerType<T>::type, T*>::value, \
|
||
|
"PointerType")
|
||
|
#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_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
|