| OLD | NEW |
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 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 "platform/heap/SafePoint.h" | 5 #include "platform/heap/SafePoint.h" |
| 6 | 6 |
| 7 #include "platform/heap/Heap.h" |
| 7 #include "wtf/Atomics.h" | 8 #include "wtf/Atomics.h" |
| 8 #include "wtf/CurrentTime.h" | 9 #include "wtf/CurrentTime.h" |
| 9 | 10 |
| 10 namespace blink { | 11 namespace blink { |
| 11 | 12 |
| 12 using PushAllRegistersCallback = void (*)(SafePointBarrier*, ThreadState*, intpt
r_t*); | 13 using PushAllRegistersCallback = void (*)(SafePointBarrier*, ThreadState*, intpt
r_t*); |
| 13 extern "C" void pushAllRegisters(SafePointBarrier*, ThreadState*, PushAllRegiste
rsCallback); | 14 extern "C" void pushAllRegisters(SafePointBarrier*, ThreadState*, PushAllRegiste
rsCallback); |
| 14 | 15 |
| 15 static double lockingTimeout() | 16 static double lockingTimeout() |
| 16 { | 17 { |
| (...skipping 10 matching lines...) Expand all Loading... |
| 27 SafePointBarrier::~SafePointBarrier() | 28 SafePointBarrier::~SafePointBarrier() |
| 28 { | 29 { |
| 29 } | 30 } |
| 30 | 31 |
| 31 bool SafePointBarrier::parkOthers() | 32 bool SafePointBarrier::parkOthers() |
| 32 { | 33 { |
| 33 ASSERT(ThreadState::current()->isAtSafePoint()); | 34 ASSERT(ThreadState::current()->isAtSafePoint()); |
| 34 | 35 |
| 35 ThreadState* current = ThreadState::current(); | 36 ThreadState* current = ThreadState::current(); |
| 36 // Lock threadAttachMutex() to prevent threads from attaching. | 37 // Lock threadAttachMutex() to prevent threads from attaching. |
| 37 ThreadState::lockThreadAttachMutex(); | 38 current->lockThreadAttachMutex(); |
| 38 ThreadState::AttachedThreadStateSet& threads = ThreadState::attachedThreads(
); | 39 const ThreadStateSet& threads = current->heap().threads(); |
| 39 | 40 |
| 40 MutexLocker locker(m_mutex); | 41 MutexLocker locker(m_mutex); |
| 41 atomicAdd(&m_unparkedThreadCount, threads.size()); | 42 atomicAdd(&m_unparkedThreadCount, threads.size()); |
| 42 releaseStore(&m_canResume, 0); | 43 releaseStore(&m_canResume, 0); |
| 43 | 44 |
| 44 for (ThreadState* state : threads) { | 45 for (ThreadState* state : threads) { |
| 45 if (state == current) | 46 if (state == current) |
| 46 continue; | 47 continue; |
| 47 | 48 |
| 48 for (auto& interruptor : state->interruptors()) | 49 for (auto& interruptor : state->interruptors()) |
| 49 interruptor->requestInterrupt(); | 50 interruptor->requestInterrupt(); |
| 50 } | 51 } |
| 51 | 52 |
| 52 while (acquireLoad(&m_unparkedThreadCount) > 0) { | 53 while (acquireLoad(&m_unparkedThreadCount) > 0) { |
| 53 double expirationTime = currentTime() + lockingTimeout(); | 54 double expirationTime = currentTime() + lockingTimeout(); |
| 54 if (!m_parked.timedWait(m_mutex, expirationTime)) { | 55 if (!m_parked.timedWait(m_mutex, expirationTime)) { |
| 55 // One of the other threads did not return to a safepoint within the
maximum | 56 // One of the other threads did not return to a safepoint within the
maximum |
| 56 // time we allow for threads to be parked. Abandon the GC and resume
the | 57 // time we allow for threads to be parked. Abandon the GC and resume
the |
| 57 // currently parked threads. | 58 // currently parked threads. |
| 58 resumeOthers(true); | 59 resumeOthers(true); |
| 59 return false; | 60 return false; |
| 60 } | 61 } |
| 61 } | 62 } |
| 62 return true; | 63 return true; |
| 63 } | 64 } |
| 64 | 65 |
| 65 void SafePointBarrier::resumeOthers(bool barrierLocked) | 66 void SafePointBarrier::resumeOthers(bool barrierLocked) |
| 66 { | 67 { |
| 67 ThreadState::AttachedThreadStateSet& threads = ThreadState::attachedThreads(
); | 68 ThreadState* current = ThreadState::current(); |
| 69 const ThreadStateSet& threads = current->heap().threads(); |
| 68 atomicSubtract(&m_unparkedThreadCount, threads.size()); | 70 atomicSubtract(&m_unparkedThreadCount, threads.size()); |
| 69 releaseStore(&m_canResume, 1); | 71 releaseStore(&m_canResume, 1); |
| 70 | 72 |
| 71 if (UNLIKELY(barrierLocked)) { | 73 if (UNLIKELY(barrierLocked)) { |
| 72 m_resume.broadcast(); | 74 m_resume.broadcast(); |
| 73 } else { | 75 } else { |
| 74 // FIXME: Resumed threads will all contend for m_mutex just | 76 // FIXME: Resumed threads will all contend for m_mutex just |
| 75 // to unlock it later which is a waste of resources. | 77 // to unlock it later which is a waste of resources. |
| 76 MutexLocker locker(m_mutex); | 78 MutexLocker locker(m_mutex); |
| 77 m_resume.broadcast(); | 79 m_resume.broadcast(); |
| 78 } | 80 } |
| 79 | 81 |
| 80 ThreadState::unlockThreadAttachMutex(); | 82 current->unlockThreadAttachMutex(); |
| 81 ASSERT(ThreadState::current()->isAtSafePoint()); | 83 ASSERT(ThreadState::current()->isAtSafePoint()); |
| 82 } | 84 } |
| 83 | 85 |
| 84 void SafePointBarrier::checkAndPark(ThreadState* state, SafePointAwareMutexLocke
r* locker) | 86 void SafePointBarrier::checkAndPark(ThreadState* state, SafePointAwareMutexLocke
r* locker) |
| 85 { | 87 { |
| 86 ASSERT(!state->sweepForbidden()); | 88 ASSERT(!state->sweepForbidden()); |
| 87 if (!acquireLoad(&m_canResume)) { | 89 if (!acquireLoad(&m_canResume)) { |
| 88 // If we are leaving the safepoint from a SafePointAwareMutexLocker | 90 // If we are leaving the safepoint from a SafePointAwareMutexLocker |
| 89 // call out to release the lock before going to sleep. This enables the | 91 // call out to release the lock before going to sleep. This enables the |
| 90 // lock to be acquired in the sweep phase, e.g. during weak processing | 92 // lock to be acquired in the sweep phase, e.g. during weak processing |
| (...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 131 // If no other thread is waiting for other threads to park then | 133 // If no other thread is waiting for other threads to park then |
| 132 // this counter can be negative: if N threads are at safe-points | 134 // this counter can be negative: if N threads are at safe-points |
| 133 // the counter will be -N. | 135 // the counter will be -N. |
| 134 if (!atomicDecrement(&m_unparkedThreadCount)) { | 136 if (!atomicDecrement(&m_unparkedThreadCount)) { |
| 135 MutexLocker locker(m_mutex); | 137 MutexLocker locker(m_mutex); |
| 136 m_parked.signal(); // Safe point reached. | 138 m_parked.signal(); // Safe point reached. |
| 137 } | 139 } |
| 138 } | 140 } |
| 139 | 141 |
| 140 } // namespace blink | 142 } // namespace blink |
| OLD | NEW |