/* * Copyright (c) 2020, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #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*); GCPtr m_cell; 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(const_cast*>(cell)))); } Handle(T* cell) { if (cell) m_impl = adopt_ref(*new HandleImpl(cell)); } Handle(T& cell) : m_impl(adopt_ref(*new HandleImpl(&cell))) { } Handle(GCPtr cell) : Handle(cell.ptr()) { } Handle(NonnullGCPtr cell) : Handle(*cell) { } T* cell() const { if (!m_impl) return nullptr; return static_cast(m_impl->cell()); } T* ptr() const { return cell(); } bool is_null() const { return m_impl.is_null(); } T* operator->() const { return cell(); } T& operator*() const { return *cell(); } bool operator!() const { return !cell(); } operator bool() const { return cell(); } operator T*() 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 inline Handle make_handle(GCPtr cell) { if (!cell) return Handle {}; return Handle::create(cell.ptr()); } template inline Handle make_handle(NonnullGCPtr cell) { return Handle::create(cell.ptr()); } 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); } }