// Copyright 2017 PDFium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef CORE_FXCRT_CFX_UNOWNED_PTR_H_ #define CORE_FXCRT_CFX_UNOWNED_PTR_H_ #include #include #include #include template class CFX_UnownedPtr { public: CFX_UnownedPtr() {} CFX_UnownedPtr(const CFX_UnownedPtr& that) : CFX_UnownedPtr(that.Get()) {} template explicit CFX_UnownedPtr(U* pObj) : m_pObj(pObj) {} // Deliberately implicit to allow returning nullptrs. // NOLINTNEXTLINE(runtime/explicit) CFX_UnownedPtr(std::nullptr_t ptr) {} ~CFX_UnownedPtr() { Probe(); } CFX_UnownedPtr& operator=(T* that) { Probe(); m_pObj = that; return *this; } CFX_UnownedPtr& operator=(const CFX_UnownedPtr& that) { Probe(); if (*this != that) m_pObj = that.Get(); return *this; } bool operator==(const CFX_UnownedPtr& that) const { return Get() == that.Get(); } bool operator!=(const CFX_UnownedPtr& that) const { return !(*this == that); } bool operator<(const CFX_UnownedPtr& that) const { return std::less()(Get(), that.Get()); } template bool operator==(const U* that) const { return Get() == that; } template bool operator!=(const U* that) const { return !(*this == that); } T* Get() const { return m_pObj; } T* Release() { Probe(); T* pTemp = nullptr; std::swap(pTemp, m_pObj); return pTemp; } explicit operator bool() const { return !!m_pObj; } T& operator*() const { return *m_pObj; } T* operator->() const { return m_pObj; } private: inline void Probe() { #if defined(MEMORY_TOOL_REPLACES_ALLOCATOR) if (m_pObj) reinterpret_cast(m_pObj)[0]; #endif } T* m_pObj = nullptr; }; template inline bool operator==(const U* lhs, const CFX_UnownedPtr& rhs) { return rhs == lhs; } template inline bool operator!=(const U* lhs, const CFX_UnownedPtr& rhs) { return rhs != lhs; } #endif // CORE_FXCRT_CFX_UNOWNED_PTR_H_