Home | History | Annotate | Download | only in fxcrt
      1 // Copyright 2016 PDFium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
      6 
      7 #ifndef CORE_FXCRT_WEAK_PTR_H_
      8 #define CORE_FXCRT_WEAK_PTR_H_
      9 
     10 #include <cstddef>
     11 #include <memory>
     12 #include <utility>
     13 
     14 #include "core/fxcrt/fx_system.h"
     15 #include "core/fxcrt/retain_ptr.h"
     16 
     17 namespace fxcrt {
     18 
     19 template <class T, class D = std::default_delete<T>>
     20 class WeakPtr {
     21  public:
     22   WeakPtr() = default;
     23   WeakPtr(const WeakPtr& that) : m_pHandle(that.m_pHandle) {}
     24   WeakPtr(WeakPtr&& that) noexcept { Swap(that); }
     25   explicit WeakPtr(std::unique_ptr<T, D> pObj)
     26       : m_pHandle(new Handle(std::move(pObj))) {}
     27 
     28   // Deliberately implicit to allow passing nullptr.
     29   // NOLINTNEXTLINE(runtime/explicit)
     30   WeakPtr(std::nullptr_t arg) {}
     31 
     32   explicit operator bool() const { return m_pHandle && !!m_pHandle->Get(); }
     33   bool HasOneRef() const { return m_pHandle && m_pHandle->HasOneRef(); }
     34   T* operator->() { return m_pHandle->Get(); }
     35   const T* operator->() const { return m_pHandle->Get(); }
     36   WeakPtr& operator=(const WeakPtr& that) {
     37     m_pHandle = that.m_pHandle;
     38     return *this;
     39   }
     40   bool operator==(const WeakPtr& that) const {
     41     return m_pHandle == that.m_pHandle;
     42   }
     43   bool operator!=(const WeakPtr& that) const { return !(*this == that); }
     44 
     45   T* Get() const { return m_pHandle ? m_pHandle->Get() : nullptr; }
     46   void DeleteObject() {
     47     if (m_pHandle) {
     48       m_pHandle->Clear();
     49       m_pHandle.Reset();
     50     }
     51   }
     52   void Reset() { m_pHandle.Reset(); }
     53   void Reset(std::unique_ptr<T, D> pObj) {
     54     m_pHandle.Reset(new Handle(std::move(pObj)));
     55   }
     56   void Swap(WeakPtr& that) { m_pHandle.Swap(that.m_pHandle); }
     57 
     58  private:
     59   class Handle {
     60    public:
     61     explicit Handle(std::unique_ptr<T, D> ptr)
     62         : m_nCount(0), m_pObj(std::move(ptr)) {}
     63     void Reset(std::unique_ptr<T, D> ptr) { m_pObj = std::move(ptr); }
     64     void Clear() {     // Now you're all weak ptrs ...
     65       m_pObj.reset();  // unique_ptr nulls first before invoking delete.
     66     }
     67     T* Get() const { return m_pObj.get(); }
     68     T* Retain() {
     69       ++m_nCount;
     70       return m_pObj.get();
     71     }
     72     void Release() {
     73       if (--m_nCount == 0)
     74         delete this;
     75     }
     76     bool HasOneRef() const { return m_nCount == 1; }
     77 
     78    private:
     79     ~Handle() = default;
     80 
     81     intptr_t m_nCount;
     82     std::unique_ptr<T, D> m_pObj;
     83   };
     84 
     85   RetainPtr<Handle> m_pHandle;
     86 };
     87 
     88 }  // namespace fxcrt
     89 
     90 using fxcrt::WeakPtr;
     91 
     92 #endif  // CORE_FXCRT_WEAK_PTR_H_
     93