#pragma once #include #include #include #include #include namespace AK { template class RefPtr; template class NonnullRefPtr; template class WeakPtr; template class CONSUMABLE(unconsumed) NonnullOwnPtr { public: typedef T ElementType; enum AdoptTag { Adopt }; RETURN_TYPESTATE(unconsumed) NonnullOwnPtr(AdoptTag, T& ptr) : m_ptr(&ptr) { } RETURN_TYPESTATE(unconsumed) NonnullOwnPtr(NonnullOwnPtr&& other) : m_ptr(other.leak_ptr()) { ASSERT(m_ptr); } template RETURN_TYPESTATE(unconsumed) NonnullOwnPtr(NonnullOwnPtr&& other) : m_ptr(static_cast(other.leak_ptr())) { ASSERT(m_ptr); } ~NonnullOwnPtr() { clear(); #ifdef SANITIZE_PTRS if constexpr (sizeof(T*) == 8) m_ptr = (T*)(0xe3e3e3e3e3e3e3e3); else m_ptr = (T*)(0xe3e3e3e3); #endif } NonnullOwnPtr(const NonnullOwnPtr&) = delete; template NonnullOwnPtr(const NonnullOwnPtr&) = delete; NonnullOwnPtr& operator=(const NonnullOwnPtr&) = delete; template NonnullOwnPtr& operator=(const NonnullOwnPtr&) = delete; template NonnullOwnPtr(const RefPtr&) = delete; template NonnullOwnPtr(const NonnullRefPtr&) = delete; template NonnullOwnPtr(const WeakPtr&) = delete; template NonnullOwnPtr& operator=(const RefPtr&) = delete; template NonnullOwnPtr& operator=(const NonnullRefPtr&) = delete; template NonnullOwnPtr& operator=(const WeakPtr&) = delete; RETURN_TYPESTATE(unconsumed) NonnullOwnPtr& operator=(NonnullOwnPtr&& other) { if (this != &other) { delete m_ptr; m_ptr = other.leak_ptr(); ASSERT(m_ptr); } return *this; } template RETURN_TYPESTATE(unconsumed) NonnullOwnPtr& operator=(NonnullOwnPtr&& other) { if (this != static_cast(&other)) { delete m_ptr; m_ptr = other.leak_ptr(); ASSERT(m_ptr); } return *this; } CALLABLE_WHEN(unconsumed) SET_TYPESTATE(consumed) T* leak_ptr() { return exchange(m_ptr, nullptr); } CALLABLE_WHEN(unconsumed) T* ptr() { return m_ptr; } CALLABLE_WHEN(unconsumed) const T* ptr() const { return m_ptr; } CALLABLE_WHEN(unconsumed) T* operator->() { return m_ptr; } CALLABLE_WHEN(unconsumed) const T* operator->() const { return m_ptr; } CALLABLE_WHEN(unconsumed) T& operator*() { return *m_ptr; } CALLABLE_WHEN(unconsumed) const T& operator*() const { return *m_ptr; } CALLABLE_WHEN(unconsumed) operator const T*() const { return m_ptr; } CALLABLE_WHEN(unconsumed) operator T*() { return m_ptr; } operator bool() const = delete; bool operator!() const = delete; private: void clear() { if (!m_ptr) return; delete m_ptr; m_ptr = nullptr; } T* m_ptr = nullptr; }; template inline NonnullOwnPtr make(Args&&... args) { return NonnullOwnPtr(NonnullOwnPtr::Adopt, *new T(forward(args)...)); } template struct Traits> : public GenericTraits> { using PeekType = const T*; static unsigned hash(const NonnullOwnPtr& p) { return int_hash((u32)p.ptr()); } static void dump(const NonnullOwnPtr& p) { kprintf("%p", p.ptr()); } static bool equals(const NonnullOwnPtr& a, const NonnullOwnPtr& b) { return a.ptr() == b.ptr(); } }; template inline const LogStream& operator<<(const LogStream& stream, const NonnullOwnPtr& value) { return stream << value.ptr(); } } using AK::make; using AK::NonnullOwnPtr;