/* * 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(); CellImpl* cell() { return m_cell; } CellImpl const* cell() const { return m_cell; } SourceLocation const& source_location() const { return m_location; } private: template friend class Handle; explicit HandleImpl(CellImpl*, SourceLocation location); GCPtr m_cell; SourceLocation m_location; IntrusiveListNode m_list_node; public: using List = IntrusiveList<&HandleImpl::m_list_node>; }; template class Handle { public: Handle() = default; static Handle create(T* cell, SourceLocation location = SourceLocation::current()) { return Handle(adopt_ref(*new HandleImpl(const_cast*>(cell), location))); } Handle(T* cell, SourceLocation location = SourceLocation::current()) { if (cell) m_impl = adopt_ref(*new HandleImpl(cell, location)); } Handle(T& cell, SourceLocation location = SourceLocation::current()) : m_impl(adopt_ref(*new HandleImpl(&cell, location))) { } Handle(GCPtr cell, SourceLocation location = SourceLocation::current()) : Handle(cell.ptr(), location) { } Handle(NonnullGCPtr cell, SourceLocation location = SourceLocation::current()) : Handle(*cell, location) { } 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(); } [[nodiscard]] 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, SourceLocation location = SourceLocation::current()) { if (!cell) return Handle {}; return Handle::create(cell, location); } template inline Handle make_handle(T& cell, SourceLocation location = SourceLocation::current()) { return Handle::create(&cell, location); } template inline Handle make_handle(GCPtr cell, SourceLocation location = SourceLocation::current()) { if (!cell) return Handle {}; return Handle::create(cell.ptr(), location); } template inline Handle make_handle(NonnullGCPtr cell, SourceLocation location = SourceLocation::current()) { return Handle::create(cell.ptr(), location); } } namespace AK { template struct Traits> : public DefaultTraits> { static unsigned hash(JS::Handle const& handle) { return Traits::hash(handle); } }; namespace Detail { template inline constexpr bool IsHashCompatible, T> = true; template inline constexpr bool IsHashCompatible> = true; } }