Commit f813494f authored by neis's avatar neis Committed by Commit bot

[debug] Don't notify listener of exceptions internal to a desugaring.

In the parser, we desugar yield* with the help of a regular yield. One
particular implementation detail of this desugaring is that when the user calls
the generator's throw method, this throws an exception that we immediately
catch. This exception should not be visible to the user, but through Devtools'
"Pause on Caught Exceptions" feature it used to be.

This CL extends the type of catch predictions with a new value for such internal
exceptions and uses that for the offending try-catch statement in yield*.  It
instruments the debugger to _not_ trigger an exception event in that case.

R=yangguo@chromium.org
TBR=littledan@chromium.org
BUG=v8:5218

Review-Url: https://codereview.chromium.org/2203803002
Cr-Commit-Position: refs/heads/master@{#38286}
parent b123476b
......@@ -3166,6 +3166,16 @@ class AstNodeFactory final BASE_EMBEDDED {
HandlerTable::PROMISE, pos);
}
TryCatchStatement* NewTryCatchStatementForDesugaring(Block* try_block,
Scope* scope,
Variable* variable,
Block* catch_block,
int pos) {
return new (local_zone_)
TryCatchStatement(local_zone_, try_block, scope, variable, catch_block,
HandlerTable::DESUGARING, pos);
}
TryFinallyStatement* NewTryFinallyStatement(Block* try_block,
Block* finally_block, int pos) {
return new (local_zone_)
......
......@@ -897,6 +897,9 @@ void AstPrinter::PrintTryStatement(TryStatement* node) {
case HandlerTable::PROMISE:
prediction = "PROMISE";
break;
case HandlerTable::DESUGARING:
prediction = "DESUGARING";
break;
}
Print(" %s\n", prediction);
}
......
......@@ -1722,8 +1722,11 @@ MaybeHandle<Object> Debug::PromiseHasUserDefinedRejectHandler(
void Debug::OnException(Handle<Object> exception, Handle<Object> promise) {
// In our prediction, try-finally is not considered to catch.
Isolate::CatchType catch_type = isolate_->PredictExceptionCatcher();
// Don't notify listener of exceptions that are internal to a desugaring.
if (catch_type == Isolate::CAUGHT_BY_DESUGARING) return;
bool uncaught = (catch_type == Isolate::NOT_CAUGHT);
if (promise->IsJSObject()) {
Handle<JSObject> jspromise = Handle<JSObject>::cast(promise);
......
......@@ -1142,8 +1142,9 @@ Object* Isolate::Throw(Object* exception, MessageLocation* location) {
// embedder didn't specify a custom uncaught exception callback,
// or if the custom callback determined that V8 should abort, then
// abort.
CatchType prediction = PredictExceptionCatcher();
if (FLAG_abort_on_uncaught_exception &&
PredictExceptionCatcher() != CAUGHT_BY_JAVASCRIPT &&
(prediction == NOT_CAUGHT || prediction == CAUGHT_BY_EXTERNAL) &&
(!abort_on_uncaught_exception_callback_ ||
abort_on_uncaught_exception_callback_(
reinterpret_cast<v8::Isolate*>(this)))) {
......@@ -1339,9 +1340,9 @@ Isolate::CatchType Isolate::PredictExceptionCatcher() {
// For JavaScript frames we perform a lookup in the handler table.
if (frame->is_java_script()) {
JavaScriptFrame* js_frame = static_cast<JavaScriptFrame*>(frame);
if (PredictException(js_frame) != HandlerTable::UNCAUGHT) {
return CAUGHT_BY_JAVASCRIPT;
}
HandlerTable::CatchPrediction prediction = PredictException(js_frame);
if (prediction == HandlerTable::DESUGARING) return CAUGHT_BY_DESUGARING;
if (prediction != HandlerTable::UNCAUGHT) return CAUGHT_BY_JAVASCRIPT;
}
// The exception has been externally caught if and only if there is an
......@@ -1750,12 +1751,16 @@ Handle<Object> Isolate::GetPromiseOnStackOnThrow() {
ThreadLocalTop* tltop = thread_local_top();
if (tltop->promise_on_stack_ == NULL) return undefined;
// Find the top-most try-catch or try-finally handler.
if (PredictExceptionCatcher() != CAUGHT_BY_JAVASCRIPT) return undefined;
CatchType prediction = PredictExceptionCatcher();
if (prediction == NOT_CAUGHT || prediction == CAUGHT_BY_EXTERNAL) {
return undefined;
}
for (JavaScriptFrameIterator it(this); !it.done(); it.Advance()) {
switch (PredictException(it.frame())) {
case HandlerTable::UNCAUGHT:
break;
case HandlerTable::CAUGHT:
case HandlerTable::DESUGARING:
return undefined;
case HandlerTable::PROMISE:
return tltop->promise_on_stack_->promise();
......
......@@ -748,7 +748,12 @@ class Isolate {
// Tries to predict whether an exception will be caught. Note that this can
// only produce an estimate, because it is undecidable whether a finally
// clause will consume or re-throw an exception.
enum CatchType { NOT_CAUGHT, CAUGHT_BY_JAVASCRIPT, CAUGHT_BY_EXTERNAL };
enum CatchType {
NOT_CAUGHT,
CAUGHT_BY_JAVASCRIPT,
CAUGHT_BY_EXTERNAL,
CAUGHT_BY_DESUGARING
};
CatchType PredictExceptionCatcher();
void ScheduleThrow(Object* exception);
......
......@@ -4466,9 +4466,13 @@ class HandlerTable : public FixedArray {
// exception or cause a re-throw to outside the code boundary. Since this is
// undecidable it is merely an approximation (e.g. useful for debugger).
enum CatchPrediction {
UNCAUGHT, // the handler will (likely) rethrow the exception.
CAUGHT, // the exception will be caught by the handler.
PROMISE // the exception will be caught and cause a promise rejection.
UNCAUGHT, // The handler will (likely) rethrow the exception.
CAUGHT, // The exception will be caught by the handler.
PROMISE, // The exception will be caught and cause a promise rejection.
DESUGARING, // The exception will be caught, but both the exception and the
// catching are part of a desugaring and should therefore not
// be visible to the user (we won't notify the debugger of such
// exceptions).
};
// Getters for handler table based on ranges.
......
......@@ -6531,7 +6531,7 @@ Expression* ParserTraits::RewriteYieldStar(
catch_scope->DeclareLocal(name, VAR, kCreatedInitialized,
Variable::NORMAL);
try_catch = factory->NewTryCatchStatement(
try_catch = factory->NewTryCatchStatementForDesugaring(
try_block, catch_scope, catch_variable, catch_block, nopos);
}
......
......@@ -67,4 +67,22 @@ assertCaught(() => {
});
// Check that an internal exception in our yield* desugaring is not observable.
{
uncaught = null;
let iter = {
next() {return {value:42, done:false}},
throw() {return {done:true}}
};
let iterable = {[Symbol.iterator]() {return iter}};
function* f() { yield* iterable }
let g = f();
g.next();
assertEquals({value: undefined, done: true}, g.throw());
assertNull(uncaught); // No exception event was generated.
}
assertFalse(error);
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment