| /* | 
 |  *  Copyright 2016 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 "rtc_base/ref_counted_object.h" | 
 |  | 
 | #include <memory> | 
 | #include <string> | 
 | #include <type_traits> | 
 | #include <utility> | 
 |  | 
 | #include "absl/strings/string_view.h" | 
 | #include "api/make_ref_counted.h" | 
 | #include "api/scoped_refptr.h" | 
 | #include "rtc_base/ref_count.h" | 
 | #include "test/gtest.h" | 
 |  | 
 | namespace rtc { | 
 |  | 
 | namespace { | 
 |  | 
 | class A { | 
 |  public: | 
 |   A() {} | 
 |  | 
 |   A(const A&) = delete; | 
 |   A& operator=(const A&) = delete; | 
 | }; | 
 |  | 
 | class RefClass : public RefCountInterface { | 
 |  public: | 
 |   RefClass() {} | 
 |  | 
 |  protected: | 
 |   ~RefClass() override {} | 
 | }; | 
 |  | 
 | class RefClassWithRvalue : public RefCountInterface { | 
 |  public: | 
 |   explicit RefClassWithRvalue(std::unique_ptr<A> a) : a_(std::move(a)) {} | 
 |  | 
 |  protected: | 
 |   ~RefClassWithRvalue() override {} | 
 |  | 
 |  public: | 
 |   std::unique_ptr<A> a_; | 
 | }; | 
 |  | 
 | class RefClassWithMixedValues : public RefCountInterface { | 
 |  public: | 
 |   RefClassWithMixedValues(std::unique_ptr<A> a, int b, absl::string_view c) | 
 |       : a_(std::move(a)), b_(b), c_(c) {} | 
 |  | 
 |  protected: | 
 |   ~RefClassWithMixedValues() override {} | 
 |  | 
 |  public: | 
 |   std::unique_ptr<A> a_; | 
 |   int b_; | 
 |   std::string c_; | 
 | }; | 
 |  | 
 | class Foo { | 
 |  public: | 
 |   Foo() {} | 
 |   Foo(int i, int j) : foo_(i + j) {} | 
 |   int foo_ = 0; | 
 | }; | 
 |  | 
 | class FooItf : public RefCountInterface { | 
 |  public: | 
 |   FooItf() {} | 
 |   FooItf(int i, int j) : foo_(i + j) {} | 
 |   int foo_ = 0; | 
 | }; | 
 |  | 
 | }  // namespace | 
 |  | 
 | TEST(RefCountedObject, HasOneRef) { | 
 |   scoped_refptr<RefCountedObject<RefClass>> aref( | 
 |       new RefCountedObject<RefClass>()); | 
 |   EXPECT_TRUE(aref->HasOneRef()); | 
 |   aref->AddRef(); | 
 |   EXPECT_FALSE(aref->HasOneRef()); | 
 |   EXPECT_EQ(aref->Release(), RefCountReleaseStatus::kOtherRefsRemained); | 
 |   EXPECT_TRUE(aref->HasOneRef()); | 
 | } | 
 |  | 
 | TEST(RefCountedObject, SupportRValuesInCtor) { | 
 |   std::unique_ptr<A> a(new A()); | 
 |   scoped_refptr<RefClassWithRvalue> ref( | 
 |       new RefCountedObject<RefClassWithRvalue>(std::move(a))); | 
 |   EXPECT_TRUE(ref->a_.get() != nullptr); | 
 |   EXPECT_TRUE(a.get() == nullptr); | 
 | } | 
 |  | 
 | TEST(RefCountedObject, SupportMixedTypesInCtor) { | 
 |   std::unique_ptr<A> a(new A()); | 
 |   int b = 9; | 
 |   std::string c = "hello"; | 
 |   scoped_refptr<RefClassWithMixedValues> ref( | 
 |       new RefCountedObject<RefClassWithMixedValues>(std::move(a), b, c)); | 
 |   EXPECT_TRUE(ref->a_.get() != nullptr); | 
 |   EXPECT_TRUE(a.get() == nullptr); | 
 |   EXPECT_EQ(b, ref->b_); | 
 |   EXPECT_EQ(c, ref->c_); | 
 | } | 
 |  | 
 | TEST(FinalRefCountedObject, CanWrapIntoScopedRefptr) { | 
 |   using WrappedTyped = FinalRefCountedObject<A>; | 
 |   static_assert(!std::is_polymorphic<WrappedTyped>::value, ""); | 
 |   scoped_refptr<WrappedTyped> ref(new WrappedTyped()); | 
 |   EXPECT_TRUE(ref.get()); | 
 |   EXPECT_TRUE(ref->HasOneRef()); | 
 |   // Test reference counter is updated on some simple operations. | 
 |   scoped_refptr<WrappedTyped> ref2 = ref; | 
 |   EXPECT_FALSE(ref->HasOneRef()); | 
 |   EXPECT_FALSE(ref2->HasOneRef()); | 
 |  | 
 |   ref = nullptr; | 
 |   EXPECT_TRUE(ref2->HasOneRef()); | 
 | } | 
 |  | 
 | TEST(FinalRefCountedObject, CanCreateFromMovedType) { | 
 |   class MoveOnly { | 
 |    public: | 
 |     MoveOnly(int a) : a_(a) {} | 
 |     MoveOnly(MoveOnly&&) = default; | 
 |  | 
 |     int a() { return a_; } | 
 |  | 
 |    private: | 
 |     int a_; | 
 |   }; | 
 |   MoveOnly foo(5); | 
 |   auto ref = make_ref_counted<MoveOnly>(std::move(foo)); | 
 |   EXPECT_EQ(ref->a(), 5); | 
 | } | 
 |  | 
 | // This test is mostly a compile-time test for scoped_refptr compatibility. | 
 | TEST(RefCounted, SmartPointers) { | 
 |   // Sanity compile-time tests. FooItf is virtual, Foo is not, FooItf inherits | 
 |   // from RefCountInterface, Foo does not. | 
 |   static_assert(std::is_base_of<RefCountInterface, FooItf>::value, ""); | 
 |   static_assert(!std::is_base_of<RefCountInterface, Foo>::value, ""); | 
 |   static_assert(std::is_polymorphic<FooItf>::value, ""); | 
 |   static_assert(!std::is_polymorphic<Foo>::value, ""); | 
 |  | 
 |   { | 
 |     // Test with FooItf, a class that inherits from RefCountInterface. | 
 |     // Check that we get a valid FooItf reference counted object. | 
 |     auto p = make_ref_counted<FooItf>(2, 3); | 
 |     EXPECT_NE(p.get(), nullptr); | 
 |     EXPECT_EQ(p->foo_, 5);  // the FooItf ctor just stores 2+3 in foo_. | 
 |  | 
 |     // Declaring what should result in the same type as `p` is of. | 
 |     scoped_refptr<FooItf> p2 = p; | 
 |   } | 
 |  | 
 |   { | 
 |     // Same for `Foo` | 
 |     auto p = make_ref_counted<Foo>(2, 3); | 
 |     EXPECT_NE(p.get(), nullptr); | 
 |     EXPECT_EQ(p->foo_, 5); | 
 |     scoped_refptr<FinalRefCountedObject<Foo>> p2 = p; | 
 |   } | 
 | } | 
 |  | 
 | }  // namespace rtc |