/* * Copyright (c) 2020, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include namespace JS { class HandleImpl : public RefCounted { AK_MAKE_NONCOPYABLE(HandleImpl); AK_MAKE_NONMOVABLE(HandleImpl); public: ~HandleImpl(); Cell* cell() { return m_cell; } Cell const* cell() const { return m_cell; } private: template friend class Handle; explicit HandleImpl(Cell*); Cell* m_cell { nullptr }; IntrusiveListNode m_list_node; public: using List = IntrusiveList<&HandleImpl::m_list_node>; }; template class Handle { public: Handle() = default; static Handle create(T* cell) { return Handle(adopt_ref(*new HandleImpl(cell))); } T* cell() { return static_cast(m_impl->cell()); } const T* cell() const { return static_cast(m_impl->cell()); } bool is_null() const { return m_impl.is_null(); } T* operator->() { return cell(); } T const* operator->() const { return cell(); } private: explicit Handle(NonnullRefPtr impl) : m_impl(move(impl)) { } RefPtr m_impl; }; template inline Handle make_handle(T* cell) { if (!cell) return Handle {}; return Handle::create(cell); } template inline Handle make_handle(T& cell) { return Handle::create(&cell); } template<> class Handle { public: Handle() = default; static Handle create(Value value) { if (value.is_cell()) return Handle(value, &value.as_cell()); return Handle(value); } auto cell() { return m_handle.cell(); } auto cell() const { return m_handle.cell(); } auto value() const { return *m_value; } bool is_null() const { return m_handle.is_null() && !m_value.has_value(); } private: explicit Handle(Value value) : m_value(value) { } explicit Handle(Value value, Cell* cell) : m_value(value) , m_handle(Handle::create(cell)) { } Optional m_value; Handle m_handle; }; inline Handle make_handle(Value value) { return Handle::create(value); } }