// Copyright 2016 PDFium 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 "core/fxcrt/weak_ptr.h" #include <memory> #include <utility> #include "core/fxcrt/fx_memory.h" #include "testing/gtest/include/gtest/gtest.h" namespace fxcrt { namespace { class PseudoDeletable; using WeakTestPtr = WeakPtr<PseudoDeletable, ReleaseDeleter<PseudoDeletable>>; using UniqueTestPtr = std::unique_ptr<PseudoDeletable, ReleaseDeleter<PseudoDeletable>>; class PseudoDeletable { public: PseudoDeletable() : delete_count_(0) {} void Release() { ++delete_count_; next_.Reset(); } void SetNext(const WeakTestPtr& next) { next_ = next; } int delete_count() const { return delete_count_; } private: int delete_count_; WeakTestPtr next_; }; } // namespace TEST(WeakPtr, Null) { WeakTestPtr ptr1; EXPECT_FALSE(ptr1); WeakTestPtr ptr2; EXPECT_TRUE(ptr1 == ptr2); EXPECT_FALSE(ptr1 != ptr2); WeakTestPtr ptr3(ptr1); EXPECT_TRUE(ptr1 == ptr3); EXPECT_FALSE(ptr1 != ptr3); WeakTestPtr ptr4 = ptr1; EXPECT_TRUE(ptr1 == ptr4); EXPECT_FALSE(ptr1 != ptr4); } TEST(WeakPtr, NonNull) { PseudoDeletable thing; EXPECT_EQ(0, thing.delete_count()); { UniqueTestPtr unique(&thing); WeakTestPtr ptr1(std::move(unique)); EXPECT_TRUE(ptr1); EXPECT_EQ(&thing, ptr1.Get()); WeakTestPtr ptr2; EXPECT_FALSE(ptr1 == ptr2); EXPECT_TRUE(ptr1 != ptr2); { WeakTestPtr ptr3(ptr1); EXPECT_TRUE(ptr1 == ptr3); EXPECT_FALSE(ptr1 != ptr3); EXPECT_EQ(&thing, ptr3.Get()); { WeakTestPtr ptr4 = ptr1; EXPECT_TRUE(ptr1 == ptr4); EXPECT_FALSE(ptr1 != ptr4); EXPECT_EQ(&thing, ptr4.Get()); } } EXPECT_EQ(0, thing.delete_count()); } EXPECT_EQ(1, thing.delete_count()); } TEST(WeakPtr, ResetNull) { PseudoDeletable thing; { UniqueTestPtr unique(&thing); WeakTestPtr ptr1(std::move(unique)); WeakTestPtr ptr2 = ptr1; ptr1.Reset(); EXPECT_FALSE(ptr1); EXPECT_EQ(nullptr, ptr1.Get()); EXPECT_TRUE(ptr2); EXPECT_EQ(&thing, ptr2.Get()); EXPECT_FALSE(ptr1 == ptr2); EXPECT_TRUE(ptr1 != ptr2); EXPECT_EQ(0, thing.delete_count()); } EXPECT_EQ(1, thing.delete_count()); } TEST(WeakPtr, ResetNonNull) { PseudoDeletable thing1; PseudoDeletable thing2; { UniqueTestPtr unique1(&thing1); WeakTestPtr ptr1(std::move(unique1)); WeakTestPtr ptr2 = ptr1; UniqueTestPtr unique2(&thing2); ptr2.Reset(std::move(unique2)); EXPECT_TRUE(ptr1); EXPECT_EQ(&thing1, ptr1.Get()); EXPECT_TRUE(ptr2); EXPECT_EQ(&thing2, ptr2.Get()); EXPECT_FALSE(ptr1 == ptr2); EXPECT_TRUE(ptr1 != ptr2); EXPECT_EQ(0, thing1.delete_count()); EXPECT_EQ(0, thing2.delete_count()); } EXPECT_EQ(1, thing1.delete_count()); EXPECT_EQ(1, thing2.delete_count()); } TEST(WeakPtr, DeleteObject) { PseudoDeletable thing; { UniqueTestPtr unique(&thing); WeakTestPtr ptr1(std::move(unique)); WeakTestPtr ptr2 = ptr1; ptr1.DeleteObject(); EXPECT_FALSE(ptr1); EXPECT_EQ(nullptr, ptr1.Get()); EXPECT_FALSE(ptr2); EXPECT_EQ(nullptr, ptr2.Get()); EXPECT_FALSE(ptr1 == ptr2); EXPECT_TRUE(ptr1 != ptr2); EXPECT_EQ(1, thing.delete_count()); } EXPECT_EQ(1, thing.delete_count()); } TEST(WeakPtr, Cyclic) { PseudoDeletable thing1; PseudoDeletable thing2; { UniqueTestPtr unique1(&thing1); UniqueTestPtr unique2(&thing2); WeakTestPtr ptr1(std::move(unique1)); WeakTestPtr ptr2(std::move(unique2)); ptr1->SetNext(ptr2); ptr2->SetNext(ptr1); } // Leaks without explicit clear. EXPECT_EQ(0, thing1.delete_count()); EXPECT_EQ(0, thing2.delete_count()); } TEST(WeakPtr, CyclicDeleteObject) { PseudoDeletable thing1; PseudoDeletable thing2; { UniqueTestPtr unique1(&thing1); UniqueTestPtr unique2(&thing2); WeakTestPtr ptr1(std::move(unique1)); WeakTestPtr ptr2(std::move(unique2)); ptr1->SetNext(ptr2); ptr2->SetNext(ptr1); ptr1.DeleteObject(); EXPECT_EQ(1, thing1.delete_count()); EXPECT_EQ(0, thing2.delete_count()); } EXPECT_EQ(1, thing1.delete_count()); EXPECT_EQ(1, thing2.delete_count()); } } // namespace fxcrt