#pragma once #include #include namespace AK { template class RetainPtr { public: enum AdoptTag { Adopt }; RetainPtr() { } RetainPtr(const T* ptr) : m_ptr(const_cast(ptr)) { retain_if_not_null(m_ptr); } RetainPtr(T* ptr) : m_ptr(ptr) { retain_if_not_null(m_ptr); } RetainPtr(T& object) : m_ptr(&object) { m_ptr->retain(); } RetainPtr(const T& object) : m_ptr(const_cast(&object)) { m_ptr->retain(); } RetainPtr(AdoptTag, T& object) : m_ptr(&object) { } RetainPtr(RetainPtr& other) : m_ptr(other.copy_ref().leak_ref()) { } RetainPtr(RetainPtr&& other) : m_ptr(other.leak_ref()) { } template RetainPtr(Retained&& other) : m_ptr(static_cast(&other.leak_ref())) { } template RetainPtr(RetainPtr&& other) : m_ptr(static_cast(other.leak_ref())) { } RetainPtr(const RetainPtr& other) : m_ptr(const_cast(other).copy_ref().leak_ref()) { } template RetainPtr(const RetainPtr& other) : m_ptr(const_cast&>(other).copy_ref().leak_ref()) { } ~RetainPtr() { clear(); #ifdef SANITIZE_PTRS if constexpr(sizeof(T*) == 8) m_ptr = (T*)(0xe0e0e0e0e0e0e0e0); else m_ptr = (T*)(0xe0e0e0e0); #endif } RetainPtr(std::nullptr_t) { } RetainPtr& operator=(RetainPtr&& other) { if (this != &other) { release_if_not_null(m_ptr); m_ptr = other.leak_ref(); } return *this; } template RetainPtr& operator=(RetainPtr&& other) { if (this != static_cast(&other)) { release_if_not_null(m_ptr); m_ptr = other.leak_ref(); } return *this; } RetainPtr& operator=(T* ptr) { if (m_ptr != ptr) release_if_not_null(m_ptr); m_ptr = ptr; retain_if_not_null(m_ptr); return *this; } RetainPtr& operator=(T& object) { if (m_ptr != &object) release_if_not_null(m_ptr); m_ptr = &object; retain_if_not_null(m_ptr); return *this; } RetainPtr& operator=(std::nullptr_t) { clear(); return *this; } RetainPtr copy_ref() const { return RetainPtr(m_ptr); } void clear() { release_if_not_null(m_ptr); m_ptr = nullptr; } bool operator!() const { return !m_ptr; } typedef T* RetainPtr::*UnspecifiedBoolType; operator UnspecifiedBoolType() const { return m_ptr ? &RetainPtr::m_ptr : nullptr; } T* leak_ref() { T* leakedPtr = m_ptr; m_ptr = nullptr; return leakedPtr; } T* ptr() { return m_ptr; } const T* ptr() const { return m_ptr; } T* operator->() { return m_ptr; } const T* operator->() const { return m_ptr; } T& operator*() { return *m_ptr; } const T& operator*() const { return *m_ptr; } operator bool() { return !!m_ptr; } bool is_null() const { return !m_ptr; } private: T* m_ptr = nullptr; }; } using AK::RetainPtr;