d4a4d69cca
This adds a bunch of tracing hooks to the module decoder and uses them to support "annotated hexdump" output for full modules in wami: $ out/x64.release/wami my_module.wasm --full-hexdump Change-Id: I5821d940b5ec236df9708eecd0124172d8893ffd Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3754741 Reviewed-by: Manos Koukoutos <manoskouk@chromium.org> Commit-Queue: Jakob Kummerow <jkummerow@chromium.org> Cr-Commit-Position: refs/heads/main@{#81791}
203 lines
6.7 KiB
JavaScript
203 lines
6.7 KiB
JavaScript
// 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 --allow-natives-syntax
|
|
|
|
d8.file.execute("test/mjsunit/wasm/wasm-module-builder.js");
|
|
|
|
function builder() {
|
|
return new WasmModuleBuilder;
|
|
}
|
|
|
|
function assertCompileError(bytes, msg) {
|
|
assertThrows(
|
|
() => new WebAssembly.Module(bytes), WebAssembly.CompileError,
|
|
'WebAssembly.Module(): ' + msg);
|
|
assertThrowsAsync(
|
|
WebAssembly.compile(bytes), WebAssembly.CompileError,
|
|
'WebAssembly.compile(): ' + msg);
|
|
}
|
|
|
|
function assertInstantiateError(error, bytes, imports = {}, msg) {
|
|
assertThrows(
|
|
() => new WebAssembly.Instance(new WebAssembly.Module(bytes), imports),
|
|
error, 'WebAssembly.Instance(): ' + msg);
|
|
assertThrowsAsync(
|
|
WebAssembly.instantiate(bytes, imports), error,
|
|
'WebAssembly.instantiate(): ' + msg);
|
|
}
|
|
|
|
// default imports to {} so we get LinkError by default, thus allowing us to
|
|
// distinguish the TypeError we want to catch
|
|
function assertTypeError(bytes, imports = {}, msg) {
|
|
assertInstantiateError(TypeError, bytes, imports, msg);
|
|
}
|
|
|
|
function assertLinkError(bytes, imports, msg) {
|
|
assertInstantiateError(WebAssembly.LinkError, bytes, imports, msg);
|
|
}
|
|
|
|
function assertConversionError(bytes, imports, msg) {
|
|
let instance =
|
|
new WebAssembly.Instance(new WebAssembly.Module(bytes), imports);
|
|
assertThrows(() => instance.exports.run(), TypeError, msg);
|
|
}
|
|
|
|
(function TestDecodingError() {
|
|
print(arguments.callee.name);
|
|
assertCompileError(bytes(), 'BufferSource argument is empty');
|
|
assertCompileError(bytes('X'), 'expected 4 bytes, fell off end @+0');
|
|
assertCompileError(
|
|
bytes('\0x00asm'),
|
|
'expected magic word 00 61 73 6d, found 00 78 30 30 @+0');
|
|
})();
|
|
|
|
(function TestValidationError() {
|
|
print(arguments.callee.name);
|
|
let error = msg => 'Compiling function #0 failed: ' + msg;
|
|
let f_error = msg => 'Compiling function #0:"f" failed: ' + msg;
|
|
assertCompileError(
|
|
(function build() {
|
|
let b = builder();
|
|
b.addType(kSig_v_v);
|
|
// Use explicit section because the builder would automatically emit
|
|
// e.g. locals declarations.
|
|
// 1 function with type 0.
|
|
b.addExplicitSection([kFunctionSectionCode, 2, 1, 0]);
|
|
// 1 function body with length 0.
|
|
b.addExplicitSection([kCodeSectionCode, 2, 1, 0]);
|
|
return b.toBuffer();
|
|
})(),
|
|
error('expected local decls count @+22'));
|
|
assertCompileError(
|
|
builder().addFunction('f', kSig_i_v).end().toBuffer(),
|
|
f_error('function body must end with "end" opcode @+24'));
|
|
assertCompileError(
|
|
builder().addFunction('f', kSig_i_v).addBody([kExprReturn])
|
|
.end().toBuffer(),
|
|
f_error('expected 1 elements on the stack for return, found 0 @+24'));
|
|
assertCompileError(builder().addFunction('f', kSig_v_v).addBody([
|
|
kExprLocalGet, 0
|
|
]).end().toBuffer(), f_error('invalid local index: 0 @+24'));
|
|
assertCompileError(
|
|
builder().addStart(0).toBuffer(),
|
|
'function index 0 out of bounds (0 entries) @+10');
|
|
})();
|
|
|
|
function import_error(index, module, func, msg) {
|
|
let full_msg = 'Import #' + index + ' module=\"' + module + '\"';
|
|
if (func !== undefined) full_msg += ' function=\"' + func + '\"';
|
|
return full_msg + ' error: ' + msg;
|
|
}
|
|
|
|
(function TestTypeError() {
|
|
print(arguments.callee.name);
|
|
let b = builder();
|
|
b.addImport('foo', 'bar', kSig_v_v);
|
|
let msg =
|
|
import_error(0, 'foo', undefined, 'module is not an object or function');
|
|
assertTypeError(b.toBuffer(), {}, msg);
|
|
|
|
b = builder();
|
|
b.addImportedGlobal('foo', 'bar', kWasmI32);
|
|
assertTypeError(b.toBuffer(), {}, msg);
|
|
|
|
b = builder();
|
|
b.addImportedMemory('foo', 'bar');
|
|
assertTypeError(b.toBuffer(), {}, msg);
|
|
})();
|
|
|
|
(function TestLinkingError() {
|
|
print(arguments.callee.name);
|
|
let b;
|
|
let msg;
|
|
|
|
b = builder();
|
|
msg = import_error(0, 'foo', 'bar', 'function import requires a callable');
|
|
b.addImport('foo', 'bar', kSig_v_v);
|
|
assertLinkError(b.toBuffer(), {foo: {}}, msg);
|
|
b = builder();
|
|
b.addImport('foo', 'bar', kSig_v_v);
|
|
assertLinkError(b.toBuffer(), {foo: {bar: 9}}, msg);
|
|
|
|
b = builder();
|
|
msg = import_error(
|
|
0, 'foo', 'bar',
|
|
'global import must be a number, valid Wasm reference, '
|
|
+ 'or WebAssembly.Global object');
|
|
b.addImportedGlobal('foo', 'bar', kWasmI32);
|
|
assertLinkError(b.toBuffer(), {foo: {}}, msg);
|
|
b = builder();
|
|
b.addImportedGlobal('foo', 'bar', kWasmI32);
|
|
assertLinkError(b.toBuffer(), {foo: {bar: ''}}, msg);
|
|
b = builder();
|
|
b.addImportedGlobal('foo', 'bar', kWasmI32);
|
|
assertLinkError(b.toBuffer(), {foo: {bar: () => 9}}, msg);
|
|
|
|
b = builder();
|
|
msg = import_error(
|
|
0, 'foo', 'bar', 'memory import must be a WebAssembly.Memory object');
|
|
b.addImportedMemory('foo', 'bar');
|
|
assertLinkError(b.toBuffer(), {foo: {}}, msg);
|
|
b = builder();
|
|
b.addImportedMemory('foo', 'bar', 1);
|
|
assertLinkError(
|
|
b.toBuffer(), {foo: {bar: () => new WebAssembly.Memory({initial: 0})}},
|
|
msg);
|
|
})();
|
|
|
|
(function TestTrapUnreachable() {
|
|
print(arguments.callee.name);
|
|
let instance = builder().addFunction('run', kSig_v_v)
|
|
.addBody([kExprUnreachable]).exportFunc().end().instantiate();
|
|
assertTraps(kTrapUnreachable, instance.exports.run);
|
|
})();
|
|
|
|
(function TestTrapDivByZero() {
|
|
print(arguments.callee.name);
|
|
let instance = builder().addFunction('run', kSig_v_v).addBody(
|
|
[kExprI32Const, 1, kExprI32Const, 0, kExprI32DivS, kExprDrop])
|
|
.exportFunc().end().instantiate();
|
|
assertTraps(kTrapDivByZero, instance.exports.run);
|
|
})();
|
|
|
|
(function TestUnreachableInStart() {
|
|
print(arguments.callee.name);
|
|
|
|
let b = builder().addFunction("start", kSig_v_v).addBody(
|
|
[kExprUnreachable]).end().addStart(0);
|
|
assertTraps(kTrapUnreachable, () => b.instantiate());
|
|
})();
|
|
|
|
(function InternalDebugTrace() {
|
|
print(arguments.callee.name);
|
|
var builder = new WasmModuleBuilder();
|
|
var sig = builder.addType(kSig_i_dd);
|
|
builder.addImport("mod", "func", sig);
|
|
builder.addFunction("main", sig)
|
|
.addBody([kExprLocalGet, 0, kExprLocalGet, 1, kExprCallFunction, 0])
|
|
.exportAs("main");
|
|
var main = builder.instantiate({
|
|
mod: {
|
|
func: ()=>{%DebugTrace();}
|
|
}
|
|
}).exports.main;
|
|
main();
|
|
})();
|
|
|
|
(function TestMultipleCorruptFunctions() {
|
|
print(arguments.callee.name);
|
|
// Generate a module with multiple corrupt functions. The error message must
|
|
// be deterministic.
|
|
var builder = new WasmModuleBuilder();
|
|
var sig = builder.addType(kSig_v_v);
|
|
for (let i = 0; i < 10; ++i) {
|
|
builder.addFunction('f' + i, sig).addBody([kExprEnd]);
|
|
}
|
|
assertCompileError(
|
|
builder.toBuffer(),
|
|
'Compiling function #0:"f0" failed: ' +
|
|
'trailing code after function end @+33');
|
|
})();
|