| // 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 <gtest/gtest.h> | 
 |  | 
 | namespace { | 
 |  | 
 | class SelfAssign : public base::RefCounted<SelfAssign> { | 
 |  protected: | 
 |   virtual ~SelfAssign() {} | 
 |  | 
 |  private: | 
 |   friend class base::RefCounted<SelfAssign>; | 
 | }; | 
 |  | 
 | class Derived : public SelfAssign { | 
 |  protected: | 
 |   ~Derived() override {} | 
 |  | 
 |  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; | 
 |  | 
 | }  // end namespace | 
 |  | 
 | TEST(RefCountedUnitTest, TestSelfAssignment) { | 
 |   SelfAssign* p = new SelfAssign; | 
 |   scoped_refptr<SelfAssign> 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<ScopedRefPtrToSelf>(); | 
 |   EXPECT_TRUE(ScopedRefPtrToSelf::was_destroyed()); | 
 | } | 
 |  | 
 | TEST(RefCountedUnitTest, BooleanTesting) { | 
 |   scoped_refptr<SelfAssign> p; | 
 |   EXPECT_FALSE(p); | 
 |   p = new SelfAssign; | 
 |   EXPECT_TRUE(p); | 
 | } | 
 |  | 
 | 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, 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, SelfMoveAssignment) { | 
 |   ScopedRefPtrCountBase::reset_count(); | 
 |  | 
 |   { | 
 |     ScopedRefPtrCountBase *raw = new ScopedRefPtrCountBase(); | 
 |     scoped_refptr<ScopedRefPtrCountBase> p(raw); | 
 |     EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count()); | 
 |     EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count()); | 
 |  | 
 |     p = p.Pass(); | 
 |     EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count()); | 
 |     EXPECT_EQ(0, ScopedRefPtrCountBase::destructor_count()); | 
 |     EXPECT_EQ(raw, p.get()); | 
 |  | 
 |     // p goes out of scope. | 
 |   } | 
 |   EXPECT_EQ(1, ScopedRefPtrCountBase::constructor_count()); | 
 |   EXPECT_EQ(1, ScopedRefPtrCountBase::destructor_count()); | 
 | } | 
 |  | 
 | 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 = p1.Pass(); | 
 |       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 = p2.Pass(); | 
 |       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 = p2.Pass(); | 
 |       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 = p1.Pass(); | 
 |       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 = p2.Pass(); | 
 |       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<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 = p2.Pass(); | 
 |       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(p1.Pass()); | 
 |       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(p1.Pass()); | 
 |       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()); | 
 | } | 
 |  |