// 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. // Flags: --expose-wasm let {session, contextGroup, Protocol} = InspectorTest.start('Tests breakable locations in wasm'); utils.load('test/mjsunit/wasm/wasm-constants.js'); utils.load('test/mjsunit/wasm/wasm-module-builder.js'); var builder = new WasmModuleBuilder(); // clang-format off var func_idx = builder.addFunction('helper', kSig_v_v) .addLocals({i32_count: 1}) .addBody([ kExprNop, kExprI32Const, 12, kExprSetLocal, 0, ]).index; builder.addFunction('main', kSig_v_i) .addBody([ kExprGetLocal, 0, kExprIf, kWasmStmt, kExprBlock, kWasmStmt, kExprCallFunction, func_idx, kExprEnd, kExprEnd ]).exportAs('main'); // clang-format on var module_bytes = builder.toArray(); function testFunction(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}); var setupCode = testFunction.toString() + ';\nvar module_bytes = ' + JSON.stringify(module_bytes) + ';\nvar instance;'; Protocol.Debugger.enable(); Protocol.Debugger.onScriptParsed(handleScriptParsed); InspectorTest.log('Running testFunction...'); evalWithUrl(setupCode, 'setup') .then(() => evalWithUrl('testFunction(module_bytes)', 'runTestFunction')) .then(getBreakableLocationsForAllWasmScripts) .then(setAllBreakableLocations) .then(() => InspectorTest.log('Running wasm code...')) .then(() => (evalWithUrl('instance.exports.main(1)', 'runWasm'), 0)) .then(waitForAllPauses) .then(() => InspectorTest.log('Finished!')) .then(InspectorTest.completeTest); var allBreakableLocations = []; var urls = {}; var numScripts = 0; var wasmScripts = []; function handleScriptParsed(messageObject) { var scriptId = messageObject.params.scriptId; var url = messageObject.params.url; urls[scriptId] = url; InspectorTest.log('Script nr ' + numScripts + ' parsed. URL: ' + url); ++numScripts; if (url.startsWith('wasm://')) { InspectorTest.log('This is a wasm script (nr ' + wasmScripts.length + ').'); wasmScripts.push(scriptId); } } function printFailure(message) { if (!message.result) { InspectorTest.logMessage(message); } return message; } function printBreakableLocations(message, expectedScriptId, source) { var lines = source.split('\n'); var locations = message.result.locations; InspectorTest.log(locations.length + ' breakable location(s):'); for (var i = 0; i < locations.length; ++i) { if (locations[i].scriptId != expectedScriptId) { InspectorTest.log( 'SCRIPT ID MISMATCH!! ' + locations[i].scriptId + ' != ' + expectedScriptId); } var line = ''; if (locations[i].lineNumber < lines.length) { line = lines[locations[i].lineNumber]; if (locations[i].columnNumber < line.length) { line = line.substr(0, locations[i].columnNumber) + '>' + line.substr(locations[i].columnNumber); } } InspectorTest.log( '[' + i + '] ' + locations[i].lineNumber + ':' + locations[i].columnNumber + ' || ' + line); } } function checkGetBreakableLocations(wasmScriptNr) { InspectorTest.log( 'Requesting all breakable locations in wasm script ' + wasmScriptNr); var scriptId = wasmScripts[wasmScriptNr]; var source; return Protocol.Debugger.getScriptSource({scriptId: scriptId}) .then(msg => source = msg.result.scriptSource) .then( () => Protocol.Debugger.getPossibleBreakpoints( {start: {lineNumber: 0, columnNumber: 0, scriptId: scriptId}})) .then(printFailure) .then(msg => (allBreakableLocations.push(...msg.result.locations), msg)) .then(msg => printBreakableLocations(msg, scriptId, source)) .then( () => InspectorTest.log( 'Requesting breakable locations in lines [0,3)')) .then(() => Protocol.Debugger.getPossibleBreakpoints({ start: {lineNumber: 0, columnNumber: 0, scriptId: scriptId}, end: {lineNumber: 3, columnNumber: 0, scriptId: scriptId} })) .then(printFailure) .then(msg => printBreakableLocations(msg, scriptId, source)) .then( () => InspectorTest.log( 'Requesting breakable locations in lines [4,6)')) .then(() => Protocol.Debugger.getPossibleBreakpoints({ start: {lineNumber: 4, columnNumber: 0, scriptId: scriptId}, end: {lineNumber: 6, columnNumber: 0, scriptId: scriptId} })) .then(printFailure) .then(msg => printBreakableLocations(msg, scriptId, source)); } function getBreakableLocationsForAllWasmScripts() { InspectorTest.log('Querying breakable locations for all wasm scripts now...'); var promise = Promise.resolve(); for (var wasmScriptNr = 0; wasmScriptNr < wasmScripts.length; ++wasmScriptNr) { promise = promise.then(checkGetBreakableLocations.bind(null, wasmScriptNr)); } return promise; } function locationMatches(loc1, loc2) { return loc1.scriptId == loc2.scriptId && loc1.lineNumber == loc2.lineNumber && loc1.columnNumber == loc2.columnNumber; } function locationStr(loc) { return urls[loc.scriptId] + ':' + loc.lineNumber + ':' + loc.columnNumber; } function setBreakpoint(loc) { InspectorTest.log('Setting at ' + locationStr(loc)); function check(msg) { if (locationMatches(loc, msg.result.actualLocation)) { InspectorTest.log("Success!"); } else { InspectorTest.log("Mismatch!"); InspectorTest.logMessage(msg); } } return Protocol.Debugger.setBreakpoint({'location': loc}) .then(printFailure) .then(check); } function setAllBreakableLocations() { InspectorTest.log('Setting a breakpoint on each breakable location...'); var promise = Promise.resolve(); for (var loc of allBreakableLocations) { promise = promise.then(setBreakpoint.bind(null, loc)); } return promise; } function removePausedLocation(msg) { var topLocation = msg.params.callFrames[0].location; InspectorTest.log('Stopped at ' + locationStr(topLocation)); for (var i = 0; i < allBreakableLocations.length; ++i) { if (locationMatches(topLocation, allBreakableLocations[i])) { allBreakableLocations.splice(i, 1); --i; } } } function waitForAllPauses() { InspectorTest.log('Missing breakpoints: ' + allBreakableLocations.length); if (allBreakableLocations.length == 0) return; return Protocol.Debugger.oncePaused() .then(removePausedLocation) .then(Protocol.Debugger.resume()) .then(waitForAllPauses); }