Commit ea818f07 authored by Michael Lippautz's avatar Michael Lippautz Committed by Commit Bot

cppgc: Fix testing APIs that enable garbage collection

The APIs require that the CppHeap is moved into a permanently detached
state that moves the heap out of a no-gc scope.

Bug: chromium:1056170
Change-Id: I1fc08451b3fdfaa4cfe58e6a1ddbe5dbed7efe5c
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2718146
Commit-Queue: Michael Lippautz <mlippautz@chromium.org>
Reviewed-by: 's avatarOmer Katz <omerkatz@chromium.org>
Cr-Commit-Position: refs/heads/master@{#73025}
parent db34c5a1
...@@ -23,13 +23,17 @@ namespace testing { ...@@ -23,13 +23,17 @@ namespace testing {
*/ */
class V8_EXPORT Heap final { class V8_EXPORT Heap final {
public: public:
explicit Heap(HeapHandle& heap_handle) : heap_handle_(heap_handle) {}
/** /**
* Atomically collects garbage on the C++ heap. * Atomically collects garbage on the C++ heap.
* *
* \param heap_handle The corresponding heap.
* \param stack_state The stack state to assume for the garbage collection. * \param stack_state The stack state to assume for the garbage collection.
*/ */
void CollectGarbage(HeapHandle& heap_handle, EmbedderStackState stack_state); void CollectGarbage(EmbedderStackState stack_state);
private:
HeapHandle& heap_handle_;
}; };
/** /**
......
...@@ -118,6 +118,13 @@ class V8_EXPORT CppHeap { ...@@ -118,6 +118,13 @@ class V8_EXPORT CppHeap {
cppgc::HeapStatistics CollectStatistics( cppgc::HeapStatistics CollectStatistics(
cppgc::HeapStatistics::DetailLevel detail_level); cppgc::HeapStatistics::DetailLevel detail_level);
/**
* Enables a detached mode that allows testing garbage collection using
* `cppgc::testing` APIs. Once used, the heap cannot be attached to an
* `Isolate` anymore.
*/
void EnableDetachedGarbageCollectionsForTesting();
private: private:
CppHeap() = default; CppHeap() = default;
......
...@@ -10,6 +10,7 @@ ...@@ -10,6 +10,7 @@
#include "include/cppgc/platform.h" #include "include/cppgc/platform.h"
#include "include/v8-platform.h" #include "include/v8-platform.h"
#include "include/v8.h" #include "include/v8.h"
#include "src/base/logging.h"
#include "src/base/macros.h" #include "src/base/macros.h"
#include "src/base/platform/time.h" #include "src/base/platform/time.h"
#include "src/execution/isolate.h" #include "src/execution/isolate.h"
...@@ -64,6 +65,11 @@ cppgc::HeapStatistics CppHeap::CollectStatistics( ...@@ -64,6 +65,11 @@ cppgc::HeapStatistics CppHeap::CollectStatistics(
detail_level); detail_level);
} }
void CppHeap::EnableDetachedGarbageCollectionsForTesting() {
return internal::CppHeap::From(this)
->EnableDetachedGarbageCollectionsForTesting();
}
void JSHeapConsistency::DijkstraMarkingBarrierSlow( void JSHeapConsistency::DijkstraMarkingBarrierSlow(
cppgc::HeapHandle& heap_handle, const TracedReferenceBase& ref) { cppgc::HeapHandle& heap_handle, const TracedReferenceBase& ref) {
auto& heap_base = cppgc::internal::HeapBase::From(heap_handle); auto& heap_base = cppgc::internal::HeapBase::From(heap_handle);
...@@ -221,6 +227,7 @@ void CppHeap::Terminate() { ...@@ -221,6 +227,7 @@ void CppHeap::Terminate() {
} }
void CppHeap::AttachIsolate(Isolate* isolate) { void CppHeap::AttachIsolate(Isolate* isolate) {
CHECK(!in_detached_testing_mode);
CHECK_NULL(isolate_); CHECK_NULL(isolate_);
isolate_ = isolate; isolate_ = isolate;
static_cast<CppgcPlatformAdapter*>(platform()) static_cast<CppgcPlatformAdapter*>(platform())
...@@ -390,5 +397,12 @@ void CppHeap::ReportBufferedAllocationSizeIfPossible() { ...@@ -390,5 +397,12 @@ void CppHeap::ReportBufferedAllocationSizeIfPossible() {
buffered_allocated_bytes_ = 0; buffered_allocated_bytes_ = 0;
} }
void CppHeap::EnableDetachedGarbageCollectionsForTesting() {
CHECK(!in_detached_testing_mode);
CHECK_NULL(isolate_);
no_gc_scope_--;
in_detached_testing_mode = true;
}
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8
...@@ -64,6 +64,8 @@ class V8_EXPORT_PRIVATE CppHeap final ...@@ -64,6 +64,8 @@ class V8_EXPORT_PRIVATE CppHeap final
void AllocatedObjectSizeDecreased(size_t) final; void AllocatedObjectSizeDecreased(size_t) final;
void ResetAllocatedObjectSize(size_t) final {} void ResetAllocatedObjectSize(size_t) final {}
void EnableDetachedGarbageCollectionsForTesting();
private: private:
void FinalizeIncrementalGarbageCollectionIfNeeded( void FinalizeIncrementalGarbageCollectionIfNeeded(
cppgc::Heap::StackState) final { cppgc::Heap::StackState) final {
...@@ -83,6 +85,8 @@ class V8_EXPORT_PRIVATE CppHeap final ...@@ -83,6 +85,8 @@ class V8_EXPORT_PRIVATE CppHeap final
int64_t buffered_allocated_bytes_ = 0; int64_t buffered_allocated_bytes_ = 0;
v8::WrapperDescriptor wrapper_descriptor_; v8::WrapperDescriptor wrapper_descriptor_;
bool in_detached_testing_mode = false;
}; };
} // namespace internal } // namespace internal
......
...@@ -153,7 +153,6 @@ void HeapBase::StandAloneGarbageCollectionForTesting( ...@@ -153,7 +153,6 @@ void HeapBase::StandAloneGarbageCollectionForTesting(
GarbageCollector::Config::CollectionType::kMajor, stack_state, GarbageCollector::Config::CollectionType::kMajor, stack_state,
GarbageCollector::Config::MarkingType::kAtomic, GarbageCollector::Config::MarkingType::kAtomic,
GarbageCollector::Config::SweepingType::kAtomic}; GarbageCollector::Config::SweepingType::kAtomic};
if (in_no_gc_scope()) return; if (in_no_gc_scope()) return;
if (!IsMarking()) { if (!IsMarking()) {
......
...@@ -10,9 +10,8 @@ ...@@ -10,9 +10,8 @@
namespace cppgc { namespace cppgc {
namespace testing { namespace testing {
void Heap::CollectGarbage(HeapHandle& heap_handle, void Heap::CollectGarbage(EmbedderStackState stack_state) {
EmbedderStackState stack_state) { auto& heap = internal::HeapBase::From(heap_handle_);
auto& heap = internal::HeapBase::From(heap_handle);
heap.StandAloneGarbageCollectionForTesting(stack_state); heap.StandAloneGarbageCollectionForTesting(stack_state);
} }
......
...@@ -6,6 +6,7 @@ ...@@ -6,6 +6,7 @@
#include "include/cppgc/garbage-collected.h" #include "include/cppgc/garbage-collected.h"
#include "include/cppgc/persistent.h" #include "include/cppgc/persistent.h"
#include "include/cppgc/platform.h" #include "include/cppgc/platform.h"
#include "include/cppgc/testing.h"
#include "include/v8-cppgc.h" #include "include/v8-cppgc.h"
#include "include/v8.h" #include "include/v8.h"
#include "src/api/api-inl.h" #include "src/api/api-inl.h"
...@@ -140,6 +141,7 @@ TEST_F(UnifiedHeapDetachedTest, AllocationBeforeConfigureHeap) { ...@@ -140,6 +141,7 @@ TEST_F(UnifiedHeapDetachedTest, AllocationBeforeConfigureHeap) {
cpp_heap.AsBase().sweeper().FinishIfRunning(); cpp_heap.AsBase().sweeper().FinishIfRunning();
EXPECT_TRUE(weak_holder); EXPECT_TRUE(weak_holder);
} }
USE(object);
{ {
js_heap.SetEmbedderStackStateForNextFinalization( js_heap.SetEmbedderStackStateForNextFinalization(
EmbedderHeapTracer::EmbedderStackState::kNoHeapPointers); EmbedderHeapTracer::EmbedderStackState::kNoHeapPointers);
...@@ -149,5 +151,27 @@ TEST_F(UnifiedHeapDetachedTest, AllocationBeforeConfigureHeap) { ...@@ -149,5 +151,27 @@ TEST_F(UnifiedHeapDetachedTest, AllocationBeforeConfigureHeap) {
} }
} }
TEST_F(UnifiedHeapDetachedTest, StandAloneCppGC) {
auto heap = v8::CppHeap::Create(
V8::GetCurrentPlatform(),
CppHeapCreateParams{{}, WrapperHelper::DefaultWrapperDescriptor()});
auto* object =
cppgc::MakeGarbageCollected<Wrappable>(heap->GetAllocationHandle());
cppgc::WeakPersistent<Wrappable> weak_holder{object};
heap->EnableDetachedGarbageCollectionsForTesting();
cppgc::testing::Heap testing_heap{heap->GetHeapHandle()};
{
testing_heap.CollectGarbage(
cppgc::EmbedderStackState::kMayContainHeapPointers);
EXPECT_TRUE(weak_holder);
}
USE(object);
{
testing_heap.CollectGarbage(cppgc::EmbedderStackState::kNoHeapPointers);
EXPECT_FALSE(weak_holder);
}
}
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment