Commit 4d7fa10c authored by kozyatinskiy's avatar kozyatinskiy Committed by Commit bot

[inspector] added restrictToFunction flag for getPossibleBreakpoints

BUG=chromium:695236
R=dgozman@chromium.org,yangguo@chromium.org

Review-Url: https://codereview.chromium.org/2710953004
Cr-Commit-Position: refs/heads/master@{#43457}
parent 7d1d83e5
...@@ -9241,7 +9241,7 @@ int GetSmiValue(i::Handle<i::FixedArray> array, int index) { ...@@ -9241,7 +9241,7 @@ int GetSmiValue(i::Handle<i::FixedArray> array, int index) {
bool debug::Script::GetPossibleBreakpoints( bool debug::Script::GetPossibleBreakpoints(
const debug::Location& start, const debug::Location& end, const debug::Location& start, const debug::Location& end,
std::vector<debug::Location>* locations) const { bool restrict_to_function, std::vector<debug::Location>* locations) const {
CHECK(!start.IsEmpty()); CHECK(!start.IsEmpty());
i::Handle<i::Script> script = Utils::OpenHandle(this); i::Handle<i::Script> script = Utils::OpenHandle(this);
if (script->type() == i::Script::TYPE_WASM) { if (script->type() == i::Script::TYPE_WASM) {
...@@ -9267,8 +9267,8 @@ bool debug::Script::GetPossibleBreakpoints( ...@@ -9267,8 +9267,8 @@ bool debug::Script::GetPossibleBreakpoints(
if (start_offset >= end_offset) return true; if (start_offset >= end_offset) return true;
std::set<int> offsets; std::set<int> offsets;
if (!isolate->debug()->GetPossibleBreakpoints(script, start_offset, if (!isolate->debug()->GetPossibleBreakpoints(
end_offset, &offsets)) { script, start_offset, end_offset, restrict_to_function, &offsets)) {
return false; return false;
} }
......
...@@ -137,6 +137,7 @@ class V8_EXPORT_PRIVATE Script { ...@@ -137,6 +137,7 @@ class V8_EXPORT_PRIVATE Script {
bool IsModule() const; bool IsModule() const;
bool GetPossibleBreakpoints(const debug::Location& start, bool GetPossibleBreakpoints(const debug::Location& start,
const debug::Location& end, const debug::Location& end,
bool restrict_to_function,
std::vector<debug::Location>* locations) const; std::vector<debug::Location>* locations) const;
private: private:
......
...@@ -1343,7 +1343,24 @@ void FindBreakablePositions(Handle<DebugInfo> debug_info, int start_position, ...@@ -1343,7 +1343,24 @@ void FindBreakablePositions(Handle<DebugInfo> debug_info, int start_position,
} // namespace } // namespace
bool Debug::GetPossibleBreakpoints(Handle<Script> script, int start_position, bool Debug::GetPossibleBreakpoints(Handle<Script> script, int start_position,
int end_position, std::set<int>* positions) { int end_position, bool restrict_to_function,
std::set<int>* positions) {
if (restrict_to_function) {
Handle<Object> result =
FindSharedFunctionInfoInScript(script, start_position);
if (result->IsUndefined(isolate_)) return false;
// Make sure the function has set up the debug info.
Handle<SharedFunctionInfo> shared =
Handle<SharedFunctionInfo>::cast(result);
if (!EnsureDebugInfo(shared)) return false;
Handle<DebugInfo> debug_info(shared->GetDebugInfo());
FindBreakablePositions(debug_info, start_position, end_position,
BREAK_POSITION_ALIGNED, positions);
return true;
}
while (true) { while (true) {
HandleScope scope(isolate_); HandleScope scope(isolate_);
List<Handle<SharedFunctionInfo>> candidates; List<Handle<SharedFunctionInfo>> candidates;
......
...@@ -312,7 +312,8 @@ class Debug { ...@@ -312,7 +312,8 @@ class Debug {
bool PrepareFunctionForBreakPoints(Handle<SharedFunctionInfo> shared); bool PrepareFunctionForBreakPoints(Handle<SharedFunctionInfo> shared);
bool GetPossibleBreakpoints(Handle<Script> script, int start_position, bool GetPossibleBreakpoints(Handle<Script> script, int start_position,
int end_position, std::set<int>* positions); int end_position, bool restrict_to_function,
std::set<int>* positions);
void RecordGenerator(Handle<JSGeneratorObject> generator_object); void RecordGenerator(Handle<JSGeneratorObject> generator_object);
......
...@@ -534,7 +534,8 @@ ...@@ -534,7 +534,8 @@
"name": "getPossibleBreakpoints", "name": "getPossibleBreakpoints",
"parameters": [ "parameters": [
{ "name": "start", "$ref": "Location", "description": "Start of range to search possible breakpoint locations in." }, { "name": "start", "$ref": "Location", "description": "Start of range to search possible breakpoint locations in." },
{ "name": "end", "$ref": "Location", "optional": true, "description": "End of range to search possible breakpoint locations in (excluding). When not specifed, end of scripts is used as end of range." } { "name": "end", "$ref": "Location", "optional": true, "description": "End of range to search possible breakpoint locations in (excluding). When not specifed, end of scripts is used as end of range." },
{ "name": "restrictToFunction", "type": "boolean", "optional": true, "description": "Only consider locations which are in the same (non-nested) function as start." }
], ],
"returns": [ "returns": [
{ "name": "locations", "type": "array", "items": { "$ref": "Location" }, "description": "List of the possible breakpoint locations." } { "name": "locations", "type": "array", "items": { "$ref": "Location" }, "description": "List of the possible breakpoint locations." }
......
...@@ -361,7 +361,7 @@ void V8DebuggerAgentImpl::removeBreakpointImpl(const String16& breakpointId) { ...@@ -361,7 +361,7 @@ void V8DebuggerAgentImpl::removeBreakpointImpl(const String16& breakpointId) {
Response V8DebuggerAgentImpl::getPossibleBreakpoints( Response V8DebuggerAgentImpl::getPossibleBreakpoints(
std::unique_ptr<protocol::Debugger::Location> start, std::unique_ptr<protocol::Debugger::Location> start,
Maybe<protocol::Debugger::Location> end, Maybe<protocol::Debugger::Location> end, Maybe<bool> restrictToFunction,
std::unique_ptr<protocol::Array<protocol::Debugger::Location>>* locations) { std::unique_ptr<protocol::Array<protocol::Debugger::Location>>* locations) {
String16 scriptId = start->getScriptId(); String16 scriptId = start->getScriptId();
...@@ -386,7 +386,8 @@ Response V8DebuggerAgentImpl::getPossibleBreakpoints( ...@@ -386,7 +386,8 @@ Response V8DebuggerAgentImpl::getPossibleBreakpoints(
if (it == m_scripts.end()) return Response::Error("Script not found"); if (it == m_scripts.end()) return Response::Error("Script not found");
std::vector<v8::debug::Location> v8Locations; std::vector<v8::debug::Location> v8Locations;
if (!it->second->getPossibleBreakpoints(v8Start, v8End, &v8Locations)) if (!it->second->getPossibleBreakpoints(
v8Start, v8End, restrictToFunction.fromMaybe(false), &v8Locations))
return Response::InternalError(); return Response::InternalError();
*locations = protocol::Array<protocol::Debugger::Location>::create(); *locations = protocol::Array<protocol::Debugger::Location>::create();
......
...@@ -66,7 +66,7 @@ class V8DebuggerAgentImpl : public protocol::Debugger::Backend { ...@@ -66,7 +66,7 @@ class V8DebuggerAgentImpl : public protocol::Debugger::Backend {
override; override;
Response getPossibleBreakpoints( Response getPossibleBreakpoints(
std::unique_ptr<protocol::Debugger::Location> start, std::unique_ptr<protocol::Debugger::Location> start,
Maybe<protocol::Debugger::Location> end, Maybe<protocol::Debugger::Location> end, Maybe<bool> restrictToFunction,
std::unique_ptr<protocol::Array<protocol::Debugger::Location>>* locations) std::unique_ptr<protocol::Array<protocol::Debugger::Location>>* locations)
override; override;
Response setScriptSource( Response setScriptSource(
......
...@@ -172,10 +172,12 @@ class ActualScript : public V8DebuggerScript { ...@@ -172,10 +172,12 @@ class ActualScript : public V8DebuggerScript {
bool getPossibleBreakpoints( bool getPossibleBreakpoints(
const v8::debug::Location& start, const v8::debug::Location& end, const v8::debug::Location& start, const v8::debug::Location& end,
bool restrictToFunction,
std::vector<v8::debug::Location>* locations) override { std::vector<v8::debug::Location>* locations) override {
v8::HandleScope scope(m_isolate); v8::HandleScope scope(m_isolate);
v8::Local<v8::debug::Script> script = m_script.Get(m_isolate); v8::Local<v8::debug::Script> script = m_script.Get(m_isolate);
return script->GetPossibleBreakpoints(start, end, locations); return script->GetPossibleBreakpoints(start, end, restrictToFunction,
locations);
} }
void resetBlackboxedStateCache() override { void resetBlackboxedStateCache() override {
...@@ -228,6 +230,7 @@ class WasmVirtualScript : public V8DebuggerScript { ...@@ -228,6 +230,7 @@ class WasmVirtualScript : public V8DebuggerScript {
bool getPossibleBreakpoints( bool getPossibleBreakpoints(
const v8::debug::Location& start, const v8::debug::Location& end, const v8::debug::Location& start, const v8::debug::Location& end,
bool restrictToFunction,
std::vector<v8::debug::Location>* locations) override { std::vector<v8::debug::Location>* locations) override {
v8::HandleScope scope(m_isolate); v8::HandleScope scope(m_isolate);
v8::Local<v8::debug::Script> script = m_script.Get(m_isolate); v8::Local<v8::debug::Script> script = m_script.Get(m_isolate);
...@@ -247,8 +250,8 @@ class WasmVirtualScript : public V8DebuggerScript { ...@@ -247,8 +250,8 @@ class WasmVirtualScript : public V8DebuggerScript {
scriptId(), v8ScriptId); scriptId(), v8ScriptId);
} }
bool success = script->GetPossibleBreakpoints(translatedStart, bool success = script->GetPossibleBreakpoints(
translatedEnd, locations); translatedStart, translatedEnd, restrictToFunction, locations);
for (v8::debug::Location& loc : *locations) { for (v8::debug::Location& loc : *locations) {
TranslateV8LocationToProtocolLocation(m_wasmTranslation, &loc, v8ScriptId, TranslateV8LocationToProtocolLocation(m_wasmTranslation, &loc, v8ScriptId,
scriptId()); scriptId());
......
...@@ -77,6 +77,7 @@ class V8DebuggerScript { ...@@ -77,6 +77,7 @@ class V8DebuggerScript {
virtual bool getPossibleBreakpoints( virtual bool getPossibleBreakpoints(
const v8::debug::Location& start, const v8::debug::Location& end, const v8::debug::Location& start, const v8::debug::Location& end,
bool ignoreNestedFunctions,
std::vector<v8::debug::Location>* locations) = 0; std::vector<v8::debug::Location>* locations) = 0;
virtual void resetBlackboxedStateCache() = 0; virtual void resetBlackboxedStateCache() = 0;
......
// Copyright 2016 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
print('Checks Debugger.getPossibleBreakpoints with ignoreNestedFunctions');
var source = `
function test() {
Array.from([1,2]).map(() => 1).filter(() => true);
function nested1() {
Array.from([1,2]).map(() => 1).filter(() => true);
}
function nested2() {
Array.from([1,2]).map(() => 1).filter(() => true);
}
nested1();
nested2();
}
//# sourceURL=test.js`;
InspectorTest.addScript(source);
var scriptId;
Protocol.Debugger.onceScriptParsed().then(message => {
if (message.params.url === 'test.js')
scriptId = message.params.scriptId;
}).then(() => InspectorTest.runTestSuite(tests));
InspectorTest.setupScriptMap();
Protocol.Debugger.onPaused(dumpBreakLocationInSourceAndResume);
Protocol.Debugger.enable();
var tests = [
function testWholeFunction(next) {
Protocol.Debugger.getPossibleBreakpoints({ start: location(1, 18), ignoreNestedFunctions: false })
.then(dumpAllLocations)
.then(next);
},
function testWholeFunctionWithoutNested(next) {
Protocol.Debugger.getPossibleBreakpoints({ start: location(1, 18), ignoreNestedFunctions: true })
.then(dumpAllLocations)
.then(next);
},
function testPartOfFunctionWithoutNested(next) {
Protocol.Debugger.getPossibleBreakpoints({ start: location(1, 18), end: location(2, 18), ignoreNestedFunctions: true })
.then(dumpAllLocations)
.then(next);
},
function testNestedFunction(next) {
Protocol.Debugger.getPossibleBreakpoints({ start: location(4, 0), ignoreNestedFunctions: true })
.then(dumpAllLocations)
.then(setAllBreakpoints)
.then(() => InspectorTest.log('Run test() to check breakpoints..'))
.then(() => Protocol.Runtime.evaluate({ expression: 'test()' }))
.then(next);
}
];
function location(lineNumber, columnNumber) {
return { lineNumber: lineNumber, columnNumber: columnNumber, scriptId: scriptId };
}
function setAllBreakpoints(message) {
var promises = [];
for (var location of message.result.locations)
promises.push(Protocol.Debugger.setBreakpoint({ location: location }).then(checkBreakpoint));
return Promise.all(promises);
}
function checkBreakpoint(message) {
if (message.error) {
InspectorTest.log('FAIL: error in setBreakpoint');
InspectorTest.logMessage(message);
return;
}
var id_data = message.result.breakpointId.split(':');
if (parseInt(id_data[1]) !== message.result.actualLocation.lineNumber || parseInt(id_data[2]) !== message.result.actualLocation.columnNumber) {
InspectorTest.log('FAIL: possible breakpoint was resolved in another location');
}
}
function dumpAllLocations(message) {
if (message.error) {
InspectorTest.logMessage(message);
return;
}
var sourceLines = source.split('\n')
var lineOffsets = Array(sourceLines.length).fill(0);
for (var location of message.result.locations) {
var lineNumber = location.lineNumber;
var columnNumber = location.columnNumber;
var line = sourceLines[lineNumber] || '';
var offset = lineOffsets[lineNumber];
line = line.slice(0, columnNumber + offset) + '#' + line.slice(columnNumber + offset);
++lineOffsets[lineNumber];
sourceLines[lineNumber] = line;
}
InspectorTest.log(sourceLines.join('\n'));
return message;
}
function dumpBreakLocationInSourceAndResume(message) {
InspectorTest.logCallFrames([ message.params.callFrames[0] ]);
var location = message.params.callFrames[0].location;
var sourceLines = source.split('\n')
var lineNumber = location.lineNumber
var columnNumber = location.columnNumber;
var line = sourceLines[lineNumber];
line = line.slice(0, columnNumber) + '^' + line.slice(columnNumber);
sourceLines[lineNumber] = line;
InspectorTest.log(sourceLines.join('\n'));
InspectorTest.log('');
Protocol.Debugger.resume();
}
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