// 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) {} 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