/* * Copyright (c) 2018-2020, Andreas Kling * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #pragma once #include #include #include #include #include namespace AK { template class OwnPtr; template class RefPtr { public: enum AdoptTag { Adopt }; RefPtr() { } RefPtr(const T* ptr) : m_ptr(const_cast(ptr)) { ref_if_not_null(m_ptr); } RefPtr(const T& object) : m_ptr(const_cast(&object)) { m_ptr->ref(); } RefPtr(AdoptTag, T& object) : m_ptr(&object) { } RefPtr(RefPtr&& other) : m_ptr(other.leak_ref()) { } ALWAYS_INLINE RefPtr(const NonnullRefPtr& other) : m_ptr(const_cast(other.ptr())) { ASSERT(m_ptr); m_ptr->ref(); } template ALWAYS_INLINE RefPtr(const NonnullRefPtr& other) : m_ptr(const_cast(other.ptr())) { ASSERT(m_ptr); m_ptr->ref(); } template ALWAYS_INLINE RefPtr(NonnullRefPtr&& other) : m_ptr(&other.leak_ref()) { ASSERT(m_ptr); } template RefPtr(RefPtr&& other) : m_ptr(other.leak_ref()) { } RefPtr(const RefPtr& other) : m_ptr(const_cast(other.ptr())) { ref_if_not_null(m_ptr); } template RefPtr(const RefPtr& other) : m_ptr(const_cast(other.ptr())) { ref_if_not_null(m_ptr); } ALWAYS_INLINE ~RefPtr() { clear(); #ifdef SANITIZE_PTRS if constexpr (sizeof(T*) == 8) m_ptr = (T*)(0xe0e0e0e0e0e0e0e0); else m_ptr = (T*)(0xe0e0e0e0); #endif } RefPtr(std::nullptr_t) { } template RefPtr(const OwnPtr&) = delete; template RefPtr& operator=(const OwnPtr&) = delete; template void swap(RefPtr& other) { ::swap(m_ptr, other.m_ptr); } ALWAYS_INLINE RefPtr& operator=(RefPtr&& other) { RefPtr tmp = move(other); swap(tmp); return *this; } template ALWAYS_INLINE RefPtr& operator=(RefPtr&& other) { RefPtr tmp = move(other); swap(tmp); return *this; } template ALWAYS_INLINE RefPtr& operator=(NonnullRefPtr&& other) { RefPtr tmp = move(other); swap(tmp); ASSERT(m_ptr); return *this; } ALWAYS_INLINE RefPtr& operator=(const NonnullRefPtr& other) { RefPtr tmp = other; swap(tmp); ASSERT(m_ptr); return *this; } template ALWAYS_INLINE RefPtr& operator=(const NonnullRefPtr& other) { RefPtr tmp = other; swap(tmp); ASSERT(m_ptr); return *this; } ALWAYS_INLINE RefPtr& operator=(const RefPtr& other) { RefPtr tmp = other; swap(tmp); return *this; } template ALWAYS_INLINE RefPtr& operator=(const RefPtr& other) { RefPtr tmp = other; swap(tmp); return *this; } ALWAYS_INLINE RefPtr& operator=(const T* ptr) { RefPtr tmp = ptr; swap(tmp); return *this; } ALWAYS_INLINE RefPtr& operator=(const T& object) { RefPtr tmp = object; swap(tmp); return *this; } RefPtr& operator=(std::nullptr_t) { clear(); return *this; } ALWAYS_INLINE void clear() { unref_if_not_null(m_ptr); m_ptr = nullptr; } bool operator!() const { return !m_ptr; } [[nodiscard]] T* leak_ref() { return exchange(m_ptr, nullptr); } NonnullRefPtr release_nonnull() { ASSERT(m_ptr); return NonnullRefPtr(NonnullRefPtr::Adopt, *leak_ref()); } ALWAYS_INLINE T* ptr() { return m_ptr; } ALWAYS_INLINE const T* ptr() const { return m_ptr; } ALWAYS_INLINE T* operator->() { ASSERT(m_ptr); return m_ptr; } ALWAYS_INLINE const T* operator->() const { ASSERT(m_ptr); return m_ptr; } ALWAYS_INLINE T& operator*() { ASSERT(m_ptr); return *m_ptr; } ALWAYS_INLINE const T& operator*() const { ASSERT(m_ptr); return *m_ptr; } ALWAYS_INLINE operator const T*() const { return m_ptr; } ALWAYS_INLINE operator T*() { return m_ptr; } operator bool() { return !!m_ptr; } bool operator==(std::nullptr_t) const { return !m_ptr; } bool operator!=(std::nullptr_t) const { return m_ptr; } bool operator==(const RefPtr& other) const { return m_ptr == other.m_ptr; } bool operator!=(const RefPtr& other) const { return m_ptr != other.m_ptr; } bool operator==(RefPtr& other) { return m_ptr == other.m_ptr; } bool operator!=(RefPtr& other) { return m_ptr != other.m_ptr; } bool operator==(const T* other) const { return m_ptr == other; } bool operator!=(const T* other) const { return m_ptr != other; } bool operator==(T* other) { return m_ptr == other; } bool operator!=(T* other) { return m_ptr != other; } bool is_null() const { return !m_ptr; } private: T* m_ptr = nullptr; }; template inline const LogStream& operator<<(const LogStream& stream, const RefPtr& value) { return stream << value.ptr(); } template struct Traits> : public GenericTraits> { using PeekType = const T*; static unsigned hash(const RefPtr& p) { return ptr_hash(p.ptr()); } static bool equals(const RefPtr& a, const RefPtr& b) { return a.ptr() == b.ptr(); } }; template inline NonnullRefPtr static_ptr_cast(const NonnullRefPtr& ptr) { return NonnullRefPtr(static_cast(*ptr)); } template inline RefPtr static_ptr_cast(const RefPtr& ptr) { return RefPtr(static_cast(ptr.ptr())); } } using AK::RefPtr; using AK::static_ptr_cast;