global-handles: Fix ASAN fake stack handling
We previously assumed that a fake stack should be mapped back to a real stack based on fake-stack offsets. This is not correct: Fake and real stack are disjoint and both contain the corresponding slot values. For global handles this means that on-stack handles must be registered using their real stack frame base to be able to purge them occasionally based on the current stack address. When dealing with a slot though, the GC can just dereference the slot for a value, indeppendent of whether the slot is in a fake or real frame. Drive-by: Fix tests that do not want stack handles by creating handles on heap. Change-Id: I2c86c8e047bd0d48c24c2642b2b4dba284a93909 Bug: chromium:1139914 Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2507720 Commit-Queue: Michael Lippautz <mlippautz@chromium.org> Reviewed-by: Ulan Degenbaev <ulan@chromium.org> Cr-Commit-Position: refs/heads/master@{#70897}
This commit is contained in:
parent
1e6fed5f06
commit
aad7b7ff33
@ -5,6 +5,7 @@
|
|||||||
#include "src/handles/global-handles.h"
|
#include "src/handles/global-handles.h"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
#include <cstdint>
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
#include "src/api/api-inl.h"
|
#include "src/api/api-inl.h"
|
||||||
@ -749,13 +750,10 @@ class GlobalHandles::OnStackTracedNodeSpace final {
|
|||||||
void SetStackStart(void* stack_start) {
|
void SetStackStart(void* stack_start) {
|
||||||
CHECK(on_stack_nodes_.empty());
|
CHECK(on_stack_nodes_.empty());
|
||||||
stack_start_ =
|
stack_start_ =
|
||||||
GetStackAddressForSlot(reinterpret_cast<uintptr_t>(stack_start));
|
GetRealStackAddressForSlot(reinterpret_cast<uintptr_t>(stack_start));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsOnStack(uintptr_t slot) const {
|
V8_INLINE bool IsOnStack(uintptr_t slot) const;
|
||||||
const uintptr_t address = GetStackAddressForSlot(slot);
|
|
||||||
return stack_start_ >= address && address > GetCurrentStackPosition();
|
|
||||||
}
|
|
||||||
|
|
||||||
void Iterate(RootVisitor* v);
|
void Iterate(RootVisitor* v);
|
||||||
TracedNode* Acquire(Object value, uintptr_t address);
|
TracedNode* Acquire(Object value, uintptr_t address);
|
||||||
@ -772,39 +770,69 @@ class GlobalHandles::OnStackTracedNodeSpace final {
|
|||||||
GlobalHandles* global_handles;
|
GlobalHandles* global_handles;
|
||||||
};
|
};
|
||||||
|
|
||||||
uintptr_t GetStackAddressForSlot(uintptr_t slot) const;
|
// Returns the real stack frame if slot is part of a fake frame, and slot
|
||||||
|
// otherwise.
|
||||||
|
V8_INLINE uintptr_t GetRealStackAddressForSlot(uintptr_t slot) const;
|
||||||
|
|
||||||
// Keeps track of registered handles and their stack address. The data
|
// Keeps track of registered handles. The data structure is cleaned on
|
||||||
// structure is cleaned on iteration and when adding new references using the
|
// iteration and when adding new references using the current stack address.
|
||||||
// current stack address.
|
// Cleaning is based on current stack address and the key of the map which is
|
||||||
|
// slightly different for ASAN configs -- see below.
|
||||||
|
#ifdef V8_USE_ADDRESS_SANITIZER
|
||||||
|
// Mapping from stack slots or real stack frames to the corresponding nodes.
|
||||||
|
// In case a reference is part of a fake frame, we map it to the real stack
|
||||||
|
// frame base instead of the actual stack slot. The list keeps all nodes for
|
||||||
|
// a particular real frame.
|
||||||
|
std::map<uintptr_t, std::list<NodeEntry>> on_stack_nodes_;
|
||||||
|
#else // !V8_USE_ADDRESS_SANITIZER
|
||||||
|
// Mapping from stack slots to the corresponding nodes. We don't expect
|
||||||
|
// aliasing with overlapping lifetimes of nodes.
|
||||||
std::map<uintptr_t, NodeEntry> on_stack_nodes_;
|
std::map<uintptr_t, NodeEntry> on_stack_nodes_;
|
||||||
|
#endif // !V8_USE_ADDRESS_SANITIZER
|
||||||
|
|
||||||
uintptr_t stack_start_ = 0;
|
uintptr_t stack_start_ = 0;
|
||||||
GlobalHandles* global_handles_ = nullptr;
|
GlobalHandles* global_handles_ = nullptr;
|
||||||
size_t acquire_count_ = 0;
|
size_t acquire_count_ = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
uintptr_t GlobalHandles::OnStackTracedNodeSpace::GetStackAddressForSlot(
|
uintptr_t GlobalHandles::OnStackTracedNodeSpace::GetRealStackAddressForSlot(
|
||||||
uintptr_t slot) const {
|
uintptr_t slot) const {
|
||||||
#ifdef V8_USE_ADDRESS_SANITIZER
|
#ifdef V8_USE_ADDRESS_SANITIZER
|
||||||
void* fake_stack = __asan_get_current_fake_stack();
|
void* real_frame = __asan_addr_is_in_fake_stack(
|
||||||
if (fake_stack) {
|
__asan_get_current_fake_stack(), reinterpret_cast<void*>(slot), nullptr,
|
||||||
void* fake_frame_start;
|
nullptr);
|
||||||
void* real_frame = __asan_addr_is_in_fake_stack(
|
return real_frame ? reinterpret_cast<uintptr_t>(real_frame) : slot;
|
||||||
fake_stack, reinterpret_cast<void*>(slot), &fake_frame_start, nullptr);
|
|
||||||
if (real_frame) {
|
|
||||||
return reinterpret_cast<uintptr_t>(real_frame) +
|
|
||||||
(slot - reinterpret_cast<uintptr_t>(fake_frame_start));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif // V8_USE_ADDRESS_SANITIZER
|
#endif // V8_USE_ADDRESS_SANITIZER
|
||||||
return slot;
|
return slot;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool GlobalHandles::OnStackTracedNodeSpace::IsOnStack(uintptr_t slot) const {
|
||||||
|
#ifdef V8_USE_ADDRESS_SANITIZER
|
||||||
|
if (__asan_addr_is_in_fake_stack(__asan_get_current_fake_stack(),
|
||||||
|
reinterpret_cast<void*>(slot), nullptr,
|
||||||
|
nullptr)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
#endif // V8_USE_ADDRESS_SANITIZER
|
||||||
|
return stack_start_ >= slot && slot > GetCurrentStackPosition();
|
||||||
|
}
|
||||||
|
|
||||||
void GlobalHandles::OnStackTracedNodeSpace::NotifyEmptyEmbedderStack() {
|
void GlobalHandles::OnStackTracedNodeSpace::NotifyEmptyEmbedderStack() {
|
||||||
on_stack_nodes_.clear();
|
on_stack_nodes_.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
void GlobalHandles::OnStackTracedNodeSpace::Iterate(RootVisitor* v) {
|
void GlobalHandles::OnStackTracedNodeSpace::Iterate(RootVisitor* v) {
|
||||||
|
#ifdef V8_USE_ADDRESS_SANITIZER
|
||||||
|
for (auto& pair : on_stack_nodes_) {
|
||||||
|
for (auto& node_entry : pair.second) {
|
||||||
|
TracedNode& node = node_entry.node;
|
||||||
|
if (node.IsRetainer()) {
|
||||||
|
v->VisitRootPointer(Root::kGlobalHandles, "on-stack TracedReference",
|
||||||
|
node.location());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#else // !V8_USE_ADDRESS_SANITIZER
|
||||||
// Handles have been cleaned from the GC entry point which is higher up the
|
// Handles have been cleaned from the GC entry point which is higher up the
|
||||||
// stack.
|
// stack.
|
||||||
for (auto& pair : on_stack_nodes_) {
|
for (auto& pair : on_stack_nodes_) {
|
||||||
@ -814,6 +842,7 @@ void GlobalHandles::OnStackTracedNodeSpace::Iterate(RootVisitor* v) {
|
|||||||
node.location());
|
node.location());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#endif // !V8_USE_ADDRESS_SANITIZER
|
||||||
}
|
}
|
||||||
|
|
||||||
GlobalHandles::TracedNode* GlobalHandles::OnStackTracedNodeSpace::Acquire(
|
GlobalHandles::TracedNode* GlobalHandles::OnStackTracedNodeSpace::Acquire(
|
||||||
@ -828,8 +857,13 @@ GlobalHandles::TracedNode* GlobalHandles::OnStackTracedNodeSpace::Acquire(
|
|||||||
NodeEntry entry;
|
NodeEntry entry;
|
||||||
entry.node.Free(nullptr);
|
entry.node.Free(nullptr);
|
||||||
entry.global_handles = global_handles_;
|
entry.global_handles = global_handles_;
|
||||||
auto pair =
|
#ifdef V8_USE_ADDRESS_SANITIZER
|
||||||
on_stack_nodes_.insert({GetStackAddressForSlot(slot), std::move(entry)});
|
auto pair = on_stack_nodes_.insert({GetRealStackAddressForSlot(slot), {}});
|
||||||
|
pair.first->second.push_back(std::move(entry));
|
||||||
|
TracedNode* result = &(pair.first->second.back().node);
|
||||||
|
#else // !V8_USE_ADDRESS_SANITIZER
|
||||||
|
auto pair = on_stack_nodes_.insert(
|
||||||
|
{GetRealStackAddressForSlot(slot), std::move(entry)});
|
||||||
if (!pair.second) {
|
if (!pair.second) {
|
||||||
// Insertion failed because there already was an entry present for that
|
// Insertion failed because there already was an entry present for that
|
||||||
// stack address. This can happen because cleanup is conservative in which
|
// stack address. This can happen because cleanup is conservative in which
|
||||||
@ -838,6 +872,7 @@ GlobalHandles::TracedNode* GlobalHandles::OnStackTracedNodeSpace::Acquire(
|
|||||||
pair.first->second.node.Free(nullptr);
|
pair.first->second.node.Free(nullptr);
|
||||||
}
|
}
|
||||||
TracedNode* result = &(pair.first->second.node);
|
TracedNode* result = &(pair.first->second.node);
|
||||||
|
#endif // !V8_USE_ADDRESS_SANITIZER
|
||||||
result->Acquire(value);
|
result->Acquire(value);
|
||||||
result->set_is_on_stack(true);
|
result->set_is_on_stack(true);
|
||||||
return result;
|
return result;
|
||||||
|
@ -324,13 +324,13 @@ void TracedGlobalTest(v8::Isolate* isolate,
|
|||||||
v8::Local<v8::Context> context = v8::Context::New(isolate);
|
v8::Local<v8::Context> context = v8::Context::New(isolate);
|
||||||
v8::Context::Scope context_scope(context);
|
v8::Context::Scope context_scope(context);
|
||||||
|
|
||||||
v8::TracedGlobal<v8::Object> global;
|
auto global = std::make_unique<v8::TracedGlobal<v8::Object>>();
|
||||||
construct_function(isolate, context, &global);
|
construct_function(isolate, context, global.get());
|
||||||
CHECK(InCorrectGeneration(isolate, global));
|
CHECK(InCorrectGeneration(isolate, *global));
|
||||||
modifier_function(global);
|
modifier_function(*global);
|
||||||
gc_function();
|
gc_function();
|
||||||
CHECK_IMPLIES(survives == SurvivalMode::kSurvives, !global.IsEmpty());
|
CHECK_IMPLIES(survives == SurvivalMode::kSurvives, !global->IsEmpty());
|
||||||
CHECK_IMPLIES(survives == SurvivalMode::kDies, global.IsEmpty());
|
CHECK_IMPLIES(survives == SurvivalMode::kDies, global->IsEmpty());
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
@ -371,33 +371,33 @@ TEST(TracedGlobalCopyWithDestructor) {
|
|||||||
i::GlobalHandles* global_handles = CcTest::i_isolate()->global_handles();
|
i::GlobalHandles* global_handles = CcTest::i_isolate()->global_handles();
|
||||||
|
|
||||||
const size_t initial_count = global_handles->handles_count();
|
const size_t initial_count = global_handles->handles_count();
|
||||||
v8::TracedGlobal<v8::Object> global1;
|
auto global1 = std::make_unique<v8::TracedGlobal<v8::Object>>();
|
||||||
{
|
{
|
||||||
v8::HandleScope scope(isolate);
|
v8::HandleScope scope(isolate);
|
||||||
global1.Reset(isolate, v8::Object::New(isolate));
|
global1->Reset(isolate, v8::Object::New(isolate));
|
||||||
}
|
}
|
||||||
v8::TracedGlobal<v8::Object> global2(global1);
|
auto global2 = std::make_unique<v8::TracedGlobal<v8::Object>>(*global1);
|
||||||
v8::TracedGlobal<v8::Object> global3;
|
auto global3 = std::make_unique<v8::TracedGlobal<v8::Object>>();
|
||||||
global3 = global2;
|
*global3 = *global2;
|
||||||
CHECK_EQ(initial_count + 3, global_handles->handles_count());
|
CHECK_EQ(initial_count + 3, global_handles->handles_count());
|
||||||
CHECK(!global1.IsEmpty());
|
CHECK(!global1->IsEmpty());
|
||||||
CHECK_EQ(global1, global2);
|
CHECK_EQ(*global1, *global2);
|
||||||
CHECK_EQ(global2, global3);
|
CHECK_EQ(*global2, *global3);
|
||||||
{
|
{
|
||||||
v8::HandleScope scope(isolate);
|
v8::HandleScope scope(isolate);
|
||||||
auto tmp = v8::Local<v8::Object>::New(isolate, global3);
|
auto tmp = v8::Local<v8::Object>::New(isolate, *global3);
|
||||||
CHECK(!tmp.IsEmpty());
|
CHECK(!tmp.IsEmpty());
|
||||||
InvokeMarkSweep();
|
InvokeMarkSweep();
|
||||||
}
|
}
|
||||||
CHECK_EQ(initial_count + 3, global_handles->handles_count());
|
CHECK_EQ(initial_count + 3, global_handles->handles_count());
|
||||||
CHECK(!global1.IsEmpty());
|
CHECK(!global1->IsEmpty());
|
||||||
CHECK_EQ(global1, global2);
|
CHECK_EQ(*global1, *global2);
|
||||||
CHECK_EQ(global2, global3);
|
CHECK_EQ(*global2, *global3);
|
||||||
InvokeMarkSweep();
|
InvokeMarkSweep();
|
||||||
CHECK_EQ(initial_count, global_handles->handles_count());
|
CHECK_EQ(initial_count, global_handles->handles_count());
|
||||||
CHECK(global1.IsEmpty());
|
CHECK(global1->IsEmpty());
|
||||||
CHECK_EQ(global1, global2);
|
CHECK_EQ(*global1, *global2);
|
||||||
CHECK_EQ(global2, global3);
|
CHECK_EQ(*global2, *global3);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST(TracedGlobalCopyNoDestructor) {
|
TEST(TracedGlobalCopyNoDestructor) {
|
||||||
@ -408,28 +408,28 @@ TEST(TracedGlobalCopyNoDestructor) {
|
|||||||
i::GlobalHandles* global_handles = CcTest::i_isolate()->global_handles();
|
i::GlobalHandles* global_handles = CcTest::i_isolate()->global_handles();
|
||||||
|
|
||||||
const size_t initial_count = global_handles->handles_count();
|
const size_t initial_count = global_handles->handles_count();
|
||||||
v8::TracedReference<v8::Value> global1;
|
auto global1 = std::make_unique<v8::TracedReference<v8::Value>>();
|
||||||
{
|
{
|
||||||
v8::HandleScope scope(isolate);
|
v8::HandleScope scope(isolate);
|
||||||
global1.Reset(isolate, v8::Object::New(isolate));
|
global1->Reset(isolate, v8::Object::New(isolate));
|
||||||
}
|
}
|
||||||
v8::TracedReference<v8::Value> global2(global1);
|
auto global2 = std::make_unique<v8::TracedReference<v8::Value>>(*global1);
|
||||||
v8::TracedReference<v8::Value> global3;
|
auto global3 = std::make_unique<v8::TracedReference<v8::Value>>();
|
||||||
global3 = global2;
|
*global3 = *global2;
|
||||||
CHECK_EQ(initial_count + 3, global_handles->handles_count());
|
CHECK_EQ(initial_count + 3, global_handles->handles_count());
|
||||||
CHECK(!global1.IsEmpty());
|
CHECK(!global1->IsEmpty());
|
||||||
CHECK_EQ(global1, global2);
|
CHECK_EQ(*global1, *global2);
|
||||||
CHECK_EQ(global2, global3);
|
CHECK_EQ(*global2, *global3);
|
||||||
{
|
{
|
||||||
v8::HandleScope scope(isolate);
|
v8::HandleScope scope(isolate);
|
||||||
auto tmp = v8::Local<v8::Value>::New(isolate, global3);
|
auto tmp = v8::Local<v8::Value>::New(isolate, *global3);
|
||||||
CHECK(!tmp.IsEmpty());
|
CHECK(!tmp.IsEmpty());
|
||||||
InvokeMarkSweep();
|
InvokeMarkSweep();
|
||||||
}
|
}
|
||||||
CHECK_EQ(initial_count + 3, global_handles->handles_count());
|
CHECK_EQ(initial_count + 3, global_handles->handles_count());
|
||||||
CHECK(!global1.IsEmpty());
|
CHECK(!global1->IsEmpty());
|
||||||
CHECK_EQ(global1, global2);
|
CHECK_EQ(*global1, *global2);
|
||||||
CHECK_EQ(global2, global3);
|
CHECK_EQ(*global2, *global3);
|
||||||
InvokeMarkSweep();
|
InvokeMarkSweep();
|
||||||
CHECK_EQ(initial_count, global_handles->handles_count());
|
CHECK_EQ(initial_count, global_handles->handles_count());
|
||||||
}
|
}
|
||||||
@ -544,15 +544,15 @@ TEST(TracedReferenceHandlesMarking) {
|
|||||||
CcTest::InitializeVM();
|
CcTest::InitializeVM();
|
||||||
v8::Isolate* isolate = CcTest::isolate();
|
v8::Isolate* isolate = CcTest::isolate();
|
||||||
v8::HandleScope scope(isolate);
|
v8::HandleScope scope(isolate);
|
||||||
v8::TracedReference<v8::Value> live;
|
auto live = std::make_unique<v8::TracedReference<v8::Value>>();
|
||||||
v8::TracedReference<v8::Value> dead;
|
auto dead = std::make_unique<v8::TracedReference<v8::Value>>();
|
||||||
live.Reset(isolate, v8::Undefined(isolate));
|
live->Reset(isolate, v8::Undefined(isolate));
|
||||||
dead.Reset(isolate, v8::Undefined(isolate));
|
dead->Reset(isolate, v8::Undefined(isolate));
|
||||||
i::GlobalHandles* global_handles = CcTest::i_isolate()->global_handles();
|
i::GlobalHandles* global_handles = CcTest::i_isolate()->global_handles();
|
||||||
{
|
{
|
||||||
TestEmbedderHeapTracer tracer;
|
TestEmbedderHeapTracer tracer;
|
||||||
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
||||||
tracer.AddReferenceForTracing(&live);
|
tracer.AddReferenceForTracing(live.get());
|
||||||
const size_t initial_count = global_handles->handles_count();
|
const size_t initial_count = global_handles->handles_count();
|
||||||
InvokeMarkSweep();
|
InvokeMarkSweep();
|
||||||
const size_t final_count = global_handles->handles_count();
|
const size_t final_count = global_handles->handles_count();
|
||||||
@ -563,7 +563,7 @@ TEST(TracedReferenceHandlesMarking) {
|
|||||||
{
|
{
|
||||||
TestEmbedderHeapTracer tracer;
|
TestEmbedderHeapTracer tracer;
|
||||||
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
||||||
tracer.AddReferenceForTracing(&live);
|
tracer.AddReferenceForTracing(live.get());
|
||||||
const size_t initial_count = global_handles->handles_count();
|
const size_t initial_count = global_handles->handles_count();
|
||||||
InvokeMarkSweep();
|
InvokeMarkSweep();
|
||||||
const size_t final_count = global_handles->handles_count();
|
const size_t final_count = global_handles->handles_count();
|
||||||
@ -579,8 +579,8 @@ TEST(TracedReferenceHandlesDoNotLeak) {
|
|||||||
CcTest::InitializeVM();
|
CcTest::InitializeVM();
|
||||||
v8::Isolate* isolate = CcTest::isolate();
|
v8::Isolate* isolate = CcTest::isolate();
|
||||||
v8::HandleScope scope(isolate);
|
v8::HandleScope scope(isolate);
|
||||||
v8::TracedReference<v8::Value> ref;
|
auto ref = std::make_unique<v8::TracedReference<v8::Value>>();
|
||||||
ref.Reset(isolate, v8::Undefined(isolate));
|
ref->Reset(isolate, v8::Undefined(isolate));
|
||||||
i::GlobalHandles* global_handles = CcTest::i_isolate()->global_handles();
|
i::GlobalHandles* global_handles = CcTest::i_isolate()->global_handles();
|
||||||
const size_t initial_count = global_handles->handles_count();
|
const size_t initial_count = global_handles->handles_count();
|
||||||
// We need two GCs because handles are black allocated.
|
// We need two GCs because handles are black allocated.
|
||||||
@ -635,10 +635,10 @@ TEST(TracedGlobalIteration) {
|
|||||||
TestEmbedderHeapTracer tracer;
|
TestEmbedderHeapTracer tracer;
|
||||||
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
||||||
|
|
||||||
v8::TracedGlobal<v8::Object> traced;
|
auto traced = std::make_unique<v8::TracedGlobal<v8::Object>>();
|
||||||
ConstructJSObject(isolate, isolate->GetCurrentContext(), &traced);
|
ConstructJSObject(isolate, isolate->GetCurrentContext(), traced.get());
|
||||||
CHECK(!traced.IsEmpty());
|
CHECK(!traced->IsEmpty());
|
||||||
traced.SetWrapperClassId(57);
|
traced->SetWrapperClassId(57);
|
||||||
TracedGlobalVisitor visitor;
|
TracedGlobalVisitor visitor;
|
||||||
{
|
{
|
||||||
v8::HandleScope scope(isolate);
|
v8::HandleScope scope(isolate);
|
||||||
@ -669,18 +669,18 @@ TEST(TracedGlobalSetFinalizationCallbackScavenge) {
|
|||||||
tracer.ConsiderTracedGlobalAsRoot(false);
|
tracer.ConsiderTracedGlobalAsRoot(false);
|
||||||
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
||||||
|
|
||||||
v8::TracedGlobal<v8::Object> traced;
|
auto traced = std::make_unique<v8::TracedGlobal<v8::Object>>();
|
||||||
ConstructJSApiObject(isolate, isolate->GetCurrentContext(), &traced);
|
ConstructJSApiObject(isolate, isolate->GetCurrentContext(), traced.get());
|
||||||
CHECK(!traced.IsEmpty());
|
CHECK(!traced->IsEmpty());
|
||||||
{
|
{
|
||||||
v8::HandleScope scope(isolate);
|
v8::HandleScope scope(isolate);
|
||||||
auto local = traced.Get(isolate);
|
auto local = traced->Get(isolate);
|
||||||
local->SetAlignedPointerInInternalField(0, reinterpret_cast<void*>(0x4));
|
local->SetAlignedPointerInInternalField(0, reinterpret_cast<void*>(0x4));
|
||||||
local->SetAlignedPointerInInternalField(1, reinterpret_cast<void*>(0x8));
|
local->SetAlignedPointerInInternalField(1, reinterpret_cast<void*>(0x8));
|
||||||
}
|
}
|
||||||
traced.SetFinalizationCallback(&traced, FinalizationCallback);
|
traced->SetFinalizationCallback(traced.get(), FinalizationCallback);
|
||||||
heap::InvokeScavenge();
|
heap::InvokeScavenge();
|
||||||
CHECK(traced.IsEmpty());
|
CHECK(traced->IsEmpty());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST(TracedGlobalSetFinalizationCallbackMarkSweep) {
|
TEST(TracedGlobalSetFinalizationCallbackMarkSweep) {
|
||||||
@ -691,18 +691,18 @@ TEST(TracedGlobalSetFinalizationCallbackMarkSweep) {
|
|||||||
TestEmbedderHeapTracer tracer;
|
TestEmbedderHeapTracer tracer;
|
||||||
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
||||||
|
|
||||||
v8::TracedGlobal<v8::Object> traced;
|
auto traced = std::make_unique<v8::TracedGlobal<v8::Object>>();
|
||||||
ConstructJSApiObject(isolate, isolate->GetCurrentContext(), &traced);
|
ConstructJSApiObject(isolate, isolate->GetCurrentContext(), traced.get());
|
||||||
CHECK(!traced.IsEmpty());
|
CHECK(!traced->IsEmpty());
|
||||||
{
|
{
|
||||||
v8::HandleScope scope(isolate);
|
v8::HandleScope scope(isolate);
|
||||||
auto local = traced.Get(isolate);
|
auto local = traced->Get(isolate);
|
||||||
local->SetAlignedPointerInInternalField(0, reinterpret_cast<void*>(0x4));
|
local->SetAlignedPointerInInternalField(0, reinterpret_cast<void*>(0x4));
|
||||||
local->SetAlignedPointerInInternalField(1, reinterpret_cast<void*>(0x8));
|
local->SetAlignedPointerInInternalField(1, reinterpret_cast<void*>(0x8));
|
||||||
}
|
}
|
||||||
traced.SetFinalizationCallback(&traced, FinalizationCallback);
|
traced->SetFinalizationCallback(traced.get(), FinalizationCallback);
|
||||||
heap::InvokeMarkSweep();
|
heap::InvokeMarkSweep();
|
||||||
CHECK(traced.IsEmpty());
|
CHECK(traced->IsEmpty());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST(TracePrologueCallingIntoV8WriteBarrier) {
|
TEST(TracePrologueCallingIntoV8WriteBarrier) {
|
||||||
|
@ -158,13 +158,13 @@ void TracedGlobalTest(v8::Isolate* isolate,
|
|||||||
NonRootingEmbedderHeapTracer tracer;
|
NonRootingEmbedderHeapTracer tracer;
|
||||||
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
heap::TemporaryEmbedderHeapTracerScope tracer_scope(isolate, &tracer);
|
||||||
|
|
||||||
TracedGlobalWrapper fp;
|
auto fp = std::make_unique<TracedGlobalWrapper>();
|
||||||
construct_function(isolate, context, &fp);
|
construct_function(isolate, context, fp.get());
|
||||||
CHECK(heap::InCorrectGeneration(isolate, fp.handle));
|
CHECK(heap::InCorrectGeneration(isolate, fp->handle));
|
||||||
modifier_function(&fp);
|
modifier_function(fp.get());
|
||||||
gc_function();
|
gc_function();
|
||||||
CHECK_IMPLIES(survives == SurvivalMode::kSurvives, !fp.handle.IsEmpty());
|
CHECK_IMPLIES(survives == SurvivalMode::kSurvives, !fp->handle.IsEmpty());
|
||||||
CHECK_IMPLIES(survives == SurvivalMode::kDies, fp.handle.IsEmpty());
|
CHECK_IMPLIES(survives == SurvivalMode::kDies, fp->handle.IsEmpty());
|
||||||
}
|
}
|
||||||
|
|
||||||
void ResurrectingFinalizer(
|
void ResurrectingFinalizer(
|
||||||
|
Loading…
Reference in New Issue
Block a user