Index: test/inspector/debugger/wasm-stepping.js |
diff --git a/test/inspector/debugger/wasm-stepping.js b/test/inspector/debugger/wasm-stepping.js |
new file mode 100644 |
index 0000000000000000000000000000000000000000..8cf14d361c048b43180efe597934fe6fcd6a8f1c |
--- /dev/null |
+++ b/test/inspector/debugger/wasm-stepping.js |
@@ -0,0 +1,160 @@ |
+// Copyright 2017 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. |
+ |
+load('test/mjsunit/wasm/wasm-constants.js'); |
+load('test/mjsunit/wasm/wasm-module-builder.js'); |
+ |
+var builder = new WasmModuleBuilder(); |
+ |
+var func_a_idx = |
+ builder.addFunction('wasm_A', kSig_v_v).addBody([kExprNop, kExprNop]).index; |
+ |
+// wasm_B calls wasm_A <param0> times. |
+builder.addFunction('wasm_B', kSig_v_i) |
+ .addBody([ |
+ // clang-format off |
+ kExprLoop, kWasmStmt, // while |
+ kExprGetLocal, 0, // - |
+ kExprIf, kWasmStmt, // if <param0> != 0 |
+ kExprGetLocal, 0, // - |
+ kExprI32Const, 1, // - |
+ kExprI32Sub, // - |
+ kExprSetLocal, 0, // decrease <param0> |
+ kExprCallFunction, func_a_idx, // - |
+ kExprBr, 1, // continue |
+ kExprEnd, // - |
+ kExprEnd, // break |
+ // clang-format on |
+ ]) |
+ .exportAs('main'); |
+ |
+var module_bytes = builder.toArray(); |
+ |
+function instantiate(bytes) { |
+ var buffer = new ArrayBuffer(bytes.length); |
+ var view = new Uint8Array(buffer); |
+ for (var i = 0; i < bytes.length; ++i) { |
+ view[i] = bytes[i] | 0; |
+ } |
+ |
+ var module = new WebAssembly.Module(buffer); |
+ // Set global variable. |
+ instance = new WebAssembly.Instance(module); |
+} |
+ |
+var evalWithUrl = (code, url) => Protocol.Runtime.evaluate( |
+ {'expression': code + '\n//# sourceURL=v8://test/' + url}); |
+ |
+Protocol.Debugger.onPaused(handlePaused); |
+var wasm_B_scriptId; |
+var step_actions = [ |
+ 'stepInto', // == stepOver, to call instruction |
+ 'stepInto', // into call to wasm_A |
+ 'stepOver', // over first nop |
+ 'stepOut', // out of wasm_A |
+ 'stepOut', // out of wasm_B, stop on breakpoint again |
+ 'stepOver', // to call |
+ 'stepOver', // over call |
+ 'resume', // to next breakpoint (third iteration) |
+ 'stepInto', // to call |
+ 'stepInto', // into wasm_A |
+ 'stepOut', // out to wasm_B |
+ // now step 9 times, until we are in wasm_A again. |
+ 'stepInto', 'stepInto', 'stepInto', 'stepInto', 'stepInto', 'stepInto', |
+ 'stepInto', 'stepInto', 'stepInto', |
+ // 3 more times, back to wasm_B. |
+ 'stepInto', 'stepInto', 'stepInto', |
+ // then just resume. |
+ 'resume' |
+]; |
+var sources = {}; |
+var urls = {}; |
+var afterTwoSourcesCallback; |
+ |
+Protocol.Debugger.enable() |
+ .then(() => InspectorTest.log('Installing code an global variable.')) |
+ .then( |
+ () => evalWithUrl('var instance;\n' + instantiate.toString(), 'setup')) |
+ .then(() => InspectorTest.log('Calling instantiate function.')) |
+ .then( |
+ () => |
+ (evalWithUrl( |
+ 'instantiate(' + JSON.stringify(module_bytes) + ')', |
+ 'callInstantiate'), |
+ 0)) |
+ .then(waitForTwoWasmScripts) |
+ .then( |
+ () => InspectorTest.log( |
+ 'Setting breakpoint on line 7 (on the setlocal before the call), url ' + |
+ urls[wasm_B_scriptId])) |
+ .then( |
+ () => Protocol.Debugger.setBreakpoint( |
+ {'location': {'scriptId': wasm_B_scriptId, 'lineNumber': 7}})) |
+ .then(printFailure) |
+ .then(msg => InspectorTest.logMessage(msg.result.actualLocation)) |
+ .then(() => evalWithUrl('instance.exports.main(4)', 'runWasm')) |
+ .then(() => InspectorTest.log('exports.main returned!')) |
+ .then(() => InspectorTest.log('Finished!')) |
+ .then(InspectorTest.completeTest); |
+ |
+function printFailure(message) { |
+ if (!message.result) { |
+ InspectorTest.logMessage(message); |
+ } |
+ return message; |
+} |
+ |
+function waitForTwoWasmScripts() { |
+ var num = 0; |
+ InspectorTest.log('Waiting for two wasm scripts to be parsed.'); |
+ var promise = new Promise(fulfill => gotBothSources = fulfill); |
+ function waitForMore() { |
+ if (num == 2) return promise; |
+ Protocol.Debugger.onceScriptParsed() |
+ .then(handleNewScript) |
+ .then(waitForMore); |
+ } |
+ function handleNewScript(msg) { |
+ var url = msg.params.url; |
+ if (!url.startsWith('wasm://')) { |
+ InspectorTest.log('Ignoring script with url ' + url); |
+ return; |
+ } |
+ num += 1; |
+ var scriptId = msg.params.scriptId; |
+ urls[scriptId] = url; |
+ InspectorTest.log('Got wasm script: ' + url); |
+ if (url.substr(-2) == '-1') wasm_B_scriptId = scriptId; |
+ InspectorTest.log('Requesting source for ' + url + '...'); |
+ Protocol.Debugger.getScriptSource({scriptId: scriptId}) |
+ .then(printFailure) |
+ .then(msg => sources[scriptId] = msg.result.scriptSource) |
+ .then(InspectorTest.log) |
+ .then(() => Object.keys(sources).length == 2 ? gotBothSources() : 0); |
+ } |
+ waitForMore(); |
+ return promise; |
+} |
+ |
+function printPauseLocation(scriptId, lineNr, columnNr) { |
+ var lines = sources[scriptId].split('\n'); |
+ var line = '<illegal line number>'; |
+ if (lineNr < lines.length) { |
+ line = lines[lineNr]; |
+ if (columnNr < line.length) { |
+ line = line.substr(0, columnNr) + '>' + line.substr(columnNr); |
+ } |
+ } |
+ InspectorTest.log( |
+ 'Paused at ' + urls[scriptId] + ':' + lineNr + ':' + columnNr + ': ' + |
+ line); |
+} |
+ |
+function handlePaused(msg) { |
+ var loc = msg.params.callFrames[0].location; |
+ printPauseLocation(loc.scriptId, loc.lineNumber, loc.columnNumber); |
+ var action = step_actions.shift(); |
+ InspectorTest.log('Step action: ' + action); |
+ Protocol.Debugger[action](); |
+} |