// Copyright 2020 the V8 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. #include #include "include/cppgc/allocation.h" #include "src/base/macros.h" #include "src/heap/cppgc/marker.h" #include "src/heap/cppgc/marking-visitor.h" #include "src/heap/cppgc/stats-collector.h" #include "test/unittests/heap/cppgc/tests.h" #include "testing/gtest/include/gtest/gtest.h" namespace cppgc { namespace internal { namespace { class WeakContainerTest : public testing::TestWithHeap { public: using Config = Marker::MarkingConfig; void StartMarking() { CHECK_EQ(0u, Heap::From(GetHeap())->AsBase().stats_collector()->marked_bytes()); Config config = {Config::CollectionType::kMajor, Config::StackState::kNoHeapPointers, Config::MarkingType::kIncremental}; GetMarkerRef() = std::make_unique( Heap::From(GetHeap())->AsBase(), GetPlatformHandle().get(), config); GetMarkerRef()->StartMarking(); } void FinishMarking(Config::StackState stack_state) { GetMarkerRef()->FinishMarking(stack_state); marked_bytes_ = Heap::From(GetHeap())->AsBase().stats_collector()->marked_bytes(); GetMarkerRef().reset(); Heap::From(GetHeap())->stats_collector()->NotifySweepingCompleted(); } size_t GetMarkedBytes() const { return marked_bytes_; } private: size_t marked_bytes_ = 0; }; template constexpr size_t SizeOf() { return RoundUp(sizeof(T) + sizeof(HeapObjectHeader)); } class TraceableGCed : public GarbageCollected { public: void Trace(cppgc::Visitor*) const { reinterpret_cast*>(&n_trace_calls) ->fetch_add(1, std::memory_order_relaxed); } static size_t n_trace_calls; }; size_t TraceableGCed::n_trace_calls = 0u; class NonTraceableGCed : public GarbageCollected { public: void Trace(cppgc::Visitor*) const { n_trace_calls++; } static size_t n_trace_calls; }; size_t NonTraceableGCed::n_trace_calls = 0u; void EmptyWeakCallback(const LivenessBroker&, const void*) {} template V8_NOINLINE T access(volatile const T& t) { return t; } } // namespace } // namespace internal template <> struct TraceTrait : public internal::TraceTraitBase { static TraceDescriptor GetWeakTraceDescriptor(const void* self) { return {self, Trace}; } }; template <> struct TraceTrait : public internal::TraceTraitBase { static TraceDescriptor GetWeakTraceDescriptor(const void* self) { return {self, nullptr}; } }; namespace internal { TEST_F(WeakContainerTest, TraceableGCedTraced) { TraceableGCed* obj = MakeGarbageCollected(GetAllocationHandle()); TraceableGCed::n_trace_calls = 0u; StartMarking(); GetMarkerRef()->Visitor().TraceWeakContainer(obj, EmptyWeakCallback, nullptr); FinishMarking(Config::StackState::kNoHeapPointers); EXPECT_NE(0u, TraceableGCed::n_trace_calls); EXPECT_EQ(SizeOf(), GetMarkedBytes()); access(obj); } TEST_F(WeakContainerTest, NonTraceableGCedNotTraced) { NonTraceableGCed* obj = MakeGarbageCollected(GetAllocationHandle()); NonTraceableGCed::n_trace_calls = 0u; StartMarking(); GetMarkerRef()->Visitor().TraceWeakContainer(obj, EmptyWeakCallback, nullptr); FinishMarking(Config::StackState::kNoHeapPointers); EXPECT_EQ(0u, NonTraceableGCed::n_trace_calls); EXPECT_EQ(SizeOf(), GetMarkedBytes()); access(obj); } TEST_F(WeakContainerTest, NonTraceableGCedNotTracedConservatively) { NonTraceableGCed* obj = MakeGarbageCollected(GetAllocationHandle()); NonTraceableGCed::n_trace_calls = 0u; StartMarking(); GetMarkerRef()->Visitor().TraceWeakContainer(obj, EmptyWeakCallback, nullptr); FinishMarking(Config::StackState::kMayContainHeapPointers); EXPECT_NE(0u, NonTraceableGCed::n_trace_calls); EXPECT_EQ(SizeOf(), GetMarkedBytes()); access(obj); } TEST_F(WeakContainerTest, PreciseGCTracesWeakContainerWhenTraced) { TraceableGCed* obj = MakeGarbageCollected(GetAllocationHandle()); TraceableGCed::n_trace_calls = 0u; StartMarking(); GetMarkerRef()->Visitor().TraceWeakContainer(obj, EmptyWeakCallback, nullptr); FinishMarking(Config::StackState::kNoHeapPointers); EXPECT_EQ(1u, TraceableGCed::n_trace_calls); EXPECT_EQ(SizeOf(), GetMarkedBytes()); access(obj); } TEST_F(WeakContainerTest, ConservativeGCTracesWeakContainer) { TraceableGCed* obj = MakeGarbageCollected(GetAllocationHandle()); TraceableGCed::n_trace_calls = 0u; StartMarking(); GetMarkerRef()->Visitor().TraceWeakContainer(obj, EmptyWeakCallback, nullptr); FinishMarking(Config::StackState::kMayContainHeapPointers); EXPECT_EQ(2u, TraceableGCed::n_trace_calls); EXPECT_EQ(SizeOf(), GetMarkedBytes()); access(obj); } TEST_F(WeakContainerTest, ConservativeGCTracesWeakContainerOnce) { NonTraceableGCed* obj = MakeGarbageCollected(GetAllocationHandle()); NonTraceableGCed* copy_obj = obj; USE(copy_obj); NonTraceableGCed* another_copy_obj = obj; USE(another_copy_obj); NonTraceableGCed::n_trace_calls = 0u; StartMarking(); GetMarkerRef()->Visitor().TraceWeakContainer(obj, EmptyWeakCallback, nullptr); FinishMarking(Config::StackState::kMayContainHeapPointers); EXPECT_EQ(1u, NonTraceableGCed::n_trace_calls); EXPECT_EQ(SizeOf(), GetMarkedBytes()); access(obj); } namespace { struct WeakCallback { static void callback(const LivenessBroker&, const void* data) { n_callback_called++; obj = data; } static size_t n_callback_called; static const void* obj; }; size_t WeakCallback::n_callback_called = 0u; const void* WeakCallback::obj = nullptr; } // namespace TEST_F(WeakContainerTest, WeakContainerWeakCallbackCalled) { TraceableGCed* obj = MakeGarbageCollected(GetAllocationHandle()); WeakCallback::n_callback_called = 0u; WeakCallback::obj = nullptr; StartMarking(); GetMarkerRef()->Visitor().TraceWeakContainer(obj, WeakCallback::callback, obj); FinishMarking(Config::StackState::kMayContainHeapPointers); EXPECT_NE(0u, WeakCallback::n_callback_called); EXPECT_EQ(SizeOf(), GetMarkedBytes()); EXPECT_EQ(obj, WeakCallback::obj); } } // namespace internal } // namespace cppgc