Changeset 185259 in webkit


Ignore:
Timestamp:
Jun 5, 2015, 11:52:12 AM (10 years ago)
Author:
mark.lam@apple.com
Message:

finally blocks should not set the exception stack trace when re-throwing the exception.
https://bugs.webkit.org/show_bug.cgi?id=145525

Reviewed by Geoffrey Garen.

Source/JavaScriptCore:

How exceptions presently work:
=============================

  1. op_throw can throw any JSValue.
  2. the VM tries to capture the stack at the throw point and propagate that as needed.
  3. finally blocks are implemented using op_catch to catch the thrown value, and throws it again using op_throw.

What's wrong with how it presently works:
========================================

  1. finally's makes for bad exception throw line numbers in the Inspector console.

The op_throw in finally will throw the value anew i.e. it captures a stack from the re-throw point.
As a result, the Inspector sees the finally block as the throw point. The original stack is lost.

  1. finally's breaks the Inspector's "Breaks on Uncaught Exception"

This is because finally blocks are indistinguishable from catch blocks. As a result, a try-finally,
which should break in the Inspector on the throw, does not because the Inspector thought the
exception was "caught".

  1. finally's yields confusing break points when the Inspector "Breaks on All Exceptions"
  1. In a try-finally scenario, the Inspector breaks 2 times: 1 at the throw, 1 at the finally.
  2. In a for-of loop (which has synthesized finallys), the Inspector will do another break. Similarly for other cases of JS code which synthesize finallys.
  3. At VM re-entry boundaries (e.g. js throws & returns to native code, which returns to js), the Inspector will do another break if there's an uncaught exception.

How this patch fixes the issues:
===============================

  1. We introduce an Exception object that wraps the thrown value and the exception stack.

When throwing an exception, the VM will check if the thrown value is an Exception
object or not. If it is not an Exception object, then we must be throwing a new
exception. The VM will create an Exception object to wrap the thrown value and
capture the current stack for it.

If the thrown value is already an Exception object, then the requested throw operation
must be a re-throw. The VM will not capture a new stack for it.

  1. op_catch will now populate 2 locals: 1 for the Exception, 1 for the thrown JSValue.

The VM is aware of the Exception object and uses it for rethrows in finally blocks.
JS source code is never aware of the Exception object.

JS code is aware of the thrown value. If it throws the caught thrown value, that
constitutes a new throw, and a new Exception object will be created for it.

  1. The VM no longer tracks the thrown JSValue and the exception stack. It will only track a m_exception field which is an Exception*.
  1. The BytecodeGenerator has already been updated in a prior patch to distinguish between Catch, Finally, and SynthesizedFinally blocks. The interpreter runtime will now report to the debugger whether we have a Catch handler, not just any handlers.

The debugger will use this detail to determine whether to break or not. "Break on
uncaught exceptions" will only break if no Catch handler was found.

This solves the issue of the debugger breaking at finally blocks, and for-of statements.

  1. The Exception object will also have a flag to indicate whether the debugger has been notified of the Exception being thrown. Once the Interpreter notifies the debugger of the Exception object, it will mark this flag and not repeat the notify the debugger again of the same Exception.

This solves the issue of the debugger breaking at VM re-entry points due to uncaught
exceptions.

  1. The life-cycle of the captured exception stack trace will now follow the life-cycle of the Exception object.

Other changes:

  1. Change all clients of the VM::exception() to expect an Exception* instead of JSValue.
  1. Fixed a few bugs where thrown exceptions are not cleared before exiting the VM.
  1. Also renamed some variables and classes to better describe what they are.
  • API/JSBase.cpp:

(JSEvaluateScript):
(JSCheckScriptSyntax):

  • API/JSObjectRef.cpp:

(handleExceptionIfNeeded):

  • The functions below all do the same exception check. Added this helper to simplify the code.

(JSClassCreate):
(JSObjectMakeFunction):
(JSObjectMakeArray):
(JSObjectMakeDate):
(JSObjectMakeError):
(JSObjectMakeRegExp):
(JSObjectGetProperty):
(JSObjectSetProperty):
(JSObjectGetPropertyAtIndex):
(JSObjectSetPropertyAtIndex):
(JSObjectDeleteProperty):
(JSObjectCallAsFunction):
(JSObjectCallAsConstructor):

  • API/JSScriptRef.cpp:
  • API/JSValue.mm:

(JSContainerConvertor::take):
(reportExceptionToInspector):

  • API/JSValueRef.cpp:

(handleExceptionIfNeeded):

  • The functions below all do the same exception check. Added this helper to simplify the code.

(evernoteHackNeeded):
(JSValueIsEqual):
(JSValueIsInstanceOfConstructor):
(JSValueCreateJSONString):
(JSValueToNumber):
(JSValueToStringCopy):
(JSValueToObject):

  • bindings/ScriptFunctionCall.cpp:

(Deprecated::ScriptFunctionCall::call):

  • bindings/ScriptFunctionCall.h:
  • bytecode/BytecodeList.json:
  • op_catch now had 2 operands: the exception register, and the thrown value register.
  • bytecode/BytecodeUseDef.h:

(JSC::computeDefsForBytecodeOffset):

  • bytecode/CodeBlock.cpp:

(JSC::CodeBlock::dumpBytecode):
(JSC::CodeBlock::handlerForBytecodeOffset):

  • bytecode/CodeBlock.h:
  • handlerForBytecodeOffset() now can look for just Catch handlers only.
  • bytecode/HandlerInfo.h:
  • Cleaned up some white space I accidentally added in a previous patch.
  • bytecompiler/BytecodeGenerator.cpp:

(JSC::BytecodeGenerator::pushTry):
(JSC::BytecodeGenerator::popTryAndEmitCatch):
(JSC::BytecodeGenerator::emitThrowReferenceError):
(JSC::BytecodeGenerator::emitEnumeration):

  • bytecompiler/BytecodeGenerator.h:

(JSC::BytecodeGenerator::emitThrow):

  • bytecompiler/NodesCodegen.cpp:

(JSC::TryNode::emitBytecode):

  • Adding support for op_catch's 2 operands.
  • debugger/Debugger.cpp:

(JSC::Debugger::hasBreakpoint):
(JSC::Debugger::pauseIfNeeded):
(JSC::Debugger::exception):

  • debugger/Debugger.h:
  • debugger/DebuggerCallFrame.cpp:

(JSC::DebuggerCallFrame::thisValue):
(JSC::DebuggerCallFrame::evaluate):

  • debugger/DebuggerCallFrame.h:

(JSC::DebuggerCallFrame::isValid):

  • inspector/InjectedScriptManager.cpp:

(Inspector::InjectedScriptManager::createInjectedScript):

  • inspector/InspectorEnvironment.h:
  • inspector/JSGlobalObjectInspectorController.cpp:

(Inspector::JSGlobalObjectInspectorController::appendAPIBacktrace):
(Inspector::JSGlobalObjectInspectorController::reportAPIException):

  • inspector/JSGlobalObjectInspectorController.h:
  • inspector/JSGlobalObjectScriptDebugServer.h:
  • inspector/JSJavaScriptCallFrame.cpp:

(Inspector::JSJavaScriptCallFrame::evaluate):

  • inspector/JavaScriptCallFrame.h:

(Inspector::JavaScriptCallFrame::vmEntryGlobalObject):
(Inspector::JavaScriptCallFrame::thisValue):
(Inspector::JavaScriptCallFrame::evaluate):

  • inspector/ScriptCallStackFactory.cpp:

(Inspector::extractSourceInformationFromException):
(Inspector::createScriptCallStackFromException):

  • inspector/ScriptCallStackFactory.h:
  • inspector/ScriptDebugServer.cpp:

(Inspector::ScriptDebugServer::evaluateBreakpointAction):
(Inspector::ScriptDebugServer::handleBreakpointHit):
(Inspector::ScriptDebugServer::handleExceptionInBreakpointCondition):

  • inspector/ScriptDebugServer.h:
  • interpreter/CallFrame.h:

(JSC::ExecState::clearException):
(JSC::ExecState::exception):
(JSC::ExecState::hadException):
(JSC::ExecState::atomicStringTable):
(JSC::ExecState::propertyNames):
(JSC::ExecState::clearSupplementaryExceptionInfo): Deleted.

  • interpreter/Interpreter.cpp:

(JSC::unwindCallFrame):
(JSC::Interpreter::stackTraceAsString):
(JSC::GetCatchHandlerFunctor::GetCatchHandlerFunctor):
(JSC::GetCatchHandlerFunctor::operator()):
(JSC::Interpreter::unwind):

  • Added a check for didNotifyInspectorOfThrow() here to prevent duplicate reports of the same Exception to the debugger.

(JSC::GetExceptionHandlerFunctor::GetExceptionHandlerFunctor): Deleted.
(JSC::GetExceptionHandlerFunctor::operator()): Deleted.

  • Renamed GetExceptionHandlerFunctor to GetCatchHandlerFunctor since the debugger is only interested in knowing whether we have Catch handlers.
  • interpreter/Interpreter.h:

(JSC::SuspendExceptionScope::SuspendExceptionScope):
(JSC::SuspendExceptionScope::~SuspendExceptionScope):
(JSC::Interpreter::sampler):
(JSC::ClearExceptionScope::ClearExceptionScope): Deleted.
(JSC::ClearExceptionScope::~ClearExceptionScope): Deleted.

  • Renamed ClearExceptionScope to SuspendExceptionScope because "clear" implies that we're purging the exception. Instead, we're merely suspending any handling of that exception for a period defined by the scope.
  • jit/AssemblyHelpers.cpp:

(JSC::AssemblyHelpers::emitExceptionCheck):

  • jit/JITExceptions.cpp:

(JSC::genericUnwind):

  • Removed the exception argument. It is always the value in VM::exception() anyway. genericUnwind() can just get it from the VM, and save everyone some work.
  • jit/JITExceptions.h:
  • jit/JITOpcodes.cpp:

(JSC::JIT::emit_op_catch):

  • jit/JITOpcodes32_64.cpp:

(JSC::JIT::privateCompileCTINativeCall):
(JSC::JIT::emit_op_catch):

  • Add support for the new op_catch operands.
  • jit/JITOperations.cpp:
  • jit/ThunkGenerators.cpp:

(JSC::nativeForGenerator):

  • jsc.cpp:

(functionRun):
(functionLoad):
(runWithScripts):
(runInteractive):

  • llint/LLIntOffsetsExtractor.cpp:
  • llint/LLIntSlowPaths.cpp:

(JSC::LLInt::LLINT_SLOW_PATH_DECL):

  • llint/LowLevelInterpreter32_64.asm:
  • llint/LowLevelInterpreter64.asm:
  • Add support for the new op_catch operands. Also update the code to handle VM::m_exception being an Exception pointer, not a JSValue.
  • parser/NodeConstructors.h:

(JSC::TryNode::TryNode):

  • parser/Nodes.h:
  • runtime/CallData.cpp:

(JSC::call):

  • runtime/CallData.h:
  • runtime/Completion.cpp:

(JSC::evaluate):

  • runtime/Completion.h:

(JSC::evaluate):

  • Change evaluate() to take a reference to the returned exception value instead of a pointer. In all but 2 or 3 cases, we want the returned exception anyway. Might as well simplify the code by requiring the reference.
  • runtime/Error.h:

(JSC::throwVMError):
(JSC::throwVMTypeError):

  • runtime/Exception.cpp: Added.

(JSC::Exception::create):
(JSC::Exception::destroy):
(JSC::Exception::createStructure):
(JSC::Exception::visitChildren):
(JSC::Exception::Exception):
(JSC::Exception::~Exception):

  • runtime/Exception.h: Added.

(JSC::Exception::valueOffset):
(JSC::Exception::cast):
(JSC::Exception::value):
(JSC::Exception::stack):
(JSC::Exception::didNotifyInspectorOfThrow):
(JSC::Exception::setDidNotifyInspectorOfThrow):

  • runtime/ExceptionHelpers.cpp:

(JSC::createTerminatedExecutionException):
(JSC::isTerminatedExecutionException):
(JSC::createStackOverflowError):

  • runtime/ExceptionHelpers.h:
  • runtime/GetterSetter.cpp:

(JSC::callGetter):

  • runtime/IteratorOperations.cpp:

(JSC::iteratorClose):

  • runtime/JSObject.cpp:
  • runtime/JSPromiseConstructor.cpp:

(JSC::constructPromise):

  • runtime/JSPromiseDeferred.cpp:

(JSC::updateDeferredFromPotentialThenable):
(JSC::abruptRejection):

  • runtime/JSPromiseReaction.cpp:

(JSC::ExecutePromiseReactionMicrotask::run):

  • runtime/VM.cpp:

(JSC::VM::VM):
(JSC::VM::releaseExecutableMemory):
(JSC::VM::throwException):
(JSC::VM::setStackPointerAtVMEntry):
(JSC::VM::getExceptionInfo): Deleted.
(JSC::VM::setExceptionInfo): Deleted.
(JSC::VM::clearException): Deleted.
(JSC::clearExceptionStack): Deleted.

  • runtime/VM.h:

(JSC::VM::targetMachinePCForThrowOffset):
(JSC::VM::clearException):
(JSC::VM::setException):
(JSC::VM::exception):
(JSC::VM::addressOfException):
(JSC::VM::exceptionStack): Deleted.

  • runtime/VMEntryScope.cpp:

(JSC::VMEntryScope::VMEntryScope):
(JSC::VMEntryScope::setEntryScopeDidPopListener):

Source/WebCore:

Update to use the new JSC::Exception object.

Test: inspector/debugger/break-on-exceptions.html

  • ForwardingHeaders/runtime/Exception.h: Added.
  • bindings/js/JSCallbackData.cpp:

(WebCore::JSCallbackData::invokeCallback):

  • bindings/js/JSCustomXPathNSResolver.cpp:

(WebCore::JSCustomXPathNSResolver::lookupNamespaceURI):

  • bindings/js/JSDOMBinding.cpp:

(WebCore::jsArray):
(WebCore::reportException):
(WebCore::reportCurrentException):

  • bindings/js/JSDOMBinding.h:
  • bindings/js/JSErrorHandler.cpp:

(WebCore::JSErrorHandler::handleEvent):

  • bindings/js/JSEventListener.cpp:

(WebCore::JSEventListener::handleEvent):

  • bindings/js/JSMainThreadExecState.cpp:

(WebCore::JSMainThreadExecState::didLeaveScriptContext):
(WebCore::functionCallHandlerFromAnyThread):
(WebCore::evaluateHandlerFromAnyThread):

  • bindings/js/JSMainThreadExecState.h:

(WebCore::JSMainThreadExecState::currentState):
(WebCore::JSMainThreadExecState::call):
(WebCore::JSMainThreadExecState::evaluate):
(WebCore::JSMainThreadExecState::runTask):

  • bindings/js/JSMediaDevicesCustom.cpp:

(WebCore::JSMediaDevices::getUserMedia):

  • Fixed a bug where the exception was not cleared before entering the VM to call JS code.
  • bindings/js/JSMutationCallback.cpp:

(WebCore::JSMutationCallback::call):

  • bindings/js/ReadableJSStream.cpp:

(WebCore::getPropertyFromObject):
(WebCore::callFunction):
(WebCore::ReadableJSStream::Source::start):

  • bindings/js/ScheduledAction.cpp:

(WebCore::ScheduledAction::executeFunctionInContext):

  • bindings/js/ScriptController.cpp:

(WebCore::ScriptController::evaluateInWorld):

  • bindings/js/SerializedScriptValue.cpp:

(WebCore::SerializedScriptValue::create):
(WebCore::SerializedScriptValue::deserialize):

  • bindings/js/WorkerScriptController.cpp:

(WebCore::WorkerScriptController::evaluate):
(WebCore::WorkerScriptController::setException):
(WebCore::WorkerScriptController::scheduleExecutionTermination):

  • bindings/js/WorkerScriptController.h:

(WebCore::WorkerScriptController::workerGlobalScopeWrapper):

  • bindings/js/WorkerScriptDebugServer.cpp:

(WebCore::WorkerScriptDebugServer::runEventLoopWhilePaused):
(WebCore::WorkerScriptDebugServer::reportException):

  • bindings/js/WorkerScriptDebugServer.h:
  • bindings/objc/WebScriptObject.mm:

(WebCore::createJSWrapper):
(WebCore::addExceptionToConsole):
(-[WebScriptObject callWebScriptMethod:withArguments:]):
(-[WebScriptObject evaluateWebScript:]):

  • Changed to call a version of JSMainThreadExecState::evaluate() that provides a stub returnedException because evaluateWebScript: doesn't need the exception.
  • inspector/PageScriptDebugServer.cpp:

(WebCore::PageScriptDebugServer::isContentScript):
(WebCore::PageScriptDebugServer::reportException):

  • inspector/PageScriptDebugServer.h:
  • workers/WorkerGlobalScope.cpp:

(WebCore::WorkerGlobalScope::importScripts):

Source/WebKit/mac:

  • WebView/WebView.mm:

(+[WebView _reportException:inContext:]):
(WebKitInitializeApplicationCachePathIfNecessary):

  • Changed to use the new Exception object.

Source/WebKit/win:

  • WebView.cpp:

(WebView::reportException):

  • Changed to use the new Exception object.

Source/WebKit2:

  • WebProcess/InjectedBundle/InjectedBundle.cpp:

(WebKit::InjectedBundle::reportException):

  • Changed to use the new Exception object.

LayoutTests:

  • fast/dom/regress-131530-expected.txt:
  • Rebased results because we now have a proper line number.
  • http/tests/inspector/inspector-test.js:

(InspectorTestProxy.clearResults):
(InspectorTestProxy.reportUncaughtException):

  • Add the feature to sanitize the url reported by reportUncaughtException() since we can have tests that do expect uncaught exceptions, and we need the test results to be invariant. Sanitization of the url, in this case means, stripping off the preceding path.
  • inspector/debugger/break-on-exception-expected.txt: Added.
  • inspector/debugger/break-on-exception.html: Added.
  • inspector/debugger/break-on-exception-catch-expected.txt: Added.
  • inspector/debugger/break-on-exception-catch.html: Added.
  • inspector/debugger/break-on-exception-finally-expected.txt: Added.
  • inspector/debugger/break-on-exception-finally.html: Added.
  • inspector/debugger/break-on-exception-native-expected.txt: Added.
  • inspector/debugger/break-on-exception-native.html: Added.
  • inspector/debugger/break-on-exception-throw-in-promise-expected.txt: Added.
  • inspector/debugger/break-on-exception-throw-in-promise.html: Added.
  • inspector/debugger/break-on-exception-throw-in-promise-with-catch-expected.txt: Added.
  • inspector/debugger/break-on-exception-throw-in-promise-with-catch.html: Added.
  • inspector/debugger/break-on-exception-throw-in-promise-then-expected.txt: Added.
  • inspector/debugger/break-on-exception-throw-in-promise-then.html: Added.
  • inspector/debugger/break-on-exception-throw-in-promise-then-with-catch-expected.txt: Added.
  • inspector/debugger/break-on-exception-throw-in-promise-then-with-catch.html: Added.
  • inspector/debugger/break-on-exception-throw-in-promise-rethrow-in-catch-expected.txt: Added.
  • inspector/debugger/break-on-exception-throw-in-promise-rethrow-in-catch.html: Added.
  • inspector/debugger/break-on-exception-window-onerror-expected.txt: Added.
  • inspector/debugger/break-on-exception-window-onerror.html: Added.
  • inspector/debugger/break-on-uncaught-exception-expected.txt: Added.
  • inspector/debugger/break-on-uncaught-exception.html: Added.
  • inspector/debugger/break-on-uncaught-exception-catch-expected.txt: Added.
  • inspector/debugger/break-on-uncaught-exception-catch.html: Added.
  • inspector/debugger/break-on-uncaught-exception-finally-expected.txt: Added.
  • inspector/debugger/break-on-uncaught-exception-finally.html: Added.
  • inspector/debugger/break-on-uncaught-exception-native-expected.txt: Added.
  • inspector/debugger/break-on-uncaught-exception-native.html: Added.
  • inspector/debugger/break-on-uncaught-exception-throw-in-promise-expected.txt: Added.
  • inspector/debugger/break-on-uncaught-exception-throw-in-promise.html: Added.
  • inspector/debugger/break-on-uncaught-exception-throw-in-promise-with-catch-expected.txt: Added.
  • inspector/debugger/break-on-uncaught-exception-throw-in-promise-with-catch.html: Added.
  • inspector/debugger/break-on-uncaught-exception-throw-in-promise-then-expected.txt: Added.
  • inspector/debugger/break-on-uncaught-exception-throw-in-promise-then.html: Added.
  • inspector/debugger/break-on-uncaught-exception-throw-in-promise-then-with-catch-expected.txt: Added.
  • inspector/debugger/break-on-uncaught-exception-throw-in-promise-then-with-catch.html: Added.
  • inspector/debugger/break-on-uncaught-exception-throw-in-promise-rethrow-in-catch-expected.txt: Added.
  • inspector/debugger/break-on-uncaught-exception-throw-in-promise-rethrow-in-catch.html: Added.
  • inspector/debugger/break-on-uncaught-exception-window-onerror-expected.txt: Added.
  • inspector/debugger/break-on-uncaught-exception-window-onerror.html: Added.
  • inspector/debugger/resources/break-on-exception-tests.js: Added.

