[debug] disable debug breaks in side-effect free debug-evaluate.

We don't want to run into the situation of breaking inside of
debug-evaluate. That would get even more confusing with throw-on-side-effect.

R=kozyatinskiy@chromium.org

Bug: v8:7592
Change-Id: I93f5de63d8943792ff000dbf7c6311df655d3793
Reviewed-on: https://chromium-review.googlesource.com/978164
Reviewed-by: Aleksey Kozyatinskiy <kozyatinskiy@chromium.org>
Commit-Queue: Yang Guo <yangguo@chromium.org>
Cr-Commit-Position: refs/heads/master@{#52227}
This commit is contained in:
Yang Guo 2018-03-26 10:35:25 +02:00 committed by Commit Bot
parent 6f52d0157f
commit cc9736a1c0
5 changed files with 57 additions and 8 deletions

View File

@ -24,6 +24,9 @@ namespace internal {
MaybeHandle<Object> DebugEvaluate::Global(Isolate* isolate,
Handle<String> source,
bool throw_on_side_effect) {
// Disable breaks in side-effect free mode.
DisableBreak disable_break_scope(isolate->debug(), throw_on_side_effect);
Handle<Context> context = isolate->native_context();
ScriptOriginOptions origin_options(false, true);
MaybeHandle<SharedFunctionInfo> maybe_function_info =

View File

@ -2388,7 +2388,7 @@ NoSideEffectScope::~NoSideEffectScope() {
isolate_->Throw(*isolate_->factory()->NewEvalError(
MessageTemplate::kNoSideEffectDebugEvaluate));
}
isolate_->set_needs_side_effect_check(old_needs_side_effect_check_);
isolate_->set_needs_side_effect_check(false);
isolate_->debug()->UpdateHookOnFunctionCall();
isolate_->debug()->side_effect_check_failed_ = false;
}

View File

@ -700,9 +700,9 @@ class ReturnValueScope {
// Stack allocated class for disabling break.
class DisableBreak BASE_EMBEDDED {
public:
explicit DisableBreak(Debug* debug)
explicit DisableBreak(Debug* debug, bool disable = true)
: debug_(debug), previous_break_disabled_(debug->break_disabled_) {
debug_->break_disabled_ = true;
debug_->break_disabled_ = disable;
}
~DisableBreak() {
debug_->break_disabled_ = previous_break_disabled_;
@ -732,10 +732,10 @@ class SuppressDebug BASE_EMBEDDED {
class NoSideEffectScope {
public:
NoSideEffectScope(Isolate* isolate, bool disallow_side_effects)
: isolate_(isolate),
old_needs_side_effect_check_(isolate->needs_side_effect_check()) {
isolate->set_needs_side_effect_check(old_needs_side_effect_check_ ||
disallow_side_effects);
: isolate_(isolate) {
// NoSideEffectScope is not re-entrant if already enabled.
CHECK(!isolate->needs_side_effect_check());
isolate->set_needs_side_effect_check(disallow_side_effects);
isolate->debug()->UpdateHookOnFunctionCall();
isolate->debug()->side_effect_check_failed_ = false;
}
@ -743,7 +743,6 @@ class NoSideEffectScope {
private:
Isolate* isolate_;
bool old_needs_side_effect_check_;
DISALLOW_COPY_AND_ASSIGN(NoSideEffectScope);
};

View File

@ -48,3 +48,26 @@ Test expression without side-effect, with throwOnSideEffect: true
}
}
}
Test that debug break triggers without throwOnSideEffect
paused
{
id : <messageId>
result : {
result : {
description : 2
type : number
value : 2
}
}
}
Test that debug break does not trigger with throwOnSideEffect
{
id : <messageId>
result : {
result : {
description : 2
type : number
value : 2
}
}
}

View File

@ -4,8 +4,19 @@
let {session, contextGroup, Protocol} = InspectorTest.start("Tests that Runtime.evaluate can run without side effects.");
session.setupScriptMap();
contextGroup.addScript(`
function f() {
return 2;
} //# sourceURL=test.js`);
Protocol.Runtime.enable();
Protocol.Debugger.enable();
Protocol.Debugger.onPaused(message => {
InspectorTest.log("paused");
Protocol.Debugger.resume();
});
(async function() {
InspectorTest.log("Test throwOnSideEffect: false");
InspectorTest.logMessage(await Protocol.Runtime.evaluate({
@ -25,5 +36,18 @@ Protocol.Debugger.enable();
throwOnSideEffect: true
}));
InspectorTest.log("Test that debug break triggers without throwOnSideEffect");
await Protocol.Debugger.setBreakpointByUrl({ url: 'test.js', lineNumber: 2 });
InspectorTest.logMessage(await Protocol.Runtime.evaluate({
expression: "f()",
throwOnSideEffect: false
}));
InspectorTest.log("Test that debug break does not trigger with throwOnSideEffect");
InspectorTest.logMessage(await Protocol.Runtime.evaluate({
expression: "f()",
throwOnSideEffect: true
}));
InspectorTest.completeTest();
})();