#pragma once #include "Types.h" namespace AK { template inline void retainIfNotNull(T* ptr) { if (ptr) ptr->retain(); } template inline void releaseIfNotNull(T* ptr) { if (ptr) ptr->release(); } template class RetainPtr { public: enum AdoptTag { Adopt }; RetainPtr() { } RetainPtr(const T* ptr) : m_ptr(const_cast(ptr)) { retainIfNotNull(m_ptr); } RetainPtr(T* ptr) : m_ptr(ptr) { retainIfNotNull(m_ptr); } RetainPtr(T& object) : m_ptr(&object) { m_ptr->retain(); } RetainPtr(AdoptTag, T& object) : m_ptr(&object) { } RetainPtr(RetainPtr&& other) : m_ptr(other.leakRef()) { } template RetainPtr(RetainPtr&& other) : m_ptr(static_cast(other.leakRef())) { } ~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) { releaseIfNotNull(m_ptr); m_ptr = other.leakRef(); } return *this; } template RetainPtr& operator=(RetainPtr&& other) { if (this != static_cast(&other)) { releaseIfNotNull(m_ptr); m_ptr = other.leakRef(); } return *this; } RetainPtr& operator=(T* ptr) { if (m_ptr != ptr) releaseIfNotNull(m_ptr); m_ptr = ptr; retainIfNotNull(m_ptr); return *this; } RetainPtr& operator=(T& object) { if (m_ptr != &object) releaseIfNotNull(m_ptr); m_ptr = &object; retainIfNotNull(m_ptr); return *this; } RetainPtr& operator=(std::nullptr_t) { clear(); return *this; } RetainPtr copyRef() const { return RetainPtr(m_ptr); } void clear() { releaseIfNotNull(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* leakRef() { 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; } private: T* m_ptr = nullptr; }; template inline RetainPtr adopt(T& object) { return RetainPtr(RetainPtr::Adopt, object); } } using AK::RetainPtr; using AK::adopt;