123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128 |
- /*
- * Copyright (c) 2020-2022, Andreas Kling <kling@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #pragma once
- #include <AK/Badge.h>
- #include <AK/Format.h>
- #include <AK/Forward.h>
- #include <AK/Noncopyable.h>
- #include <AK/StringView.h>
- #include <LibJS/Forward.h>
- #include <LibJS/Heap/GCPtr.h>
- #include <LibJS/Heap/Internals.h>
- namespace JS {
- #define JS_CELL(class_, base_class) \
- public: \
- using Base = base_class; \
- virtual StringView class_name() const override \
- { \
- return #class_##sv; \
- } \
- friend class JS::Heap;
- class Cell {
- AK_MAKE_NONCOPYABLE(Cell);
- AK_MAKE_NONMOVABLE(Cell);
- public:
- virtual void initialize(Realm&);
- virtual ~Cell() = default;
- bool is_marked() const { return m_mark; }
- void set_marked(bool b) { m_mark = b; }
- enum class State : bool {
- Live,
- Dead,
- };
- State state() const { return m_state; }
- void set_state(State state) { m_state = state; }
- virtual StringView class_name() const = 0;
- class Visitor {
- public:
- void visit(Cell* cell)
- {
- if (cell)
- visit_impl(*cell);
- }
- void visit(Cell& cell)
- {
- visit_impl(cell);
- }
- template<typename T>
- void visit(GCPtr<T> cell)
- {
- if (cell)
- visit_impl(const_cast<RemoveConst<T>&>(*cell.ptr()));
- }
- template<typename T>
- void visit(NonnullGCPtr<T> cell)
- {
- visit_impl(const_cast<RemoveConst<T>&>(*cell.ptr()));
- }
- void visit(Value value);
- // Allow explicitly ignoring a GC-allocated member in a visit_edges implementation instead
- // of just not using it.
- template<typename T>
- void ignore(T const&)
- {
- }
- virtual void visit_possible_values(ReadonlyBytes) = 0;
- protected:
- virtual void visit_impl(Cell&) = 0;
- virtual ~Visitor() = default;
- };
- virtual void visit_edges(Visitor&) { }
- // This will be called on unmarked objects by the garbage collector in a separate pass before destruction.
- virtual void finalize() { }
- // This allows cells to survive GC by choice, even if nothing points to them.
- // It's used to implement special rules in the web platform.
- // NOTE: Cells must call set_overrides_must_survive_garbage_collection() for this to be honored.
- virtual bool must_survive_garbage_collection() const { return false; }
- bool overrides_must_survive_garbage_collection(Badge<Heap>) const { return m_overrides_must_survive_garbage_collection; }
- ALWAYS_INLINE Heap& heap() const { return HeapBlockBase::from_cell(this)->heap(); }
- ALWAYS_INLINE VM& vm() const { return bit_cast<HeapBase*>(&heap())->vm(); }
- protected:
- Cell() = default;
- void set_overrides_must_survive_garbage_collection(bool b) { m_overrides_must_survive_garbage_collection = b; }
- private:
- bool m_mark : 1 { false };
- bool m_overrides_must_survive_garbage_collection : 1 { false };
- State m_state : 1 { State::Live };
- };
- }
- template<>
- struct AK::Formatter<JS::Cell> : AK::Formatter<FormatString> {
- ErrorOr<void> format(FormatBuilder& builder, JS::Cell const* cell)
- {
- if (!cell)
- return builder.put_string("Cell{nullptr}"sv);
- return Formatter<FormatString>::format(builder, "{}({})"sv, cell->class_name(), cell);
- }
- };
|