blob: 8a1551f1a22e4a0b6aea70f819c81d6443f2f2e0 [file] [log] [blame]
// 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.
#ifndef V8_HEAP_MARKING_WORKLIST_INL_H_
#define V8_HEAP_MARKING_WORKLIST_INL_H_
#include <unordered_map>
#include "src/heap/cppgc-js/cpp-marking-state-inl.h"
#include "src/heap/marking-worklist.h"
#include "src/objects/embedder-data-slot.h"
#include "src/objects/js-objects-inl.h"
namespace v8 {
namespace internal {
template <typename Callback>
void MarkingWorklists::Update(Callback callback) {
shared_.Update(callback);
on_hold_.Update(callback);
wrapper_.Update(callback);
other_.Update(callback);
for (auto cw : context_worklists_) {
if (cw.context == kSharedContext || cw.context == kOtherContext) {
// These contexts were updated above.
continue;
}
cw.worklist->Update(callback);
}
}
void MarkingWorklists::Local::Push(HeapObject object) { active_.Push(object); }
bool MarkingWorklists::Local::Pop(HeapObject* object) {
if (active_.Pop(object)) return true;
if (!is_per_context_mode_) return false;
// The active worklist is empty. Find any other non-empty worklist and
// switch the active worklist to it.
return PopContext(object);
}
void MarkingWorklists::Local::PushOnHold(HeapObject object) {
on_hold_.Push(object);
}
bool MarkingWorklists::Local::PopOnHold(HeapObject* object) {
return on_hold_.Pop(object);
}
bool MarkingWorklists::Local::SupportsExtractWrapper() {
return cpp_marking_state_.get();
}
bool MarkingWorklists::Local::ExtractWrapper(Map map, JSObject object,
WrapperSnapshot& snapshot) {
DCHECK_NOT_NULL(cpp_marking_state_);
return cpp_marking_state_->ExtractEmbedderDataSnapshot(map, object, snapshot);
}
void MarkingWorklists::Local::PushExtractedWrapper(
const WrapperSnapshot& snapshot) {
DCHECK_NOT_NULL(cpp_marking_state_);
cpp_marking_state_->MarkAndPush(snapshot);
}
void MarkingWorklists::Local::PushWrapper(HeapObject object) {
DCHECK_NULL(cpp_marking_state_);
wrapper_.Push(object);
}
bool MarkingWorklists::Local::PopWrapper(HeapObject* object) {
DCHECK_NULL(cpp_marking_state_);
return wrapper_.Pop(object);
}
Address MarkingWorklists::Local::SwitchToContext(Address context) {
if (context == active_context_) return context;
return SwitchToContextSlow(context);
}
Address MarkingWorklists::Local::SwitchToShared() {
return SwitchToContext(kSharedContext);
}
void MarkingWorklists::Local::SwitchToContext(
Address context, MarkingWorklist::Local* worklist) {
// Save the current worklist.
*active_owner_ = std::move(active_);
// Switch to the new worklist.
active_owner_ = worklist;
active_ = std::move(*worklist);
active_context_ = context;
}
bool MarkingWorklists::Local::PublishWrapper() {
if (!cpp_marking_state_) return false;
cpp_marking_state_->Publish();
return true;
}
} // namespace internal
} // namespace v8
#endif // V8_HEAP_MARKING_WORKLIST_INL_H_