| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523 | /* * * Copyright 2017 gRPC authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * *     http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * */#include "src/core/lib/gprpp/ref_counted_ptr.h"#include <gtest/gtest.h>#include <grpc/support/log.h>#include "src/core/lib/gprpp/dual_ref_counted.h"#include "src/core/lib/gprpp/memory.h"#include "src/core/lib/gprpp/ref_counted.h"#include "test/core/util/test_config.h"namespace grpc_core {namespace testing {namespace {//// RefCountedPtr<> tests//class Foo : public RefCounted<Foo> { public:  Foo() : value_(0) {}  explicit Foo(int value) : value_(value) {}  int value() const { return value_; } private:  int value_;};TEST(RefCountedPtr, DefaultConstructor) { RefCountedPtr<Foo> foo; }TEST(RefCountedPtr, ExplicitConstructorEmpty) {  RefCountedPtr<Foo> foo(nullptr);}TEST(RefCountedPtr, ExplicitConstructor) { RefCountedPtr<Foo> foo(new Foo()); }TEST(RefCountedPtr, MoveConstructor) {  RefCountedPtr<Foo> foo(new Foo());  RefCountedPtr<Foo> foo2(std::move(foo));  EXPECT_EQ(nullptr, foo.get());  // NOLINT  EXPECT_NE(nullptr, foo2.get());}TEST(RefCountedPtr, MoveAssignment) {  RefCountedPtr<Foo> foo(new Foo());  RefCountedPtr<Foo> foo2 = std::move(foo);  EXPECT_EQ(nullptr, foo.get());  // NOLINT  EXPECT_NE(nullptr, foo2.get());}TEST(RefCountedPtr, CopyConstructor) {  RefCountedPtr<Foo> foo(new Foo());  RefCountedPtr<Foo> foo2(foo);  EXPECT_NE(nullptr, foo.get());  EXPECT_EQ(foo.get(), foo2.get());}TEST(RefCountedPtr, CopyAssignment) {  RefCountedPtr<Foo> foo(new Foo());  RefCountedPtr<Foo> foo2 = foo;  EXPECT_NE(nullptr, foo.get());  EXPECT_EQ(foo.get(), foo2.get());}TEST(RefCountedPtr, CopyAssignmentWhenEmpty) {  RefCountedPtr<Foo> foo;  RefCountedPtr<Foo> foo2;  foo2 = foo;  EXPECT_EQ(nullptr, foo.get());  EXPECT_EQ(nullptr, foo2.get());}TEST(RefCountedPtr, CopyAssignmentToSelf) {  RefCountedPtr<Foo> foo(new Foo());  foo = *&foo;  // The "*&" avoids warnings from LLVM -Wself-assign.}TEST(RefCountedPtr, EnclosedScope) {  RefCountedPtr<Foo> foo(new Foo());  {    RefCountedPtr<Foo> foo2(std::move(foo));    EXPECT_EQ(nullptr, foo.get());    EXPECT_NE(nullptr, foo2.get());  }  EXPECT_EQ(nullptr, foo.get());}TEST(RefCountedPtr, ResetFromNullToNonNull) {  RefCountedPtr<Foo> foo;  EXPECT_EQ(nullptr, foo.get());  foo.reset(new Foo());  EXPECT_NE(nullptr, foo.get());}TEST(RefCountedPtr, ResetFromNonNullToNonNull) {  RefCountedPtr<Foo> foo(new Foo());  EXPECT_NE(nullptr, foo.get());  Foo* original = foo.get();  foo.reset(new Foo());  EXPECT_NE(nullptr, foo.get());  EXPECT_NE(original, foo.get());}TEST(RefCountedPtr, ResetFromNonNullToNull) {  RefCountedPtr<Foo> foo(new Foo());  EXPECT_NE(nullptr, foo.get());  foo.reset();  EXPECT_EQ(nullptr, foo.get());}TEST(RefCountedPtr, ResetFromNullToNull) {  RefCountedPtr<Foo> foo;  EXPECT_EQ(nullptr, foo.get());  foo.reset();  EXPECT_EQ(nullptr, foo.get());}TEST(RefCountedPtr, DerefernceOperators) {  RefCountedPtr<Foo> foo(new Foo());  foo->value();  Foo& foo_ref = *foo;  foo_ref.value();}TEST(RefCountedPtr, EqualityOperators) {  RefCountedPtr<Foo> foo(new Foo());  RefCountedPtr<Foo> bar = foo;  RefCountedPtr<Foo> empty;  // Test equality between RefCountedPtrs.  EXPECT_EQ(foo, bar);  EXPECT_NE(foo, empty);  // Test equality with bare pointers.  EXPECT_EQ(foo, foo.get());  EXPECT_EQ(empty, nullptr);  EXPECT_NE(foo, nullptr);}TEST(RefCountedPtr, Swap) {  Foo* foo = new Foo();  Foo* bar = new Foo();  RefCountedPtr<Foo> ptr1(foo);  RefCountedPtr<Foo> ptr2(bar);  ptr1.swap(ptr2);  EXPECT_EQ(foo, ptr2.get());  EXPECT_EQ(bar, ptr1.get());  RefCountedPtr<Foo> ptr3;  ptr3.swap(ptr2);  EXPECT_EQ(nullptr, ptr2.get());  EXPECT_EQ(foo, ptr3.get());}TEST(MakeRefCounted, NoArgs) {  RefCountedPtr<Foo> foo = MakeRefCounted<Foo>();  EXPECT_EQ(0, foo->value());}TEST(MakeRefCounted, Args) {  RefCountedPtr<Foo> foo = MakeRefCounted<Foo>(3);  EXPECT_EQ(3, foo->value());}TraceFlag foo_tracer(true, "foo");class FooWithTracing : public RefCounted<FooWithTracing> { public:  FooWithTracing() : RefCounted(&foo_tracer) {}};TEST(RefCountedPtr, RefCountedWithTracing) {  RefCountedPtr<FooWithTracing> foo(new FooWithTracing());  RefCountedPtr<FooWithTracing> foo2 = foo->Ref(DEBUG_LOCATION, "foo");  foo2.release();  foo->Unref(DEBUG_LOCATION, "foo");}class BaseClass : public RefCounted<BaseClass> { public:  BaseClass() {}};class Subclass : public BaseClass { public:  Subclass() {}};TEST(RefCountedPtr, ConstructFromSubclass) {  RefCountedPtr<BaseClass> p(new Subclass());}TEST(RefCountedPtr, CopyAssignFromSubclass) {  RefCountedPtr<BaseClass> b;  EXPECT_EQ(nullptr, b.get());  RefCountedPtr<Subclass> s = MakeRefCounted<Subclass>();  b = s;  EXPECT_NE(nullptr, b.get());}TEST(RefCountedPtr, MoveAssignFromSubclass) {  RefCountedPtr<BaseClass> b;  EXPECT_EQ(nullptr, b.get());  RefCountedPtr<Subclass> s = MakeRefCounted<Subclass>();  b = std::move(s);  EXPECT_NE(nullptr, b.get());}TEST(RefCountedPtr, ResetFromSubclass) {  RefCountedPtr<BaseClass> b;  EXPECT_EQ(nullptr, b.get());  b.reset(new Subclass());  EXPECT_NE(nullptr, b.get());}TEST(RefCountedPtr, EqualityWithSubclass) {  Subclass* s = new Subclass();  RefCountedPtr<BaseClass> b(s);  EXPECT_EQ(b, s);}void FunctionTakingBaseClass(RefCountedPtr<BaseClass> p) {  p.reset();  // To appease clang-tidy.}TEST(RefCountedPtr, CanPassSubclassToFunctionExpectingBaseClass) {  RefCountedPtr<Subclass> p = MakeRefCounted<Subclass>();  FunctionTakingBaseClass(p);}void FunctionTakingSubclass(RefCountedPtr<Subclass> p) {  p.reset();  // To appease clang-tidy.}TEST(RefCountedPtr, CanPassSubclassToFunctionExpectingSubclass) {  RefCountedPtr<Subclass> p = MakeRefCounted<Subclass>();  FunctionTakingSubclass(p);}//// WeakRefCountedPtr<> tests//class Bar : public DualRefCounted<Bar> { public:  Bar() : value_(0) {}  explicit Bar(int value) : value_(value) {}  ~Bar() { GPR_ASSERT(shutting_down_); }  void Orphan() override { shutting_down_ = true; }  int value() const { return value_; } private:  int value_;  bool shutting_down_ = false;};TEST(WeakRefCountedPtr, DefaultConstructor) { WeakRefCountedPtr<Bar> bar; }TEST(WeakRefCountedPtr, ExplicitConstructorEmpty) {  WeakRefCountedPtr<Bar> bar(nullptr);}TEST(WeakRefCountedPtr, ExplicitConstructor) {  RefCountedPtr<Bar> bar_strong(new Bar());  bar_strong->WeakRef().release();  WeakRefCountedPtr<Bar> bar(bar_strong.get());}TEST(WeakRefCountedPtr, MoveConstructor) {  RefCountedPtr<Bar> bar_strong(new Bar());  WeakRefCountedPtr<Bar> bar = bar_strong->WeakRef();  WeakRefCountedPtr<Bar> bar2(std::move(bar));  EXPECT_EQ(nullptr, bar.get());  // NOLINT  EXPECT_NE(nullptr, bar2.get());}TEST(WeakRefCountedPtr, MoveAssignment) {  RefCountedPtr<Bar> bar_strong(new Bar());  WeakRefCountedPtr<Bar> bar = bar_strong->WeakRef();  WeakRefCountedPtr<Bar> bar2 = std::move(bar);  EXPECT_EQ(nullptr, bar.get());  // NOLINT  EXPECT_NE(nullptr, bar2.get());}TEST(WeakRefCountedPtr, CopyConstructor) {  RefCountedPtr<Bar> bar_strong(new Bar());  WeakRefCountedPtr<Bar> bar = bar_strong->WeakRef();  WeakRefCountedPtr<Bar> bar2(bar);  EXPECT_NE(nullptr, bar.get());  EXPECT_EQ(bar.get(), bar2.get());}TEST(WeakRefCountedPtr, CopyAssignment) {  RefCountedPtr<Bar> bar_strong(new Bar());  WeakRefCountedPtr<Bar> bar = bar_strong->WeakRef();  WeakRefCountedPtr<Bar> bar2 = bar;  EXPECT_NE(nullptr, bar.get());  EXPECT_EQ(bar.get(), bar2.get());}TEST(WeakRefCountedPtr, CopyAssignmentWhenEmpty) {  WeakRefCountedPtr<Bar> bar;  WeakRefCountedPtr<Bar> bar2;  bar2 = bar;  EXPECT_EQ(nullptr, bar.get());  EXPECT_EQ(nullptr, bar2.get());}TEST(WeakRefCountedPtr, CopyAssignmentToSelf) {  RefCountedPtr<Bar> bar_strong(new Bar());  WeakRefCountedPtr<Bar> bar = bar_strong->WeakRef();  bar = *&bar;  // The "*&" avoids warnings from LLVM -Wself-assign.}TEST(WeakRefCountedPtr, EnclosedScope) {  RefCountedPtr<Bar> bar_strong(new Bar());  WeakRefCountedPtr<Bar> bar = bar_strong->WeakRef();  {    WeakRefCountedPtr<Bar> bar2(std::move(bar));    EXPECT_EQ(nullptr, bar.get());    EXPECT_NE(nullptr, bar2.get());  }  EXPECT_EQ(nullptr, bar.get());}TEST(WeakRefCountedPtr, ResetFromNullToNonNull) {  RefCountedPtr<Bar> bar_strong(new Bar());  WeakRefCountedPtr<Bar> bar;  EXPECT_EQ(nullptr, bar.get());  bar_strong->WeakRef().release();  bar.reset(bar_strong.get());  EXPECT_NE(nullptr, bar.get());}TEST(WeakRefCountedPtr, ResetFromNonNullToNonNull) {  RefCountedPtr<Bar> bar_strong(new Bar());  RefCountedPtr<Bar> bar2_strong(new Bar());  WeakRefCountedPtr<Bar> bar = bar_strong->WeakRef();  EXPECT_NE(nullptr, bar.get());  bar2_strong->WeakRef().release();  bar.reset(bar2_strong.get());  EXPECT_NE(nullptr, bar.get());  EXPECT_NE(bar_strong.get(), bar.get());}TEST(WeakRefCountedPtr, ResetFromNonNullToNull) {  RefCountedPtr<Bar> bar_strong(new Bar());  WeakRefCountedPtr<Bar> bar = bar_strong->WeakRef();  EXPECT_NE(nullptr, bar.get());  bar.reset();  EXPECT_EQ(nullptr, bar.get());}TEST(WeakRefCountedPtr, ResetFromNullToNull) {  WeakRefCountedPtr<Bar> bar;  EXPECT_EQ(nullptr, bar.get());  bar.reset();  EXPECT_EQ(nullptr, bar.get());}TEST(WeakRefCountedPtr, DerefernceOperators) {  RefCountedPtr<Bar> bar_strong(new Bar());  WeakRefCountedPtr<Bar> bar = bar_strong->WeakRef();  bar->value();  Bar& bar_ref = *bar;  bar_ref.value();}TEST(WeakRefCountedPtr, EqualityOperators) {  RefCountedPtr<Bar> bar_strong(new Bar());  WeakRefCountedPtr<Bar> bar = bar_strong->WeakRef();  WeakRefCountedPtr<Bar> bar2 = bar;  WeakRefCountedPtr<Bar> empty;  // Test equality between RefCountedPtrs.  EXPECT_EQ(bar, bar2);  EXPECT_NE(bar, empty);  // Test equality with bare pointers.  EXPECT_EQ(bar, bar.get());  EXPECT_EQ(empty, nullptr);  EXPECT_NE(bar, nullptr);}TEST(WeakRefCountedPtr, Swap) {  RefCountedPtr<Bar> bar_strong(new Bar());  RefCountedPtr<Bar> bar2_strong(new Bar());  WeakRefCountedPtr<Bar> bar = bar_strong->WeakRef();  WeakRefCountedPtr<Bar> bar2 = bar2_strong->WeakRef();  bar.swap(bar2);  EXPECT_EQ(bar_strong.get(), bar2.get());  EXPECT_EQ(bar2_strong.get(), bar.get());  WeakRefCountedPtr<Bar> bar3;  bar3.swap(bar2);  EXPECT_EQ(nullptr, bar2.get());  EXPECT_EQ(bar_strong.get(), bar3.get());}TraceFlag bar_tracer(true, "bar");class BarWithTracing : public DualRefCounted<BarWithTracing> { public:  BarWithTracing() : DualRefCounted(&bar_tracer) {}  ~BarWithTracing() { GPR_ASSERT(shutting_down_); }  void Orphan() override { shutting_down_ = true; } private:  bool shutting_down_ = false;};TEST(WeakRefCountedPtr, RefCountedWithTracing) {  RefCountedPtr<BarWithTracing> bar_strong(new BarWithTracing());  WeakRefCountedPtr<BarWithTracing> bar = bar_strong->WeakRef();  WeakRefCountedPtr<BarWithTracing> bar2 = bar->WeakRef(DEBUG_LOCATION, "bar");  bar2.release();  bar->WeakUnref(DEBUG_LOCATION, "bar");}class WeakBaseClass : public DualRefCounted<WeakBaseClass> { public:  WeakBaseClass() {}  ~WeakBaseClass() { GPR_ASSERT(shutting_down_); }  void Orphan() override { shutting_down_ = true; } private:  bool shutting_down_ = false;};class WeakSubclass : public WeakBaseClass { public:  WeakSubclass() {}};TEST(WeakRefCountedPtr, ConstructFromWeakSubclass) {  RefCountedPtr<WeakSubclass> strong(new WeakSubclass());  WeakRefCountedPtr<WeakBaseClass> p(strong->WeakRef().release());}TEST(WeakRefCountedPtr, CopyAssignFromWeakSubclass) {  RefCountedPtr<WeakSubclass> strong(new WeakSubclass());  WeakRefCountedPtr<WeakBaseClass> b;  EXPECT_EQ(nullptr, b.get());  WeakRefCountedPtr<WeakSubclass> s = strong->WeakRef();  b = s;  EXPECT_NE(nullptr, b.get());}TEST(WeakRefCountedPtr, MoveAssignFromWeakSubclass) {  RefCountedPtr<WeakSubclass> strong(new WeakSubclass());  WeakRefCountedPtr<WeakBaseClass> b;  EXPECT_EQ(nullptr, b.get());  WeakRefCountedPtr<WeakSubclass> s = strong->WeakRef();  b = std::move(s);  EXPECT_NE(nullptr, b.get());}TEST(WeakRefCountedPtr, ResetFromWeakSubclass) {  RefCountedPtr<WeakSubclass> strong(new WeakSubclass());  WeakRefCountedPtr<WeakBaseClass> b;  EXPECT_EQ(nullptr, b.get());  b.reset(strong->WeakRef().release());  EXPECT_NE(nullptr, b.get());}TEST(WeakRefCountedPtr, EqualityWithWeakSubclass) {  RefCountedPtr<WeakSubclass> strong(new WeakSubclass());  WeakRefCountedPtr<WeakBaseClass> b = strong->WeakRef();  EXPECT_EQ(b, strong.get());}void FunctionTakingWeakBaseClass(WeakRefCountedPtr<WeakBaseClass> p) {  p.reset();  // To appease clang-tidy.}TEST(WeakRefCountedPtr, CanPassWeakSubclassToFunctionExpectingWeakBaseClass) {  RefCountedPtr<WeakSubclass> strong(new WeakSubclass());  WeakRefCountedPtr<WeakSubclass> p = strong->WeakRef();  FunctionTakingWeakBaseClass(p);}void FunctionTakingWeakSubclass(WeakRefCountedPtr<WeakSubclass> p) {  p.reset();  // To appease clang-tidy.}TEST(WeakRefCountedPtr, CanPassWeakSubclassToFunctionExpectingWeakSubclass) {  RefCountedPtr<WeakSubclass> strong(new WeakSubclass());  WeakRefCountedPtr<WeakSubclass> p = strong->WeakRef();  FunctionTakingWeakSubclass(p);}}  // namespace}  // namespace testing}  // namespace grpc_coreint main(int argc, char** argv) {  grpc::testing::TestEnvironment env(argc, argv);  ::testing::InitGoogleTest(&argc, argv);  return RUN_ALL_TESTS();}
 |