You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
672 lines
21 KiB
672 lines
21 KiB
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
#include "base/memory/ref_counted.h"
|
|
|
|
#include <type_traits>
|
|
#include <utility>
|
|
|
|
#include "base/test/gtest_util.h"
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
|
|
|
namespace {
|
|
|
|
class SelfAssign : public base::RefCounted<SelfAssign> {
|
|
protected:
|
|
virtual ~SelfAssign() = default;
|
|
|
|
private:
|
|
friend class base::RefCounted<SelfAssign>;
|
|
};
|
|
|
|
class Derived : public SelfAssign {
|
|
protected:
|
|
~Derived() override = default;
|
|
|
|
private:
|
|
friend class base::RefCounted<Derived>;
|
|
};
|
|
|
|
class CheckDerivedMemberAccess : public scoped_refptr<SelfAssign> {
|
|
public:
|
|
CheckDerivedMemberAccess() {
|
|
// This shouldn't compile if we don't have access to the member variable.
|
|
SelfAssign** pptr = &ptr_;
|
|
EXPECT_EQ(*pptr, ptr_);
|
|
}
|
|
};
|
|
|
|
class ScopedRefPtrToSelf : public base::RefCounted<ScopedRefPtrToSelf> {
|
|
public:
|
|
ScopedRefPtrToSelf() : self_ptr_(this) {}
|
|
|
|
static bool was_destroyed() { return was_destroyed_; }
|
|
|
|
static void reset_was_destroyed() { was_destroyed_ = false; }
|
|
|
|
scoped_refptr<ScopedRefPtrToSelf> self_ptr_;
|
|
|
|
private:
|
|
friend class base::RefCounted<ScopedRefPtrToSelf>;
|
|
~ScopedRefPtrToSelf() { was_destroyed_ = true; }
|
|
|
|
static bool was_destroyed_;
|
|
};
|
|
|
|
bool ScopedRefPtrToSelf::was_destroyed_ = false;
|
|
|
|
class ScopedRefPtrCountBase : public base::RefCounted<ScopedRefPtrCountBase> {
|
|
public:
|
|
ScopedRefPtrCountBase() { ++constructor_count_; }
|
|
|
|
static int constructor_count() { return constructor_count_; }
|
|
|
|
static int destructor_count() { return destructor_count_; }
|
|
|
|
static void reset_count() {
|
|
constructor_count_ = 0;
|
|
destructor_count_ = 0;
|
|
}
|
|
|
|
protected:
|
|
virtual ~ScopedRefPtrCountBase() { ++destructor_count_; }
|
|
|
|
private:
|
|
friend class base::RefCounted<ScopedRefPtrCountBase>;
|
|
|
|
static int constructor_count_;
|
|
static int destructor_count_;
|
|
};
|
|
|
|
int ScopedRefPtrCountBase::constructor_count_ = 0;
|
|
int ScopedRefPtrCountBase::destructor_count_ = 0;
|
|
|
|
class ScopedRefPtrCountDerived : public ScopedRefPtrCountBase {
|
|
public:
|
|
ScopedRefPtrCountDerived() { ++constructor_count_; }
|
|
|
|
static int constructor_count() { return constructor_count_; }
|
|
|
|
static int destructor_count() { return destructor_count_; }
|
|
|
|
static void reset_count() {
|
|
constructor_count_ = 0;
|
|
destructor_count_ = 0;
|
|
}
|
|
|
|
protected:
|
|
~ScopedRefPtrCountDerived() override { ++destructor_count_; }
|
|
|
|
private:
|
|
friend class base::RefCounted<ScopedRefPtrCountDerived>;
|
|
|
|
static int constructor_count_;
|
|
static int destructor_count_;
|
|
};
|
|
|
|
int ScopedRefPtrCountDerived::constructor_count_ = 0;
|
|
int ScopedRefPtrCountDerived::destructor_count_ = 0;
|
|
|
|
class Other : public base::RefCounted<Other> {
|
|
private:
|
|
friend class base::RefCounted<Other>;
|
|
|
|
~Other() = default;
|
|
};
|
|
|
|
class HasPrivateDestructorWithDeleter;
|
|
|
|
struct Deleter {
|
|
static void Destruct(const HasPrivateDestructorWithDeleter* x);
|
|
};
|
|
|
|
class HasPrivateDestructorWithDeleter
|
|
: public base::RefCounted<HasPrivateDestructorWithDeleter, Deleter> {
|
|
public:
|
|
HasPrivateDestructorWithDeleter() = default;
|
|
|
|
private:
|
|
friend struct Deleter;
|
|
~HasPrivateDestructorWithDeleter() = default;
|
|
};
|
|
|
|
void Deleter::Destruct(const HasPrivateDestructorWithDeleter* x) {
|
|
delete x;
|
|
}
|
|
|
|
scoped_refptr<Other> Overloaded(scoped_refptr<Other> other) {
|
|
return other;
|
|
}
|
|
|
|
scoped_refptr<SelfAssign> Overloaded(scoped_refptr<SelfAssign> self_assign) {
|
|
return self_assign;
|
|
}
|
|
|
|
class InitialRefCountIsOne : public base::RefCounted<InitialRefCountIsOne> {
|
|
public:
|
|
REQUIRE_ADOPTION_FOR_REFCOUNTED_TYPE();
|
|
|
|
InitialRefCountIsOne() = default;
|
|
|
|
private:
|
|
friend class base::RefCounted<InitialRefCountIsOne>;
|
|
~InitialRefCountIsOne() = default;
|
|
};
|
|
|
|
// Checks that the scoped_refptr is null before the reference counted object is
|
|
// destroyed.
|
|
class CheckRefptrNull : public base::RefCounted<CheckRefptrNull> {
|
|
public:
|
|
// Set the last scoped_refptr that will have a reference to this object.
|
|
void set_scoped_refptr(scoped_refptr<CheckRefptrNull>* ptr) { ptr_ = ptr; }
|
|
|
|
protected:
|
|
virtual ~CheckRefptrNull() {
|
|
EXPECT_NE(ptr_, nullptr);
|
|
EXPECT_EQ(ptr_->get(), nullptr);
|
|
}
|
|
|
|
private:
|
|
friend class base::RefCounted<CheckRefptrNull>;
|
|
|
|
scoped_refptr<CheckRefptrNull>* ptr_ = nullptr;
|
|
};
|
|
|
|
} // end namespace
|
|
|
|
TEST(RefCountedUnitTest, TestSelfAssignment) {
|
|
SelfAssign* p = new SelfAssign;
|
|
scoped_refptr<SelfAssign> var(p);
|
|
var = *&var; // The *& defeats Clang's -Wself-assign warning.
|
|
EXPECT_EQ(var.get(), p);
|
|
var = std::move(var);
|
|
EXPECT_EQ(var.get(), p);
|
|
var.swap(var);
|
|
EXPECT_EQ(var.get(), p);
|
|
swap(var, var);
|
|
EXPECT_EQ(var.get(), p);
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, ScopedRefPtrMemberAccess) {
|
|
CheckDerivedMemberAccess check;
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, ScopedRefPtrToSelfPointerAssignment) {
|
|
ScopedRefPtrToSelf::reset_was_destroyed();
|
|
|
|
ScopedRefPtrToSelf* check = new ScopedRefPtrToSelf();
|
|
EXPECT_FALSE(ScopedRefPtrToSelf::was_destroyed());
|
|
check->self_ptr_ = nullptr;
|
|
EXPECT_TRUE(ScopedRefPtrToSelf::was_destroyed());
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, ScopedRefPtrToSelfMoveAssignment) {
|
|
ScopedRefPtrToSelf::reset_was_destroyed();
|
|
|
|
ScopedRefPtrToSelf* check = new ScopedRefPtrToSelf();
|
|
EXPECT_FALSE(ScopedRefPtrToSelf::was_destroyed());
|
|
// Releasing |check->self_ptr_| will delete |check|.
|
|
// The move assignment operator must assign |check->self_ptr_| first then
|
|
// release |check->self_ptr_|.
|
|
check->self_ptr_ = scoped_refptr<ScopedRefPtrToSelf>();
|
|
EXPECT_TRUE(ScopedRefPtrToSelf::was_destroyed());
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, BooleanTesting) {
|
|
scoped_refptr<SelfAssign> ptr_to_an_instance = new SelfAssign;
|
|
EXPECT_TRUE(ptr_to_an_instance);
|
|
EXPECT_FALSE(!ptr_to_an_instance);
|
|
|
|
if (ptr_to_an_instance) {
|
|
} else {
|
|
ADD_FAILURE() << "Pointer to an instance should result in true.";
|
|
}
|
|
|
|
if (!ptr_to_an_instance) { // check for operator!().
|
|
ADD_FAILURE() << "Pointer to an instance should result in !x being false.";
|
|
}
|
|
|
|
scoped_refptr<SelfAssign> null_ptr;
|
|
EXPECT_FALSE(null_ptr);
|
|
EXPECT_TRUE(!null_ptr);
|
|
|
|
if (null_ptr) {
|
|
ADD_FAILURE() << "Null pointer should result in false.";
|
|
}
|
|
|
|
if (!null_ptr) { // check for operator!().
|
|
} else {
|
|
ADD_FAILURE() << "Null pointer should result in !x being true.";
|
|
}
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, Equality) {
|
|
scoped_refptr<SelfAssign> p1(new SelfAssign);
|
|
scoped_refptr<SelfAssign> p2(new SelfAssign);
|
|
|
|
EXPECT_EQ(p1, p1);
|
|
EXPECT_EQ(p2, p2);
|
|
|
|
EXPECT_NE(p1, p2);
|
|
EXPECT_NE(p2, p1);
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, NullptrEquality) {
|
|
scoped_refptr<SelfAssign> ptr_to_an_instance(new SelfAssign);
|
|
scoped_refptr<SelfAssign> ptr_to_nullptr;
|
|
|
|
EXPECT_NE(nullptr, ptr_to_an_instance);
|
|
EXPECT_NE(ptr_to_an_instance, nullptr);
|
|
EXPECT_EQ(nullptr, ptr_to_nullptr);
|
|
EXPECT_EQ(ptr_to_nullptr, nullptr);
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, ConvertibleEquality) {
|
|
scoped_refptr<Derived> p1(new Derived);
|
|
scoped_refptr<SelfAssign> p2;
|
|
|
|
EXPECT_NE(p1, p2);
|
|
EXPECT_NE(p2, p1);
|
|
|
|
p2 = p1;
|
|
|
|
EXPECT_EQ(p1, p2);
|
|
EXPECT_EQ(p2, p1);
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, MoveAssignment1) {
|
|
ScopedRefPtrCountBase::reset_count();
|
|
|
|
{
|
|
ScopedRefPtrCountBase *raw = new ScopedRefPtrCountBase();
|
|
scoped_refptr<ScopedRefPtrCountBase> p1(raw);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
{
|
|
scoped_refptr<ScopedRefPtrCountBase> p2;
|
|
|
|
p2 = std::move(p1);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(nullptr, p1.get());
|
|
EXPECT_EQ(raw, p2.get());
|
|
|
|
// p2 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
// p1 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, MoveAssignment2) {
|
|
ScopedRefPtrCountBase::reset_count();
|
|
|
|
{
|
|
ScopedRefPtrCountBase *raw = new ScopedRefPtrCountBase();
|
|
scoped_refptr<ScopedRefPtrCountBase> p1;
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
{
|
|
scoped_refptr<ScopedRefPtrCountBase> p2(raw);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
p1 = std::move(p2);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(raw, p1.get());
|
|
EXPECT_EQ(nullptr, p2.get());
|
|
|
|
// p2 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
// p1 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, MoveAssignmentSameInstance1) {
|
|
ScopedRefPtrCountBase::reset_count();
|
|
|
|
{
|
|
ScopedRefPtrCountBase *raw = new ScopedRefPtrCountBase();
|
|
scoped_refptr<ScopedRefPtrCountBase> p1(raw);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
{
|
|
scoped_refptr<ScopedRefPtrCountBase> p2(p1);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
p1 = std::move(p2);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(raw, p1.get());
|
|
EXPECT_EQ(nullptr, p2.get());
|
|
|
|
// p2 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
// p1 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, MoveAssignmentSameInstance2) {
|
|
ScopedRefPtrCountBase::reset_count();
|
|
|
|
{
|
|
ScopedRefPtrCountBase *raw = new ScopedRefPtrCountBase();
|
|
scoped_refptr<ScopedRefPtrCountBase> p1(raw);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
{
|
|
scoped_refptr<ScopedRefPtrCountBase> p2(p1);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
p2 = std::move(p1);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(nullptr, p1.get());
|
|
EXPECT_EQ(raw, p2.get());
|
|
|
|
// p2 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
// p1 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, MoveAssignmentDifferentInstances) {
|
|
ScopedRefPtrCountBase::reset_count();
|
|
|
|
{
|
|
ScopedRefPtrCountBase *raw1 = new ScopedRefPtrCountBase();
|
|
scoped_refptr<ScopedRefPtrCountBase> p1(raw1);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
{
|
|
ScopedRefPtrCountBase *raw2 = new ScopedRefPtrCountBase();
|
|
scoped_refptr<ScopedRefPtrCountBase> p2(raw2);
|
|
EXPECT_EQ(2, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
p1 = std::move(p2);
|
|
EXPECT_EQ(2, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(raw2, p1.get());
|
|
EXPECT_EQ(nullptr, p2.get());
|
|
|
|
// p2 goes out of scope.
|
|
}
|
|
EXPECT_EQ(2, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
// p1 goes out of scope.
|
|
}
|
|
EXPECT_EQ(2, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(2, ScopedRefPtrCountBase::destructor_count());
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, MoveAssignmentSelfMove) {
|
|
ScopedRefPtrCountBase::reset_count();
|
|
|
|
{
|
|
ScopedRefPtrCountBase* raw = new ScopedRefPtrCountBase;
|
|
scoped_refptr<ScopedRefPtrCountBase> p1(raw);
|
|
scoped_refptr<ScopedRefPtrCountBase>& p1_ref = p1;
|
|
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
p1 = std::move(p1_ref);
|
|
|
|
// |p1| is "valid but unspecified", so don't bother inspecting its
|
|
// contents, just ensure that we don't crash.
|
|
}
|
|
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, MoveAssignmentDerived) {
|
|
ScopedRefPtrCountBase::reset_count();
|
|
ScopedRefPtrCountDerived::reset_count();
|
|
|
|
{
|
|
ScopedRefPtrCountBase *raw1 = new ScopedRefPtrCountBase();
|
|
scoped_refptr<ScopedRefPtrCountBase> p1(raw1);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountDerived::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountDerived::destructor_count());
|
|
|
|
{
|
|
ScopedRefPtrCountDerived *raw2 = new ScopedRefPtrCountDerived();
|
|
scoped_refptr<ScopedRefPtrCountDerived> p2(raw2);
|
|
EXPECT_EQ(2, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountDerived::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountDerived::destructor_count());
|
|
|
|
p1 = std::move(p2);
|
|
EXPECT_EQ(2, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountDerived::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountDerived::destructor_count());
|
|
EXPECT_EQ(raw2, p1.get());
|
|
EXPECT_EQ(nullptr, p2.get());
|
|
|
|
// p2 goes out of scope.
|
|
}
|
|
EXPECT_EQ(2, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountDerived::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountDerived::destructor_count());
|
|
|
|
// p1 goes out of scope.
|
|
}
|
|
EXPECT_EQ(2, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(2, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountDerived::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountDerived::destructor_count());
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, MoveConstructor) {
|
|
ScopedRefPtrCountBase::reset_count();
|
|
|
|
{
|
|
ScopedRefPtrCountBase *raw = new ScopedRefPtrCountBase();
|
|
scoped_refptr<ScopedRefPtrCountBase> p1(raw);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
{
|
|
scoped_refptr<ScopedRefPtrCountBase> p2(std::move(p1));
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(nullptr, p1.get());
|
|
EXPECT_EQ(raw, p2.get());
|
|
|
|
// p2 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
|
|
// p1 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, MoveConstructorDerived) {
|
|
ScopedRefPtrCountBase::reset_count();
|
|
ScopedRefPtrCountDerived::reset_count();
|
|
|
|
{
|
|
ScopedRefPtrCountDerived *raw1 = new ScopedRefPtrCountDerived();
|
|
scoped_refptr<ScopedRefPtrCountDerived> p1(raw1);
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountDerived::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountDerived::destructor_count());
|
|
|
|
{
|
|
scoped_refptr<ScopedRefPtrCountBase> p2(std::move(p1));
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountDerived::constructor_count());
|
|
EXPECT_EQ(0, ScopedRefPtrCountDerived::destructor_count());
|
|
EXPECT_EQ(nullptr, p1.get());
|
|
EXPECT_EQ(raw1, p2.get());
|
|
|
|
// p2 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountDerived::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountDerived::destructor_count());
|
|
|
|
// p1 goes out of scope.
|
|
}
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountDerived::constructor_count());
|
|
EXPECT_EQ(1, ScopedRefPtrCountDerived::destructor_count());
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, TestOverloadResolutionCopy) {
|
|
const scoped_refptr<Derived> derived(new Derived);
|
|
const scoped_refptr<SelfAssign> expected(derived);
|
|
EXPECT_EQ(expected, Overloaded(derived));
|
|
|
|
const scoped_refptr<Other> other(new Other);
|
|
EXPECT_EQ(other, Overloaded(other));
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, TestOverloadResolutionMove) {
|
|
scoped_refptr<Derived> derived(new Derived);
|
|
const scoped_refptr<SelfAssign> expected(derived);
|
|
EXPECT_EQ(expected, Overloaded(std::move(derived)));
|
|
|
|
scoped_refptr<Other> other(new Other);
|
|
const scoped_refptr<Other> other2(other);
|
|
EXPECT_EQ(other2, Overloaded(std::move(other)));
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, TestMakeRefCounted) {
|
|
scoped_refptr<Derived> derived = new Derived;
|
|
EXPECT_TRUE(derived->HasOneRef());
|
|
derived.reset();
|
|
|
|
scoped_refptr<Derived> derived2 = base::MakeRefCounted<Derived>();
|
|
EXPECT_TRUE(derived2->HasOneRef());
|
|
derived2.reset();
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, TestInitialRefCountIsOne) {
|
|
scoped_refptr<InitialRefCountIsOne> obj =
|
|
base::MakeRefCounted<InitialRefCountIsOne>();
|
|
EXPECT_TRUE(obj->HasOneRef());
|
|
obj.reset();
|
|
|
|
scoped_refptr<InitialRefCountIsOne> obj2 =
|
|
base::AdoptRef(new InitialRefCountIsOne);
|
|
EXPECT_TRUE(obj2->HasOneRef());
|
|
obj2.reset();
|
|
|
|
scoped_refptr<Other> obj3 = base::MakeRefCounted<Other>();
|
|
EXPECT_TRUE(obj3->HasOneRef());
|
|
obj3.reset();
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, TestPrivateDestructorWithDeleter) {
|
|
// Ensure that RefCounted doesn't need the access to the pointee dtor when
|
|
// a custom deleter is given.
|
|
scoped_refptr<HasPrivateDestructorWithDeleter> obj =
|
|
base::MakeRefCounted<HasPrivateDestructorWithDeleter>();
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, TestReset) {
|
|
ScopedRefPtrCountBase::reset_count();
|
|
|
|
// Create ScopedRefPtrCountBase that is referenced by |obj1| and |obj2|.
|
|
scoped_refptr<ScopedRefPtrCountBase> obj1 =
|
|
base::MakeRefCounted<ScopedRefPtrCountBase>();
|
|
scoped_refptr<ScopedRefPtrCountBase> obj2 = obj1;
|
|
EXPECT_NE(obj1.get(), nullptr);
|
|
EXPECT_NE(obj2.get(), nullptr);
|
|
EXPECT_EQ(ScopedRefPtrCountBase::constructor_count(), 1);
|
|
EXPECT_EQ(ScopedRefPtrCountBase::destructor_count(), 0);
|
|
|
|
// Check that calling reset() on |obj1| resets it. |obj2| still has a
|
|
// reference to the ScopedRefPtrCountBase so it shouldn't be reset.
|
|
obj1.reset();
|
|
EXPECT_EQ(obj1.get(), nullptr);
|
|
EXPECT_EQ(ScopedRefPtrCountBase::constructor_count(), 1);
|
|
EXPECT_EQ(ScopedRefPtrCountBase::destructor_count(), 0);
|
|
|
|
// Check that calling reset() on |obj2| resets it and causes the deletion of
|
|
// the ScopedRefPtrCountBase.
|
|
obj2.reset();
|
|
EXPECT_EQ(obj2.get(), nullptr);
|
|
EXPECT_EQ(ScopedRefPtrCountBase::constructor_count(), 1);
|
|
EXPECT_EQ(ScopedRefPtrCountBase::destructor_count(), 1);
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, TestResetAlreadyNull) {
|
|
// Check that calling reset() on a null scoped_refptr does nothing.
|
|
scoped_refptr<ScopedRefPtrCountBase> obj;
|
|
obj.reset();
|
|
// |obj| should still be null after calling reset().
|
|
EXPECT_EQ(obj.get(), nullptr);
|
|
}
|
|
|
|
TEST(RefCountedUnitTest, CheckScopedRefptrNullBeforeObjectDestruction) {
|
|
scoped_refptr<CheckRefptrNull> obj = base::MakeRefCounted<CheckRefptrNull>();
|
|
obj->set_scoped_refptr(&obj);
|
|
|
|
// Check that when reset() is called the scoped_refptr internal pointer is set
|
|
// to null before the reference counted object is destroyed. This check is
|
|
// done by the CheckRefptrNull destructor.
|
|
obj.reset();
|
|
EXPECT_EQ(obj.get(), nullptr);
|
|
}
|
|
|
|
TEST(RefCountedDeathTest, TestAdoptRef) {
|
|
// Check that WrapRefCounted() DCHECKs if passed a type that defines
|
|
// REQUIRE_ADOPTION_FOR_REFCOUNTED_TYPE.
|
|
EXPECT_DCHECK_DEATH(base::WrapRefCounted(new InitialRefCountIsOne));
|
|
|
|
// Check that AdoptRef() DCHECKs if passed a nullptr.
|
|
InitialRefCountIsOne* ptr = nullptr;
|
|
EXPECT_DCHECK_DEATH(base::AdoptRef(ptr));
|
|
|
|
// Check that AdoptRef() DCHECKs if passed an object that doesn't need to be
|
|
// adopted.
|
|
scoped_refptr<InitialRefCountIsOne> obj =
|
|
base::MakeRefCounted<InitialRefCountIsOne>();
|
|
EXPECT_DCHECK_DEATH(base::AdoptRef(obj.get()));
|
|
}
|