| // Copyright (c) 2010 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/threading/thread_collision_warner.h" | 
 |  | 
 | #include "base/logging.h" | 
 | #include "base/threading/platform_thread.h" | 
 |  | 
 | namespace base { | 
 |  | 
 | void DCheckAsserter::warn() { | 
 |   NOTREACHED() << "Thread Collision"; | 
 | } | 
 |  | 
 | static subtle::Atomic32 CurrentThread() { | 
 |   const PlatformThreadId current_thread_id = PlatformThread::CurrentId(); | 
 |   // We need to get the thread id into an atomic data type. This might be a | 
 |   // truncating conversion, but any loss-of-information just increases the | 
 |   // chance of a fault negative, not a false positive. | 
 |   const subtle::Atomic32 atomic_thread_id = | 
 |       static_cast<subtle::Atomic32>(current_thread_id); | 
 |  | 
 |   return atomic_thread_id; | 
 | } | 
 |  | 
 | void ThreadCollisionWarner::EnterSelf() { | 
 |   // If the active thread is 0 then I'll write the current thread ID | 
 |   // if two or more threads arrive here only one will succeed to | 
 |   // write on valid_thread_id_ the current thread ID. | 
 |   subtle::Atomic32 current_thread_id = CurrentThread(); | 
 |  | 
 |   int previous_value = subtle::NoBarrier_CompareAndSwap(&valid_thread_id_, | 
 |                                                         0, | 
 |                                                         current_thread_id); | 
 |   if (previous_value != 0 && previous_value != current_thread_id) { | 
 |     // gotcha! a thread is trying to use the same class and that is | 
 |     // not current thread. | 
 |     asserter_->warn(); | 
 |   } | 
 |  | 
 |   subtle::NoBarrier_AtomicIncrement(&counter_, 1); | 
 | } | 
 |  | 
 | void ThreadCollisionWarner::Enter() { | 
 |   subtle::Atomic32 current_thread_id = CurrentThread(); | 
 |  | 
 |   if (subtle::NoBarrier_CompareAndSwap(&valid_thread_id_, | 
 |                                        0, | 
 |                                        current_thread_id) != 0) { | 
 |     // gotcha! another thread is trying to use the same class. | 
 |     asserter_->warn(); | 
 |   } | 
 |  | 
 |   subtle::NoBarrier_AtomicIncrement(&counter_, 1); | 
 | } | 
 |  | 
 | void ThreadCollisionWarner::Leave() { | 
 |   if (subtle::Barrier_AtomicIncrement(&counter_, -1) == 0) { | 
 |     subtle::NoBarrier_Store(&valid_thread_id_, 0); | 
 |   } | 
 | } | 
 |  | 
 | }  // namespace base |