7f58be6b38
If a WebAssembly function is exported, its js-to-wasm wrapper has a field which contains a reference to the WebAssembly function. Originally this reference was an index into the export table, which then contains an index into the function table, which then contains the metadata of the WebAssembly function. With this CL we use the index into the function table directly as the reference to the WebAssembly function. TEST=mjsunit/wasm/test-import-export-wrapper R=rossberg@chromium.org, mtrofin@chromium.org CC=titzer@chromium.org Review-Url: https://codereview.chromium.org/2472103002 Cr-Commit-Position: refs/heads/master@{#40729}
376 lines
9.7 KiB
JavaScript
376 lines
9.7 KiB
JavaScript
// Copyright 2015 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
|
|
|
|
function bytes() {
|
|
var buffer = new ArrayBuffer(arguments.length);
|
|
var view = new Uint8Array(buffer);
|
|
for (var i = 0; i < arguments.length; i++) {
|
|
var val = arguments[i];
|
|
if ((typeof val) == "string") val = val.charCodeAt(0);
|
|
view[i] = val | 0;
|
|
}
|
|
return buffer;
|
|
}
|
|
|
|
// Header declaration constants
|
|
var kWasmH0 = 0;
|
|
var kWasmH1 = 0x61;
|
|
var kWasmH2 = 0x73;
|
|
var kWasmH3 = 0x6d;
|
|
|
|
var kWasmV0 = 0xD;
|
|
var kWasmV1 = 0;
|
|
var kWasmV2 = 0;
|
|
var kWasmV3 = 0;
|
|
|
|
var kHeaderSize = 8;
|
|
var kPageSize = 65536;
|
|
|
|
function bytesWithHeader() {
|
|
var buffer = new ArrayBuffer(kHeaderSize + arguments.length);
|
|
var view = new Uint8Array(buffer);
|
|
view[0] = kWasmH0;
|
|
view[1] = kWasmH1;
|
|
view[2] = kWasmH2;
|
|
view[3] = kWasmH3;
|
|
view[4] = kWasmV0;
|
|
view[5] = kWasmV1;
|
|
view[6] = kWasmV2;
|
|
view[7] = kWasmV3;
|
|
for (var i = 0; i < arguments.length; i++) {
|
|
var val = arguments[i];
|
|
if ((typeof val) == "string") val = val.charCodeAt(0);
|
|
view[kHeaderSize + i] = val | 0;
|
|
}
|
|
return buffer;
|
|
}
|
|
|
|
let kDeclNoLocals = 0;
|
|
|
|
// Section declaration constants
|
|
let kUnknownSectionCode = 0;
|
|
let kTypeSectionCode = 1; // Function signature declarations
|
|
let kImportSectionCode = 2; // Import declarations
|
|
let kFunctionSectionCode = 3; // Function declarations
|
|
let kTableSectionCode = 4; // Indirect function table and other tables
|
|
let kMemorySectionCode = 5; // Memory attributes
|
|
let kGlobalSectionCode = 6; // Global declarations
|
|
let kExportSectionCode = 7; // Exports
|
|
let kStartSectionCode = 8; // Start function declaration
|
|
let kElementSectionCode = 9; // Elements section
|
|
let kCodeSectionCode = 10; // Function code
|
|
let kDataSectionCode = 11; // Data segments
|
|
let kNameSectionCode = 12; // Name section (encoded as string)
|
|
|
|
let kWasmFunctionTypeForm = 0x60;
|
|
let kWasmAnyFunctionTypeForm = 0x70;
|
|
|
|
let kResizableMaximumFlag = 1;
|
|
|
|
// Function declaration flags
|
|
let kDeclFunctionName = 0x01;
|
|
let kDeclFunctionImport = 0x02;
|
|
let kDeclFunctionLocals = 0x04;
|
|
let kDeclFunctionExport = 0x08;
|
|
|
|
// Local types
|
|
let kAstStmt = 0x40;
|
|
let kAstI32 = 0x7f;
|
|
let kAstI64 = 0x7e;
|
|
let kAstF32 = 0x7d;
|
|
let kAstF64 = 0x7c;
|
|
let kAstS128 = 0x7b;
|
|
|
|
let kExternalFunction = 0;
|
|
let kExternalTable = 1;
|
|
let kExternalMemory = 2;
|
|
let kExternalGlobal = 3;
|
|
|
|
let kTableZero = 0;
|
|
let kMemoryZero = 0;
|
|
|
|
// Useful signatures
|
|
let kSig_i_i = makeSig([kAstI32], [kAstI32]);
|
|
let kSig_l_l = makeSig([kAstI64], [kAstI64]);
|
|
let kSig_i_l = makeSig([kAstI64], [kAstI32]);
|
|
let kSig_i_ii = makeSig([kAstI32, kAstI32], [kAstI32]);
|
|
let kSig_i_iii = makeSig([kAstI32, kAstI32, kAstI32], [kAstI32]);
|
|
let kSig_d_dd = makeSig([kAstF64, kAstF64], [kAstF64]);
|
|
let kSig_l_ll = makeSig([kAstI64, kAstI64], [kAstI64]);
|
|
let kSig_i_dd = makeSig([kAstF64, kAstF64], [kAstI32]);
|
|
let kSig_v_v = makeSig([], []);
|
|
let kSig_i_v = makeSig([], [kAstI32]);
|
|
let kSig_l_v = makeSig([], [kAstI64]);
|
|
let kSig_f_v = makeSig([], [kAstF64]);
|
|
let kSig_d_v = makeSig([], [kAstF64]);
|
|
let kSig_v_i = makeSig([kAstI32], []);
|
|
let kSig_v_ii = makeSig([kAstI32, kAstI32], []);
|
|
let kSig_v_iii = makeSig([kAstI32, kAstI32, kAstI32], []);
|
|
let kSig_v_l = makeSig([kAstI64], []);
|
|
let kSig_v_d = makeSig([kAstF64], []);
|
|
let kSig_v_dd = makeSig([kAstF64, kAstF64], []);
|
|
let kSig_v_ddi = makeSig([kAstF64, kAstF64, kAstI32], []);
|
|
let kSig_s_v = makeSig([], [kAstS128]);
|
|
|
|
function makeSig(params, results) {
|
|
return {params: params, results: results};
|
|
}
|
|
|
|
function makeSig_v_x(x) {
|
|
return makeSig([x], []);
|
|
}
|
|
|
|
function makeSig_v_xx(x) {
|
|
return makeSig([x, x], []);
|
|
}
|
|
|
|
function makeSig_r_v(r) {
|
|
return makeSig([], [r]);
|
|
}
|
|
|
|
function makeSig_r_x(r, x) {
|
|
return makeSig([x], [r]);
|
|
}
|
|
|
|
function makeSig_r_xx(r, x) {
|
|
return makeSig([x, x], [r]);
|
|
}
|
|
|
|
// Opcodes
|
|
let kExprUnreachable = 0x00;
|
|
let kExprNop = 0x01;
|
|
let kExprBlock = 0x02;
|
|
let kExprLoop = 0x03;
|
|
let kExprIf = 0x04;
|
|
let kExprElse = 0x05;
|
|
let kExprTry = 0x06;
|
|
let kExprCatch = 0x07;
|
|
let kExprThrow = 0x08;
|
|
let kExprEnd = 0x0b;
|
|
let kExprBr = 0x0c;
|
|
let kExprBrIf = 0x0d;
|
|
let kExprBrTable = 0x0e;
|
|
let kExprReturn = 0x0f;
|
|
let kExprCallFunction = 0x10;
|
|
let kExprCallIndirect = 0x11;
|
|
let kExprDrop = 0x1a;
|
|
let kExprSelect = 0x1b;
|
|
let kExprGetLocal = 0x20;
|
|
let kExprSetLocal = 0x21;
|
|
let kExprTeeLocal = 0x22;
|
|
let kExprGetGlobal = 0x23;
|
|
let kExprSetGlobal = 0x24;
|
|
let kExprI32Const = 0x41;
|
|
let kExprI64Const = 0x42;
|
|
let kExprF32Const = 0x43;
|
|
let kExprF64Const = 0x44;
|
|
let kExprI8Const = 0xcb;
|
|
let kExprI32LoadMem = 0x28;
|
|
let kExprI64LoadMem = 0x29;
|
|
let kExprF32LoadMem = 0x2a;
|
|
let kExprF64LoadMem = 0x2b;
|
|
let kExprI32LoadMem8S = 0x2c;
|
|
let kExprI32LoadMem8U = 0x2d;
|
|
let kExprI32LoadMem16S = 0x2e;
|
|
let kExprI32LoadMem16U = 0x2f;
|
|
let kExprI64LoadMem8S = 0x30;
|
|
let kExprI64LoadMem8U = 0x31;
|
|
let kExprI64LoadMem16S = 0x32;
|
|
let kExprI64LoadMem16U = 0x33;
|
|
let kExprI64LoadMem32S = 0x34;
|
|
let kExprI64LoadMem32U = 0x35;
|
|
let kExprI32StoreMem = 0x36;
|
|
let kExprI64StoreMem = 0x37;
|
|
let kExprF32StoreMem = 0x38;
|
|
let kExprF64StoreMem = 0x39;
|
|
let kExprI32StoreMem8 = 0x3a;
|
|
let kExprI32StoreMem16 = 0x3b;
|
|
let kExprI64StoreMem8 = 0x3c;
|
|
let kExprI64StoreMem16 = 0x3d;
|
|
let kExprI64StoreMem32 = 0x3e;
|
|
let kExprMemorySize = 0x3f;
|
|
let kExprGrowMemory = 0x40;
|
|
let kExprI32Eqz = 0x45;
|
|
let kExprI32Eq = 0x46;
|
|
let kExprI32Ne = 0x47;
|
|
let kExprI32LtS = 0x48;
|
|
let kExprI32LtU = 0x49;
|
|
let kExprI32GtS = 0x4a;
|
|
let kExprI32GtU = 0x4b;
|
|
let kExprI32LeS = 0x4c;
|
|
let kExprI32LeU = 0x4d;
|
|
let kExprI32GeS = 0x4e;
|
|
let kExprI32GeU = 0x4f;
|
|
let kExprI64Eqz = 0x50;
|
|
let kExprI64Eq = 0x51;
|
|
let kExprI64Ne = 0x52;
|
|
let kExprI64LtS = 0x53;
|
|
let kExprI64LtU = 0x54;
|
|
let kExprI64GtS = 0x55;
|
|
let kExprI64GtU = 0x56;
|
|
let kExprI64LeS = 0x57;
|
|
let kExprI64LeU = 0x58;
|
|
let kExprI64GeS = 0x59;
|
|
let kExprI64GeU = 0x5a;
|
|
let kExprF32Eq = 0x5b;
|
|
let kExprF32Ne = 0x5c;
|
|
let kExprF32Lt = 0x5d;
|
|
let kExprF32Gt = 0x5e;
|
|
let kExprF32Le = 0x5f;
|
|
let kExprF32Ge = 0x60;
|
|
let kExprF64Eq = 0x61;
|
|
let kExprF64Ne = 0x62;
|
|
let kExprF64Lt = 0x63;
|
|
let kExprF64Gt = 0x64;
|
|
let kExprF64Le = 0x65;
|
|
let kExprF64Ge = 0x66;
|
|
let kExprI32Clz = 0x67;
|
|
let kExprI32Ctz = 0x68;
|
|
let kExprI32Popcnt = 0x69;
|
|
let kExprI32Add = 0x6a;
|
|
let kExprI32Sub = 0x6b;
|
|
let kExprI32Mul = 0x6c;
|
|
let kExprI32DivS = 0x6d;
|
|
let kExprI32DivU = 0x6e;
|
|
let kExprI32RemS = 0x6f;
|
|
let kExprI32RemU = 0x70;
|
|
let kExprI32And = 0x71;
|
|
let kExprI32Ior = 0x72;
|
|
let kExprI32Xor = 0x73;
|
|
let kExprI32Shl = 0x74;
|
|
let kExprI32ShrS = 0x75;
|
|
let kExprI32ShrU = 0x76;
|
|
let kExprI32Rol = 0x77;
|
|
let kExprI32Ror = 0x78;
|
|
let kExprI64Clz = 0x79;
|
|
let kExprI64Ctz = 0x7a;
|
|
let kExprI64Popcnt = 0x7b;
|
|
let kExprI64Add = 0x7c;
|
|
let kExprI64Sub = 0x7d;
|
|
let kExprI64Mul = 0x7e;
|
|
let kExprI64DivS = 0x7f;
|
|
let kExprI64DivU = 0x80;
|
|
let kExprI64RemS = 0x81;
|
|
let kExprI64RemU = 0x82;
|
|
let kExprI64And = 0x83;
|
|
let kExprI64Ior = 0x84;
|
|
let kExprI64Xor = 0x85;
|
|
let kExprI64Shl = 0x86;
|
|
let kExprI64ShrS = 0x87;
|
|
let kExprI64ShrU = 0x88;
|
|
let kExprI64Rol = 0x89;
|
|
let kExprI64Ror = 0x8a;
|
|
let kExprF32Abs = 0x8b;
|
|
let kExprF32Neg = 0x8c;
|
|
let kExprF32Ceil = 0x8d;
|
|
let kExprF32Floor = 0x8e;
|
|
let kExprF32Trunc = 0x8f;
|
|
let kExprF32NearestInt = 0x90;
|
|
let kExprF32Sqrt = 0x91;
|
|
let kExprF32Add = 0x92;
|
|
let kExprF32Sub = 0x93;
|
|
let kExprF32Mul = 0x94;
|
|
let kExprF32Div = 0x95;
|
|
let kExprF32Min = 0x96;
|
|
let kExprF32Max = 0x97;
|
|
let kExprF32CopySign = 0x98;
|
|
let kExprF64Abs = 0x99;
|
|
let kExprF64Neg = 0x9a;
|
|
let kExprF64Ceil = 0x9b;
|
|
let kExprF64Floor = 0x9c;
|
|
let kExprF64Trunc = 0x9d;
|
|
let kExprF64NearestInt = 0x9e;
|
|
let kExprF64Sqrt = 0x9f;
|
|
let kExprF64Add = 0xa0;
|
|
let kExprF64Sub = 0xa1;
|
|
let kExprF64Mul = 0xa2;
|
|
let kExprF64Div = 0xa3;
|
|
let kExprF64Min = 0xa4;
|
|
let kExprF64Max = 0xa5;
|
|
let kExprF64CopySign = 0xa6;
|
|
let kExprI32ConvertI64 = 0xa7;
|
|
let kExprI32SConvertF32 = 0xa8;
|
|
let kExprI32UConvertF32 = 0xa9;
|
|
let kExprI32SConvertF64 = 0xaa;
|
|
let kExprI32UConvertF64 = 0xab;
|
|
let kExprI64SConvertI32 = 0xac;
|
|
let kExprI64UConvertI32 = 0xad;
|
|
let kExprI64SConvertF32 = 0xae;
|
|
let kExprI64UConvertF32 = 0xaf;
|
|
let kExprI64SConvertF64 = 0xb0;
|
|
let kExprI64UConvertF64 = 0xb1;
|
|
let kExprF32SConvertI32 = 0xb2;
|
|
let kExprF32UConvertI32 = 0xb3;
|
|
let kExprF32SConvertI64 = 0xb4;
|
|
let kExprF32UConvertI64 = 0xb5;
|
|
let kExprF32ConvertF64 = 0xb6;
|
|
let kExprF64SConvertI32 = 0xb7;
|
|
let kExprF64UConvertI32 = 0xb8;
|
|
let kExprF64SConvertI64 = 0xb9;
|
|
let kExprF64UConvertI64 = 0xba;
|
|
let kExprF64ConvertF32 = 0xbb;
|
|
let kExprI32ReinterpretF32 = 0xbc;
|
|
let kExprI64ReinterpretF64 = 0xbd;
|
|
let kExprF32ReinterpretI32 = 0xbe;
|
|
let kExprF64ReinterpretI64 = 0xbf;
|
|
|
|
let kTrapUnreachable = 0;
|
|
let kTrapMemOutOfBounds = 1;
|
|
let kTrapDivByZero = 2;
|
|
let kTrapDivUnrepresentable = 3;
|
|
let kTrapRemByZero = 4;
|
|
let kTrapFloatUnrepresentable = 5;
|
|
let kTrapFuncInvalid = 6;
|
|
let kTrapFuncSigMismatch = 7;
|
|
let kTrapInvalidIndex = 8;
|
|
|
|
let kTrapMsgs = [
|
|
"unreachable",
|
|
"memory access out of bounds",
|
|
"divide by zero",
|
|
"divide result unrepresentable",
|
|
"remainder by zero",
|
|
"integer result unrepresentable",
|
|
"invalid function",
|
|
"function signature mismatch",
|
|
"invalid index into function table"
|
|
];
|
|
|
|
function assertTraps(trap, code) {
|
|
var threwException = true;
|
|
try {
|
|
if (typeof code === 'function') {
|
|
code();
|
|
} else {
|
|
eval(code);
|
|
}
|
|
threwException = false;
|
|
} catch (e) {
|
|
assertEquals("object", typeof e);
|
|
assertEquals(kTrapMsgs[trap], e.message);
|
|
// Success.
|
|
return;
|
|
}
|
|
throw new MjsUnitAssertionError("Did not trap, expected: " + kTrapMsgs[trap]);
|
|
}
|
|
|
|
function assertWasmThrows(value, code) {
|
|
assertEquals("number", typeof(value));
|
|
try {
|
|
if (typeof code === 'function') {
|
|
code();
|
|
} else {
|
|
eval(code);
|
|
}
|
|
} catch (e) {
|
|
assertEquals("number", typeof e);
|
|
assertEquals(value, e);
|
|
// Success.
|
|
return;
|
|
}
|
|
throw new MjsUnitAssertionError("Did not throw at all, expected: " + value);
|
|
}
|