OLD | NEW |
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "base/memory/singleton.h" | 5 #include "base/memory/singleton.h" |
6 #include "base/threading/platform_thread.h" | 6 #include "base/threading/platform_thread.h" |
7 | 7 |
8 namespace base { | 8 namespace base { |
9 namespace internal { | 9 namespace internal { |
10 | 10 |
11 subtle::AtomicWord WaitForInstance(subtle::AtomicWord* instance) { | 11 subtle::AtomicWord WaitForInstance(subtle::AtomicWord* instance) { |
12 // Handle the race. Another thread beat us and either: | 12 // Handle the race. Another thread beat us and either: |
13 // - Has the object in BeingCreated state | 13 // - Has the object in BeingCreated state |
14 // - Already has the object created... | 14 // - Already has the object created... |
15 // We know value != NULL. It could be kBeingCreatedMarker, or a valid ptr. | 15 // We know value != NULL. It could be kBeingCreatedMarker, or a valid ptr. |
16 // Unless your constructor can be very time consuming, it is very unlikely | 16 // Unless your constructor can be very time consuming, it is very unlikely |
17 // to hit this race. When it does, we just spin and yield the thread until | 17 // to hit this race. When it does, we just spin and yield the thread until |
18 // the object has been created. | 18 // the object has been created. |
19 subtle::AtomicWord value; | 19 subtle::AtomicWord value; |
20 while (true) { | 20 while (true) { |
21 value = subtle::NoBarrier_Load(instance); | 21 // The load has acquire memory ordering as the thread which reads the |
| 22 // instance pointer must acquire visibility over the associated data. |
| 23 // The pairing Release_Store operation is in Singleton::get(). |
| 24 value = subtle::Acquire_Load(instance); |
22 if (value != kBeingCreatedMarker) | 25 if (value != kBeingCreatedMarker) |
23 break; | 26 break; |
24 PlatformThread::YieldCurrentThread(); | 27 PlatformThread::YieldCurrentThread(); |
25 } | 28 } |
26 return value; | 29 return value; |
27 } | 30 } |
28 | 31 |
29 } // namespace internal | 32 } // namespace internal |
30 } // namespace base | 33 } // namespace base |
31 | 34 |
OLD | NEW |