#pragma once #include namespace AK { template class OwnPtr { public: OwnPtr() {} explicit OwnPtr(T* ptr) : m_ptr(ptr) { } OwnPtr(OwnPtr&& other) : m_ptr(other.leak_ptr()) { } template OwnPtr(NonnullOwnPtr&& other) : m_ptr(static_cast(other.leak_ptr())) { } template OwnPtr(OwnPtr&& other) : m_ptr(static_cast(other.leak_ptr())) { } OwnPtr(std::nullptr_t) {}; ~OwnPtr() { clear(); #ifdef SANITIZE_PTRS if constexpr (sizeof(T*) == 8) m_ptr = (T*)(0xe1e1e1e1e1e1e1e1); else m_ptr = (T*)(0xe1e1e1e1); #endif } OwnPtr(const OwnPtr&) = delete; template OwnPtr(const OwnPtr&) = delete; OwnPtr& operator=(const OwnPtr&) = delete; template OwnPtr& operator=(const OwnPtr&) = delete; template OwnPtr(const NonnullOwnPtr&) = delete; template OwnPtr& operator=(const NonnullOwnPtr&) = delete; template OwnPtr(const RefPtr&) = delete; template OwnPtr(const NonnullRefPtr&) = delete; template OwnPtr(const WeakPtr&) = delete; template OwnPtr& operator=(const RefPtr&) = delete; template OwnPtr& operator=(const NonnullRefPtr&) = delete; template OwnPtr& operator=(const WeakPtr&) = delete; OwnPtr& operator=(OwnPtr&& other) { if (this != &other) { delete m_ptr; m_ptr = other.leak_ptr(); } return *this; } template OwnPtr& operator=(OwnPtr&& other) { if (this != static_cast(&other)) { delete m_ptr; m_ptr = other.leak_ptr(); } return *this; } template OwnPtr& operator=(NonnullOwnPtr&& other) { ASSERT(m_ptr != other.ptr()); delete m_ptr; m_ptr = other.leak_ptr(); return *this; } OwnPtr& operator=(T* ptr) { if (m_ptr != ptr) delete m_ptr; m_ptr = ptr; return *this; } OwnPtr& operator=(std::nullptr_t) { clear(); return *this; } void clear() { delete m_ptr; m_ptr = nullptr; } bool operator!() const { return !m_ptr; } T* leak_ptr() { T* leaked_ptr = m_ptr; m_ptr = nullptr; return leaked_ptr; } NonnullOwnPtr release_nonnull() { ASSERT(m_ptr); return NonnullOwnPtr(NonnullOwnPtr::Adopt, *leak_ptr()); } T* ptr() { return m_ptr; } const T* ptr() const { return m_ptr; } T* operator->() { ASSERT(m_ptr); return m_ptr; } const T* operator->() const { ASSERT(m_ptr); return m_ptr; } T& operator*() { ASSERT(m_ptr); return *m_ptr; } const T& operator*() const { ASSERT(m_ptr); return *m_ptr; } operator const T*() const { return m_ptr; } operator T*() { return m_ptr; } operator bool() { return !!m_ptr; } private: T* m_ptr = nullptr; }; template struct Traits> : public GenericTraits> { using PeekType = const T*; static unsigned hash(const OwnPtr& p) { return int_hash((u32)p.ptr()); } static void dump(const OwnPtr& p) { kprintf("%p", p.ptr()); } static bool equals(const OwnPtr& a, const OwnPtr& b) { return a.ptr() == b.ptr(); } }; template inline const LogStream& operator<<(const LogStream& stream, const OwnPtr& value) { return stream << value.ptr(); } } using AK::OwnPtr;