| // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "base/memory/weak_ptr.h" |
| |
| namespace base { |
| namespace internal { |
| |
| WeakReference::Flag::Flag() { |
| // Flags only become bound when checked for validity, or invalidated, |
| // so that we can check that later validity/invalidation operations on |
| // the same Flag take place on the same sequenced thread. |
| DETACH_FROM_SEQUENCE(sequence_checker_); |
| } |
| |
| void WeakReference::Flag::Invalidate() { |
| // The flag being invalidated with a single ref implies that there are no |
| // weak pointers in existence. Allow deletion on other thread in this case. |
| #if DCHECK_IS_ON() |
| DCHECK(sequence_checker_.CalledOnValidSequence() || HasOneRef()) |
| << "WeakPtrs must be invalidated on the same sequenced thread."; |
| #endif |
| invalidated_.Set(); |
| } |
| |
| bool WeakReference::Flag::IsValid() const { |
| DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_) |
| << "WeakPtrs must be checked on the same sequenced thread."; |
| return !invalidated_.IsSet(); |
| } |
| |
| bool WeakReference::Flag::MaybeValid() const { |
| return !invalidated_.IsSet(); |
| } |
| |
| WeakReference::Flag::~Flag() = default; |
| |
| WeakReference::WeakReference() = default; |
| |
| WeakReference::WeakReference(const scoped_refptr<Flag>& flag) : flag_(flag) {} |
| |
| WeakReference::~WeakReference() = default; |
| |
| WeakReference::WeakReference(WeakReference&& other) = default; |
| |
| WeakReference::WeakReference(const WeakReference& other) = default; |
| |
| bool WeakReference::IsValid() const { |
| return flag_ && flag_->IsValid(); |
| } |
| |
| bool WeakReference::MaybeValid() const { |
| return flag_ && flag_->MaybeValid(); |
| } |
| |
| WeakReferenceOwner::WeakReferenceOwner() = default; |
| |
| WeakReferenceOwner::~WeakReferenceOwner() { |
| Invalidate(); |
| } |
| |
| WeakReference WeakReferenceOwner::GetRef() const { |
| // If we hold the last reference to the Flag then create a new one. |
| if (!HasRefs()) |
| flag_ = new WeakReference::Flag(); |
| |
| return WeakReference(flag_); |
| } |
| |
| void WeakReferenceOwner::Invalidate() { |
| if (flag_) { |
| flag_->Invalidate(); |
| flag_ = nullptr; |
| } |
| } |
| |
| WeakPtrBase::WeakPtrBase() : ptr_(0) {} |
| |
| WeakPtrBase::~WeakPtrBase() = default; |
| |
| WeakPtrBase::WeakPtrBase(const WeakReference& ref, uintptr_t ptr) |
| : ref_(ref), ptr_(ptr) { |
| DCHECK(ptr_); |
| } |
| |
| WeakPtrFactoryBase::WeakPtrFactoryBase(uintptr_t ptr) : ptr_(ptr) { |
| DCHECK(ptr_); |
| } |
| |
| WeakPtrFactoryBase::~WeakPtrFactoryBase() { |
| ptr_ = 0; |
| } |
| |
| } // namespace internal |
| } // namespace base |