|  | // Copyright (c) 2012 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/supports_user_data.h" | 
|  |  | 
|  | namespace base { | 
|  |  | 
|  | SupportsUserData::SupportsUserData() { | 
|  | // Harmless to construct on a different execution sequence to subsequent | 
|  | // usage. | 
|  | sequence_checker_.DetachFromSequence(); | 
|  | } | 
|  |  | 
|  | SupportsUserData::Data* SupportsUserData::GetUserData(const void* key) const { | 
|  | DCHECK(sequence_checker_.CalledOnValidSequence()); | 
|  | // Avoid null keys; they are too vulnerable to collision. | 
|  | DCHECK(key); | 
|  | DataMap::const_iterator found = user_data_.find(key); | 
|  | if (found != user_data_.end()) | 
|  | return found->second.get(); | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | void SupportsUserData::SetUserData(const void* key, | 
|  | std::unique_ptr<Data> data) { | 
|  | DCHECK(sequence_checker_.CalledOnValidSequence()); | 
|  | // Avoid null keys; they are too vulnerable to collision. | 
|  | DCHECK(key); | 
|  | user_data_[key] = std::move(data); | 
|  | } | 
|  |  | 
|  | void SupportsUserData::RemoveUserData(const void* key) { | 
|  | DCHECK(sequence_checker_.CalledOnValidSequence()); | 
|  | user_data_.erase(key); | 
|  | } | 
|  |  | 
|  | void SupportsUserData::DetachFromSequence() { | 
|  | sequence_checker_.DetachFromSequence(); | 
|  | } | 
|  |  | 
|  | SupportsUserData::~SupportsUserData() { | 
|  | DCHECK(sequence_checker_.CalledOnValidSequence() || user_data_.empty()); | 
|  | DataMap local_user_data; | 
|  | user_data_.swap(local_user_data); | 
|  | // Now this->user_data_ is empty, and any destructors called transitively from | 
|  | // the destruction of |local_user_data| will see it that way instead of | 
|  | // examining a being-destroyed object. | 
|  | } | 
|  |  | 
|  | }  // namespace base |