Changeset 189009 in webkit
- Timestamp:
- Aug 26, 2015, 7:49:52 PM (10 years ago)
- Location:
- trunk
- Files:
-
- 1 added
- 22 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/LayoutTests/ChangeLog
r189004 r189009 1 2015-08-26 Mark Lam <mark.lam@apple.com> 2 3 watchdog m_didFire state erroneously retained. 4 https://bugs.webkit.org/show_bug.cgi?id=131082 5 6 Reviewed by Geoffrey Garen. 7 8 * fast/workers/worker-terminate-forever-expected.txt: 9 * fast/workers/worker-terminate-forever.html: 10 - Updated to check if the worker actually did terminate. 11 1 12 2015-08-26 Andy Estes <aestes@apple.com> 2 13 -
trunk/LayoutTests/fast/workers/worker-terminate-forever-expected.txt
r57192 r189009 1 CONSOLE MESSAGE: line 15: Worker was started 2 CONSOLE MESSAGE: line 34: Worker was terminated 1 3 Test Worker.terminate() for a worker that tries to run forever. -
trunk/LayoutTests/fast/workers/worker-terminate-forever.html
r124680 r189009 2 2 <p>Test Worker.terminate() for a worker that tries to run forever.</p> 3 3 <script> 4 if (window.testRunner) 4 if (window.testRunner) { 5 5 testRunner.dumpAsText(); 6 testRunner.waitUntilDone(); 7 } 6 8 7 9 var worker = new Worker('resources/worker-run-forever.js'); 8 worker.terminate(); 10 11 function waitForWorkerToStart() { 12 var startTime = Date.now(); 13 function checkIfWorkerStarted() { 14 if (internals.workerThreadCount == 1) { 15 console.log("Worker was started"); 16 worker.terminate(); 17 setTimeout(waitForWorkerToStop, 0); 18 19 } else if (Date.now() - startTime < 5000) { 20 setTimeout(checkIfWorkerStarted, 0); 21 22 } else { 23 console.log("Worker did not show up"); 24 testRunner.notifyDone(); 25 } 26 } 27 setTimeout(checkIfWorkerStarted, 0); 28 } 29 30 function waitForWorkerToStop() { 31 var startTime = Date.now(); 32 function checkIfWorkerStopped() { 33 if (internals.workerThreadCount == 0) { 34 console.log("Worker was terminated"); 35 testRunner.notifyDone(); 36 37 } else if (Date.now() - startTime < 5000) { 38 setTimeout(checkIfWorkerStopped, 0); 39 40 } else { 41 console.log("Did not see worker terminate"); 42 testRunner.notifyDone(); 43 } 44 } 45 setTimeout(checkIfWorkerStopped, 0); 46 } 47 48 window.setTimeout(waitForWorkerToStart, 0); 49 9 50 </script> 10 51 </body> -
trunk/Source/JavaScriptCore/API/JSContextRef.cpp
r188338 r189009 100 100 if (callback) { 101 101 void* callbackPtr = reinterpret_cast<void*>(callback); 102 watchdog.setTimeLimit( vm,std::chrono::duration_cast<std::chrono::microseconds>(std::chrono::duration<double>(limit)), internalScriptTimeoutCallback, callbackPtr, callbackData);102 watchdog.setTimeLimit(std::chrono::duration_cast<std::chrono::microseconds>(std::chrono::duration<double>(limit)), internalScriptTimeoutCallback, callbackPtr, callbackData); 103 103 } else 104 watchdog.setTimeLimit( vm,std::chrono::duration_cast<std::chrono::microseconds>(std::chrono::duration<double>(limit)));104 watchdog.setTimeLimit(std::chrono::duration_cast<std::chrono::microseconds>(std::chrono::duration<double>(limit))); 105 105 } 106 106 … … 110 110 JSLockHolder locker(&vm); 111 111 if (vm.watchdog) 112 vm.watchdog->setTimeLimit( vm,Watchdog::noTimeLimit);112 vm.watchdog->setTimeLimit(Watchdog::noTimeLimit); 113 113 } 114 114 -
trunk/Source/JavaScriptCore/API/tests/ExecutionTimeLimitTest.cpp
r188689 r189009 84 84 }; 85 85 86 static void testResetAfterTimeout(bool& failed) 87 { 88 JSValueRef v = nullptr; 89 JSValueRef exception = nullptr; 90 const char* reentryScript = "100"; 91 JSStringRef script = JSStringCreateWithUTF8CString(reentryScript); 92 v = JSEvaluateScript(context, script, nullptr, nullptr, 1, &exception); 93 if (exception) { 94 printf("FAIL: Watchdog timeout was not reset.\n"); 95 failed = true; 96 } else if (!JSValueIsNumber(context, v) || JSValueToNumber(context, v, nullptr) != 100) { 97 printf("FAIL: Script result is not as expected.\n"); 98 failed = true; 99 } 100 } 101 86 102 int testExecutionTimeLimit() 87 103 { … … 153 169 failed = true; 154 170 } 171 172 testResetAfterTimeout(failed); 155 173 } 156 174 … … 188 206 failed = true; 189 207 } 208 209 testResetAfterTimeout(failed); 190 210 } 191 211 … … 223 243 failed = true; 224 244 } 245 246 testResetAfterTimeout(failed); 225 247 } 226 248 -
trunk/Source/JavaScriptCore/ChangeLog
r189002 r189009 1 2015-08-26 Mark Lam <mark.lam@apple.com> 2 3 watchdog m_didFire state erroneously retained. 4 https://bugs.webkit.org/show_bug.cgi?id=131082 5 6 Reviewed by Geoffrey Garen. 7 8 The watchdog can fire for 2 reasons: 9 1. an external controlling entity (i.e. another thread) has scheduled termination 10 of the script thread via watchdog::terminateSoon(). 11 2. the allowed CPU time has expired. 12 13 For case 1, we're doing away with the m_didFire flag. Watchdog::terminateSoon() 14 will set the timer deadlines and m_timeLimit to 0, and m_timerDidFire to true. 15 This will get the script thread to check Watchdog::didFire() and terminate 16 execution. 17 18 Note: the watchdog only guarantees that script execution will terminate as soon 19 as possible due to a time limit of 0. Once we've exited the VM, the client of the 20 VM is responsible from keeping a flag to prevent new script execution. 21 22 In a race condition, if terminateSoon() is called just after execution has gotten 23 past the client's reentry check and the client is in the process of re-entering, 24 the worst that can happen is that we will schedule the watchdog timer to fire 25 after a period of 0. This will terminate script execution quickly, and thereafter 26 the client's check should be able to prevent further entry into the VM. 27 28 The correctness (i.e. has no race condition) of this type of termination relies 29 on the termination state being sticky. Once the script thread is terminated this 30 way, the VM will continue to terminate scripts quickly until the client sets the 31 time limit to a non-zero value (or clears it which sets the time limit to 32 noTimeLimit). 33 34 For case 2, the watchdog does not alter m_timeLimit. If the CPU deadline has 35 been reached, the script thread will terminate execution and exit the VM. 36 37 If the client of the VM starts new script execution, the watchdog will allow 38 execution for the specified m_timeLimit. In this case, since m_timeLimit is not 39 0, the script gets a fresh allowance of CPU time to execute. Hence, terminations 40 due to watchdog time outs are no longer sticky. 41 42 * API/JSContextRef.cpp: 43 (JSContextGroupSetExecutionTimeLimit): 44 (JSContextGroupClearExecutionTimeLimit): 45 * API/tests/ExecutionTimeLimitTest.cpp: 46 - Add test scenarios to verify that the watchdog is automatically reset by the VM 47 upon throwing the TerminatedExecutionException. 48 49 (testResetAfterTimeout): 50 (testExecutionTimeLimit): 51 * JavaScriptCore.vcxproj/JavaScriptCore.vcxproj: 52 * JavaScriptCore.vcxproj/JavaScriptCore.vcxproj.filters: 53 * JavaScriptCore.xcodeproj/project.pbxproj: 54 * dfg/DFGByteCodeParser.cpp: 55 (JSC::DFG::ByteCodeParser::parseBlock): 56 * interpreter/Interpreter.cpp: 57 (JSC::Interpreter::execute): 58 (JSC::Interpreter::executeCall): 59 (JSC::Interpreter::executeConstruct): 60 * jit/JITOpcodes.cpp: 61 (JSC::JIT::emit_op_loop_hint): 62 (JSC::JIT::emitSlow_op_loop_hint): 63 * jit/JITOperations.cpp: 64 * llint/LLIntSlowPaths.cpp: 65 (JSC::LLInt::LLINT_SLOW_PATH_DECL): 66 * runtime/VM.cpp: 67 (JSC::VM::VM): 68 (JSC::VM::ensureWatchdog): 69 * runtime/VM.h: 70 * runtime/VMInlines.h: Added. 71 (JSC::VM::shouldTriggerTermination): 72 * runtime/Watchdog.cpp: 73 (JSC::Watchdog::Watchdog): 74 (JSC::Watchdog::setTimeLimit): 75 (JSC::Watchdog::terminateSoon): 76 (JSC::Watchdog::didFireSlow): 77 (JSC::Watchdog::hasTimeLimit): 78 (JSC::Watchdog::enteredVM): 79 (JSC::Watchdog::exitedVM): 80 (JSC::Watchdog::startTimer): 81 (JSC::Watchdog::stopTimer): 82 (JSC::Watchdog::hasStartedTimer): Deleted. 83 (JSC::Watchdog::fire): Deleted. 84 * runtime/Watchdog.h: 85 (JSC::Watchdog::didFire): 86 (JSC::Watchdog::timerDidFireAddress): 87 1 88 2015-08-26 Joseph Pecoraro <pecoraro@apple.com> 2 89 -
trunk/Source/JavaScriptCore/JavaScriptCore.vcxproj/JavaScriptCore.vcxproj
r188979 r189009 1771 1771 <ClInclude Include="..\runtime\Uint8Array.h" /> 1772 1772 <ClInclude Include="..\runtime\VM.h" /> 1773 <ClInclude Include="..\runtime\VMInlines.h" /> 1773 1774 <ClInclude Include="..\runtime\VMEntryScope.h" /> 1774 1775 <ClInclude Include="..\runtime\VarOffset.h" /> -
trunk/Source/JavaScriptCore/JavaScriptCore.vcxproj/JavaScriptCore.vcxproj.filters
r188796 r189009 3268 3268 <Filter>runtime</Filter> 3269 3269 </ClInclude> 3270 <ClInclude Include="..\runtime\VMInlines.h"> 3271 <Filter>runtime</Filter> 3272 </ClInclude> 3270 3273 <ClInclude Include="..\runtime\VMEntryScope.h"> 3271 3274 <Filter>runtime</Filter> -
trunk/Source/JavaScriptCore/JavaScriptCore.xcodeproj/project.pbxproj
r188979 r189009 3638 3638 FE7BA60D1A1A7CEC00F1F7B4 /* HeapVerifier.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = HeapVerifier.cpp; sourceTree = "<group>"; }; 3639 3639 FE7BA60E1A1A7CEC00F1F7B4 /* HeapVerifier.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = HeapVerifier.h; sourceTree = "<group>"; }; 3640 FE90BB3A1B7CF64E006B3F03 /* VMInlines.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = VMInlines.h; sourceTree = "<group>"; }; 3640 3641 FEA0861E182B7A0400F6D851 /* Breakpoint.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = Breakpoint.h; sourceTree = "<group>"; }; 3641 3642 FEA0861F182B7A0400F6D851 /* DebuggerPrimitives.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = DebuggerPrimitives.h; sourceTree = "<group>"; }; … … 4999 5000 E18E3A570DF9278C00D90B34 /* VM.cpp */, 5000 5001 E18E3A560DF9278C00D90B34 /* VM.h */, 5002 FE90BB3A1B7CF64E006B3F03 /* VMInlines.h */, 5001 5003 FE5932A5183C5A2600A1ECCC /* VMEntryScope.cpp */, 5002 5004 FE5932A6183C5A2600A1ECCC /* VMEntryScope.h */, -
trunk/Source/JavaScriptCore/dfg/DFGByteCodeParser.cpp
r188979 r189009 4108 4108 addToGraph(LoopHint); 4109 4109 4110 if (m_vm->watchdog && m_vm->watchdog->hasTimeLimit())4110 if (m_vm->watchdog) 4111 4111 addToGraph(CheckWatchdogTimer); 4112 4112 -
trunk/Source/JavaScriptCore/interpreter/Interpreter.cpp
r188583 r189009 72 72 #include "Symbol.h" 73 73 #include "VMEntryScope.h" 74 #include "VMInlines.h" 74 75 #include "VirtualRegister.h" 75 #include "Watchdog.h"76 76 77 77 #include <limits.h> … … 866 866 ProgramCodeBlock* codeBlock = program->codeBlock(); 867 867 868 if (UNLIKELY(vm. watchdog && vm.watchdog->didFire(callFrame)))868 if (UNLIKELY(vm.shouldTriggerTermination(callFrame))) 869 869 return throwTerminatedExecutionException(callFrame); 870 870 … … 929 929 newCodeBlock = 0; 930 930 931 if (UNLIKELY(vm. watchdog && vm.watchdog->didFire(callFrame)))931 if (UNLIKELY(vm.shouldTriggerTermination(callFrame))) 932 932 return throwTerminatedExecutionException(callFrame); 933 933 … … 999 999 newCodeBlock = 0; 1000 1000 1001 if (UNLIKELY(vm. watchdog && vm.watchdog->didFire(callFrame)))1001 if (UNLIKELY(vm.shouldTriggerTermination(callFrame))) 1002 1002 return throwTerminatedExecutionException(callFrame); 1003 1003 … … 1072 1072 profiler->willExecute(closure.oldCallFrame, closure.function); 1073 1073 1074 if (UNLIKELY(vm. watchdog && vm.watchdog->didFire(closure.oldCallFrame)))1074 if (UNLIKELY(vm.shouldTriggerTermination(closure.oldCallFrame))) 1075 1075 return throwTerminatedExecutionException(closure.oldCallFrame); 1076 1076 … … 1153 1153 } 1154 1154 1155 if (UNLIKELY(vm. watchdog && vm.watchdog->didFire(callFrame)))1155 if (UNLIKELY(vm.shouldTriggerTermination(callFrame))) 1156 1156 return throwTerminatedExecutionException(callFrame); 1157 1157 -
trunk/Source/JavaScriptCore/jit/JITOpcodes.cpp
r188545 r189009 916 916 917 917 // Emit the watchdog timer check: 918 if (m_vm->watchdog && m_vm->watchdog->hasTimeLimit())918 if (m_vm->watchdog) 919 919 addSlowCase(branchTest8(NonZero, AbsoluteAddress(m_vm->watchdog->timerDidFireAddress()))); 920 920 } … … 942 942 943 943 // Emit the slow path of the watchdog timer check: 944 if (m_vm->watchdog && m_vm->watchdog->hasTimeLimit()) {944 if (m_vm->watchdog) { 945 945 linkSlowCase(iter); 946 946 callOperation(operationHandleWatchdogTimer); -
trunk/Source/JavaScriptCore/jit/JITOperations.cpp
r188932 r189009 59 59 #include "TestRunnerUtils.h" 60 60 #include "TypeProfilerLog.h" 61 #include " Watchdog.h"61 #include "VMInlines.h" 62 62 #include <wtf/InlineASM.h> 63 63 … … 1095 1095 NativeCallFrameTracer tracer(&vm, exec); 1096 1096 1097 if (UNLIKELY(vm. watchdog && vm.watchdog->didFire(exec)))1097 if (UNLIKELY(vm.shouldTriggerTermination(exec))) 1098 1098 vm.throwException(exec, createTerminatedExecutionException(&vm)); 1099 1099 -
trunk/Source/JavaScriptCore/llint/LLIntSlowPaths.cpp
r188545 r189009 55 55 #include "ProtoCallFrame.h" 56 56 #include "StructureRareDataInlines.h" 57 #include " Watchdog.h"57 #include "VMInlines.h" 58 58 #include <wtf/StringPrintStream.h> 59 59 … … 1304 1304 LLINT_BEGIN_NO_SET_PC(); 1305 1305 ASSERT(vm.watchdog); 1306 if (UNLIKELY(vm. watchdog->didFire(exec)))1306 if (UNLIKELY(vm.shouldTriggerTermination(exec))) 1307 1307 LLINT_THROW(createTerminatedExecutionException(&vm)); 1308 1308 LLINT_RETURN_TWO(0, exec); -
trunk/Source/JavaScriptCore/runtime/VM.cpp
r188884 r189009 293 293 std::chrono::milliseconds timeoutMillis(Options::watchdog()); 294 294 Watchdog& watchdog = ensureWatchdog(); 295 watchdog.setTimeLimit( *this,timeoutMillis);295 watchdog.setTimeLimit(timeoutMillis); 296 296 } 297 297 } … … 389 389 // same as a plain C++ pointer, and loads the address of Watchdog from it. 390 390 RELEASE_ASSERT(*reinterpret_cast<Watchdog**>(&watchdog) == watchdog.get()); 391 392 // And if we've previously compiled any functions, we need to revert 393 // them because they don't have the needed polling checks for the watchdog 394 // yet. 395 deleteAllCode(); 391 396 } 392 397 return *watchdog; -
trunk/Source/JavaScriptCore/runtime/VM.h
r188846 r189009 237 237 JS_EXPORT_PRIVATE ~VM(); 238 238 239 Watchdog& ensureWatchdog();239 JS_EXPORT_PRIVATE Watchdog& ensureWatchdog(); 240 240 241 241 private: … … 566 566 JS_EXPORT_PRIVATE void drainMicrotasks(); 567 567 568 inline bool shouldTriggerTermination(ExecState*); 569 568 570 private: 569 571 friend class LLIntOffsetsExtractor; -
trunk/Source/JavaScriptCore/runtime/Watchdog.cpp
r188394 r189009 43 43 Watchdog::Watchdog() 44 44 : m_timerDidFire(false) 45 , m_didFire(false)46 45 , m_timeLimit(noTimeLimit) 47 46 , m_cpuDeadline(noTimeLimit) … … 53 52 { 54 53 m_timerHandler = [this] { 54 LockHolder locker(m_lock); 55 55 this->m_timerDidFire = true; 56 56 this->deref(); … … 58 58 } 59 59 60 inline bool Watchdog::hasStartedTimer() 61 { 62 return m_cpuDeadline != noTimeLimit; 63 } 64 65 void Watchdog::setTimeLimit(VM& vm, std::chrono::microseconds limit, 60 void Watchdog::setTimeLimit(std::chrono::microseconds limit, 66 61 ShouldTerminateCallback callback, void* data1, void* data2) 67 62 { 68 bool hadTimeLimit = hasTimeLimit(); 69 70 m_didFire = false; // Reset the watchdog. 63 LockHolder locker(m_lock); 71 64 72 65 m_timeLimit = limit; … … 75 68 m_callbackData2 = data2; 76 69 77 // If this is the first time that time limit is being enabled, then any 78 // previously JIT compiled code will not have the needed polling checks. 79 // Hence, we need to flush all the pre-existing compiled code. 80 // 81 // However, if the time limit is already enabled, and we're just changing the 82 // time limit value, then any existing JITted code will have the appropriate 83 // polling checks. Hence, there is no need to re-do this flushing. 84 if (!hadTimeLimit) { 85 // And if we've previously compiled any functions, we need to revert 86 // them because they don't have the needed polling checks yet. 87 vm.deleteAllCode(); 88 } 70 if (m_hasEnteredVM && hasTimeLimit()) 71 startTimer(locker, m_timeLimit); 72 } 89 73 90 if (m_hasEnteredVM && hasTimeLimit()) 91 startTimer(m_timeLimit); 74 JS_EXPORT_PRIVATE void Watchdog::terminateSoon() 75 { 76 LockHolder locker(m_lock); 77 78 m_timeLimit = std::chrono::microseconds(0); 79 m_cpuDeadline = std::chrono::microseconds(0); 80 m_wallClockDeadline = std::chrono::microseconds(0); 81 m_timerDidFire = true; 92 82 } 93 83 94 84 bool Watchdog::didFireSlow(ExecState* exec) 95 85 { 96 ASSERT(m_timerDidFire);97 m_timerDidFire = false;86 { 87 LockHolder locker(m_lock); 98 88 99 // A legitimate timer is the timer which woke up watchdog and can caused didFireSlow() to be 100 // called after m_wallClockDeadline has expired. All other timers are considered to be stale, 101 // and their wake ups are considered to be spurious and should be ignored. 102 // 103 // didFireSlow() will race against m_timerHandler to clear and set m_timerDidFire respectively. 104 // We need to deal with a variety of scenarios where we can have stale timers and legitimate 105 // timers firing in close proximity to each other. 106 // 107 // Consider the following scenarios: 108 // 109 // 1. A stale timer fires long before m_wallClockDeadline expires. 110 // 111 // In this case, the m_wallClockDeadline check below will fail and the stale timer will be 112 // ignored as spurious. We just need to make sure that we clear m_timerDidFire before we 113 // check m_wallClockDeadline and return below. 114 // 115 // 2. A stale timer fires just before m_wallClockDeadline expires. 116 // Before the watchdog can gets to the clearing m_timerDidFire above, the legit timer also fires. 117 // 118 // In this case, m_timerDidFire was set twice by the 2 timers, but we only clear need to clear 119 // it once. The m_wallClockDeadline below will pass and this invocation of didFireSlow() will 120 // be treated as the response to the legit timer. The spurious timer is effectively ignored. 121 // 122 // 3. A state timer fires just before m_wallClockDeadline expires. 123 // Right after the watchdog clears m_timerDidFire above, the legit timer also fires. 124 // 125 // The fact that the legit timer fails means that the m_wallClockDeadline check will pass. 126 // As long as we clear m_timerDidFire before we do the check, we are safe. This is the same 127 // scenario as 2 above. 128 // 129 // 4. A stale timer fires just before m_wallClockDeadline expires. 130 // Right after we do the m_wallClockDeadline check below, the legit timer fires. 131 // 132 // The fact that the legit timer fires after the m_wallClockDeadline check means that 133 // the m_wallClockDeadline check will have failed. This is the same scenario as 1 above. 134 // 135 // 5. A legit timer fires. 136 // A stale timer also fires before we clear m_timerDidFire above. 137 // 138 // This is the same scenario as 2 above. 139 // 140 // 6. A legit timer fires. 141 // A stale timer fires right after we clear m_timerDidFire above. 142 // 143 // In this case, the m_wallClockDeadline check will pass, and we fire the watchdog 144 // though m_timerDidFire remains set. This just means that didFireSlow() will be called again due 145 // to m_timerDidFire being set. The subsequent invocation of didFireSlow() will end up handling 146 // the stale timer and ignoring it. This is the same scenario as 1 above. 147 // 148 // 7. A legit timer fires. 149 // A stale timer fires right after we do the m_wallClockDeadline check. 150 // 151 // This is the same as 6, which means it's the same as 1 above. 152 // 153 // In all these cases, we just need to ensure that we clear m_timerDidFire before we do the 154 // m_wallClockDeadline check below. Hence, a storeLoad fence is needed to ensure that these 2 155 // operations do not get re-ordered. 89 ASSERT(m_timerDidFire); 90 m_timerDidFire = false; 156 91 157 WTF::storeLoadFence(); 92 if (currentWallClockTime() < m_wallClockDeadline) 93 return false; // Just a stale timer firing. Nothing to do. 158 94 159 if (currentWallClockTime() < m_wallClockDeadline) 160 return false; // Just a stale timer firing. Nothing to do. 95 // Set m_wallClockDeadline to noTimeLimit here so that we can reject all future 96 // spurious wakes. 97 m_wallClockDeadline = noTimeLimit; 161 98 162 m_wallClockDeadline = noTimeLimit; 99 auto cpuTime = currentCPUTime(); 100 if (cpuTime < m_cpuDeadline) { 101 auto remainingCPUTime = m_cpuDeadline - cpuTime; 102 startTimer(locker, remainingCPUTime); 103 return false; 104 } 105 } 163 106 164 if (currentCPUTime() >= m_cpuDeadline) {165 // Case 1: the allowed CPU time has elapsed.107 // Note: we should not be holding the lock while calling the callbacks. The callbacks may 108 // call setTimeLimit() which will try to lock as well. 166 109 167 // If m_callback is not set, then we terminate by default. 168 // Else, we let m_callback decide if we should terminate or not. 169 bool needsTermination = !m_callback 170 || m_callback(exec, m_callbackData1, m_callbackData2); 171 if (needsTermination) { 172 m_didFire = true; 173 return true; 174 } 110 // If m_callback is not set, then we terminate by default. 111 // Else, we let m_callback decide if we should terminate or not. 112 bool needsTermination = !m_callback 113 || m_callback(exec, m_callbackData1, m_callbackData2); 114 if (needsTermination) 115 return true; 116 117 { 118 LockHolder locker(m_lock); 175 119 176 120 // If we get here, then the callback above did not want to terminate execution. As a … … 185 129 186 130 ASSERT(m_hasEnteredVM); 187 if (hasTimeLimit() && !hasStartedTimer()) 188 startTimer(m_timeLimit); 189 190 } else { 191 // Case 2: the allowed CPU time has NOT elapsed. 192 auto remainingCPUTime = m_cpuDeadline - currentCPUTime(); 193 startTimer(remainingCPUTime); 131 bool callbackAlreadyStartedTimer = (m_cpuDeadline != noTimeLimit); 132 if (hasTimeLimit() && !callbackAlreadyStartedTimer) 133 startTimer(locker, m_timeLimit); 194 134 } 195 196 135 return false; 197 136 } … … 202 141 } 203 142 204 void Watchdog::fire()205 {206 m_didFire = true;207 }208 209 143 void Watchdog::enteredVM() 210 144 { 211 145 m_hasEnteredVM = true; 212 if (hasTimeLimit()) 213 startTimer(m_timeLimit); 146 if (hasTimeLimit()) { 147 LockHolder locker(m_lock); 148 startTimer(locker, m_timeLimit); 149 } 214 150 } 215 151 … … 217 153 { 218 154 ASSERT(m_hasEnteredVM); 219 stopTimer(); 155 LockHolder locker(m_lock); 156 stopTimer(locker); 220 157 m_hasEnteredVM = false; 221 158 } 222 159 223 void Watchdog::startTimer( std::chrono::microseconds timeLimit)160 void Watchdog::startTimer(LockHolder&, std::chrono::microseconds timeLimit) 224 161 { 225 162 ASSERT(m_hasEnteredVM); 226 163 ASSERT(hasTimeLimit()); 164 ASSERT(timeLimit <= m_timeLimit); 227 165 228 166 m_cpuDeadline = currentCPUTime() + timeLimit; … … 231 169 232 170 if ((wallClockTime < m_wallClockDeadline) 233 && (m_wallClockDeadline <= wallClockDeadline)) {171 && (m_wallClockDeadline <= wallClockDeadline)) 234 172 return; // Wait for the current active timer to expire before starting a new one. 235 }236 173 237 174 // Else, the current active timer won't fire soon enough. So, start a new timer. 238 175 this->ref(); // m_timerHandler will deref to match later. 239 176 m_wallClockDeadline = wallClockDeadline; 240 m_timerDidFire = false;241 242 // We clear m_timerDidFire because we're starting a new timer. However, we need to make sure243 // that the clearing occurs before the timer thread is started. Thereafter, only didFireSlow()244 // should clear m_timerDidFire (unless we start yet another timer). Hence, we need a storeStore245 // fence here to ensure these operations do not get re-ordered.246 WTF::storeStoreFence();247 177 248 178 m_timerQueue->dispatchAfter(std::chrono::nanoseconds(timeLimit), m_timerHandler); 249 179 } 250 180 251 void Watchdog::stopTimer( )181 void Watchdog::stopTimer(LockHolder&) 252 182 { 253 183 m_cpuDeadline = noTimeLimit; -
trunk/Source/JavaScriptCore/runtime/Watchdog.h
r188329 r189009 27 27 #define Watchdog_h 28 28 29 #include < mutex>29 #include <wtf/Lock.h> 30 30 #include <wtf/Ref.h> 31 31 #include <wtf/ThreadSafeRefCounted.h> … … 45 45 46 46 typedef bool (*ShouldTerminateCallback)(ExecState*, void* data1, void* data2); 47 void setTimeLimit(VM&, std::chrono::microseconds limit, ShouldTerminateCallback = 0, void* data1 = 0, void* data2 = 0); 47 void setTimeLimit(std::chrono::microseconds limit, ShouldTerminateCallback = 0, void* data1 = 0, void* data2 = 0); 48 JS_EXPORT_PRIVATE void terminateSoon(); 48 49 49 // This version of didFire() will check the elapsed CPU time and call the50 // callback (if needed) to determine if the watchdog should fire.51 50 bool didFire(ExecState* exec) 52 51 { 53 if (m_didFire)54 return true;55 52 if (!m_timerDidFire) 56 53 return false; … … 62 59 void exitedVM(); 63 60 64 // This version of didFire() is a more efficient version for when we want65 // to know if the watchdog has fired in the past, and not whether it should66 // fire right now.67 bool didFire() { return m_didFire; }68 JS_EXPORT_PRIVATE void fire();69 70 61 void* timerDidFireAddress() { return &m_timerDidFire; } 71 62 … … 73 64 74 65 private: 75 void startTimer(std::chrono::microseconds timeLimit); 76 void stopTimer(); 77 78 inline bool hasStartedTimer(); 66 void startTimer(LockHolder&, std::chrono::microseconds timeLimit); 67 void stopTimer(LockHolder&); 79 68 80 69 bool didFireSlow(ExecState*); … … 86 75 // (probably from another thread) but is only cleared in the script thread. 87 76 bool m_timerDidFire; 88 bool m_didFire;89 77 90 78 std::chrono::microseconds m_timeLimit; … … 92 80 std::chrono::microseconds m_cpuDeadline; 93 81 std::chrono::microseconds m_wallClockDeadline; 82 83 // Writes to m_timerDidFire and m_timeLimit, and Reads+Writes to m_cpuDeadline and m_wallClockDeadline 84 // must be guarded by this lock. 85 Lock m_lock; 94 86 95 87 bool m_hasEnteredVM { false }; -
trunk/Source/WebCore/ChangeLog
r189005 r189009 1 2015-08-26 Mark Lam <mark.lam@apple.com> 2 3 watchdog m_didFire state erroneously retained. 4 https://bugs.webkit.org/show_bug.cgi?id=131082 5 6 Reviewed by Geoffrey Garen. 7 8 No new tests. The new code is covered by the JSC API tests and an existing test: 9 fast/workers/worker-terminate-forever.html 10 11 * bindings/js/JSEventListener.cpp: 12 (WebCore::JSEventListener::handleEvent): 13 * bindings/js/WorkerScriptController.cpp: 14 (WebCore::WorkerScriptController::WorkerScriptController): 15 - Always create a watchdog for the Web Worker's VM. We need this in order to support 16 Worker.terminate(). 17 (WebCore::WorkerScriptController::evaluate): 18 (WebCore::WorkerScriptController::scheduleExecutionTermination): 19 (WebCore::WorkerScriptController::isTerminatingExecution): 20 (WebCore::WorkerScriptController::forbidExecution): 21 (WebCore::WorkerScriptController::isExecutionTerminating): Deleted. 22 * bindings/js/WorkerScriptController.h: 23 1 24 2015-08-26 Myles C. Maxfield <mmaxfield@apple.com> 2 25 -
trunk/Source/WebCore/bindings/js/JSEventListener.cpp
r188329 r189009 136 136 137 137 if (is<WorkerGlobalScope>(*scriptExecutionContext)) { 138 auto scriptController = downcast<WorkerGlobalScope>(*scriptExecutionContext).script(); 138 139 bool terminatorCausedException = (exec->hadException() && isTerminatedExecutionException(exec->exception())); 139 if (terminatorCausedException || (vm.watchdog && vm.watchdog->didFire()))140 downcast<WorkerGlobalScope>(*scriptExecutionContext).script()->forbidExecution();140 if (terminatorCausedException || scriptController->isTerminatingExecution()) 141 scriptController->forbidExecution(); 141 142 } 142 143 -
trunk/Source/WebCore/bindings/js/WorkerScriptController.cpp
r188594 r189009 57 57 , m_executionForbidden(false) 58 58 { 59 m_vm->ensureWatchdog(); 59 60 initNormalWorldClientData(m_vm.get()); 60 61 } … … 122 123 123 124 VM& vm = exec->vm(); 124 if ((returnedException && isTerminatedExecutionException(returnedException)) 125 || (vm.watchdog && vm.watchdog->didFire())) { 125 if ((returnedException && isTerminatedExecutionException(returnedException)) || isTerminatingExecution()) { 126 126 forbidExecution(); 127 127 return; … … 150 150 { 151 151 // The mutex provides a memory barrier to ensure that once 152 // termination is scheduled, is ExecutionTerminatingwill152 // termination is scheduled, isTerminatingExecution() will 153 153 // accurately reflect that state when called from another thread. 154 154 LockHolder locker(m_scheduledTerminationMutex); 155 if (m_vm->watchdog) 156 m_vm->watchdog->fire(); 155 m_isTerminatingExecution = true; 156 157 ASSERT(m_vm->watchdog); 158 m_vm->watchdog->terminateSoon(); 157 159 } 158 160 159 bool WorkerScriptController::is ExecutionTerminating() const161 bool WorkerScriptController::isTerminatingExecution() const 160 162 { 161 163 // See comments in scheduleExecutionTermination regarding mutex usage. 162 164 LockHolder locker(m_scheduledTerminationMutex); 163 if (m_vm->watchdog) 164 return m_vm->watchdog->didFire(); 165 return false; 165 return m_isTerminatingExecution; 166 166 } 167 167 -
trunk/Source/WebCore/bindings/js/WorkerScriptController.h
r188594 r189009 72 72 // Can be called from any thread. 73 73 void scheduleExecutionTermination(); 74 bool is ExecutionTerminating() const;74 bool isTerminatingExecution() const; 75 75 76 76 // Called on Worker thread when JS exits with termination exception caused by forbidExecution() request, … … 98 98 JSC::Strong<JSWorkerGlobalScope> m_workerGlobalScopeWrapper; 99 99 bool m_executionForbidden; 100 bool m_isTerminatingExecution { false }; 100 101 mutable Lock m_scheduledTerminationMutex; 101 102 };
Note:
See TracChangeset
for help on using the changeset viewer.