// 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 "base/test/opaque_ref_counted.h" #include "testing/gtest/include/gtest/gtest.h" namespace { class SelfAssign : public base::RefCounted { protected: virtual ~SelfAssign() {} private: friend class base::RefCounted; }; class Derived : public SelfAssign { protected: ~Derived() override {} private: friend class base::RefCounted; }; class CheckDerivedMemberAccess : public scoped_refptr { 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 { public: ScopedRefPtrToSelf() : self_ptr_(this) {} static bool was_destroyed() { return was_destroyed_; } static void reset_was_destroyed() { was_destroyed_ = false; } scoped_refptr self_ptr_; private: friend class base::RefCounted; ~ScopedRefPtrToSelf() { was_destroyed_ = true; } static bool was_destroyed_; }; bool ScopedRefPtrToSelf::was_destroyed_ = false; class ScopedRefPtrCountBase : public base::RefCounted { 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; 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; static int constructor_count_; static int destructor_count_; }; int ScopedRefPtrCountDerived::constructor_count_ = 0; int ScopedRefPtrCountDerived::destructor_count_ = 0; } // end namespace TEST(RefCountedUnitTest, TestSelfAssignment) { SelfAssign* p = new SelfAssign; scoped_refptr var(p); 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(); EXPECT_TRUE(ScopedRefPtrToSelf::was_destroyed()); } TEST(RefCountedUnitTest, ScopedRefPtrToOpaque) { scoped_refptr p = base::MakeOpaqueRefCounted(); base::TestOpaqueRefCounted(p); scoped_refptr q; q = p; base::TestOpaqueRefCounted(p); base::TestOpaqueRefCounted(q); } TEST(RefCountedUnitTest, BooleanTesting) { scoped_refptr p; EXPECT_FALSE(p); p = new SelfAssign; EXPECT_TRUE(p); } TEST(RefCountedUnitTest, Equality) { scoped_refptr p1(new SelfAssign); scoped_refptr p2(new SelfAssign); EXPECT_EQ(p1, p1); EXPECT_EQ(p2, p2); EXPECT_NE(p1, p2); EXPECT_NE(p2, p1); } TEST(RefCountedUnitTest, ConvertibleEquality) { scoped_refptr p1(new Derived); scoped_refptr 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 p1(raw); EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count()); EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count()); { scoped_refptr 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 p1; EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count()); EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count()); { scoped_refptr 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 p1(raw); EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count()); EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count()); { scoped_refptr 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 p1(raw); EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count()); EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count()); { scoped_refptr 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 p1(raw1); EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count()); EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count()); { ScopedRefPtrCountBase *raw2 = new ScopedRefPtrCountBase(); scoped_refptr 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, MoveAssignmentDerived) { ScopedRefPtrCountBase::reset_count(); ScopedRefPtrCountDerived::reset_count(); { ScopedRefPtrCountBase *raw1 = new ScopedRefPtrCountBase(); scoped_refptr 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 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 p1(raw); EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count()); EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count()); { scoped_refptr 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 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 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()); }