| // Copyright 2023 The Dawn & Tint Authors |
| // |
| // 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. |
| // |
| // 3. Neither the name of the copyright holder nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // 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. |
| |
| #ifndef SRC_DAWN_COMMON_WEAKREF_H_ |
| #define SRC_DAWN_COMMON_WEAKREF_H_ |
| |
| #include <utility> |
| |
| #include "dawn/common/Ref.h" |
| #include "dawn/common/WeakRefSupport.h" |
| |
| namespace dawn { |
| |
| template <typename T> |
| class WeakRef; |
| |
| template < |
| typename T, |
| typename = typename std::enable_if<std::is_base_of_v<detail::WeakRefSupportBase, T>>::type> |
| WeakRef<T> GetWeakRef(T* obj) { |
| return WeakRef<T>(obj); |
| } |
| |
| template < |
| typename T, |
| typename = typename std::enable_if<std::is_base_of_v<detail::WeakRefSupportBase, T>>::type> |
| WeakRef<T> GetWeakRef(const Ref<T>& obj) { |
| return GetWeakRef(obj.Get()); |
| } |
| |
| template <typename T> |
| class WeakRef { |
| public: |
| WeakRef() {} |
| |
| // Constructors from nullptr. |
| // NOLINTNEXTLINE(runtime/explicit) |
| constexpr WeakRef(std::nullptr_t) : WeakRef() {} |
| |
| // Constructors from a WeakRef<U>, where U can also equal T. |
| template <typename U, typename = typename std::enable_if<std::is_base_of_v<T, U>>::type> |
| WeakRef(const WeakRef<U>& other) : mData(other.mData) {} |
| template <typename U, typename = typename std::enable_if<std::is_base_of_v<T, U>>::type> |
| WeakRef<T>& operator=(const WeakRef<U>& other) { |
| mData = other.mData; |
| return *this; |
| } |
| template <typename U, typename = typename std::enable_if<std::is_base_of_v<T, U>>::type> |
| WeakRef(WeakRef<U>&& other) : mData(std::move(other.mData)) {} |
| template <typename U, typename = typename std::enable_if<std::is_base_of_v<T, U>>::type> |
| WeakRef<T>& operator=(WeakRef<U>&& other) { |
| if (&other != this) { |
| mData = std::move(other.mData); |
| } |
| return *this; |
| } |
| |
| // Constructor from explicit WeakRefSupport<T>* is allowed. |
| // NOLINTNEXTLINE(runtime/explicit) |
| WeakRef(WeakRefSupport<T>* support) : mData(support->mData) {} |
| |
| // Promotes a WeakRef to a Ref. Access to the raw pointer is not allowed because a raw pointer |
| // could become invalid after being retrieved. |
| Ref<T> Promote() const { |
| if (mData != nullptr) { |
| return AcquireRef(static_cast<T*>(mData->TryGetRef().Detach())); |
| } |
| return nullptr; |
| } |
| |
| // Returns the raw pointer to the RefCountedT if it has not been invalidated. Note that this |
| // function is not thread-safe since the returned pointer can become invalid after being |
| // retrieved. |
| T* UnsafeGet() const { |
| if (mData != nullptr) { |
| return static_cast<T*>(mData->UnsafeGet()); |
| } |
| return nullptr; |
| } |
| |
| friend WeakRef GetWeakRef<>(T* obj); |
| friend WeakRef GetWeakRef<>(const Ref<T>& obj); |
| |
| private: |
| // Friend is needed so that we can access the data ref in conversions. |
| template <typename U> |
| friend class WeakRef; |
| // Friend is needed to access the private constructor. |
| template <typename U> |
| friend class Ref; |
| |
| // Constructor from data should only be allowed via the GetWeakRef function. |
| explicit WeakRef(detail::WeakRefSupportBase* data) : mData(data->mData) {} |
| |
| Ref<detail::WeakRefData> mData = nullptr; |
| }; |
| |
| } // namespace dawn |
| |
| #endif // SRC_DAWN_COMMON_WEAKREFCOUNTED_H_ |