cee5ec390b
Weak containers are retraced if they are found through the stack using the conservative scanner, possibly resulting in a race with the concurrent marker. Bug: v8:12648 Change-Id: I0936a2953e3e2151cea4191f335a091b0e334e6b Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3474678 Auto-Submit: Michael Lippautz <mlippautz@chromium.org> Reviewed-by: Omer Katz <omerkatz@chromium.org> Commit-Queue: Omer Katz <omerkatz@chromium.org> Cr-Commit-Position: refs/heads/main@{#79184}
206 lines
6.5 KiB
C++
206 lines
6.5 KiB
C++
// 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 <atomic>
|
|
|
|
#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<Marker>(
|
|
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 <typename T>
|
|
constexpr size_t SizeOf() {
|
|
return RoundUp<kAllocationGranularity>(sizeof(T) + sizeof(HeapObjectHeader));
|
|
}
|
|
|
|
class TraceableGCed : public GarbageCollected<TraceableGCed> {
|
|
public:
|
|
void Trace(cppgc::Visitor*) const {
|
|
reinterpret_cast<std::atomic<size_t>*>(&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<NonTraceableGCed> {
|
|
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 <typename T>
|
|
V8_NOINLINE T access(volatile const T& t) {
|
|
return t;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
} // namespace internal
|
|
|
|
template <>
|
|
struct TraceTrait<internal::TraceableGCed>
|
|
: public internal::TraceTraitBase<internal::TraceableGCed> {
|
|
static TraceDescriptor GetWeakTraceDescriptor(const void* self) {
|
|
return {self, Trace};
|
|
}
|
|
};
|
|
|
|
template <>
|
|
struct TraceTrait<internal::NonTraceableGCed>
|
|
: public internal::TraceTraitBase<internal::NonTraceableGCed> {
|
|
static TraceDescriptor GetWeakTraceDescriptor(const void* self) {
|
|
return {self, nullptr};
|
|
}
|
|
};
|
|
|
|
namespace internal {
|
|
|
|
TEST_F(WeakContainerTest, TraceableGCedTraced) {
|
|
TraceableGCed* obj =
|
|
MakeGarbageCollected<TraceableGCed>(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<TraceableGCed>(), GetMarkedBytes());
|
|
access(obj);
|
|
}
|
|
|
|
TEST_F(WeakContainerTest, NonTraceableGCedNotTraced) {
|
|
NonTraceableGCed* obj =
|
|
MakeGarbageCollected<NonTraceableGCed>(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<NonTraceableGCed>(), GetMarkedBytes());
|
|
access(obj);
|
|
}
|
|
|
|
TEST_F(WeakContainerTest, NonTraceableGCedNotTracedConservatively) {
|
|
NonTraceableGCed* obj =
|
|
MakeGarbageCollected<NonTraceableGCed>(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<NonTraceableGCed>(), GetMarkedBytes());
|
|
access(obj);
|
|
}
|
|
|
|
TEST_F(WeakContainerTest, PreciseGCTracesWeakContainerWhenTraced) {
|
|
TraceableGCed* obj =
|
|
MakeGarbageCollected<TraceableGCed>(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<TraceableGCed>(), GetMarkedBytes());
|
|
access(obj);
|
|
}
|
|
|
|
TEST_F(WeakContainerTest, ConservativeGCTracesWeakContainer) {
|
|
TraceableGCed* obj =
|
|
MakeGarbageCollected<TraceableGCed>(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<TraceableGCed>(), GetMarkedBytes());
|
|
access(obj);
|
|
}
|
|
|
|
TEST_F(WeakContainerTest, ConservativeGCTracesWeakContainerOnce) {
|
|
NonTraceableGCed* obj =
|
|
MakeGarbageCollected<NonTraceableGCed>(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<NonTraceableGCed>(), 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<TraceableGCed>(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<TraceableGCed>(), GetMarkedBytes());
|
|
EXPECT_EQ(obj, WeakCallback::obj);
|
|
}
|
|
|
|
} // namespace internal
|
|
} // namespace cppgc
|