(doThrow):
(testCatch):
(testFinally):
(testThrowingThruNativeCode):
(testThrowingInPromise):
(testThrowingInPromiseWithCatch):
(testThrowingInPromiseThen):
(testThrowingInPromiseThenWithCatch):
(testThrowingInPromiseWithRethrowInCatch):

Location:
trunk
Files:
44 added
101 edited

Legend:

Unmodified
Added
Removed
  • TabularUnified trunk/LayoutTests/ChangeLog

    r185258 r185259  
     12015-06-05  Mark Lam  <mark.lam@apple.com>
     2
     3        finally blocks should not set the exception stack trace when re-throwing the exception.
     4        https://bugs.webkit.org/show_bug.cgi?id=145525
     5
     6        Reviewed by Geoffrey Garen.
     7
     8        * TestExpectations:
     9        - Skip the new tests until webkit.org/b/145090 is fixed.
     10
     11        * fast/dom/regress-131530-expected.txt:
     12        - Rebased results because we now have a proper line number.
     13
     14        * http/tests/inspector/inspector-test.js:
     15        (InspectorTestProxy.clearResults):
     16        (InspectorTestProxy.reportUncaughtException):
     17        - Add the feature to sanitize the url reported by reportUncaughtException() since
     18          we can have tests that do expect uncaught exceptions, and we need the test
     19          results to be invariant.  Sanitization of the url, in this case means, stripping
     20          off the preceding path.
     21
     22        * inspector/debugger/break-on-exception-expected.txt: Added.
     23        * inspector/debugger/break-on-exception.html: Added.
     24        * inspector/debugger/break-on-exception-catch-expected.txt: Added.
     25        * inspector/debugger/break-on-exception-catch.html: Added.
     26        * inspector/debugger/break-on-exception-finally-expected.txt: Added.
     27        * inspector/debugger/break-on-exception-finally.html: Added.
     28        * inspector/debugger/break-on-exception-native-expected.txt: Added.
     29        * inspector/debugger/break-on-exception-native.html: Added.
     30
     31        * inspector/debugger/break-on-exception-throw-in-promise-expected.txt: Added.
     32        * inspector/debugger/break-on-exception-throw-in-promise.html: Added.
     33        * inspector/debugger/break-on-exception-throw-in-promise-with-catch-expected.txt: Added.
     34        * inspector/debugger/break-on-exception-throw-in-promise-with-catch.html: Added.
     35        * inspector/debugger/break-on-exception-throw-in-promise-then-expected.txt: Added.
     36        * inspector/debugger/break-on-exception-throw-in-promise-then.html: Added.
     37        * inspector/debugger/break-on-exception-throw-in-promise-then-with-catch-expected.txt: Added.
     38        * inspector/debugger/break-on-exception-throw-in-promise-then-with-catch.html: Added.
     39        * inspector/debugger/break-on-exception-throw-in-promise-rethrow-in-catch-expected.txt: Added.
     40        * inspector/debugger/break-on-exception-throw-in-promise-rethrow-in-catch.html: Added.
     41
     42        * inspector/debugger/break-on-exception-window-onerror-expected.txt: Added.
     43        * inspector/debugger/break-on-exception-window-onerror.html: Added.
     44
     45        * inspector/debugger/break-on-uncaught-exception-expected.txt: Added.
     46        * inspector/debugger/break-on-uncaught-exception.html: Added.
     47        * inspector/debugger/break-on-uncaught-exception-catch-expected.txt: Added.
     48        * inspector/debugger/break-on-uncaught-exception-catch.html: Added.
     49        * inspector/debugger/break-on-uncaught-exception-finally-expected.txt: Added.
     50        * inspector/debugger/break-on-uncaught-exception-finally.html: Added.
     51        * inspector/debugger/break-on-uncaught-exception-native-expected.txt: Added.
     52        * inspector/debugger/break-on-uncaught-exception-native.html: Added.
     53
     54        * inspector/debugger/break-on-uncaught-exception-throw-in-promise-expected.txt: Added.
     55        * inspector/debugger/break-on-uncaught-exception-throw-in-promise.html: Added.
     56        * inspector/debugger/break-on-uncaught-exception-throw-in-promise-with-catch-expected.txt: Added.
     57        * inspector/debugger/break-on-uncaught-exception-throw-in-promise-with-catch.html: Added.
     58        * inspector/debugger/break-on-uncaught-exception-throw-in-promise-then-expected.txt: Added.
     59        * inspector/debugger/break-on-uncaught-exception-throw-in-promise-then.html: Added.
     60        * inspector/debugger/break-on-uncaught-exception-throw-in-promise-then-with-catch-expected.txt: Added.
     61        * inspector/debugger/break-on-uncaught-exception-throw-in-promise-then-with-catch.html: Added.
     62        * inspector/debugger/break-on-uncaught-exception-throw-in-promise-rethrow-in-catch-expected.txt: Added.
     63        * inspector/debugger/break-on-uncaught-exception-throw-in-promise-rethrow-in-catch.html: Added.
     64
     65        * inspector/debugger/break-on-uncaught-exception-window-onerror-expected.txt: Added.
     66        * inspector/debugger/break-on-uncaught-exception-window-onerror.html: Added.
     67
     68        * inspector/debugger/resources/break-on-exception-tests.js: Added.
     69        (doThrow):
     70        (testCatch):
     71        (testFinally):
     72        (testThrowingThruNativeCode):
     73        (testThrowingInPromise):
     74        (testThrowingInPromiseWithCatch):
     75        (testThrowingInPromiseThen):
     76        (testThrowingInPromiseThenWithCatch):
     77        (testThrowingInPromiseWithRethrowInCatch):
     78
    1792015-06-05  Eric Carlson  <eric.carlson@apple.com>
    280
  • TabularUnified trunk/LayoutTests/TestExpectations

    r185258 r185259  
    526526
    527527webkit.org/b/145390 storage/indexeddb/deleteIndex-bug110792.html [ Pass Failure ]
     528
     529webkit.org/b/145090 inspector/debugger/break-on-exception.html [ Skip ]
     530webkit.org/b/145090 inspector/debugger/break-on-exception-catch.html [ Skip ]
     531webkit.org/b/145090 inspector/debugger/break-on-exception-finally.html [ Skip ]
     532webkit.org/b/145090 inspector/debugger/break-on-exception-native.html [ Skip ]
     533webkit.org/b/145090 inspector/debugger/break-on-exception-throw-in-promise.html [ Skip ]
     534webkit.org/b/145090 inspector/debugger/break-on-exception-throw-in-promise-with-catch.html [ Skip ]
     535webkit.org/b/145090 inspector/debugger/break-on-exception-throw-in-promise-then.html [ Skip ]
     536webkit.org/b/145090 inspector/debugger/break-on-exception-throw-in-promise-then-with-catch.html [ Skip ]
     537webkit.org/b/145090 inspector/debugger/break-on-exception-throw-in-promise-rethrow-in-catch.html [ Skip ]
     538webkit.org/b/145090 inspector/debugger/break-on-exception-window-onerror.html [ Skip ]
     539webkit.org/b/145090 inspector/debugger/break-on-uncaught-exception.html [ Skip ]
     540webkit.org/b/145090 inspector/debugger/break-on-uncaught-exception-catch.html [ Skip ]
     541webkit.org/b/145090 inspector/debugger/break-on-uncaught-exception-finally.html [ Skip ]
     542webkit.org/b/145090 inspector/debugger/break-on-uncaught-exception-native.html [ Skip ]
     543webkit.org/b/145090 inspector/debugger/break-on-uncaught-exception-throw-in-promise.html [ Skip ]
     544webkit.org/b/145090 inspector/debugger/break-on-uncaught-exception-throw-in-promise-with-catch.html [ Skip ]
     545webkit.org/b/145090 inspector/debugger/break-on-uncaught-exception-throw-in-promise-then.html [ Skip ]
     546webkit.org/b/145090 inspector/debugger/break-on-uncaught-exception-throw-in-promise-then-with-catch.html [ Skip ]
     547webkit.org/b/145090 inspector/debugger/break-on-uncaught-exception-throw-in-promise-rethrow-in-catch.html [ Skip ]
     548webkit.org/b/145090 inspector/debugger/break-on-uncaught-exception-window-onerror.html [ Skip ]
     549
  • TabularUnified trunk/LayoutTests/fast/dom/regress-131530-expected.txt

    r167142 r185259  
    11CONSOLE MESSAGE: line 5: Exception to trigger unwinding in MutationObserver
    2 CONSOLE MESSAGE: Pending exception before MutationObservers are called.
     2CONSOLE MESSAGE: line 22: Pending exception before MutationObservers are called.
    33Regression test for https://webkit.org/b/131530. This test should not crash.
    44
  • TabularUnified trunk/LayoutTests/http/tests/inspector/inspector-test.js

    r166901 r185259  
    144144}
    145145
     146InspectorTestProxy.needToSanitizeUncaughtExceptionURLs = false;
     147
    146148InspectorTestProxy.reportUncaughtException = function(message, url, lineNumber)
    147149{
     150    if (InspectorTestProxy.needToSanitizeUncaughtExceptionURLs) {
     151        if (typeof url == "string") {
     152            var lastSlash = url.lastIndexOf("/");
     153            var lastBackSlash = url.lastIndexOf("\\");
     154            var lastPathSeparator = Math.max(lastSlash, lastBackSlash);
     155            if (lastPathSeparator > 0)
     156                url = url.substr(lastPathSeparator + 1);
     157        }
     158    }
     159
    148160    var result = "Uncaught exception in test page: " + message + " [" + url + ":" + lineNumber + "]";
    149161    InspectorTestProxy.addResult(result);
  • TabularUnified trunk/Source/JavaScriptCore/API/JSBase.cpp

    r181411 r185259  
    3131#include "CallFrame.h"
    3232#include "Completion.h"
     33#include "Exception.h"
    3334#include "GCActivityCallback.h"
    3435#include "InitializeThreading.h"
     
    6465    SourceCode source = makeSource(script->string(), sourceURL ? sourceURL->string() : String(), TextPosition(OrdinalNumber::fromOneBasedInt(startingLineNumber), OrdinalNumber::first()));
    6566
    66     JSValue evaluationException;
    67     JSValue returnValue = evaluate(globalObject->globalExec(), source, jsThisObject, &evaluationException);
     67    Exception* evaluationException;
     68    JSValue returnValue = evaluate(globalObject->globalExec(), source, jsThisObject, evaluationException);
    6869
    6970    if (evaluationException) {
    7071        if (exception)
    71             *exception = toRef(exec, evaluationException);
     72            *exception = toRef(exec, evaluationException->value());
    7273#if ENABLE(REMOTE_INSPECTOR)
    7374        // FIXME: If we have a debugger attached we could learn about ParseError exceptions through
     
    108109            *exception = toRef(exec, syntaxException);
    109110#if ENABLE(REMOTE_INSPECTOR)
    110         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, syntaxException);
     111        Exception* exception = Exception::create(exec->vm(), syntaxException);
     112        exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exception);
    111113#endif
    112114        return false;
  • TabularUnified trunk/Source/JavaScriptCore/API/JSObjectRef.cpp

    r182280 r185259  
    3535#include "DateConstructor.h"
    3636#include "ErrorConstructor.h"
     37#include "Exception.h"
    3738#include "FunctionConstructor.h"
    3839#include "Identifier.h"
     
    6263using namespace JSC;
    6364
     65enum class ExceptionStatus {
     66    DidThrow,
     67    DidNotThrow
     68};
     69
     70static ExceptionStatus handleExceptionIfNeeded(ExecState* exec, JSValueRef* returnedExceptionRef)
     71{
     72    if (exec->hadException()) {
     73        Exception* exception = exec->exception();
     74        if (returnedExceptionRef)
     75            *returnedExceptionRef = toRef(exec, exception->value());
     76        exec->clearException();
     77#if ENABLE(REMOTE_INSPECTOR)
     78        exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exception);
     79#endif
     80        return ExceptionStatus::DidThrow;
     81    }
     82    return ExceptionStatus::DidNotThrow;
     83}
     84
    6485JSClassRef JSClassCreate(const JSClassDefinition* definition)
    6586{
     
    149170
    150171    JSObject* result = constructFunction(exec, exec->lexicalGlobalObject(), args, nameID, sourceURL ? sourceURL->string() : String(), TextPosition(OrdinalNumber::fromOneBasedInt(startingLineNumber), OrdinalNumber::first()));
    151     if (exec->hadException()) {
    152         JSValue exceptionValue = exec->exception();
    153         if (exception)
    154             *exception = toRef(exec, exceptionValue);
    155         exec->clearException();
    156 #if ENABLE(REMOTE_INSPECTOR)
    157         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    158 #endif
     172    if (handleExceptionIfNeeded(exec, exception) == ExceptionStatus::DidThrow)
    159173        result = 0;
    160     }
    161174    return toRef(result);
    162175}
     
    181194        result = constructEmptyArray(exec, 0);
    182195
    183     if (exec->hadException()) {
    184         JSValue exceptionValue = exec->exception();
    185         if (exception)
    186             *exception = toRef(exec, exceptionValue);
    187         exec->clearException();
    188 #if ENABLE(REMOTE_INSPECTOR)
    189         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    190 #endif
     196    if (handleExceptionIfNeeded(exec, exception) == ExceptionStatus::DidThrow)
    191197        result = 0;
    192     }
    193198
    194199    return toRef(result);
     
    209214
    210215    JSObject* result = constructDate(exec, exec->lexicalGlobalObject(), argList);
    211     if (exec->hadException()) {
    212         JSValue exceptionValue = exec->exception();
    213         if (exception)
    214             *exception = toRef(exec, exceptionValue);
    215         exec->clearException();
    216 #if ENABLE(REMOTE_INSPECTOR)
    217         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    218 #endif
     216    if (handleExceptionIfNeeded(exec, exception) == ExceptionStatus::DidThrow)
    219217        result = 0;
    220     }
    221218
    222219    return toRef(result);
     
    236233    JSObject* result = ErrorInstance::create(exec, errorStructure, message);
    237234
    238     if (exec->hadException()) {
    239         JSValue exceptionValue = exec->exception();
    240         if (exception)
    241             *exception = toRef(exec, exceptionValue);
    242         exec->clearException();
    243 #if ENABLE(REMOTE_INSPECTOR)
    244         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    245 #endif
     235    if (handleExceptionIfNeeded(exec, exception) == ExceptionStatus::DidThrow)
    246236        result = 0;
    247     }
    248237
    249238    return toRef(result);
     
    264253
    265254    JSObject* result = constructRegExp(exec, exec->lexicalGlobalObject(),  argList);
    266     if (exec->hadException()) {
    267         JSValue exceptionValue = exec->exception();
    268         if (exception)
    269             *exception = toRef(exec, exceptionValue);
    270         exec->clearException();
    271 #if ENABLE(REMOTE_INSPECTOR)
    272         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    273 #endif
     255    if (handleExceptionIfNeeded(exec, exception) == ExceptionStatus::DidThrow)
    274256        result = 0;
    275     }
    276257   
    277258    return toRef(result);
     
    340321
    341322    JSValue jsValue = jsObject->get(exec, propertyName->identifier(&exec->vm()));
    342     if (exec->hadException()) {
    343         JSValue exceptionValue = exec->exception();
    344         if (exception)
    345             *exception = toRef(exec, exceptionValue);
    346         exec->clearException();
    347 #if ENABLE(REMOTE_INSPECTOR)
    348         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    349 #endif
    350     }
     323    handleExceptionIfNeeded(exec, exception);
    351324    return toRef(exec, jsValue);
    352325}
     
    373346    }
    374347
    375     if (exec->hadException()) {
    376         JSValue exceptionValue = exec->exception();
    377         if (exception)
    378             *exception = toRef(exec, exceptionValue);
    379         exec->clearException();
    380 #if ENABLE(REMOTE_INSPECTOR)
    381         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    382 #endif
    383     }
     348    handleExceptionIfNeeded(exec, exception);
    384349}
    385350
     
    396361
    397362    JSValue jsValue = jsObject->get(exec, propertyIndex);
    398     if (exec->hadException()) {
    399         JSValue exceptionValue = exec->exception();
    400         if (exception)
    401             *exception = toRef(exec, exceptionValue);
    402         exec->clearException();
    403 #if ENABLE(REMOTE_INSPECTOR)
    404         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    405 #endif
    406     }
     363    handleExceptionIfNeeded(exec, exception);
    407364    return toRef(exec, jsValue);
    408365}
     
    422379   
    423380    jsObject->methodTable()->putByIndex(jsObject, exec, propertyIndex, jsValue, false);
    424     if (exec->hadException()) {
    425         JSValue exceptionValue = exec->exception();
    426         if (exception)
    427             *exception = toRef(exec, exceptionValue);
    428         exec->clearException();
    429 #if ENABLE(REMOTE_INSPECTOR)
    430         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    431 #endif
    432     }
     381    handleExceptionIfNeeded(exec, exception);
    433382}
    434383
     
    445394
    446395    bool result = jsObject->methodTable()->deleteProperty(jsObject, exec, propertyName->identifier(&exec->vm()));
    447     if (exec->hadException()) {
    448         JSValue exceptionValue = exec->exception();
    449         if (exception)
    450             *exception = toRef(exec, exceptionValue);
    451         exec->clearException();
    452 #if ENABLE(REMOTE_INSPECTOR)
    453         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    454 #endif
    455     }
     396    handleExceptionIfNeeded(exec, exception);
    456397    return result;
    457398}
     
    617558
    618559    JSValueRef result = toRef(exec, call(exec, jsObject, callType, callData, jsThisObject, argList));
    619     if (exec->hadException()) {
    620         JSValue exceptionValue = exec->exception();
    621         if (exception)
    622             *exception = toRef(exec, exceptionValue);
    623         exec->clearException();
    624 #if ENABLE(REMOTE_INSPECTOR)
    625         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    626 #endif
     560    if (handleExceptionIfNeeded(exec, exception) == ExceptionStatus::DidThrow)
    627561        result = 0;
    628     }
    629562    return result;
    630563}
     
    658591        argList.append(toJS(exec, arguments[i]));
    659592    JSObjectRef result = toRef(construct(exec, jsObject, constructType, constructData, argList));
    660     if (exec->hadException()) {
    661         JSValue exceptionValue = exec->exception();
    662         if (exception)
    663             *exception = toRef(exec, exceptionValue);
    664         exec->clearException();
    665 #if ENABLE(REMOTE_INSPECTOR)
    666         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    667 #endif
     593    if (handleExceptionIfNeeded(exec, exception) == ExceptionStatus::DidThrow)
    668594        result = 0;
    669     }
    670595    return result;
    671596}
  • TabularUnified trunk/Source/JavaScriptCore/API/JSScriptRef.cpp

    r181664 r185259  
    2828#include "APICast.h"
    2929#include "Completion.h"
     30#include "Exception.h"
    3031#include "JSBasePrivate.h"
    3132#include "VM.h"
     
    143144        return 0;
    144145    }
    145     JSValue internalException;
     146    Exception* internalException;
    146147    JSValue thisValue = thisValueRef ? toJS(exec, thisValueRef) : jsUndefined();
    147     JSValue result = evaluate(exec, SourceCode(script), thisValue, &internalException);
     148    JSValue result = evaluate(exec, SourceCode(script), thisValue, internalException);
    148149    if (internalException) {
    149150        if (exception)
    150             *exception = toRef(exec, internalException);
     151            *exception = toRef(exec, internalException->value());
    151152        return 0;
    152153    }
  • TabularUnified trunk/Source/JavaScriptCore/API/JSValue.mm

    r182297 r185259  
    2929#import "DateInstance.h"
    3030#import "Error.h"
     31#import "Exception.h"
    3132#import "JavaScriptCore.h"
    3233#import "JSContextInternal.h"
     
    646647
    647648#if ENABLE(REMOTE_INSPECTOR)
    648 static void reportExceptionToInspector(JSGlobalContextRef context, JSC::JSValue exception)
     649static void reportExceptionToInspector(JSGlobalContextRef context, JSC::JSValue exceptionValue)
    649650{
    650651    JSC::ExecState* exec = toJS(context);
     652    JSC::Exception* exception = JSC::Exception::create(exec->vm(), exceptionValue);
    651653    exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exception);
    652654}
  • TabularUnified trunk/Source/JavaScriptCore/API/JSValueRef.cpp

    r182297 r185259  
    2929#include "APICast.h"
    3030#include "DateInstance.h"
     31#include "Exception.h"
    3132#include "JSAPIWrapperObject.h"
    3233#include "JSCInlines.h"
     
    5354using namespace JSC;
    5455
     56enum class ExceptionStatus {
     57    DidThrow,
     58    DidNotThrow
     59};
     60
     61static ExceptionStatus handleExceptionIfNeeded(ExecState* exec, JSValueRef* returnedExceptionRef)
     62{
     63    if (exec->hadException()) {
     64        Exception* exception = exec->exception();
     65        if (returnedExceptionRef)
     66            *returnedExceptionRef = toRef(exec, exception->value());
     67        exec->clearException();
     68#if ENABLE(REMOTE_INSPECTOR)
     69        exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exception);
     70#endif
     71        return ExceptionStatus::DidThrow;
     72    }
     73    return ExceptionStatus::DidNotThrow;
     74}
     75
    5576#if PLATFORM(MAC)
    5677static bool evernoteHackNeeded()
     
    225246
    226247    bool result = JSValue::equal(exec, jsA, jsB); // false if an exception is thrown
    227     if (exec->hadException()) {
    228         JSValue exceptionValue = exec->exception();
    229         if (exception)
    230             *exception = toRef(exec, exceptionValue);
    231         exec->clearException();
    232 #if ENABLE(REMOTE_INSPECTOR)
    233         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    234 #endif
    235     }
     248    handleExceptionIfNeeded(exec, exception);
     249   
    236250    return result;
    237251}
     
    267281        return false;
    268282    bool result = jsConstructor->hasInstance(exec, jsValue); // false if an exception is thrown
    269     if (exec->hadException()) {
    270         JSValue exceptionValue = exec->exception();
    271         if (exception)
    272             *exception = toRef(exec, exceptionValue);
    273         exec->clearException();
    274 #if ENABLE(REMOTE_INSPECTOR)
    275         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    276 #endif
    277     }
     283    handleExceptionIfNeeded(exec, exception);
    278284    return result;
    279285}
     
    369375    if (exception)
    370376        *exception = 0;
    371     if (exec->hadException()) {
    372         JSValue exceptionValue = exec->exception();
    373         if (exception)
    374             *exception = toRef(exec, exceptionValue);
    375         exec->clearException();
    376 #if ENABLE(REMOTE_INSPECTOR)
    377         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    378 #endif
    379         return 0;
    380     }
     377    if (handleExceptionIfNeeded(exec, exception) == ExceptionStatus::DidThrow)
     378        return 0;
    381379    return OpaqueJSString::create(result).leakRef();
    382380}
     
    407405
    408406    double number = jsValue.toNumber(exec);
    409     if (exec->hadException()) {
    410         JSValue exceptionValue = exec->exception();
    411         if (exception)
    412             *exception = toRef(exec, exceptionValue);
    413         exec->clearException();
    414 #if ENABLE(REMOTE_INSPECTOR)
    415         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    416 #endif
     407    if (handleExceptionIfNeeded(exec, exception) == ExceptionStatus::DidThrow)
    417408        number = PNaN;
    418     }
    419409    return number;
    420410}
     
    432422   
    433423    RefPtr<OpaqueJSString> stringRef(OpaqueJSString::create(jsValue.toString(exec)->value(exec)));
    434     if (exec->hadException()) {
    435         JSValue exceptionValue = exec->exception();
    436         if (exception)
    437             *exception = toRef(exec, exceptionValue);
    438         exec->clearException();
    439 #if ENABLE(REMOTE_INSPECTOR)
    440         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    441 #endif
     424    if (handleExceptionIfNeeded(exec, exception) == ExceptionStatus::DidThrow)
    442425        stringRef.clear();
    443     }
    444426    return stringRef.release().leakRef();
    445427}
     
    457439   
    458440    JSObjectRef objectRef = toRef(jsValue.toObject(exec));
    459     if (exec->hadException()) {
    460         JSValue exceptionValue = exec->exception();
    461         if (exception)
    462             *exception = toRef(exec, exceptionValue);
    463         exec->clearException();
    464 #if ENABLE(REMOTE_INSPECTOR)
    465         exec->vmEntryGlobalObject()->inspectorController().reportAPIException(exec, exceptionValue);
    466 #endif
     441    if (handleExceptionIfNeeded(exec, exception) == ExceptionStatus::DidThrow)
    467442        objectRef = 0;
    468     }
    469443    return objectRef;
    470444}
  • TabularUnified trunk/Source/JavaScriptCore/CMakeLists.txt

    r184776 r185259  
    455455    runtime/ErrorInstance.cpp
    456456    runtime/ErrorPrototype.cpp
     457    runtime/Exception.cpp
    457458    runtime/ExceptionFuzz.cpp
    458459    runtime/ExceptionHelpers.cpp
  • TabularUnified trunk/Source/JavaScriptCore/ChangeLog

    r185240 r185259  
     12015-06-05  Mark Lam  <mark.lam@apple.com>
     2
     3        finally blocks should not set the exception stack trace when re-throwing the exception.
     4        https://bugs.webkit.org/show_bug.cgi?id=145525
     5
     6        Reviewed by Geoffrey Garen.
     7
     8        How exceptions presently work:
     9        =============================
     10        1. op_throw can throw any JSValue.
     11        2. the VM tries to capture the stack at the throw point and propagate that as needed.
     12        3. finally blocks are implemented using op_catch to catch the thrown value, and throws it again using op_throw.
     13
     14        What's wrong with how it presently works:
     15        ========================================
     16        1. finally's makes for bad exception throw line numbers in the Inspector console.
     17
     18           The op_throw in finally will throw the value anew i.e. it captures a stack from the re-throw point.
     19           As a result, the Inspector sees the finally block as the throw point.  The original stack is lost.
     20
     21        2. finally's breaks the Inspector's "Breaks on Uncaught Exception"
     22
     23           This is because finally blocks are indistinguishable from catch blocks.  As a result, a try-finally,
     24           which should break in the Inspector on the throw, does not because the Inspector thought the
     25           exception was "caught".
     26
     27        3. finally's yields confusing break points when the Inspector "Breaks on All Exceptions"
     28
     29           a. In a try-finally scenario, the Inspector breaks 2 times: 1 at the throw, 1 at the finally.
     30           b. In a for-of loop (which has synthesized finallys), the Inspector will do another break.
     31              Similarly for other cases of JS code which synthesize finallys.
     32           c. At VM re-entry boundaries (e.g. js throws & returns to native code, which returns to js),
     33              the Inspector will do another break if there's an uncaught exception.
     34
     35        How this patch fixes the issues:
     36        ===============================
     37        1. We introduce an Exception object that wraps the thrown value and the exception stack.
     38
     39           When throwing an exception, the VM will check if the thrown value is an Exception
     40           object or not.  If it is not an Exception object, then we must be throwing a new
     41           exception.  The VM will create an Exception object to wrap the thrown value and
     42           capture the current stack for it.
     43
     44           If the thrown value is already an Exception object, then the requested throw operation
     45           must be a re-throw.  The VM will not capture a new stack for it.
     46
     47        2. op_catch will now populate 2 locals: 1 for the Exception, 1 for the thrown JSValue.
     48
     49           The VM is aware of the Exception object and uses it for rethrows in finally blocks.
     50           JS source code is never aware of the Exception object.
     51
     52           JS code is aware of the thrown value.  If it throws the caught thrown value, that
     53           constitutes a new throw, and a new Exception object will be created for it.
     54
     55        3. The VM no longer tracks the thrown JSValue and the exception stack.  It will only
     56           track a m_exception field which is an Exception*.
     57
     58        4. The BytecodeGenerator has already been updated in a prior patch to distinguish
     59           between Catch, Finally, and SynthesizedFinally blocks.  The interpreter runtime will
     60           now report to the debugger whether we have a Catch handler, not just any handlers.
     61
     62           The debugger will use this detail to determine whether to break or not.  "Break on
     63           uncaught exceptions" will only break if no Catch handler was found.
     64
     65           This solves the issue of the debugger breaking at finally blocks, and for-of statements.
     66
     67        5. The Exception object will also have a flag to indicate whether the debugger has been
     68           notified of the Exception being thrown.  Once the Interpreter notifies the debugger
     69           of the Exception object, it will mark this flag and not repeat the notify the debugger
     70           again of the same Exception.
     71
     72           This solves the issue of the debugger breaking at VM re-entry points due to uncaught
     73           exceptions.
     74
     75        6. The life-cycle of the captured exception stack trace will now follow the life-cycle
     76           of the Exception object.
     77
     78        Other changes:
     79        7. Change all clients of the VM::exception() to expect an Exception* instead of JSValue.
     80
     81        8. Fixed a few bugs where thrown exceptions are not cleared before exiting the VM.
     82
     83        9. Also renamed some variables and classes to better describe what they are.
     84
     85        * API/JSBase.cpp:
     86        (JSEvaluateScript):
     87        (JSCheckScriptSyntax):
     88
     89        * API/JSObjectRef.cpp:
     90        (handleExceptionIfNeeded):
     91        - The functions below all do the same exception check.  Added this helper
     92          to simplify the code.
     93        (JSClassCreate):
     94        (JSObjectMakeFunction):
     95        (JSObjectMakeArray):
     96        (JSObjectMakeDate):
     97        (JSObjectMakeError):
     98        (JSObjectMakeRegExp):
     99        (JSObjectGetProperty):
     100        (JSObjectSetProperty):
     101        (JSObjectGetPropertyAtIndex):
     102        (JSObjectSetPropertyAtIndex):
     103        (JSObjectDeleteProperty):
     104        (JSObjectCallAsFunction):
     105        (JSObjectCallAsConstructor):
     106
     107        * API/JSScriptRef.cpp:
     108        * API/JSValue.mm:
     109        (JSContainerConvertor::take):
     110        (reportExceptionToInspector):
     111
     112        * API/JSValueRef.cpp:
     113        (handleExceptionIfNeeded):
     114        - The functions below all do the same exception check.  Added this helper
     115          to simplify the code.
     116        (evernoteHackNeeded):
     117        (JSValueIsEqual):
     118        (JSValueIsInstanceOfConstructor):
     119        (JSValueCreateJSONString):
     120        (JSValueToNumber):
     121        (JSValueToStringCopy):
     122        (JSValueToObject):
     123
     124        * CMakeLists.txt:
     125        * JavaScriptCore.vcxproj/JavaScriptCore.vcxproj:
     126        * JavaScriptCore.vcxproj/JavaScriptCore.vcxproj.filters:
     127        * JavaScriptCore.xcodeproj/project.pbxproj:
     128        - Added new files Exception.h and Exception.cpp.
     129
     130        * bindings/ScriptFunctionCall.cpp:
     131        (Deprecated::ScriptFunctionCall::call):
     132        * bindings/ScriptFunctionCall.h:
     133
     134        * bytecode/BytecodeList.json:
     135        - op_catch now had 2 operands: the exception register, and the thrown value register.
     136
     137        * bytecode/BytecodeUseDef.h:
     138        (JSC::computeDefsForBytecodeOffset):
     139        * bytecode/CodeBlock.cpp:
     140        (JSC::CodeBlock::dumpBytecode):
     141        (JSC::CodeBlock::handlerForBytecodeOffset):
     142        * bytecode/CodeBlock.h:
     143        - handlerForBytecodeOffset() now can look for just Catch handlers only.
     144
     145        * bytecode/HandlerInfo.h:
     146        - Cleaned up some white space I accidentally added in a previous patch.
     147
     148        * bytecompiler/BytecodeGenerator.cpp:
     149        (JSC::BytecodeGenerator::pushTry):
     150        (JSC::BytecodeGenerator::popTryAndEmitCatch):
     151        (JSC::BytecodeGenerator::emitThrowReferenceError):
     152        (JSC::BytecodeGenerator::emitEnumeration):
     153        * bytecompiler/BytecodeGenerator.h:
     154        (JSC::BytecodeGenerator::emitThrow):
     155        * bytecompiler/NodesCodegen.cpp:
     156        (JSC::TryNode::emitBytecode):
     157        - Adding support for op_catch's 2 operands.
     158
     159        * debugger/Debugger.cpp:
     160        (JSC::Debugger::hasBreakpoint):
     161        (JSC::Debugger::pauseIfNeeded):
     162        (JSC::Debugger::exception):
     163        * debugger/Debugger.h:
     164        * debugger/DebuggerCallFrame.cpp:
     165        (JSC::DebuggerCallFrame::thisValue):
     166        (JSC::DebuggerCallFrame::evaluate):
     167        * debugger/DebuggerCallFrame.h:
     168        (JSC::DebuggerCallFrame::isValid):
     169        * inspector/InjectedScriptManager.cpp:
     170        (Inspector::InjectedScriptManager::createInjectedScript):
     171        * inspector/InspectorEnvironment.h:
     172        * inspector/JSGlobalObjectInspectorController.cpp:
     173        (Inspector::JSGlobalObjectInspectorController::appendAPIBacktrace):
     174        (Inspector::JSGlobalObjectInspectorController::reportAPIException):
     175        * inspector/JSGlobalObjectInspectorController.h:
     176        * inspector/JSGlobalObjectScriptDebugServer.h:
     177        * inspector/JSJavaScriptCallFrame.cpp:
     178        (Inspector::JSJavaScriptCallFrame::evaluate):
     179        * inspector/JavaScriptCallFrame.h:
     180        (Inspector::JavaScriptCallFrame::vmEntryGlobalObject):
     181        (Inspector::JavaScriptCallFrame::thisValue):
     182        (Inspector::JavaScriptCallFrame::evaluate):
     183        * inspector/ScriptCallStackFactory.cpp:
     184        (Inspector::extractSourceInformationFromException):
     185        (Inspector::createScriptCallStackFromException):
     186        * inspector/ScriptCallStackFactory.h:
     187        * inspector/ScriptDebugServer.cpp:
     188        (Inspector::ScriptDebugServer::evaluateBreakpointAction):
     189        (Inspector::ScriptDebugServer::handleBreakpointHit):
     190        (Inspector::ScriptDebugServer::handleExceptionInBreakpointCondition):
     191        * inspector/ScriptDebugServer.h:
     192        * interpreter/CallFrame.h:
     193        (JSC::ExecState::clearException):
     194        (JSC::ExecState::exception):
     195        (JSC::ExecState::hadException):
     196        (JSC::ExecState::atomicStringTable):
     197        (JSC::ExecState::propertyNames):
     198        (JSC::ExecState::clearSupplementaryExceptionInfo): Deleted.
     199
     200        * interpreter/Interpreter.cpp:
     201        (JSC::unwindCallFrame):
     202        (JSC::Interpreter::stackTraceAsString):
     203        (JSC::GetCatchHandlerFunctor::GetCatchHandlerFunctor):
     204        (JSC::GetCatchHandlerFunctor::operator()):
     205        (JSC::Interpreter::unwind):
     206        - Added a check for didNotifyInspectorOfThrow() here to prevent duplicate reports
     207          of the same Exception to the debugger.
     208
     209        (JSC::GetExceptionHandlerFunctor::GetExceptionHandlerFunctor): Deleted.
     210        (JSC::GetExceptionHandlerFunctor::operator()): Deleted.
     211        - Renamed GetExceptionHandlerFunctor to GetCatchHandlerFunctor since the debugger
     212          is only interested in knowing whether we have Catch handlers.
     213
     214        * interpreter/Interpreter.h:
     215        (JSC::SuspendExceptionScope::SuspendExceptionScope):
     216        (JSC::SuspendExceptionScope::~SuspendExceptionScope):
     217        (JSC::Interpreter::sampler):
     218        (JSC::ClearExceptionScope::ClearExceptionScope): Deleted.
     219        (JSC::ClearExceptionScope::~ClearExceptionScope): Deleted.
     220        - Renamed ClearExceptionScope to SuspendExceptionScope because "clear" implies that
     221          we're purging the exception.  Instead, we're merely suspending any handling of
     222          that exception for a period defined by the scope.
     223
     224        * jit/AssemblyHelpers.cpp:
     225        (JSC::AssemblyHelpers::emitExceptionCheck):
     226
     227        * jit/JITExceptions.cpp:
     228        (JSC::genericUnwind):
     229        - Removed the exception argument.  It is always the value in VM::exception() anyway.
     230          genericUnwind() can just get it from the VM, and save everyone some work.
     231
     232        * jit/JITExceptions.h:
     233        * jit/JITOpcodes.cpp:
     234        (JSC::JIT::emit_op_catch):
     235        * jit/JITOpcodes32_64.cpp:
     236        (JSC::JIT::privateCompileCTINativeCall):
     237        (JSC::JIT::emit_op_catch):
     238        - Add support for the new op_catch operands.
     239
     240        * jit/JITOperations.cpp:
     241        * jit/ThunkGenerators.cpp:
     242        (JSC::nativeForGenerator):
     243        * jsc.cpp:
     244        (functionRun):
     245        (functionLoad):
     246        (runWithScripts):
     247        (runInteractive):
     248        * llint/LLIntOffsetsExtractor.cpp:
     249        * llint/LLIntSlowPaths.cpp:
     250        (JSC::LLInt::LLINT_SLOW_PATH_DECL):
     251
     252        * llint/LowLevelInterpreter32_64.asm:
     253        * llint/LowLevelInterpreter64.asm:
     254        - Add support for the new op_catch operands.  Also update the code to handle
     255          VM::m_exception being an Exception pointer, not a JSValue.
     256
     257        * parser/NodeConstructors.h:
     258        (JSC::TryNode::TryNode):
     259        * parser/Nodes.h:
     260        * runtime/CallData.cpp:
     261        (JSC::call):
     262        * runtime/CallData.h:
     263
     264        * runtime/Completion.cpp:
     265        (JSC::evaluate):
     266        * runtime/Completion.h:
     267        (JSC::evaluate):
     268        - Change evaluate() to take a reference to the returned exception value instead
     269          of a pointer.  In all but 2 or 3 cases, we want the returned exception anyway.
     270          Might as well simplify the code by requiring the reference.
     271
     272        * runtime/Error.h:
     273        (JSC::throwVMError):
     274        (JSC::throwVMTypeError):
     275
     276        * runtime/Exception.cpp: Added.
     277        (JSC::Exception::create):
     278        (JSC::Exception::destroy):
     279        (JSC::Exception::createStructure):
     280        (JSC::Exception::visitChildren):
     281        (JSC::Exception::Exception):
     282        (JSC::Exception::~Exception):
     283        * runtime/Exception.h: Added.
     284        (JSC::Exception::valueOffset):
     285        (JSC::Exception::cast):
     286        (JSC::Exception::value):
     287        (JSC::Exception::stack):
     288        (JSC::Exception::didNotifyInspectorOfThrow):
     289        (JSC::Exception::setDidNotifyInspectorOfThrow):
     290
     291        * runtime/ExceptionHelpers.cpp:
     292        (JSC::createTerminatedExecutionException):
     293        (JSC::isTerminatedExecutionException):
     294        (JSC::createStackOverflowError):
     295        * runtime/ExceptionHelpers.h:
     296        * runtime/GetterSetter.cpp:
     297        (JSC::callGetter):
     298        * runtime/IteratorOperations.cpp:
     299        (JSC::iteratorClose):
     300        * runtime/JSObject.cpp:
     301        * runtime/JSPromiseConstructor.cpp:
     302        (JSC::constructPromise):
     303        * runtime/JSPromiseDeferred.cpp:
     304        (JSC::updateDeferredFromPotentialThenable):
     305        (JSC::abruptRejection):
     306        * runtime/JSPromiseReaction.cpp:
     307        (JSC::ExecutePromiseReactionMicrotask::run):
     308
     309        * runtime/VM.cpp:
     310        (JSC::VM::VM):
     311        (JSC::VM::releaseExecutableMemory):
     312        (JSC::VM::throwException):
     313        (JSC::VM::setStackPointerAtVMEntry):
     314        (JSC::VM::getExceptionInfo): Deleted.
     315        (JSC::VM::setExceptionInfo): Deleted.
     316        (JSC::VM::clearException): Deleted.
     317        (JSC::clearExceptionStack): Deleted.
     318        * runtime/VM.h:
     319        (JSC::VM::targetMachinePCForThrowOffset):
     320        (JSC::VM::clearException):
     321        (JSC::VM::setException):
     322        (JSC::VM::exception):
     323        (JSC::VM::addressOfException):
     324        (JSC::VM::exceptionStack): Deleted.
     325        * runtime/VMEntryScope.cpp:
     326        (JSC::VMEntryScope::VMEntryScope):
     327        (JSC::VMEntryScope::setEntryScopeDidPopListener):
     328
    13292015-06-04  Benjamin Poulain  <bpoulain@apple.com>
    2330
  • TabularUnified trunk/Source/JavaScriptCore/JavaScriptCore.vcxproj/JavaScriptCore.vcxproj

    r184776 r185259  
    713713    <ClCompile Include="..\runtime\ErrorInstance.cpp" />
    714714    <ClCompile Include="..\runtime\ErrorPrototype.cpp" />
     715    <ClCompile Include="..\runtime\Exception.cpp" />
    715716    <ClCompile Include="..\runtime\ExceptionFuzz.cpp" />
    716717    <ClCompile Include="..\runtime\ExceptionHelpers.cpp" />
     
    15041505    <ClInclude Include="..\runtime\ErrorInstance.h" />
    15051506    <ClInclude Include="..\runtime\ErrorPrototype.h" />
     1507    <ClInclude Include="..\runtime\Exception.h" />
    15061508    <ClInclude Include="..\runtime\ExceptionFuzz.h" />
    15071509    <ClInclude Include="..\runtime\ExceptionHelpers.h" />
  • TabularUnified trunk/Source/JavaScriptCore/JavaScriptCore.vcxproj/JavaScriptCore.vcxproj.filters

    r184776 r185259  
    16991699      <Filter>dfg</Filter>
    17001700    </ClCompile>
     1701    <ClCompile Include="..\runtime\Exception.cpp">
     1702      <Filter>runtime</Filter>
     1703    </ClCompile>
    17011704    <ClCompile Include="..\runtime\ExceptionFuzz.cpp">
    17021705      <Filter>runtime</Filter>
     
    41904193    <ClInclude Include="..\dfg\DFGValueStrength.h">
    41914194      <Filter>dfg</Filter>
     4195    </ClInclude>
     4196    <ClInclude Include="..\runtime\Exception.h">
     4197      <Filter>runtime</Filter>
    41924198    </ClInclude>
    41934199    <ClInclude Include="..\runtime\ExceptionFuzz.h">
  • TabularUnified trunk/Source/JavaScriptCore/JavaScriptCore.xcodeproj/project.pbxproj

    r184776 r185259  
    16611661                FE0D4A061AB8DD0A002F54BF /* ExecutionTimeLimitTest.cpp in Sources */ = {isa = PBXBuildFile; fileRef = FE0D4A041AB8DD0A002F54BF /* ExecutionTimeLimitTest.cpp */; };
    16621662                FE0D4A091ABA2437002F54BF /* GlobalContextWithFinalizerTest.cpp in Sources */ = {isa = PBXBuildFile; fileRef = FE0D4A071ABA2437002F54BF /* GlobalContextWithFinalizerTest.cpp */; };
     1663                FE1C0FFD1B193E9800B53FCA /* Exception.h in Headers */ = {isa = PBXBuildFile; fileRef = FE1C0FFC1B193E9800B53FCA /* Exception.h */; settings = {ATTRIBUTES = (Private, ); }; };
     1664                FE1C0FFF1B194FD100B53FCA /* Exception.cpp in Sources */ = {isa = PBXBuildFile; fileRef = FE1C0FFE1B194FD100B53FCA /* Exception.cpp */; };
    16631665                FE20CE9D15F04A9500DF3430 /* LLIntCLoop.cpp in Sources */ = {isa = PBXBuildFile; fileRef = FE20CE9B15F04A9500DF3430 /* LLIntCLoop.cpp */; };
    16641666                FE20CE9E15F04A9500DF3430 /* LLIntCLoop.h in Headers */ = {isa = PBXBuildFile; fileRef = FE20CE9C15F04A9500DF3430 /* LLIntCLoop.h */; settings = {ATTRIBUTES = (Private, ); }; };
     
    34643466                FE0D4A071ABA2437002F54BF /* GlobalContextWithFinalizerTest.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = GlobalContextWithFinalizerTest.cpp; path = API/tests/GlobalContextWithFinalizerTest.cpp; sourceTree = "<group>"; };
    34653467                FE0D4A081ABA2437002F54BF /* GlobalContextWithFinalizerTest.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = GlobalContextWithFinalizerTest.h; path = API/tests/GlobalContextWithFinalizerTest.h; sourceTree = "<group>"; };
     3468                FE1C0FFC1B193E9800B53FCA /* Exception.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = Exception.h; sourceTree = "<group>"; };
     3469                FE1C0FFE1B194FD100B53FCA /* Exception.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = Exception.cpp; sourceTree = "<group>"; };
    34663470                FE20CE9B15F04A9500DF3430 /* LLIntCLoop.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = LLIntCLoop.cpp; path = llint/LLIntCLoop.cpp; sourceTree = "<group>"; };
    34673471                FE20CE9C15F04A9500DF3430 /* LLIntCLoop.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = LLIntCLoop.h; path = llint/LLIntCLoop.h; sourceTree = "<group>"; };
     
    44284432                                BC02E9060E1839DB000F9297 /* ErrorPrototype.cpp */,
    44294433                                BC02E9070E1839DB000F9297 /* ErrorPrototype.h */,
     4434                                FE1C0FFC1B193E9800B53FCA /* Exception.h */,
     4435                                FE1C0FFE1B194FD100B53FCA /* Exception.cpp */,
    44304436                                0F12DE0D1979D5FD0006FF4E /* ExceptionFuzz.cpp */,
    44314437                                0F12DE0E1979D5FD0006FF4E /* ExceptionFuzz.h */,
     
    56255631                                FE5068651AE246390009DAB7 /* DeferredSourceDump.h in Headers */,
    56265632                                C442CB251A6CDB8C005D3D7C /* JSInputs.json in Headers */,
     5633                                FE1C0FFD1B193E9800B53FCA /* Exception.h in Headers */,
    56275634                                52678F911A04177C006A306D /* ControlFlowProfiler.h in Headers */,
    56285635                                52678F8F1A031009006A306D /* BasicBlockLocation.h in Headers */,
     
    74097416                                7C008CD2186F8A9300955C24 /* JSPromiseFunctions.cpp in Sources */,
    74107417                                7C184E1E17BEE22E007CB63A /* JSPromisePrototype.cpp in Sources */,
     7418                                FE1C0FFF1B194FD100B53FCA /* Exception.cpp in Sources */,
    74117419                                709FB86B1AE335C60039D069 /* WeakSetPrototype.cpp in Sources */,
    74127420                                7C008CDE1871258D00955C24 /* JSPromiseReaction.cpp in Sources */,
  • TabularUnified trunk/Source/JavaScriptCore/bindings/ScriptFunctionCall.cpp

    r182205 r185259  
    134134
    135135    JSValue result;
    136     JSValue exception;
     136    Exception* exception;
    137137    if (m_callHandler)
    138         result = m_callHandler(m_exec, function, callType, callData, thisObject, m_arguments, &exception);
     138        result = m_callHandler(m_exec, function, callType, callData, thisObject, m_arguments, exception);
    139139    else
    140         result = JSC::call(m_exec, function, callType, callData, thisObject, m_arguments, &exception);
     140        result = JSC::call(m_exec, function, callType, callData, thisObject, m_arguments, exception);
    141141
    142142    if (exception) {
  • TabularUnified trunk/Source/JavaScriptCore/bindings/ScriptFunctionCall.h

    r167142 r185259  
    7272class JS_EXPORT_PRIVATE ScriptFunctionCall : public ScriptCallArgumentHandler {
    7373public:
    74     typedef JSC::JSValue (*ScriptFunctionCallHandler)(JSC::ExecState* exec, JSC::JSValue functionObject, JSC::CallType callType, const JSC::CallData& callData, JSC::JSValue thisValue, const JSC::ArgList& args, JSC::JSValue* exception);
     74    typedef JSC::JSValue (*ScriptFunctionCallHandler)(JSC::ExecState* exec, JSC::JSValue functionObject, JSC::CallType callType, const JSC::CallData& callData, JSC::JSValue thisValue, const JSC::ArgList& args, JSC::Exception*& exception);
    7575    ScriptFunctionCall(const ScriptObject& thisObject, const String& name, ScriptFunctionCallHandler handler = nullptr);
    7676    ScriptValue call(bool& hadException);
  • TabularUnified trunk/Source/JavaScriptCore/bytecode/BytecodeList.json

    r184328 r185259  
    114114            { "name" : "op_pop_scope", "length" : 2 },
    115115            { "name" : "op_push_name_scope", "length" : 5 },
    116             { "name" : "op_catch", "length" : 2 },
     116            { "name" : "op_catch", "length" : 3 },
    117117            { "name" : "op_throw", "length" : 2 },
    118118            { "name" : "op_throw_static_error", "length" : 3 },
  • TabularUnified trunk/Source/JavaScriptCore/bytecode/BytecodeUseDef.h

    r184324 r185259  
    304304    case op_strcat:
    305305    case op_to_primitive:
    306     case op_catch:
    307306    case op_create_this:
    308307    case op_new_array:
     
    375374        return;
    376375    }
     376    case op_catch:
    377377    case op_create_lexical_environment: {
    378378        functor(codeBlock, instruction, opcodeID, instruction[1].u.operand);
  • TabularUnified trunk/Source/JavaScriptCore/bytecode/CodeBlock.cpp

    r185083 r185259  
    14851485        case op_catch: {
    14861486            int r0 = (++it)->u.operand;
    1487             printLocationOpAndRegisterOperand(out, exec, location, it, "catch", r0);
     1487            int r1 = (++it)->u.operand;
     1488            printLocationAndOp(out, exec, location, it, "catch");
     1489            out.printf("%s, %s", registerName(r0).data(), registerName(r1).data());
    14881490            break;
    14891491        }
     
    28792881#endif
    28802882
    2881 HandlerInfo* CodeBlock::handlerForBytecodeOffset(unsigned bytecodeOffset)
     2883HandlerInfo* CodeBlock::handlerForBytecodeOffset(unsigned bytecodeOffset, RequiredHandler requiredHandler)
    28822884{
    28832885    RELEASE_ASSERT(bytecodeOffset < instructions().size());
     
    28892891    for (size_t i = 0; i < exceptionHandlers.size(); ++i) {
    28902892        HandlerInfo& handler = exceptionHandlers[i];
     2893        if ((requiredHandler == RequiredHandler::CatchHandler) && !handler.isCatchHandler())
     2894            continue;
     2895
    28912896        // Handlers are ordered innermost first, so the first handler we encounter
    28922897        // that contains the source address is the correct handler to use.
  • TabularUnified trunk/Source/JavaScriptCore/bytecode/CodeBlock.h

    r183570 r185259  
    186186    }
    187187
    188     HandlerInfo* handlerForBytecodeOffset(unsigned bytecodeOffset);
     188    enum class RequiredHandler {
     189        CatchHandler,
     190        AnyHandler
     191    };
     192    HandlerInfo* handlerForBytecodeOffset(unsigned bytecodeOffset, RequiredHandler = RequiredHandler::AnyHandler);
    189193    unsigned lineNumberForBytecodeOffset(unsigned bytecodeOffset);
    190194    unsigned columnNumberForBytecodeOffset(unsigned bytecodeOffset);
  • TabularUnified trunk/Source/JavaScriptCore/bytecode/HandlerInfo.h

    r185083 r185259  
    4141    HandlerType type() const { return static_cast<HandlerType>(typeBits); }
    4242    void setType(HandlerType type) { typeBits = static_cast<uint32_t>(type); }
    43    
     43
    4444    const char* typeName()
    4545    {
     
    5656        return nullptr;
    5757    }
    58    
     58
    5959    bool isCatchHandler() const { return type() == HandlerType::Catch; }
    60    
     60
    6161    uint32_t start;
    6262    uint32_t end;
  • TabularUnified trunk/Source/JavaScriptCore/bytecompiler/BytecodeGenerator.cpp

    r185083 r185259  
    25292529}
    25302530
    2531 RegisterID* BytecodeGenerator::popTryAndEmitCatch(TryData* tryData, RegisterID* targetRegister, Label* end, HandlerType handlerType)
     2531void BytecodeGenerator::popTryAndEmitCatch(TryData* tryData, RegisterID* exceptionRegister, RegisterID* thrownValueRegister, Label* end, HandlerType handlerType)
    25322532{
    25332533    m_usesExceptions = true;
     
    25472547
    25482548    emitOpcode(op_catch);
    2549     instructions().append(targetRegister->index());
    2550     return targetRegister;
     2549    instructions().append(exceptionRegister->index());
     2550    instructions().append(thrownValueRegister->index());
    25512551}
    25522552
     
    27612761        {
    27622762            RefPtr<Label> catchHere = emitLabel(newLabel().get());
    2763             RefPtr<RegisterID> exceptionRegister = popTryAndEmitCatch(tryData, newTemporary(), catchHere.get(), HandlerType::SynthesizedFinally);
     2763            RefPtr<RegisterID> exceptionRegister = newTemporary();
     2764            RefPtr<RegisterID> thrownValueRegister = newTemporary();
     2765            popTryAndEmitCatch(tryData, exceptionRegister.get(),
     2766                thrownValueRegister.get(), catchHere.get(), HandlerType::SynthesizedFinally);
     2767
    27642768            RefPtr<Label> catchDone = newLabel();
    27652769
     
    27792783
    27802784            // Absorb exception.
    2781             popTryAndEmitCatch(returnCallTryData, newTemporary(), catchDone.get(), HandlerType::SynthesizedFinally);
     2785            popTryAndEmitCatch(returnCallTryData, newTemporary(),
     2786                newTemporary(), catchDone.get(), HandlerType::SynthesizedFinally);
    27822787            emitThrow(exceptionRegister.get());
    27832788        }
  • TabularUnified trunk/Source/JavaScriptCore/bytecompiler/BytecodeGenerator.h

    r185083 r185259  
    554554        TryData* pushTry(Label* start);
    555555        // End a try block. 'end' must have been emitted.
    556         RegisterID* popTryAndEmitCatch(TryData*, RegisterID* targetRegister, Label* end, HandlerType);
     556        void popTryAndEmitCatch(TryData*, RegisterID* exceptionRegister, RegisterID* thrownValueRegister, Label* end, HandlerType);
    557557
    558558        void emitThrow(RegisterID* exc)
  • TabularUnified trunk/Source/JavaScriptCore/bytecompiler/NodesCodegen.cpp

    r185083 r185259  
    28822882        // Uncaught exception path: the catch block.
    28832883        RefPtr<Label> here = generator.emitLabel(generator.newLabel().get());
    2884         RefPtr<RegisterID> exceptionRegister = generator.popTryAndEmitCatch(tryData, generator.newTemporary(), here.get(), HandlerType::Catch);
     2884        RefPtr<RegisterID> exceptionRegister = generator.newTemporary();
     2885        RefPtr<RegisterID> thrownValueRegister = generator.newTemporary();
     2886        generator.popTryAndEmitCatch(tryData, exceptionRegister.get(), thrownValueRegister.get(), here.get(), HandlerType::Catch);
    28852887       
    28862888        if (m_finallyBlock) {
     
    28902892        }
    28912893
    2892         generator.emitPushCatchScope(generator.scopeRegister(), m_exceptionIdent, exceptionRegister.get(), DontDelete);
     2894        generator.emitPushCatchScope(generator.scopeRegister(), m_thrownValueIdent, thrownValueRegister.get(), DontDelete);
    28932895        generator.emitProfileControlFlow(m_tryBlock->endOffset() + 1);
    28942896        generator.emitNode(dst, m_catchBlock);
     
    29132915
    29142916        // Uncaught exception path: invoke the finally block, then re-throw the exception.
    2915         RefPtr<RegisterID> tempExceptionRegister = generator.popTryAndEmitCatch(tryData, generator.newTemporary(), preFinallyLabel.get(), HandlerType::Finally);
     2917        RefPtr<RegisterID> exceptionRegister = generator.newTemporary();
     2918        RefPtr<RegisterID> thrownValueRegister = generator.newTemporary();
     2919        generator.popTryAndEmitCatch(tryData, exceptionRegister.get(), thrownValueRegister.get(), preFinallyLabel.get(), HandlerType::Finally);
    29162920        generator.emitProfileControlFlow(finallyStartOffset);
    29172921        generator.emitNode(dst, m_finallyBlock);
    2918         generator.emitThrow(tempExceptionRegister.get());
     2922        generator.emitThrow(exceptionRegister.get());
    29192923
    29202924        generator.emitLabel(finallyEndLabel.get());
  • TabularUnified trunk/Source/JavaScriptCore/debugger/Debugger.cpp

    r183124 r185259  
    488488    TemporaryPausedState pausedState(*this);
    489489
    490     JSValue exception;
     490    Exception* exception;
    491491    DebuggerCallFrame* debuggerCallFrame = currentDebuggerCallFrame();
    492492    JSValue result = debuggerCallFrame->evaluate(breakpoint->condition, exception);
     
    696696}
    697697
    698 void Debugger::exception(CallFrame* callFrame, JSValue exception, bool hasHandler)
     698void Debugger::exception(CallFrame* callFrame, JSValue exception, bool hasCatchHandler)
    699699{
    700700    if (m_isPaused)
     
    702702
    703703    PauseReasonDeclaration reason(*this, PausedForException);
    704     if (m_pauseOnExceptionsState == PauseOnAllExceptions || (m_pauseOnExceptionsState == PauseOnUncaughtExceptions && !hasHandler)) {
     704    if (m_pauseOnExceptionsState == PauseOnAllExceptions || (m_pauseOnExceptionsState == PauseOnUncaughtExceptions && !hasCatchHandler)) {
    705705        m_pauseOnNextStatement = true;
    706706        setSteppingMode(SteppingModeEnabled);
  • TabularUnified trunk/Source/JavaScriptCore/debugger/Debugger.h

    r178137 r185259  
    110110    virtual void sourceParsed(ExecState*, SourceProvider*, int errorLineNumber, const WTF::String& errorMessage) = 0;
    111111
    112     void exception(CallFrame*, JSValue exceptionValue, bool hasHandler);
     112    void exception(CallFrame*, JSValue exceptionValue, bool hasCatchHandler);
    113113    void atStatement(CallFrame*);
    114114    void callEvent(CallFrame*);
     
    125125    virtual bool needPauseHandling(JSGlobalObject*) { return false; }
    126126    virtual void handleBreakpointHit(JSGlobalObject*, const Breakpoint&) { }
    127     virtual void handleExceptionInBreakpointCondition(ExecState*, JSValue exception) const { UNUSED_PARAM(exception); }
     127    virtual void handleExceptionInBreakpointCondition(ExecState*, Exception*) const { }
    128128    virtual void handlePause(JSGlobalObject*, ReasonForPause) { }
    129129    virtual void notifyDoneProcessingDebuggerEvents() { }
  • TabularUnified trunk/Source/JavaScriptCore/debugger/DebuggerCallFrame.cpp

    r182198 r185259  
    3333#include "DebuggerEvalEnabler.h"
    3434#include "DebuggerScope.h"
     35#include "Exception.h"
    3536#include "Interpreter.h"
    3637#include "JSFunction.h"
     
    177178
    178179// Evaluate some JavaScript code in the scope of this frame.
    179 JSValue DebuggerCallFrame::evaluate(const String& script, JSValue& exception)
     180JSValue DebuggerCallFrame::evaluate(const String& script, Exception*& exception)
    180181{
    181182    ASSERT(isValid());
  • TabularUnified trunk/Source/JavaScriptCore/debugger/DebuggerCallFrame.h

    r173892 r185259  
    3939
    4040class DebuggerScope;
     41class Exception;
    4142class ExecState;
    4243typedef ExecState CallFrame;
     
    6768    JS_EXPORT_PRIVATE Type type() const;
    6869    JS_EXPORT_PRIVATE JSValue thisValue() const;
    69     JSValue evaluate(const String&, JSValue& exception);
     70    JSValue evaluate(const String&, Exception*&);
    7071
    7172    bool isValid() const { return !!m_callFrame; }
  • TabularUnified trunk/Source/JavaScriptCore/inspector/InjectedScriptManager.cpp

    r181358 r185259  
    140140    JSValue globalThisValue = scriptState->globalThisValue();
    141141
    142     JSValue evaluationException;
     142    Exception* evaluationException;
    143143    InspectorEvaluateHandler evaluateHandler = m_environment.evaluateHandler();
    144     JSValue functionValue = evaluateHandler(scriptState, sourceCode, globalThisValue, &evaluationException);
     144    JSValue functionValue = evaluateHandler(scriptState, sourceCode, globalThisValue, evaluationException);
    145145    if (evaluationException)
    146146        return Deprecated::ScriptObject();
  • TabularUnified trunk/Source/JavaScriptCore/inspector/InspectorEnvironment.h

    r178060 r185259  
    3434
    3535namespace JSC {
     36class Exception;
    3637class SourceCode;
    3738}
     
    3940namespace Inspector {
    4041
    41 typedef JSC::JSValue (*InspectorFunctionCallHandler)(JSC::ExecState* exec, JSC::JSValue functionObject, JSC::CallType callType, const JSC::CallData& callData, JSC::JSValue thisValue, const JSC::ArgList& args, JSC::JSValue* exception);
    42 typedef JSC::JSValue (*InspectorEvaluateHandler)(JSC::ExecState*, const JSC::SourceCode&, JSC::JSValue thisValue, JSC::JSValue* exception);
     42typedef JSC::JSValue (*InspectorFunctionCallHandler)(JSC::ExecState* exec, JSC::JSValue functionObject, JSC::CallType callType, const JSC::CallData& callData, JSC::JSValue thisValue, const JSC::ArgList& args, JSC::Exception*& returnedException);
     43typedef JSC::JSValue (*InspectorEvaluateHandler)(JSC::ExecState*, const JSC::SourceCode&, JSC::JSValue thisValue, JSC::Exception*& returnedException);
    4344
    4445class InspectorEnvironment {
  • TabularUnified trunk/Source/JavaScriptCore/inspector/JSGlobalObjectInspectorController.cpp

    r183319 r185259  
    3030#include "ConsoleMessage.h"
    3131#include "ErrorHandlingScope.h"
     32#include "Exception.h"
    3233#include "InjectedScriptHost.h"
    3334#include "InjectedScriptManager.h"
     
    189190}
    190191
    191 void JSGlobalObjectInspectorController::reportAPIException(ExecState* exec, JSValue exception)
     192void JSGlobalObjectInspectorController::reportAPIException(ExecState* exec, Exception* exception)
    192193{
    193194    if (isTerminatedExecutionException(exception))
     
    202203    // FIXME: <http://webkit.org/b/115087> Web Inspector: Should not evaluate JavaScript handling exceptions
    203204    // If this is a custom exception object, call toString on it to try and get a nice string representation for the exception.
    204     String errorMessage = exception.toString(exec)->value(exec);
     205    String errorMessage = exception->value().toString(exec)->value(exec);
    205206    exec->clearException();
    206207
  • TabularUnified trunk/Source/JavaScriptCore/inspector/JSGlobalObjectInspectorController.h

    r183319 r185259  
    4444namespace JSC {
    4545class ConsoleClient;
     46class Exception;
    4647class ExecState;
    4748class JSGlobalObject;
     
    8283
    8384    void pause();
    84     void reportAPIException(JSC::ExecState*, JSC::JSValue exception);
     85    void reportAPIException(JSC::ExecState*, JSC::Exception*);
    8586
    8687    JSC::ConsoleClient* consoleClient() const;
  • TabularUnified trunk/Source/JavaScriptCore/inspector/JSGlobalObjectScriptDebugServer.h

    r178820 r185259  
    5555    // Until a time comes where an exception can be caused outside of the API (e.g. setTimeout
    5656    // or some other async operation in a pure JSContext) we can ignore exceptions reported here.
    57     virtual void reportException(JSC::ExecState*, JSC::JSValue) const override { }
     57    virtual void reportException(JSC::ExecState*, JSC::Exception*) const override { }
    5858
    5959    ListenerSet m_listeners;
  • TabularUnified trunk/Source/JavaScriptCore/inspector/JSJavaScriptCallFrame.cpp

    r184316 r185259  
    2929#include "DebuggerScope.h"
    3030#include "Error.h"
     31#include "Exception.h"
    3132#include "JSCJSValue.h"
    3233#include "JSCellInlines.h"
     
    7677JSValue JSJavaScriptCallFrame::evaluate(ExecState* exec)
    7778{
    78     JSValue exception;
     79    Exception* exception;
    7980    JSValue result = impl().evaluate(exec->argument(0).toString(exec)->value(exec), exception);
    8081    if (exception)
  • TabularUnified trunk/Source/JavaScriptCore/inspector/JavaScriptCallFrame.h

    r178820 r185259  
    5656
    5757    JSC::JSValue thisValue() const { return m_debuggerCallFrame->thisValue(); }
    58     JSC::JSValue evaluate(const String& script, JSC::JSValue& exception) const  { return m_debuggerCallFrame->evaluate(script, exception); }
     58    JSC::JSValue evaluate(const String& script, JSC::Exception*& exception) const  { return m_debuggerCallFrame->evaluate(script, exception); }
    5959
    6060private:
  • TabularUnified trunk/Source/JavaScriptCore/inspector/ScriptCallStackFactory.cpp

    r182205 r185259  
    3535
    3636#include "CallFrame.h"
     37#include "Exception.h"
    3738#include "JSCJSValue.h"
    3839#include "JSCInlines.h"
     
    129130}
    130131
    131 PassRefPtr<ScriptCallStack> createScriptCallStackFromException(JSC::ExecState* exec, JSC::JSValue& exception, size_t maxStackSize)
     132PassRefPtr<ScriptCallStack> createScriptCallStackFromException(JSC::ExecState* exec, JSC::Exception* exception, size_t maxStackSize)
    132133{
    133134    Vector<ScriptCallFrame> frames;
    134     RefCountedArray<StackFrame> stackTrace = exec->vm().exceptionStack();
     135    RefCountedArray<StackFrame> stackTrace = exception->stack();
    135136    for (size_t i = 0; i < stackTrace.size() && i < maxStackSize; i++) {
    136137        unsigned line;
     
    142143
    143144    // Fallback to getting at least the line and sourceURL from the exception object if it has values and the exceptionStack doesn't.
    144     JSObject* exceptionObject = exception.toObject(exec);
    145     if (exception.isObject()) {
     145    if (exception->value().isObject()) {
     146        JSObject* exceptionObject = exception->value().toObject(exec);
    146147        int lineNumber;
    147148        int columnNumber;
  • TabularUnified trunk/Source/JavaScriptCore/inspector/ScriptCallStackFactory.h

    r164824 r185259  
    3636
    3737namespace JSC {
     38class Exception;
    3839class ExecState;
    3940class JSValue;
     
    4849JS_EXPORT_PRIVATE PassRefPtr<ScriptCallStack> createScriptCallStack(JSC::ExecState*, size_t maxStackSize);
    4950JS_EXPORT_PRIVATE PassRefPtr<ScriptCallStack> createScriptCallStackForConsole(JSC::ExecState*, size_t maxStackSize);
    50 JS_EXPORT_PRIVATE PassRefPtr<ScriptCallStack> createScriptCallStackFromException(JSC::ExecState*, JSC::JSValue& exception, size_t maxStackSize);
     51JS_EXPORT_PRIVATE PassRefPtr<ScriptCallStack> createScriptCallStackFromException(JSC::ExecState*, JSC::Exception*, size_t maxStackSize);
    5152JS_EXPORT_PRIVATE PassRefPtr<ScriptArguments> createScriptArguments(JSC::ExecState*, unsigned skipArgumentCount);
    5253
  • TabularUnified trunk/Source/JavaScriptCore/inspector/ScriptDebugServer.cpp

    r178820 r185259  
    3434#include "DebuggerCallFrame.h"
    3535#include "DebuggerScope.h"
     36#include "Exception.h"
    3637#include "JSJavaScriptCallFrame.h"
    3738#include "JSLock.h"
     
    9495    }
    9596    case ScriptBreakpointActionTypeEvaluate: {
    96         JSValue exception;
     97        Exception* exception;
    9798        debuggerCallFrame->evaluate(breakpointAction.data, exception);
    9899        if (exception)
     
    104105        break;
    105106    case ScriptBreakpointActionTypeProbe: {
    106         JSValue exception;
     107        Exception* exception;
    107108        JSValue result = debuggerCallFrame->evaluate(breakpointAction.data, exception);
    108109        if (exception)
     
    110111       
    111112        JSC::ExecState* state = debuggerCallFrame->scope()->globalObject()->globalExec();
    112         Deprecated::ScriptValue wrappedResult = Deprecated::ScriptValue(state->vm(), exception ? exception : result);
     113        Deprecated::ScriptValue wrappedResult = Deprecated::ScriptValue(state->vm(), exception ? exception->value() : result);
    113114        dispatchBreakpointActionProbe(state, breakpointAction, wrappedResult);
    114115        break;
     
    303304}
    304305
    305 void ScriptDebugServer::handleExceptionInBreakpointCondition(JSC::ExecState* exec, JSC::JSValue exception) const
     306void ScriptDebugServer::handleExceptionInBreakpointCondition(JSC::ExecState* exec, JSC::Exception* exception) const
    306307{
    307308    reportException(exec, exception);
  • TabularUnified trunk/Source/JavaScriptCore/inspector/ScriptDebugServer.h

    r180130 r185259  
    8080    virtual void runEventLoopWhilePaused() = 0;
    8181    virtual bool isContentScript(JSC::ExecState*) const = 0;
    82     virtual void reportException(JSC::ExecState*, JSC::JSValue) const = 0;
     82    virtual void reportException(JSC::ExecState*, JSC::Exception*) const = 0;
    8383
    8484    bool evaluateBreakpointAction(const ScriptBreakpointAction&);
     
    102102    virtual bool needPauseHandling(JSC::JSGlobalObject*) override final { return true; }
    103103    virtual void handleBreakpointHit(JSC::JSGlobalObject*, const JSC::Breakpoint&) override final;
    104     virtual void handleExceptionInBreakpointCondition(JSC::ExecState*, JSC::JSValue exception) const override final;
     104    virtual void handleExceptionInBreakpointCondition(JSC::ExecState*, JSC::Exception*) const override final;
    105105    virtual void handlePause(JSC::JSGlobalObject*, JSC::Debugger::ReasonForPause) override final;
    106106    virtual void notifyDoneProcessingDebuggerEvents() override final;
  • TabularUnified trunk/Source/JavaScriptCore/interpreter/CallFrame.h

    r183935 r185259  
    2525
    2626#include "AbstractPC.h"
    27 #include "VM.h"
    2827#include "JSStack.h"
    2928#include "MacroAssemblerCodeRef.h"
    3029#include "Register.h"
    3130#include "StackVisitor.h"
     31#include "VM.h"
    3232#include "VMEntryRecord.h"
    3333
     
    7676
    7777        void clearException() { vm().clearException(); }
    78         void clearSupplementaryExceptionInfo()
    79         {
    80             vm().clearExceptionStack();
    81         }
    82 
    83         JSValue exception() const { return vm().exception(); }
    84         bool hadException() const { return !vm().exception().isEmpty(); }
     78
     79        Exception* exception() const { return vm().exception(); }
     80        bool hadException() const { return !!vm().exception(); }
    8581
    8682        AtomicStringTable* atomicStringTable() const { return vm().atomicStringTable(); }
  • TabularUnified trunk/Source/JavaScriptCore/interpreter/Interpreter.cpp

    r184640 r185259  
    4242#include "ErrorInstance.h"
    4343#include "EvalCodeCache.h"
     44#include "Exception.h"
    4445#include "ExceptionHelpers.h"
    4546#include "GetterSetter.h"
     
    436437    CallFrame* callFrame = visitor->callFrame();
    437438    if (Debugger* debugger = callFrame->vmEntryGlobalObject()->debugger()) {
    438         ClearExceptionScope scope(&callFrame->vm());
     439        SuspendExceptionScope scope(&callFrame->vm());
    439440        if (jsDynamicCast<JSFunction*>(callFrame->callee()))
    440441            debugger->returnEvent(callFrame);
     
    584585}
    585586
    586 class GetExceptionHandlerFunctor {
     587class GetCatchHandlerFunctor {
    587588public:
    588     GetExceptionHandlerFunctor()
     589    GetCatchHandlerFunctor()
    589590        : m_handler(0)
    590591    {
     
    600601
    601602        unsigned bytecodeOffset = visitor->bytecodeOffset();
    602         m_handler = codeBlock->handlerForBytecodeOffset(bytecodeOffset);
     603        m_handler = codeBlock->handlerForBytecodeOffset(bytecodeOffset, CodeBlock::RequiredHandler::CatchHandler);
    603604        if (m_handler)
    604605            return StackVisitor::Done;
     
    650651};
    651652
    652 NEVER_INLINE HandlerInfo* Interpreter::unwind(VMEntryFrame*& vmEntryFrame, CallFrame*& callFrame, JSValue& exceptionValue)
     653NEVER_INLINE HandlerInfo* Interpreter::unwind(VMEntryFrame*& vmEntryFrame, CallFrame*& callFrame, Exception* exception)
    653654{
    654655    CodeBlock* codeBlock = callFrame->codeBlock();
    655656    bool isTermination = false;
    656657
     658    JSValue exceptionValue = exception->value();
    657659    ASSERT(!exceptionValue.isEmpty());
    658660    ASSERT(!exceptionValue.isCell() || exceptionValue.asCell());
     
    663665
    664666    if (exceptionValue.isObject())
    665         isTermination = isTerminatedExecutionException(asObject(exceptionValue));
    666 
    667     ASSERT(callFrame->vm().exceptionStack().size());
     667        isTermination = isTerminatedExecutionException(exception);
     668
     669    ASSERT(callFrame->vm().exception() && callFrame->vm().exception()->stack().size());
    668670
    669671    Debugger* debugger = callFrame->vmEntryGlobalObject()->debugger();
    670     if (debugger && debugger->needsExceptionCallbacks()) {
    671         // We need to clear the exception and the exception stack here in order to see if a new exception happens.
     672    if (debugger && debugger->needsExceptionCallbacks() && !exception->didNotifyInspectorOfThrow()) {
     673        // We need to clear the exception here in order to see if a new exception happens.
    672674        // Afterwards, the values are put back to continue processing this error.
    673         ClearExceptionScope scope(&callFrame->vm());
     675        SuspendExceptionScope scope(&callFrame->vm());
    674676        // This code assumes that if the debugger is enabled then there is no inlining.
    675677        // If that assumption turns out to be false then we'll ignore the inlined call
     
    677679        // https://bugs.webkit.org/show_bug.cgi?id=121754
    678680
    679         bool hasHandler;
     681        bool hasCatchHandler;
    680682        if (isTermination)
    681             hasHandler = false;
     683            hasCatchHandler = false;
    682684        else {
    683             GetExceptionHandlerFunctor functor;
     685            GetCatchHandlerFunctor functor;
    684686            callFrame->iterate(functor);
    685             hasHandler = !!functor.handler();
    686         }
    687 
    688         debugger->exception(callFrame, exceptionValue, hasHandler);
     687            HandlerInfo* handler = functor.handler();
     688            ASSERT(!handler || handler->isCatchHandler());
     689            hasCatchHandler = !!handler;
     690        }
     691
     692        debugger->exception(callFrame, exceptionValue, hasCatchHandler);
    689693        ASSERT(!callFrame->hadException());
    690694    }
     695    exception->setDidNotifyInspectorOfThrow();
    691696
    692697    // Calculate an exception handler vPC, unwinding call frames as necessary.
  • TabularUnified trunk/Source/JavaScriptCore/interpreter/Interpreter.h

    r182759 r185259  
    9999    };
    100100
    101     class ClearExceptionScope {
    102     public:
    103         ClearExceptionScope(VM* vm): m_vm(vm)
    104         {
    105             vm->getExceptionInfo(oldException, oldExceptionStack);
     101    class SuspendExceptionScope {
     102    public:
     103        SuspendExceptionScope(VM* vm)
     104            : m_vm(vm)
     105        {
     106            oldException = vm->exception();
    106107            vm->clearException();
    107108        }
    108         ~ClearExceptionScope()
    109         {
    110             m_vm->setExceptionInfo(oldException, oldExceptionStack);
    111         }
    112     private:
    113         JSC::JSValue oldException;
    114         RefCountedArray<JSC::StackFrame> oldExceptionStack;
     109        ~SuspendExceptionScope()
     110        {
     111            m_vm->setException(oldException);
     112        }
     113    private:
     114        Exception* oldException;
    115115        VM* m_vm;
    116116    };
     
    216216        SamplingTool* sampler() { return m_sampler.get(); }
    217217
    218         NEVER_INLINE HandlerInfo* unwind(VMEntryFrame*&, CallFrame*&, JSValue&);
     218        NEVER_INLINE HandlerInfo* unwind(VMEntryFrame*&, CallFrame*&, Exception*);
    219219        NEVER_INLINE void debug(CallFrame*, DebugHookID);
    220220        JSString* stackTraceAsString(ExecState*, Vector<StackFrame>);
  • TabularUnified trunk/Source/JavaScriptCore/jit/AssemblyHelpers.cpp

    r180279 r185259  
    224224    result = branchTest64(kind == NormalExceptionCheck ? NonZero : Zero, AbsoluteAddress(vm()->addressOfException()));
    225225#elif USE(JSVALUE32_64)
    226     result = branch32(kind == NormalExceptionCheck ? NotEqual : Equal, AbsoluteAddress(reinterpret_cast<char*>(vm()->addressOfException()) + OBJECT_OFFSETOF(JSValue, u.asBits.tag)), TrustedImm32(JSValue::EmptyValueTag));
     226    result = branch32(kind == NormalExceptionCheck ? NotEqual : Equal, AbsoluteAddress(vm()->addressOfException()), TrustedImm32(0));
    227227#endif
    228228   
  • TabularUnified trunk/Source/JavaScriptCore/jit/JITExceptions.cpp

    r172867 r185259  
    4141namespace JSC {
    4242
    43 void genericUnwind(VM* vm, ExecState* callFrame, JSValue exceptionValue)
     43void genericUnwind(VM* vm, ExecState* callFrame)
    4444{
    4545    if (Options::breakOnThrow()) {
     
    4848    }
    4949   
    50     RELEASE_ASSERT(exceptionValue);
     50    Exception* exception = vm->exception();
     51    RELEASE_ASSERT(exception);
    5152    VMEntryFrame* vmEntryFrame = vm->topVMEntryFrame;
    52     HandlerInfo* handler = vm->interpreter->unwind(vmEntryFrame, callFrame, exceptionValue); // This may update vmEntryFrame and callFrame.
     53    HandlerInfo* handler = vm->interpreter->unwind(vmEntryFrame, callFrame, exception); // This may update vmEntryFrame and callFrame.
    5354
    5455    void* catchRoutine;
  • TabularUnified trunk/Source/JavaScriptCore/jit/JITExceptions.h

    r170147 r185259  
    3434class VM;
    3535
    36 void genericUnwind(VM*, ExecState*, JSValue exceptionValue);
     36void genericUnwind(VM*, ExecState*);
    3737
    3838} // namespace JSC
  • TabularUnified trunk/Source/JavaScriptCore/jit/JITOpcodes.cpp

    r185240 r185259  
    3232#include "CopiedSpaceInlines.h"
    3333#include "Debugger.h"
     34#include "Exception.h"
    3435#include "Heap.h"
    3536#include "JITInlines.h"
     
    530531    store64(TrustedImm64(JSValue::encode(JSValue())), Address(regT3, VM::exceptionOffset()));
    531532    emitPutVirtualRegister(currentInstruction[1].u.operand);
     533
     534    load64(Address(regT0, Exception::valueOffset()), regT0);
     535    emitPutVirtualRegister(currentInstruction[2].u.operand);
    532536}
    533537
  • TabularUnified trunk/Source/JavaScriptCore/jit/JITOpcodes32_64.cpp

    r185240 r185259  
    3333#include "CCallHelpers.h"
    3434#include "Debugger.h"
     35#include "Exception.h"
    3536#include "JITInlines.h"
    3637#include "JSArray.h"
     
    9899
    99100    // Check for an exception
    100     Jump sawException = branch32(NotEqual, AbsoluteAddress(reinterpret_cast<char*>(vm->addressOfException()) + OBJECT_OFFSETOF(JSValue, u.asBits.tag)), TrustedImm32(JSValue::EmptyValueTag));
     101    Jump sawException = branch32(NotEqual, AbsoluteAddress(vm->addressOfException()), TrustedImm32(0));
    101102
    102103    emitFunctionEpilogue();
     
    830831
    831832    // Now store the exception returned by operationThrow.
    832     load32(Address(regT3, VM::exceptionOffset() + OBJECT_OFFSETOF(JSValue, u.asBits.payload)), regT0);
    833     load32(Address(regT3, VM::exceptionOffset() + OBJECT_OFFSETOF(JSValue, u.asBits.tag)), regT1);
    834     store32(TrustedImm32(JSValue().payload()), Address(regT3, VM::exceptionOffset() + OBJECT_OFFSETOF(JSValue, u.asBits.payload)));
    835     store32(TrustedImm32(JSValue().tag()), Address(regT3, VM::exceptionOffset() + OBJECT_OFFSETOF(JSValue, u.asBits.tag)));
     833    load32(Address(regT3, VM::exceptionOffset()), regT2);
     834    move(TrustedImm32(JSValue::CellTag), regT1);
     835
     836    store32(TrustedImm32(0), Address(regT3, VM::exceptionOffset()));
    836837
    837838    unsigned exception = currentInstruction[1].u.operand;
    838     emitStore(exception, regT1, regT0);
     839    emitStore(exception, regT1, regT2);
     840
     841    load32(Address(regT2, Exception::valueOffset() + OBJECT_OFFSETOF(JSValue, u.asBits.payload)), regT0);
     842    load32(Address(regT2, Exception::valueOffset() + OBJECT_OFFSETOF(JSValue, u.asBits.tag)), regT1);
     843
     844    unsigned thrownValue = currentInstruction[2].u.operand;
     845    emitStore(thrownValue, regT1, regT0);
    839846}
    840847
  • TabularUnified trunk/Source/JavaScriptCore/jit/JITOperations.cpp

    r185240 r185259  
    19191919
    19201920    // Results stored out-of-band in vm.targetMachinePCForThrow, vm.callFrameForThrow & vm.vmEntryFrameForThrow
    1921     genericUnwind(vm, exec, exceptionValue);
     1921    genericUnwind(vm, exec);
    19221922}
    19231923
     
    19581958{
    19591959    NativeCallFrameTracer tracer(vm, exec);
    1960 
    1961     JSValue exceptionValue = vm->exception();
    1962     ASSERT(exceptionValue);
    1963    
    1964     genericUnwind(vm, exec, exceptionValue);
     1960    genericUnwind(vm, exec);
    19651961    ASSERT(vm->targetMachinePCForThrow);
    19661962}
     
    19731969
    19741970    NativeCallFrameTracerWithRestore tracer(vm, vmEntryFrame, callerFrame);
    1975 
    1976     JSValue exceptionValue = vm->exception();
    1977     ASSERT(exceptionValue);
    1978    
    1979     genericUnwind(vm, callerFrame, exceptionValue);
     1971    genericUnwind(vm, callerFrame);
    19801972    ASSERT(vm->targetMachinePCForThrow);
    19811973}
     
    19851977    VM* vm = &exec->vm();
    19861978    NativeCallFrameTracer tracer(vm, exec);
    1987 
    1988     genericUnwind(vm, exec, vm->exception());
     1979    genericUnwind(vm, exec);
    19891980}
    19901981
  • TabularUnified trunk/Source/JavaScriptCore/jit/ThunkGenerators.cpp

    r183963 r185259  
    350350    JSInterfaceJIT::Jump exceptionHandler = jit.branch32(
    351351        JSInterfaceJIT::NotEqual,
    352         JSInterfaceJIT::AbsoluteAddress(reinterpret_cast<char*>(vm->addressOfException()) + OBJECT_OFFSETOF(EncodedValueDescriptor, asBits.tag)),
    353         JSInterfaceJIT::TrustedImm32(JSValue::EmptyValueTag));
     352        JSInterfaceJIT::AbsoluteAddress(vm->addressOfException()),
     353        JSInterfaceJIT::TrustedImm32(0));
    354354#endif
    355355
  • TabularUnified trunk/Source/JavaScriptCore/jsc.cpp

    r184617 r185259  
    3030#include "CopiedSpaceInlines.h"
    3131#include "Disassembler.h"
     32#include "Exception.h"
    3233#include "ExceptionHelpers.h"
    3334#include "HeapStatistics.h"
     
    900901        exec->vm(), Identifier::fromString(globalObject->globalExec(), "arguments"), array);
    901902
    902     JSValue exception;
     903    Exception* exception;
    903904    StopWatch stopWatch;
    904905    stopWatch.start();
    905     evaluate(globalObject->globalExec(), jscSource(script.data(), fileName), JSValue(), &exception);
     906    evaluate(globalObject->globalExec(), jscSource(script.data(), fileName), JSValue(), exception);
    906907    stopWatch.stop();
    907908
    908     if (!!exception) {
     909    if (exception) {
    909910        exec->vm().throwException(globalObject->globalExec(), exception);
    910911        return JSValue::encode(jsUndefined());
     
    923924    JSGlobalObject* globalObject = exec->lexicalGlobalObject();
    924925   
    925     JSValue evaluationException;
    926     JSValue result = evaluate(globalObject->globalExec(), jscSource(script.data(), fileName), JSValue(), &evaluationException);
     926    Exception* evaluationException;
     927    JSValue result = evaluate(globalObject->globalExec(), jscSource(script.data(), fileName), JSValue(), evaluationException);
    927928    if (evaluationException)
    928929        exec->vm().throwException(exec, evaluationException);
     
    12861287        vm.startSampling();
    12871288
    1288         JSValue evaluationException;
    1289         JSValue returnValue = evaluate(globalObject->globalExec(), jscSource(script, fileName), JSValue(), &evaluationException);
     1289        Exception* evaluationException;
     1290        JSValue returnValue = evaluate(globalObject->globalExec(), jscSource(script, fileName), JSValue(), evaluationException);
    12901291        success = success && !evaluationException;
    12911292        if (dump && !evaluationException)
    12921293            printf("End: %s\n", returnValue.toString(globalObject->globalExec())->value(globalObject->globalExec()).utf8().data());
    12931294        if (evaluationException) {
    1294             printf("Exception: %s\n", evaluationException.toString(globalObject->globalExec())->value(globalObject->globalExec()).utf8().data());
     1295            printf("Exception: %s\n", evaluationException->value().toString(globalObject->globalExec())->value(globalObject->globalExec()).utf8().data());
    12951296            Identifier stackID = Identifier::fromString(globalObject->globalExec(), "stack");
    1296             JSValue stackValue = evaluationException.get(globalObject->globalExec(), stackID);
     1297            JSValue stackValue = evaluationException->value().get(globalObject->globalExec(), stackID);
    12971298            if (!stackValue.isUndefinedOrNull())
    12981299                printf("%s\n", stackValue.toString(globalObject->globalExec())->value(globalObject->globalExec()).utf8().data());
     
    13501351       
    13511352       
    1352         JSValue evaluationException;
    1353         JSValue returnValue = evaluate(globalObject->globalExec(), makeSource(source, interpreterName), JSValue(), &evaluationException);
     1353        Exception* evaluationException;
     1354        JSValue returnValue = evaluate(globalObject->globalExec(), makeSource(source, interpreterName), JSValue(), evaluationException);
    13541355#else
    13551356        printf("%s", interactivePrompt);
     
    13661367        line.append('\0');
    13671368
    1368         JSValue evaluationException;
    1369         JSValue returnValue = evaluate(globalObject->globalExec(), jscSource(line.data(), interpreterName), JSValue(), &evaluationException);
     1369        Exception* evaluationException;
     1370        JSValue returnValue = evaluate(globalObject->globalExec(), jscSource(line.data(), interpreterName), JSValue(), evaluationException);
    13701371#endif
    13711372        if (evaluationException)
    1372             printf("Exception: %s\n", evaluationException.toString(globalObject->globalExec())->value(globalObject->globalExec()).utf8().data());
     1373            printf("Exception: %s\n", evaluationException->value().toString(globalObject->globalExec())->value(globalObject->globalExec()).utf8().data());
    13731374        else
    13741375            printf("%s\n", returnValue.toString(globalObject->globalExec())->value(globalObject->globalExec()).utf8().data());
  • TabularUnified trunk/Source/JavaScriptCore/llint/LLIntOffsetsExtractor.cpp

    r181993 r185259  
    3131#include "Debugger.h"
    3232#include "DirectArguments.h"
     33#include "Exception.h"
    3334#include "Executable.h"
    3435#include "Heap.h"
  • TabularUnified trunk/Source/JavaScriptCore/llint/LLIntSlowPaths.cpp

    r185109 r185259  
    13761376{
    13771377    LLINT_BEGIN_NO_SET_PC();
    1378     ASSERT(vm.exception());
    1379     genericUnwind(&vm, exec, vm.exception());
     1378    genericUnwind(&vm, exec);
    13801379    LLINT_END_IMPL();
    13811380}
  • TabularUnified trunk/Source/JavaScriptCore/llint/LowLevelInterpreter32_64.asm

    r184328 r185259  
    694694    andp MarkedBlockMask, t3
    695695    loadp MarkedBlock::m_weakSet + WeakSet::m_vm[t3], t3
    696     bieq VM::m_exception + TagOffset[t3], EmptyValueTag, .noException
     696    btiz VM::m_exception[t3], .noException
    697697    jmp label
    698698.noException:
     
    19531953
    19541954    loadi VM::targetInterpreterPCForThrow[t3], PC
    1955     loadi VM::m_exception + PayloadOffset[t3], t0
    1956     loadi VM::m_exception + TagOffset[t3], t1
    1957     storei 0, VM::m_exception + PayloadOffset[t3]
    1958     storei EmptyValueTag, VM::m_exception + TagOffset[t3]
     1955    loadi VM::m_exception[t3], t0
     1956    storei 0, VM::m_exception[t3]
    19591957    loadi 4[PC], t2
    19601958    storei t0, PayloadOffset[cfr, t2, 8]
     1959    storei CellTag, TagOffset[cfr, t2, 8]
     1960
     1961    loadi Exception::m_value + TagOffset[t0], t1
     1962    loadi Exception::m_value + PayloadOffset[t0], t0
     1963    loadi 8[PC], t2
     1964    storei t0, PayloadOffset[cfr, t2, 8]
    19611965    storei t1, TagOffset[cfr, t2, 8]
     1966
    19621967    traceExecution()  # This needs to be here because we don't want to clobber t0, t1, t2, t3 above.
    1963     dispatch(2)
     1968    dispatch(3)
    19641969
    19651970_llint_op_end:
     
    20392044   
    20402045    functionEpilogue()
    2041     bineq VM::m_exception + TagOffset[t3], EmptyValueTag, .handleException
     2046    btinz VM::m_exception[t3], .handleException
    20422047    ret
    20432048
  • TabularUnified trunk/Source/JavaScriptCore/llint/LowLevelInterpreter64.asm

    r184328 r185259  
    18211821    subp PB, PC
    18221822    rshiftp 3, PC
     1823
    18231824    loadq VM::m_exception[t3], t0
    18241825    storeq 0, VM::m_exception[t3]
    18251826    loadisFromInstruction(1, t2)
    18261827    storeq t0, [cfr, t2, 8]
    1827     traceExecution()
    1828     dispatch(2)
     1828
     1829    loadq Exception::m_value[t0], t3
     1830    loadisFromInstruction(2, t2)
     1831    storeq t3, [cfr, t2, 8]
     1832
     1833    traceExecution()
     1834    dispatch(3)
    18291835
    18301836
  • TabularUnified trunk/Source/JavaScriptCore/parser/NodeConstructors.h

    r184337 r185259  
    812812    }
    813813
    814     inline TryNode::TryNode(const JSTokenLocation& location, StatementNode* tryBlock, const Identifier& exceptionIdent, StatementNode* catchBlock, StatementNode* finallyBlock)
     814    inline TryNode::TryNode(const JSTokenLocation& location, StatementNode* tryBlock, const Identifier& thrownValueIdent, StatementNode* catchBlock, StatementNode* finallyBlock)
    815815        : StatementNode(location)
    816816        , m_tryBlock(tryBlock)
    817         , m_exceptionIdent(exceptionIdent)
     817        , m_thrownValueIdent(thrownValueIdent)
    818818        , m_catchBlock(catchBlock)
    819819        , m_finallyBlock(finallyBlock)
  • TabularUnified trunk/Source/JavaScriptCore/parser/Nodes.h

    r184828 r185259  
    15121512
    15131513        StatementNode* m_tryBlock;
    1514         const Identifier& m_exceptionIdent;
     1514        const Identifier& m_thrownValueIdent;
    15151515        StatementNode* m_catchBlock;
    15161516        StatementNode* m_finallyBlock;
  • TabularUnified trunk/Source/JavaScriptCore/runtime/CallData.cpp

    r167142 r185259  
    2727#include "CallData.h"
    2828
     29#include "Exception.h"
    2930#include "Executable.h"
    3031#include "Interpreter.h"
     
    4041}
    4142
    42 JSValue call(ExecState* exec, JSValue functionObject, CallType callType, const CallData& callData, JSValue thisValue, const ArgList& args, JSValue* exception)
     43JSValue call(ExecState* exec, JSValue functionObject, CallType callType, const CallData& callData, JSValue thisValue, const ArgList& args, Exception*& returnedException)
    4344{
    4445    JSValue result = call(exec, functionObject, callType, callData, thisValue, args);
    4546    if (exec->hadException()) {
    46         if (exception)
    47             *exception = exec->exception();
     47        returnedException = exec->exception();
    4848        exec->clearException();
    4949        return jsUndefined();
    50     }
     50    } else
     51        returnedException = nullptr;
    5152    RELEASE_ASSERT(result);
    5253    return result;
  • TabularUnified trunk/Source/JavaScriptCore/runtime/CallData.h

    r167142 r185259  
    3535
    3636class ArgList;
     37class Exception;
    3738class ExecState;
    3839class FunctionExecutable;
     
    5960
    6061JS_EXPORT_PRIVATE JSValue call(ExecState*, JSValue functionObject, CallType, const CallData&, JSValue thisValue, const ArgList&);
    61 JS_EXPORT_PRIVATE JSValue call(ExecState*, JSValue functionObject, CallType, const CallData&, JSValue thisValue, const ArgList&, JSValue* exception);
     62JS_EXPORT_PRIVATE JSValue call(ExecState*, JSValue functionObject, CallType, const CallData&, JSValue thisValue, const ArgList&, Exception*& returnedException);
    6263
    6364} // namespace JSC
  • TabularUnified trunk/Source/JavaScriptCore/runtime/Completion.cpp

    r181664 r185259  
    2727#include "CodeProfiling.h"
    2828#include "Debugger.h"
     29#include "Exception.h"
    2930#include "Interpreter.h"
    3031#include "JSGlobalObject.h"
     
    6162}
    6263
    63 JSValue evaluate(ExecState* exec, const SourceCode& source, JSValue thisValue, JSValue* returnedException)
     64JSValue evaluate(ExecState* exec, const SourceCode& source, JSValue thisValue, Exception*& returnedException)
    6465{
    6566    JSLockHolder lock(exec);
    6667    RELEASE_ASSERT(exec->vm().atomicStringTable() == wtfThreadData().atomicStringTable());
    6768    RELEASE_ASSERT(!exec->vm().isCollectorBusy());
     69    returnedException = nullptr;
    6870
    6971    CodeProfiling profile(source);
     
    7173    ProgramExecutable* program = ProgramExecutable::create(exec, source);
    7274    if (!program) {
    73         if (returnedException)
    74             *returnedException = exec->vm().exception();
    75 
     75        returnedException = exec->vm().exception();
    7676        exec->vm().clearException();
    7777        return jsUndefined();
     
    8484
    8585    if (exec->hadException()) {
    86         if (returnedException)
    87             *returnedException = exec->exception();
    88 
     86        returnedException = exec->exception();
    8987        exec->clearException();
    9088        return jsUndefined();
  • TabularUnified trunk/Source/JavaScriptCore/runtime/Completion.h

    r180637 r185259  
    2727
    2828namespace JSC {
    29    
     29
     30class Exception;
    3031class ExecState;
    3132class JSScope;
     
    3637JS_EXPORT_PRIVATE bool checkSyntax(VM&, const SourceCode&, ParserError&);
    3738JS_EXPORT_PRIVATE bool checkSyntax(ExecState*, const SourceCode&, JSValue* exception = 0);
    38 JS_EXPORT_PRIVATE JSValue evaluate(ExecState*, const SourceCode&, JSValue thisValue = JSValue(), JSValue* exception = 0);
     39JS_EXPORT_PRIVATE JSValue evaluate(ExecState*, const SourceCode&, JSValue thisValue, Exception*& returnedException);
     40inline JSValue evaluate(ExecState* exec, const SourceCode& sourceCode, JSValue thisValue = JSValue())
     41{
     42    Exception* unused;
     43    return evaluate(exec, sourceCode, thisValue, unused);
     44}
    3945
    4046} // namespace JSC
  • TabularUnified trunk/Source/JavaScriptCore/runtime/Error.h

    r184651 r185259  
    7777
    7878// Convenience wrappers, wrap result as an EncodedJSValue.
     79inline void throwVMError(ExecState* exec, Exception* exception) { exec->vm().throwException(exec, exception); }
    7980inline EncodedJSValue throwVMError(ExecState* exec, JSValue error) { return JSValue::encode(exec->vm().throwException(exec, error)); }
    8081inline EncodedJSValue throwVMTypeError(ExecState* exec) { return JSValue::encode(throwTypeError(exec)); }
  • TabularUnified trunk/Source/JavaScriptCore/runtime/ExceptionHelpers.cpp

    r182577 r185259  
    3333#include "CallFrame.h"
    3434#include "ErrorHandlingScope.h"
     35#include "Exception.h"
    3536#include "JSGlobalObjectFunctions.h"
    3637#include "JSNotAnObject.h"
     
    6061}
    6162
    62 bool isTerminatedExecutionException(JSObject* object)
    63 {
    64     return object->inherits(TerminatedExecutionError::info());
    65 }
    66 
    67 bool isTerminatedExecutionException(JSValue value)
    68 {
    69     return value.inherits(TerminatedExecutionError::info());
    70 }
    71 
     63bool isTerminatedExecutionException(Exception* exception)
     64{
     65    return exception->value().inherits(TerminatedExecutionError::info());
     66}
    7267
    7368JSObject* createStackOverflowError(ExecState* exec)
  • TabularUnified trunk/Source/JavaScriptCore/runtime/ExceptionHelpers.h

    r182747 r185259  
    3838
    3939JSObject* createTerminatedExecutionException(VM*);
    40 bool isTerminatedExecutionException(JSObject*);
    41 JS_EXPORT_PRIVATE bool isTerminatedExecutionException(JSValue);
     40JS_EXPORT_PRIVATE bool isTerminatedExecutionException(Exception*);
    4241JS_EXPORT_PRIVATE JSObject* createError(ExecState*, JSValue, const String&, ErrorInstance::SourceAppender);
    4342JS_EXPORT_PRIVATE JSObject* createStackOverflowError(ExecState*);
  • TabularUnified trunk/Source/JavaScriptCore/runtime/GetterSetter.cpp

    r177030 r185259  
    2525
    2626#include "Error.h"
     27#include "Exception.h"
    2728#include "JSObject.h"
    2829#include "JSCInlines.h"
     
    7677    // We work around that by checking here.
    7778    if (exec->hadException())
    78         return exec->exception();
     79        return exec->exception()->value();
    7980
    8081    JSObject* getter = jsCast<GetterSetter*>(getterSetter)->getter();
  • TabularUnified trunk/Source/JavaScriptCore/runtime/IteratorOperations.cpp

    r184586 r185259  
    9090void iteratorClose(ExecState* exec, JSValue iterator)
    9191{
    92     JSValue exception;
     92    Exception* exception = nullptr;
    9393    if (exec->hadException()) {
    9494        exception = exec->exception();
     
    100100
    101101    if (returnFunction.isUndefined()) {
    102         if (!exception.isEmpty())
     102        if (exception)
    103103            exec->vm().throwException(exec, exception);
    104104        return;
     
    108108    CallType returnFunctionCallType = getCallData(returnFunction, returnFunctionCallData);
    109109    if (returnFunctionCallType == CallTypeNone) {
    110         if (!exception.isEmpty())
     110        if (exception)
    111111            exec->vm().throwException(exec, exception);
    112112        else
     
    118118    JSValue innerResult = call(exec, returnFunction, returnFunctionCallType, returnFunctionCallData, iterator, returnFunctionArguments);
    119119
    120     if (!exception.isEmpty()) {
     120    if (exception) {
    121121        exec->vm().throwException(exec, exception);
    122122        return;
  • TabularUnified trunk/Source/JavaScriptCore/runtime/JSObject.cpp

    r184407 r185259  
    3232#include "DatePrototype.h"
    3333#include "ErrorConstructor.h"
     34#include "Exception.h"
    3435#include "Executable.h"
    3536#include "GetterSetter.h"
  • TabularUnified trunk/Source/JavaScriptCore/runtime/JSPromiseConstructor.cpp

    r182205 r185259  
    3030
    3131#include "Error.h"
     32#include "Exception.h"
    3233#include "IteratorOperations.h"
    3334#include "JSCJSValueInlines.h"
     
    137138    // 14. If result is an abrupt completion, call PromiseReject(promise, result.[[value]]).
    138139    if (exec->hadException()) {
    139         JSValue exception = exec->exception();
     140        JSValue exception = exec->exception()->value();
    140141        exec->clearException();
    141142
  • TabularUnified trunk/Source/JavaScriptCore/runtime/JSPromiseDeferred.cpp

    r173410 r185259  
    3030
    3131#include "Error.h"
     32#include "Exception.h"
    3233#include "JSCJSValueInlines.h"
    3334#include "JSCellInlines.h"
     
    151152        //    deferred.[[Reject]] with undefined as thisArgument and a List containing
    152153        //    then.[[value]] as argumentsList.
    153         JSValue exception = exec->exception();
     154        JSValue exception = exec->exception()->value();
    154155        exec->clearException();
    155156
     
    186187        //    deferred.[[Reject]] with undefined as thisArgument and a List containing
    187188        //    thenCallResult.[[value]] as argumentsList.
    188         JSValue exception = exec->exception();
     189        JSValue exception = exec->exception()->value();
    189190        exec->clearException();
    190191
     
    229230{
    230231    ASSERT(exec->hadException());
    231     JSValue argument = exec->exception();
     232    JSValue argument = exec->exception()->value();
    232233    exec->clearException();
    233234
  • TabularUnified trunk/Source/JavaScriptCore/runtime/JSPromiseReaction.cpp

    r173410 r185259  
    3030
    3131#include "Error.h"
     32#include "Exception.h"
    3233#include "JSCJSValueInlines.h"
    3334#include "JSCellInlines.h"
     
    8990    //    and a List containing handlerResult.[[value]] as argumentsList.
    9091    if (exec->hadException()) {
    91         JSValue exception = exec->exception();
     92        JSValue exception = exec->exception()->value();
    9293        exec->clearException();
    9394
  • TabularUnified trunk/Source/JavaScriptCore/runtime/VM.cpp

    r184337 r185259  
    4343#include "Disassembler.h"
    4444#include "ErrorInstance.h"
     45#include "Exception.h"
    4546#include "FTLThunks.h"
    4647#include "FunctionConstructor.h"
     
    236237    inferredValueStructure.set(*this, InferredValue::createStructure(*this, 0, jsNull()));
    237238    functionRareDataStructure.set(*this, FunctionRareData::createStructure(*this, 0, jsNull()));
     239    exceptionStructure.set(*this, Exception::createStructure(*this, 0, jsNull()));
    238240#if ENABLE(PROMISES)
    239241    promiseDeferredStructure.set(*this, JSPromiseDeferred::createStructure(*this, 0, jsNull()));
     
    546548}
    547549
    548 JSValue VM::throwException(ExecState* exec, JSValue error)
     550void VM::throwException(ExecState* exec, Exception* exception)
    549551{
    550552    if (Options::breakOnThrow()) {
     
    554556   
    555557    ASSERT(exec == topCallFrame || exec == exec->lexicalGlobalObject()->globalExec() || exec == exec->vmEntryGlobalObject()->globalExec());
    556    
    557     Vector<StackFrame> stackTrace;
    558     interpreter->getStackTrace(stackTrace);
    559     m_exceptionStack = RefCountedArray<StackFrame>(stackTrace);
    560     m_exception = error;
    561 
    562     return error;
    563 }
    564    
     558    setException(exception);
     559}
     560
     561JSValue VM::throwException(ExecState* exec, JSValue thrownValue)
     562{
     563    Exception* exception = jsDynamicCast<Exception*>(thrownValue);
     564    if (!exception)
     565        exception = Exception::create(*this, thrownValue);
     566
     567    throwException(exec, exception);
     568    return JSValue(exception);
     569}
     570
    565571JSObject* VM::throwException(ExecState* exec, JSObject* error)
    566572{
    567573    return asObject(throwException(exec, JSValue(error)));
    568 }
    569 void VM::getExceptionInfo(JSValue& exception, RefCountedArray<StackFrame>& exceptionStack)
    570 {
    571     exception = m_exception;
    572     exceptionStack = m_exceptionStack;
    573 }
    574 void VM::setExceptionInfo(JSValue& exception, RefCountedArray<StackFrame>& exceptionStack)
    575 {
    576     m_exception = exception;
    577     m_exceptionStack = exceptionStack;
    578 }
    579 
    580 void VM::clearException()
    581 {
    582     m_exception = JSValue();
    583 }
    584 void VM:: clearExceptionStack()
    585 {
    586     m_exceptionStack = RefCountedArray<StackFrame>();
    587574}
    588575
  • TabularUnified trunk/Source/JavaScriptCore/runtime/VM.h

    r184447 r185259  
    5959#include <wtf/HashMap.h>
    6060#include <wtf/HashSet.h>
    61 #include <wtf/RefCountedArray.h>
    6261#include <wtf/SimpleStats.h>
    6362#include <wtf/StackBounds.h>
     
    7978class CommonIdentifiers;
    8079class ExecState;
     80class Exception;
    8181class HandleStack;
    8282class TypeProfiler;
     
    275275    Strong<Structure> inferredValueStructure;
    276276    Strong<Structure> functionRareDataStructure;
     277    Strong<Structure> exceptionStructure;
    277278#if ENABLE(PROMISES)
    278279    Strong<Structure> promiseDeferredStructure;
     
    373374    }
    374375
    375     JS_EXPORT_PRIVATE void clearException();
    376     JS_EXPORT_PRIVATE void clearExceptionStack();
    377     void getExceptionInfo(JSValue& exception, RefCountedArray<StackFrame>& exceptionStack);
    378     void setExceptionInfo(JSValue& exception, RefCountedArray<StackFrame>& exceptionStack);
    379     JSValue exception() const { return m_exception; }
    380     JSValue* addressOfException() { return &m_exception; }
    381     const RefCountedArray<StackFrame>& exceptionStack() const { return m_exceptionStack; }
    382 
     376    void clearException() { m_exception = nullptr; }
     377    void setException(Exception* exception) { m_exception = exception; }
     378
     379    Exception* exception() const { return m_exception; }
     380    JSCell** addressOfException() { return reinterpret_cast<JSCell**>(&m_exception); }
     381
     382    JS_EXPORT_PRIVATE void throwException(ExecState*, Exception*);
    383383    JS_EXPORT_PRIVATE JSValue throwException(ExecState*, JSValue);
    384384    JS_EXPORT_PRIVATE JSObject* throwException(ExecState*, JSObject*);
     
    570570#endif
    571571    void* m_lastStackTop;
    572     JSValue m_exception;
     572    Exception* m_exception { nullptr };
    573573    bool m_inDefineOwnProperty;
    574574    std::unique_ptr<CodeCache> m_codeCache;
    575575    LegacyProfiler* m_enabledProfiler;
    576576    std::unique_ptr<BuiltinExecutables> m_builtinExecutables;
    577     RefCountedArray<StackFrame> m_exceptionStack;
    578577    HashMap<String, RefPtr<WatchpointSet>> m_impurePropertyWatchpointSets;
    579578    std::unique_ptr<TypeProfiler> m_typeProfiler;
  • TabularUnified trunk/Source/JavaScriptCore/runtime/VMEntryScope.cpp

    r172324 r185259  
    5050        vm.resetDateCache();
    5151    }
    52 
    53     // Clear the captured exception stack between entries
    54     vm.clearExceptionStack();
    5552}
    5653
  • TabularUnified trunk/Source/WebCore/ChangeLog

    r185258 r185259  
     12015-06-05  Mark Lam  <mark.lam@apple.com>
     2
     3        finally blocks should not set the exception stack trace when re-throwing the exception.
     4        https://bugs.webkit.org/show_bug.cgi?id=145525
     5
     6        Reviewed by Geoffrey Garen.
     7
     8        Update to use the new JSC::Exception object.
     9
     10        Test: inspector/debugger/break-on-exceptions.html
     11
     12        * ForwardingHeaders/runtime/Exception.h: Added.
     13        * bindings/js/JSCallbackData.cpp:
     14        (WebCore::JSCallbackData::invokeCallback):
     15        * bindings/js/JSCustomXPathNSResolver.cpp:
     16        (WebCore::JSCustomXPathNSResolver::lookupNamespaceURI):
     17        * bindings/js/JSDOMBinding.cpp:
     18        (WebCore::jsArray):
     19        (WebCore::reportException):
     20        (WebCore::reportCurrentException):
     21        * bindings/js/JSDOMBinding.h:
     22        * bindings/js/JSErrorHandler.cpp:
     23        (WebCore::JSErrorHandler::handleEvent):
     24        * bindings/js/JSEventListener.cpp:
     25        (WebCore::JSEventListener::handleEvent):
     26        * bindings/js/JSMainThreadExecState.cpp:
     27        (WebCore::JSMainThreadExecState::didLeaveScriptContext):
     28        (WebCore::functionCallHandlerFromAnyThread):
     29        (WebCore::evaluateHandlerFromAnyThread):
     30        * bindings/js/JSMainThreadExecState.h:
     31        (WebCore::JSMainThreadExecState::currentState):
     32        (WebCore::JSMainThreadExecState::call):
     33        (WebCore::JSMainThreadExecState::evaluate):
     34        (WebCore::JSMainThreadExecState::runTask):
     35
     36        * bindings/js/JSMediaDevicesCustom.cpp:
     37        (WebCore::JSMediaDevices::getUserMedia):
     38        - Fixed a bug where the exception was not cleared before entering the VM to
     39          call JS code.
     40
     41        * bindings/js/JSMutationCallback.cpp:
     42        (WebCore::JSMutationCallback::call):
     43        * bindings/js/ReadableJSStream.cpp:
     44        (WebCore::getPropertyFromObject):
     45        (WebCore::callFunction):
     46        (WebCore::ReadableJSStream::Source::start):
     47        * bindings/js/ScheduledAction.cpp:
     48        (WebCore::ScheduledAction::executeFunctionInContext):
     49        * bindings/js/ScriptController.cpp:
     50        (WebCore::ScriptController::evaluateInWorld):
     51        * bindings/js/SerializedScriptValue.cpp:
     52        (WebCore::SerializedScriptValue::create):
     53        (WebCore::SerializedScriptValue::deserialize):
     54        * bindings/js/WorkerScriptController.cpp:
     55        (WebCore::WorkerScriptController::evaluate):
     56        (WebCore::WorkerScriptController::setException):
     57        (WebCore::WorkerScriptController::scheduleExecutionTermination):
     58        * bindings/js/WorkerScriptController.h:
     59        (WebCore::WorkerScriptController::workerGlobalScopeWrapper):
     60        * bindings/js/WorkerScriptDebugServer.cpp:
     61        (WebCore::WorkerScriptDebugServer::runEventLoopWhilePaused):
     62        (WebCore::WorkerScriptDebugServer::reportException):
     63        * bindings/js/WorkerScriptDebugServer.h:
     64        * bindings/objc/WebScriptObject.mm:
     65        (WebCore::createJSWrapper):
     66        (WebCore::addExceptionToConsole):
     67        (-[WebScriptObject callWebScriptMethod:withArguments:]):
     68        (-[WebScriptObject evaluateWebScript:]):
     69        - Changed to call a version of JSMainThreadExecState::evaluate() that provides
     70          a stub returnedException because evaluateWebScript: doesn't need the exception.
     71
     72        * inspector/PageScriptDebugServer.cpp:
     73        (WebCore::PageScriptDebugServer::isContentScript):
     74        (WebCore::PageScriptDebugServer::reportException):
     75        * inspector/PageScriptDebugServer.h:
     76        * workers/WorkerGlobalScope.cpp:
     77        (WebCore::WorkerGlobalScope::importScripts):
     78
    1792015-06-05  Eric Carlson  <eric.carlson@apple.com>
    280
  • TabularUnified trunk/Source/WebCore/bindings/js/JSCallbackData.cpp

    r182205 r185259  
    7474    InspectorInstrumentationCookie cookie = JSMainThreadExecState::instrumentFunctionCall(context, callType, callData);
    7575
    76     JSValue exception;
     76    Exception* exception;
    7777    JSValue result = context->isDocument()
    78         ? JSMainThreadExecState::call(exec, function, callType, callData, thisValue, args, &exception)
    79         : JSC::call(exec, function, callType, callData, thisValue, args, &exception);
     78        ? JSMainThreadExecState::call(exec, function, callType, callData, thisValue, args, exception)
     79        : JSC::call(exec, function, callType, callData, thisValue, args, exception);
    8080
    8181    InspectorInstrumentation::didCallFunction(cookie, context);
  • TabularUnified trunk/Source/WebCore/bindings/js/JSCustomXPathNSResolver.cpp

    r184543 r185259  
    9494    args.append(jsStringWithCache(exec, prefix));
    9595
    96     JSValue exception;
    97     JSValue retval = JSMainThreadExecState::call(exec, function, callType, callData, m_customResolver.get(), args, &exception);
     96    Exception* exception;
     97    JSValue retval = JSMainThreadExecState::call(exec, function, callType, callData, m_customResolver.get(), args, exception);
    9898
    9999    String result;
  • TabularUnified trunk/Source/WebCore/bindings/js/JSDOMBinding.cpp

    r184651 r185259  
    4141#include <runtime/Error.h>
    4242#include <runtime/ErrorHandlingScope.h>
     43#include <runtime/Exception.h>
    4344#include <runtime/ExceptionHelpers.h>
    4445#include <runtime/JSFunction.h>
     
    139140}
    140141
    141 void reportException(ExecState* exec, JSValue exception, CachedScript* cachedScript)
     142void reportException(ExecState* exec, Exception* exception, CachedScript* cachedScript)
    142143{
    143144    RELEASE_ASSERT(exec->vm().currentThreadIsHoldingAPILock());
     
    149150    RefPtr<ScriptCallStack> callStack(createScriptCallStackFromException(exec, exception, ScriptCallStack::maxCallStackSizeToCapture));
    150151    exec->clearException();
    151     exec->clearSupplementaryExceptionInfo();
    152152
    153153    JSDOMGlobalObject* globalObject = jsCast<JSDOMGlobalObject*>(exec->lexicalGlobalObject());
     
    167167
    168168    String errorMessage;
    169     if (ExceptionBase* exceptionBase = toExceptionBase(exception))
     169    if (ExceptionBase* exceptionBase = toExceptionBase(exception->value()))
    170170        errorMessage = exceptionBase->message() + ": "  + exceptionBase->description();
    171171    else {
    172172        // FIXME: <http://webkit.org/b/115087> Web Inspector: WebCore::reportException should not evaluate JavaScript handling exceptions
    173173        // If this is a custon exception object, call toString on it to try and get a nice string representation for the exception.
    174         errorMessage = exception.toString(exec)->value(exec);
     174        errorMessage = exception->value().toString(exec)->value(exec);
    175175        exec->clearException();
    176         exec->clearSupplementaryExceptionInfo();
    177176    }
    178177
     
    183182void reportCurrentException(ExecState* exec)
    184183{
    185     JSValue exception = exec->exception();
     184    Exception* exception = exec->exception();
    186185    exec->clearException();
    187186    reportException(exec, exception);
  • TabularUnified trunk/Source/WebCore/bindings/js/JSDOMBinding.h

    r184651 r185259  
    248248const JSC::HashTable& getHashTableForGlobalData(JSC::VM&, const JSC::HashTable& staticTable);
    249249
    250 WEBCORE_EXPORT void reportException(JSC::ExecState*, JSC::JSValue exception, CachedScript* = nullptr);
     250WEBCORE_EXPORT void reportException(JSC::ExecState*, JSC::Exception*, CachedScript* = nullptr);
    251251void reportCurrentException(JSC::ExecState*);
    252252
  • TabularUnified trunk/Source/WebCore/bindings/js/JSErrorHandler.cpp

    r174225 r185259  
    9999        VMEntryScope entryScope(vm, vm.entryScope ? vm.entryScope->globalObject() : globalObject);
    100100
    101         JSValue exception;
     101        Exception* exception;
    102102        JSValue returnValue = scriptExecutionContext->isDocument()
    103             ? JSMainThreadExecState::call(exec, jsFunction, callType, callData, globalObject, args, &exception)
    104             : JSC::call(exec, jsFunction, callType, callData, globalObject, args, &exception);
     103            ? JSMainThreadExecState::call(exec, jsFunction, callType, callData, globalObject, args, exception)
     104            : JSC::call(exec, jsFunction, callType, callData, globalObject, args, exception);
    105105
    106106        globalObject->setCurrentEvent(savedEvent);
  • TabularUnified trunk/Source/WebCore/bindings/js/JSEventListener.cpp

    r184616 r185259  
    125125
    126126        JSValue thisValue = handleEventFunction == jsFunction ? toJS(exec, globalObject, event->currentTarget()) : jsFunction;
    127         JSValue exception;
     127        Exception* exception;
    128128        JSValue retval = scriptExecutionContext->isDocument()
    129             ? JSMainThreadExecState::call(exec, handleEventFunction, callType, callData, thisValue, args, &exception)
    130             : JSC::call(exec, handleEventFunction, callType, callData, thisValue, args, &exception);
     129            ? JSMainThreadExecState::call(exec, handleEventFunction, callType, callData, thisValue, args, exception)
     130            : JSC::call(exec, handleEventFunction, callType, callData, thisValue, args, exception);
    131131
    132132        InspectorInstrumentation::didCallFunction(cookie, scriptExecutionContext);
  • TabularUnified trunk/Source/WebCore/bindings/js/JSMainThreadExecState.cpp

    r167142 r185259  
    4747}
    4848
    49 JSC::JSValue functionCallHandlerFromAnyThread(JSC::ExecState* exec, JSC::JSValue functionObject, JSC::CallType callType, const JSC::CallData& callData, JSC::JSValue thisValue, const JSC::ArgList& args, JSC::JSValue* exception)
     49JSC::JSValue functionCallHandlerFromAnyThread(JSC::ExecState* exec, JSC::JSValue functionObject, JSC::CallType callType, const JSC::CallData& callData, JSC::JSValue thisValue, const JSC::ArgList& args, JSC::Exception*& returnedException)
    5050{
    5151    if (isMainThread())
    52         return JSMainThreadExecState::call(exec, functionObject, callType, callData, thisValue, args, exception);
    53     return JSC::call(exec, functionObject, callType, callData, thisValue, args, exception);
     52        return JSMainThreadExecState::call(exec, functionObject, callType, callData, thisValue, args, returnedException);
     53    return JSC::call(exec, functionObject, callType, callData, thisValue, args, returnedException);
    5454}
    5555
    56 JSC::JSValue evaluateHandlerFromAnyThread(JSC::ExecState* exec, const JSC::SourceCode& source, JSC::JSValue thisValue, JSC::JSValue* exception)
     56JSC::JSValue evaluateHandlerFromAnyThread(JSC::ExecState* exec, const JSC::SourceCode& source, JSC::JSValue thisValue, JSC::Exception*& returnedException)
    5757{
    5858    if (isMainThread())
    59         return JSMainThreadExecState::evaluate(exec, source, thisValue, exception);
    60     return JSC::evaluate(exec, source, thisValue, exception);
     59        return JSMainThreadExecState::evaluate(exec, source, thisValue, returnedException);
     60    return JSC::evaluate(exec, source, thisValue, returnedException);
    6161}
    6262
  • TabularUnified trunk/Source/WebCore/bindings/js/JSMainThreadExecState.h

    r167142 r185259  
    5151    };
    5252   
    53     static JSC::JSValue call(JSC::ExecState* exec, JSC::JSValue functionObject, JSC::CallType callType, const JSC::CallData& callData, JSC::JSValue thisValue, const JSC::ArgList& args, JSC::JSValue* exception)
     53    static JSC::JSValue call(JSC::ExecState* exec, JSC::JSValue functionObject, JSC::CallType callType, const JSC::CallData& callData, JSC::JSValue thisValue, const JSC::ArgList& args, JSC::Exception*& returnedException)
    5454    {
    5555        JSMainThreadExecState currentState(exec);
    56         return JSC::call(exec, functionObject, callType, callData, thisValue, args, exception);
     56        return JSC::call(exec, functionObject, callType, callData, thisValue, args, returnedException);
    5757    };
    5858
    59     static JSC::JSValue evaluate(JSC::ExecState* exec, const JSC::SourceCode& source, JSC::JSValue thisValue, JSC::JSValue* exception)
     59    static JSC::JSValue evaluate(JSC::ExecState* exec, const JSC::SourceCode& source, JSC::JSValue thisValue, JSC::Exception*& returnedException)
    6060    {
    6161        JSMainThreadExecState currentState(exec);
    62         return JSC::evaluate(exec, source, thisValue, exception);
     62        return JSC::evaluate(exec, source, thisValue, returnedException);
     63    };
     64
     65    static JSC::JSValue evaluate(JSC::ExecState* exec, const JSC::SourceCode& source, JSC::JSValue thisValue = JSC::JSValue())
     66    {
     67        JSC::Exception* unused;
     68        return evaluate(exec, source, thisValue, unused);
    6369    };
    6470
     
    122128};
    123129
    124 JSC::JSValue functionCallHandlerFromAnyThread(JSC::ExecState*, JSC::JSValue functionObject, JSC::CallType, const JSC::CallData&, JSC::JSValue thisValue, const JSC::ArgList& args, JSC::JSValue* exception);
    125 JSC::JSValue evaluateHandlerFromAnyThread(JSC::ExecState*, const JSC::SourceCode&, JSC::JSValue thisValue, JSC::JSValue* exception);
     130JSC::JSValue functionCallHandlerFromAnyThread(JSC::ExecState*, JSC::JSValue functionObject, JSC::CallType, const JSC::CallData&, JSC::JSValue thisValue, const JSC::ArgList& args, JSC::Exception*& returnedException);
     131JSC::JSValue evaluateHandlerFromAnyThread(JSC::ExecState*, const JSC::SourceCode&, JSC::JSValue thisValue, JSC::Exception*& returnedException);
    126132
    127133} // namespace WebCore
  • TabularUnified trunk/Source/WebCore/bindings/js/JSMediaDevicesCustom.cpp

    r184984 r185259  
    4040#include "JSMediaStream.h"
    4141#include "JSNavigatorUserMediaError.h"
     42#include <runtime/Exception.h>
    4243
    4344using namespace JSC;
     
    5152    Dictionary options(exec, exec->argument(0));
    5253    if (exec->hadException()) {
    53         wrapper.reject(exec->exception());
     54        Exception* exception = exec->exception();
     55        exec->clearException();
     56        wrapper.reject(exception->value());
    5457        return wrapper.promise();
    5558    }
  • TabularUnified trunk/Source/WebCore/bindings/js/JSMutationCallback.cpp

    r167142 r185259  
    8888    InspectorInstrumentationCookie cookie = JSMainThreadExecState::instrumentFunctionCall(context, callType, callData);
    8989
    90     JSValue exception;
    91     JSMainThreadExecState::call(exec, callback, callType, callData, jsObserver, args, &exception);
     90    Exception* exception;
     91    JSMainThreadExecState::call(exec, callback, callType, callData, jsObserver, args, exception);
    9292
    9393    InspectorInstrumentation::didCallFunction(cookie, context);
  • TabularUnified trunk/Source/WebCore/bindings/js/ReadableJSStream.cpp

    r185197 r185259  
    5353}
    5454
    55 static inline JSValue callFunction(ExecState* exec, JSValue jsFunction, JSValue thisValue, const ArgList& arguments, JSValue* exception)
     55static inline JSValue callFunction(ExecState* exec, JSValue jsFunction, JSValue thisValue, const ArgList& arguments, Exception*& exception)
    5656{
    5757    CallData callData;
     
    8989    arguments.append(jsController(exec, globalObject()));
    9090
    91     JSValue exception;
    92     callFunction(&exec, startFunction, m_source.get(), arguments, &exception);
     91    Exception* exception;
     92    callFunction(&exec, startFunction, m_source.get(), arguments, exception);
    9393
    9494    if (exception) {
  • TabularUnified trunk/Source/WebCore/bindings/js/ScheduledAction.cpp

    r182266 r185259  
    100100    InspectorInstrumentationCookie cookie = JSMainThreadExecState::instrumentFunctionCall(&context, callType, callData);
    101101
    102     JSValue exception;
     102    Exception* exception;
    103103    if (is<Document>(context))
    104         JSMainThreadExecState::call(exec, m_function.get(), callType, callData, thisValue, args, &exception);
     104        JSMainThreadExecState::call(exec, m_function.get(), callType, callData, thisValue, args, exception);
    105105    else
    106         JSC::call(exec, m_function.get(), callType, callData, thisValue, args, &exception);
     106        JSC::call(exec, m_function.get(), callType, callData, thisValue, args, exception);
    107107
    108108    InspectorInstrumentation::didCallFunction(cookie, &context);
  • TabularUnified trunk/Source/WebCore/bindings/js/ScriptController.cpp

    r182243 r185259  
    161161    InspectorInstrumentationCookie cookie = InspectorInstrumentation::willEvaluateScript(m_frame, sourceURL, sourceCode.startLine());
    162162
    163     JSValue evaluationException;
    164 
    165     JSValue returnValue = JSMainThreadExecState::evaluate(exec, jsSourceCode, shell, &evaluationException);
     163    Exception* evaluationException;
     164    JSValue returnValue = JSMainThreadExecState::evaluate(exec, jsSourceCode, shell, evaluationException);
    166165
    167166    InspectorInstrumentation::didEvaluateScript(cookie, m_frame);
  • TabularUnified trunk/Source/WebCore/bindings/js/SerializedScriptValue.cpp

    r184828 r185259  
    5656#include <runtime/DateInstance.h>
    5757#include <runtime/Error.h>
     58#include <runtime/Exception.h>
    5859#include <runtime/ExceptionHelpers.h>
    5960#include <runtime/JSArrayBuffer.h>
     
    27032704    if (exec->hadException()) {
    27042705        if (exception)
    2705             *exception = toRef(exec, exec->exception());
     2706            *exception = toRef(exec, exec->exception()->value());
    27062707        exec->clearException();
    27072708        return 0;
     
    27332734    if (exec->hadException()) {
    27342735        if (exception)
    2735             *exception = toRef(exec, exec->exception());
     2736            *exception = toRef(exec, exec->exception()->value());
    27362737        exec->clearException();
    27372738        return nullptr;
  • TabularUnified trunk/Source/WebCore/bindings/js/WorkerScriptController.cpp

    r179353 r185259  
    4141#include <interpreter/Interpreter.h>
    4242#include <runtime/Completion.h>
     43#include <runtime/Exception.h>
    4344#include <runtime/ExceptionHelpers.h>
    4445#include <runtime/Error.h>
     
    99100        return;
    100101
    101     Deprecated::ScriptValue exception;
    102     evaluate(sourceCode, &exception);
    103     if (exception.jsValue()) {
     102    Exception* exception;
     103    evaluate(sourceCode, exception);
     104    if (exception) {
    104105        JSLockHolder lock(vm());
    105         reportException(m_workerGlobalScopeWrapper->globalExec(), exception.jsValue());
    106     }
    107 }
    108 
    109 void WorkerScriptController::evaluate(const ScriptSourceCode& sourceCode, Deprecated::ScriptValue* exception)
     106        reportException(m_workerGlobalScopeWrapper->globalExec(), exception);
     107    }
     108}
     109
     110void WorkerScriptController::evaluate(const ScriptSourceCode& sourceCode, JSC::Exception*& returnedException)
    110111{
    111112    if (isExecutionForbidden())
     
    117118    JSLockHolder lock(exec);
    118119
    119     JSValue evaluationException;
    120     JSC::evaluate(exec, sourceCode.jsSourceCode(), m_workerGlobalScopeWrapper->globalThis(), &evaluationException);
     120    JSC::Exception* evaluationException;
     121    JSC::evaluate(exec, sourceCode.jsSourceCode(), m_workerGlobalScopeWrapper->globalThis(), evaluationException);
    121122
    122123    VM& vm = exec->vm();
     
    132133        int columnNumber = 0;
    133134        String sourceURL = sourceCode.url().string();
    134         if (m_workerGlobalScope->sanitizeScriptError(errorMessage, lineNumber, columnNumber, sourceURL, sourceCode.cachedScript()))
    135             *exception = Deprecated::ScriptValue(*m_vm, exec->vm().throwException(exec, createError(exec, errorMessage.impl())));
    136         else
    137             *exception = Deprecated::ScriptValue(*m_vm, evaluationException);
    138     }
    139 }
    140 
    141 void WorkerScriptController::setException(const Deprecated::ScriptValue& exception)
    142 {
    143     m_workerGlobalScopeWrapper->globalExec()->vm().throwException(m_workerGlobalScopeWrapper->globalExec(), exception.jsValue());
     135        if (m_workerGlobalScope->sanitizeScriptError(errorMessage, lineNumber, columnNumber, sourceURL, sourceCode.cachedScript())) {
     136            vm.throwException(exec, createError(exec, errorMessage.impl()));
     137            evaluationException = vm.exception();
     138            vm.clearException();
     139        }
     140    }
     141    returnedException = evaluationException;
     142}
     143
     144void WorkerScriptController::setException(JSC::Exception* exception)
     145{
     146    JSC::ExecState* exec = m_workerGlobalScopeWrapper->globalExec();
     147    exec->vm().throwException(exec, exception);
    144148}
    145149
  • TabularUnified trunk/Source/WebCore/bindings/js/WorkerScriptController.h

    r165710 r185259  
    6161
    6262        void evaluate(const ScriptSourceCode&);
    63         void evaluate(const ScriptSourceCode&, Deprecated::ScriptValue* exception);
     63        void evaluate(const ScriptSourceCode&, JSC::Exception*& returnedException);
    6464
    65         void setException(const Deprecated::ScriptValue&);
     65        void setException(JSC::Exception*);
    6666
    6767        // Async request to terminate a JS run execution. Eventually causes termination
  • TabularUnified trunk/Source/WebCore/bindings/js/WorkerScriptDebugServer.cpp

    r178820 r185259  
    100100}
    101101
    102 void WorkerScriptDebugServer::reportException(JSC::ExecState* exec, JSC::JSValue exception) const
     102void WorkerScriptDebugServer::reportException(JSC::ExecState* exec, JSC::Exception* exception) const
    103103{
    104104    WebCore::reportException(exec, exception);
  • TabularUnified trunk/Source/WebCore/bindings/js/WorkerScriptDebugServer.h

    r178820 r185259  
    5757    virtual void runEventLoopWhilePaused() override;
    5858    virtual bool isContentScript(JSC::ExecState*) const override { return false; }
    59     virtual void reportException(JSC::ExecState*, JSC::JSValue) const override;
     59    virtual void reportException(JSC::ExecState*, JSC::Exception*) const override;
    6060
    6161    WorkerGlobalScope* m_workerGlobalScope;
  • TabularUnified trunk/Source/WebCore/bindings/objc/WebScriptObject.mm

    r182205 r185259  
    122122}
    123123
    124 static void addExceptionToConsole(ExecState* exec, JSC::JSValue& exception)
     124static void addExceptionToConsole(ExecState* exec, JSC::Exception* exception)
    125125{
    126126    JSDOMWindow* window = asJSDOMWindow(exec->vmEntryGlobalObject());
     
    132132static void addExceptionToConsole(ExecState* exec)
    133133{
    134     JSC::JSValue exception = exec->exception();
     134    JSC::Exception* exception = exec->exception();
    135135    exec->clearException();
    136136    addExceptionToConsole(exec, exception);
     
    343343        return nil;
    344344
    345     JSC::JSValue exception;
    346     JSC::JSValue result = JSMainThreadExecState::call(exec, function, callType, callData, [self _imp], argList, &exception);
     345    JSC::Exception* exception;
     346    JSC::JSValue result = JSMainThreadExecState::call(exec, function, callType, callData, [self _imp], argList, exception);
    347347
    348348    if (exception) {
     
    367367    JSLockHolder lock(exec);
    368368   
    369     JSC::JSValue returnValue = JSMainThreadExecState::evaluate(exec, makeSource(String(script)), JSC::JSValue(), 0);
     369    JSC::JSValue returnValue = JSMainThreadExecState::evaluate(exec, makeSource(String(script)), JSC::JSValue());
    370370
    371371    id resultObj = [WebScriptObject _convertValueToObjcValue:returnValue originRootObject:[self _originRootObject] rootObject:[self _rootObject]];
  • TabularUnified trunk/Source/WebCore/inspector/PageScriptDebugServer.cpp

    r178820 r185259  
    138138}
    139139
    140 void PageScriptDebugServer::reportException(ExecState* exec, JSValue exception) const
     140void PageScriptDebugServer::reportException(ExecState* exec, Exception* exception) const
    141141{
    142142    WebCore::reportException(exec, exception);
  • TabularUnified trunk/Source/WebCore/inspector/PageScriptDebugServer.h

    r178820 r185259  
    5555    virtual void runEventLoopWhilePaused() override;
    5656    virtual bool isContentScript(JSC::ExecState*) const override;
    57     virtual void reportException(JSC::ExecState*, JSC::JSValue) const override;
     57    virtual void reportException(JSC::ExecState*, JSC::Exception*) const override;
    5858
    5959    void runEventLoopWhilePausedInternal();
  • TabularUnified trunk/Source/WebCore/workers/WorkerGlobalScope.cpp

    r184990 r185259  
    198198        InspectorInstrumentation::scriptImported(scriptExecutionContext(), scriptLoader->identifier(), scriptLoader->script());
    199199
    200         Deprecated::ScriptValue exception;
    201         m_script->evaluate(ScriptSourceCode(scriptLoader->script(), scriptLoader->responseURL()), &exception);
    202         if (!exception.hasNoValue()) {
     200        JSC::Exception* exception;
     201        m_script->evaluate(ScriptSourceCode(scriptLoader->script(), scriptLoader->responseURL()), exception);
     202        if (exception) {
    203203            m_script->setException(exception);
    204204            return;
  • TabularUnified trunk/Source/WebKit/mac/ChangeLog

    r185168 r185259  
     12015-06-05  Mark Lam  <mark.lam@apple.com>
     2
     3        finally blocks should not set the exception stack trace when re-throwing the exception.
     4        https://bugs.webkit.org/show_bug.cgi?id=145525
     5
     6        Reviewed by Geoffrey Garen.
     7
     8        * WebView/WebView.mm:
     9        (+[WebView _reportException:inContext:]):
     10        (WebKitInitializeApplicationCachePathIfNecessary):
     11        - Changed to use the new Exception object.
     12
    1132015-06-03  Anders Carlsson  <andersca@apple.com>
    214
  • TabularUnified trunk/Source/WebKit/mac/WebView/WebView.mm

    r184946 r185259  
    115115#import <Foundation/NSURLConnection.h>
    116116#import <JavaScriptCore/APICast.h>
     117#import <JavaScriptCore/Exception.h>
    117118#import <JavaScriptCore/JSValueRef.h>
    118119#import <WebCore/AlternativeTextUIController.h>
     
    725726        return;
    726727
    727     reportException(execState, toJS(execState, exception));
     728    Exception* vmException = Exception::cast(toJS(execState, exception));
     729    reportException(execState, vmException);
    728730}
    729731
  • TabularUnified trunk/Source/WebKit/win/ChangeLog

    r185111 r185259  
     12015-06-05  Mark Lam  <mark.lam@apple.com>
     2
     3        finally blocks should not set the exception stack trace when re-throwing the exception.
     4        https://bugs.webkit.org/show_bug.cgi?id=145525
     5
     6        Reviewed by Geoffrey Garen.
     7
     8        * WebView.cpp:
     9        (WebView::reportException):
     10        - Changed to use the new Exception object.
     11
    1122015-06-02  Brady Eidson  <beidson@apple.com>
    213
  • TabularUnified trunk/Source/WebKit/win/WebView.cpp

    r184813 r185259  
    7373#include "resource.h"
    7474#include <JavaScriptCore/APICast.h>
     75#include <JavaScriptCore/Exception.h>
    7576#include <JavaScriptCore/InitializeThreading.h>
    7677#include <JavaScriptCore/JSCJSValue.h>
     
    60306031        return E_FAIL;
    60316032
    6032     WebCore::reportException(execState, toJS(execState, exception));
     6033    JSC::Exception* vmException = JSC::Exception::cast(toJS(execState, exception));
     6034    WebCore::reportException(execState, vmException);
    60336035    return S_OK;
    60346036}
  • TabularUnified trunk/Source/WebKit2/ChangeLog

    r185251 r185259  
     12015-06-05  Mark Lam  <mark.lam@apple.com>
     2
     3        finally blocks should not set the exception stack trace when re-throwing the exception.
     4        https://bugs.webkit.org/show_bug.cgi?id=145525
     5
     6        Reviewed by Geoffrey Garen.
     7
     8        * WebProcess/InjectedBundle/InjectedBundle.cpp:
     9        (WebKit::InjectedBundle::reportException):
     10        - Changed to use the new Exception object.
     11
    1122015-06-05  Anders Carlsson  <andersca@apple.com>
    213
  • TabularUnified trunk/Source/WebKit2/WebProcess/InjectedBundle/InjectedBundle.cpp

    r184950 r185259  
    5151#include "WebProcessPoolMessages.h"
    5252#include <JavaScriptCore/APICast.h>
     53#include <JavaScriptCore/Exception.h>
    5354#include <JavaScriptCore/JSLock.h>
    5455#include <WebCore/ApplicationCache.h>
     
    523524        return;
    524525
    525     WebCore::reportException(execState, toJS(execState, exception));
     526    WebCore::reportException(execState, Exception::cast(toJS(execState, exception)));
    526527}
    527528
Note: See TracChangeset for help on using the changeset viewer.