UNPKG

chrome-devtools-frontend

Version:
1,436 lines (1,413 loc) • 67.4 kB
/* Copyright 2016 Mozilla Foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { BinaryReader, BinaryReaderState, bytesToString, ElementMode, ExternalKind, IDataSegmentBody, IElementSegment, IElementSegmentBody, IExportEntry, IFunctionEntry, IFunctionInformation, IFunctionNameEntry, IFunctionType, IGlobalNameEntry, IGlobalType, IGlobalVariable, IImportEntry, ILocalNameEntry, IMemoryAddress, IMemoryNameEntry, IMemoryType, INameEntry, Int64, IOperatorInformation, IResizableLimits, ISectionInformation, IStartEntry, isTypeIndex, ITableNameEntry, ITableType, ITypeNameEntry, NameType, OperatorCode, OperatorCodeNames, SectionCode, Type, } from "./WasmParser.js"; const NAME_SECTION_NAME = "name"; const INVALID_NAME_SYMBOLS_REGEX = /[^0-9A-Za-z!#$%&'*+.:<=>?@^_`|~\/\-]/; const INVALID_NAME_SYMBOLS_REGEX_GLOBAL = new RegExp( INVALID_NAME_SYMBOLS_REGEX.source, "g" ); function typeToString(type: number): string { switch (type) { case Type.i32: return "i32"; case Type.i64: return "i64"; case Type.f32: return "f32"; case Type.f64: return "f64"; case Type.v128: return "v128"; case Type.funcref: return "funcref"; case Type.externref: return "externref"; default: throw new Error(`Unexpected type ${type}`); } } function formatFloat32(n: number): string { if (n === 0) return 1 / n < 0 ? "-0.0" : "0.0"; if (isFinite(n)) return n.toString(); if (!isNaN(n)) return n < 0 ? "-inf" : "inf"; var view = new DataView(new ArrayBuffer(8)); view.setFloat32(0, n, true); var data = view.getInt32(0, true); var payload = data & 0x7fffff; const canonicalBits = 4194304; // 0x800..0 if (data > 0 && payload === canonicalBits) return "nan"; // canonical NaN; else if (payload === canonicalBits) return "-nan"; return (data < 0 ? "-" : "+") + "nan:0x" + payload.toString(16); } function formatFloat64(n: number): string { if (n === 0) return 1 / n < 0 ? "-0.0" : "0.0"; if (isFinite(n)) return n.toString(); if (!isNaN(n)) return n < 0 ? "-inf" : "inf"; var view = new DataView(new ArrayBuffer(8)); view.setFloat64(0, n, true); var data1 = view.getUint32(0, true); var data2 = view.getInt32(4, true); var payload = data1 + (data2 & 0xfffff) * 4294967296; const canonicalBits = 524288 * 4294967296; // 0x800..0 if (data2 > 0 && payload === canonicalBits) return "nan"; // canonical NaN; else if (payload === canonicalBits) return "-nan"; return (data2 < 0 ? "-" : "+") + "nan:0x" + payload.toString(16); } function formatI32Array(bytes, count) { var dv = new DataView(bytes.buffer, bytes.byteOffset, bytes.byteLength); var result = []; for (var i = 0; i < count; i++) result.push(`0x${formatHex(dv.getInt32(i << 2, true), 8)}`); return result.join(" "); } function formatI8Array(bytes, count) { var dv = new DataView(bytes.buffer, bytes.byteOffset, bytes.byteLength); var result = []; for (var i = 0; i < count; i++) result.push(`${dv.getInt8(i)}`); return result.join(" "); } function memoryAddressToString( address: IMemoryAddress, code: OperatorCode ): string { var defaultAlignFlags; switch (code) { case OperatorCode.v128_load: case OperatorCode.i16x8_load8x8_s: case OperatorCode.i16x8_load8x8_u: case OperatorCode.i32x4_load16x4_s: case OperatorCode.i32x4_load16x4_u: case OperatorCode.i64x2_load32x2_s: case OperatorCode.i64x2_load32x2_u: case OperatorCode.v8x16_load_splat: case OperatorCode.v16x8_load_splat: case OperatorCode.v32x4_load_splat: case OperatorCode.v64x2_load_splat: case OperatorCode.v128_store: defaultAlignFlags = 4; break; case OperatorCode.i64_load: case OperatorCode.i64_store: case OperatorCode.f64_load: case OperatorCode.f64_store: case OperatorCode.i64_atomic_wait: case OperatorCode.i64_atomic_load: case OperatorCode.i64_atomic_store: case OperatorCode.i64_atomic_rmw_add: case OperatorCode.i64_atomic_rmw_sub: case OperatorCode.i64_atomic_rmw_and: case OperatorCode.i64_atomic_rmw_or: case OperatorCode.i64_atomic_rmw_xor: case OperatorCode.i64_atomic_rmw_xchg: case OperatorCode.i64_atomic_rmw_cmpxchg: case OperatorCode.v128_load64_zero: defaultAlignFlags = 3; break; case OperatorCode.i32_load: case OperatorCode.i64_load32_s: case OperatorCode.i64_load32_u: case OperatorCode.i32_store: case OperatorCode.i64_store32: case OperatorCode.f32_load: case OperatorCode.f32_store: case OperatorCode.atomic_notify: case OperatorCode.i32_atomic_wait: case OperatorCode.i32_atomic_load: case OperatorCode.i64_atomic_load32_u: case OperatorCode.i32_atomic_store: case OperatorCode.i64_atomic_store32: case OperatorCode.i32_atomic_rmw_add: case OperatorCode.i64_atomic_rmw32_add_u: case OperatorCode.i32_atomic_rmw_sub: case OperatorCode.i64_atomic_rmw32_sub_u: case OperatorCode.i32_atomic_rmw_and: case OperatorCode.i64_atomic_rmw32_and_u: case OperatorCode.i32_atomic_rmw_or: case OperatorCode.i64_atomic_rmw32_or_u: case OperatorCode.i32_atomic_rmw_xor: case OperatorCode.i64_atomic_rmw32_xor_u: case OperatorCode.i32_atomic_rmw_xchg: case OperatorCode.i64_atomic_rmw32_xchg_u: case OperatorCode.i32_atomic_rmw_cmpxchg: case OperatorCode.i64_atomic_rmw32_cmpxchg_u: case OperatorCode.v128_load32_zero: defaultAlignFlags = 2; break; case OperatorCode.i32_load16_s: case OperatorCode.i32_load16_u: case OperatorCode.i64_load16_s: case OperatorCode.i64_load16_u: case OperatorCode.i32_store16: case OperatorCode.i64_store16: case OperatorCode.i32_atomic_load16_u: case OperatorCode.i64_atomic_load16_u: case OperatorCode.i32_atomic_store16: case OperatorCode.i64_atomic_store16: case OperatorCode.i32_atomic_rmw16_add_u: case OperatorCode.i64_atomic_rmw16_add_u: case OperatorCode.i32_atomic_rmw16_sub_u: case OperatorCode.i64_atomic_rmw16_sub_u: case OperatorCode.i32_atomic_rmw16_and_u: case OperatorCode.i64_atomic_rmw16_and_u: case OperatorCode.i32_atomic_rmw16_or_u: case OperatorCode.i64_atomic_rmw16_or_u: case OperatorCode.i32_atomic_rmw16_xor_u: case OperatorCode.i64_atomic_rmw16_xor_u: case OperatorCode.i32_atomic_rmw16_xchg_u: case OperatorCode.i64_atomic_rmw16_xchg_u: case OperatorCode.i32_atomic_rmw16_cmpxchg_u: case OperatorCode.i64_atomic_rmw16_cmpxchg_u: defaultAlignFlags = 1; break; case OperatorCode.i32_load8_s: case OperatorCode.i32_load8_u: case OperatorCode.i64_load8_s: case OperatorCode.i64_load8_u: case OperatorCode.i32_store8: case OperatorCode.i64_store8: case OperatorCode.i32_atomic_load8_u: case OperatorCode.i64_atomic_load8_u: case OperatorCode.i32_atomic_store8: case OperatorCode.i64_atomic_store8: case OperatorCode.i32_atomic_rmw8_add_u: case OperatorCode.i64_atomic_rmw8_add_u: case OperatorCode.i32_atomic_rmw8_sub_u: case OperatorCode.i64_atomic_rmw8_sub_u: case OperatorCode.i32_atomic_rmw8_and_u: case OperatorCode.i64_atomic_rmw8_and_u: case OperatorCode.i32_atomic_rmw8_or_u: case OperatorCode.i64_atomic_rmw8_or_u: case OperatorCode.i32_atomic_rmw8_xor_u: case OperatorCode.i64_atomic_rmw8_xor_u: case OperatorCode.i32_atomic_rmw8_xchg_u: case OperatorCode.i64_atomic_rmw8_xchg_u: case OperatorCode.i32_atomic_rmw8_cmpxchg_u: case OperatorCode.i64_atomic_rmw8_cmpxchg_u: defaultAlignFlags = 0; break; } if (address.flags == defaultAlignFlags) // hide default flags return !address.offset ? null : `offset=${address.offset}`; if (!address.offset) // hide default offset return `align=${1 << address.flags}`; return `offset=${address.offset | 0} align=${1 << address.flags}`; } function globalTypeToString(type: IGlobalType): string { const typeStr = typeToString(type.contentType); return type.mutability ? `(mut ${typeStr})` : typeStr; } function limitsToString(limits: IResizableLimits): string { return ( limits.initial + (limits.maximum !== undefined ? " " + limits.maximum : "") ); } var paddingCache = ["0", "00", "000"]; function formatHex(n: number, width?: number): string { var s = (n >>> 0).toString(16).toUpperCase(); if (width === undefined || s.length >= width) return s; var paddingIndex = width - s.length - 1; while (paddingIndex >= paddingCache.length) paddingCache.push(paddingCache[paddingCache.length - 1] + "0"); return paddingCache[paddingIndex] + s; } const IndentIncrement = " "; function isValidName(name: string) { return !INVALID_NAME_SYMBOLS_REGEX.test(name); } export interface IExportMetadata { getFunctionExportNames(index: number): string[]; getGlobalExportNames(index: number): string[]; getMemoryExportNames(index: number): string[]; getTableExportNames(index: number): string[]; } export interface INameResolver { getTypeName(index: number, isRef: boolean): string; getTableName(index: number, isRef: boolean): string; getMemoryName(index: number, isRef: boolean): string; getGlobalName(index: number, isRef: boolean): string; getElementName(index: number, isRef: boolean): string; getFunctionName(index: number, isImport: boolean, isRef: boolean): string; getVariableName(funcIndex: number, index: number, isRef: boolean): string; getLabel(index: number): string; } export class DefaultNameResolver implements INameResolver { public getTypeName(index: number, isRef: boolean): string { return "$type" + index; } public getTableName(index: number, isRef: boolean): string { return "$table" + index; } public getMemoryName(index: number, isRef: boolean): string { return "$memory" + index; } public getGlobalName(index: number, isRef: boolean): string { return "$global" + index; } public getElementName(index: number, isRef: boolean): string { return `$elem${index}`; } public getFunctionName( index: number, isImport: boolean, isRef: boolean ): string { return (isImport ? "$import" : "$func") + index; } public getVariableName( funcIndex: number, index: number, isRef: boolean ): string { return "$var" + index; } public getLabel(index: number): string { return "$label" + index; } } const EMPTY_STRING_ARRAY: string[] = []; class DevToolsExportMetadata implements IExportMetadata { private readonly _functionExportNames: string[][]; private readonly _globalExportNames: string[][]; private readonly _memoryExportNames: string[][]; private readonly _tableExportNames: string[][]; constructor( functionExportNames: string[][], globalExportNames: string[][], memoryExportNames: string[][], tableExportNames: string[][] ) { this._functionExportNames = functionExportNames; this._globalExportNames = globalExportNames; this._memoryExportNames = memoryExportNames; this._tableExportNames = tableExportNames; } public getFunctionExportNames(index: number) { return this._functionExportNames[index] ?? EMPTY_STRING_ARRAY; } public getGlobalExportNames(index: number) { return this._globalExportNames[index] ?? EMPTY_STRING_ARRAY; } public getMemoryExportNames(index: number) { return this._memoryExportNames[index] ?? EMPTY_STRING_ARRAY; } public getTableExportNames(index: number) { return this._tableExportNames[index] ?? EMPTY_STRING_ARRAY; } } export class NumericNameResolver implements INameResolver { public getTypeName(index: number, isRef: boolean): string { return isRef ? "" + index : `(;${index};)`; } public getTableName(index: number, isRef: boolean): string { return isRef ? "" + index : `(;${index};)`; } public getMemoryName(index: number, isRef: boolean): string { return isRef ? "" + index : `(;${index};)`; } public getGlobalName(index: number, isRef: boolean): string { return isRef ? "" + index : `(;${index};)`; } public getElementName(index: number, isRef: boolean): string { return isRef ? "" + index : `(;${index};)`; } public getFunctionName( index: number, isImport: boolean, isRef: boolean ): string { return isRef ? "" + index : `(;${index};)`; } public getVariableName( funcIndex: number, index: number, isRef: boolean ): string { return isRef ? "" + index : `(;${index};)`; } public getLabel(index: number): string { return null; } } export enum LabelMode { Depth, WhenUsed, Always, } // The breakable range is [start, end). export interface IFunctionBodyOffset { start: number; end: number; } export interface IDisassemblerResult { lines: Array<string>; offsets?: Array<number>; done: boolean; functionBodyOffsets?: Array<IFunctionBodyOffset>; } export class WasmDisassembler { private _lines: Array<string>; private _offsets: Array<number>; private _buffer: string; private _types: Array<IFunctionType>; private _funcIndex: number; private _funcTypes: Array<number>; private _importCount: number; private _globalCount: number; private _memoryCount: number; private _tableCount: number; private _elementCount: number; private _expression: Array<IOperatorInformation>; private _backrefLabels: Array<{ line: number; position: number; useLabel: boolean; label: string; }>; private _labelIndex: number; private _indent: string; private _indentLevel: number; private _addOffsets: boolean; private _skipTypes = true; private _done: boolean; private _currentPosition: number; private _nameResolver: INameResolver; private _exportMetadata: IExportMetadata = null; private _labelMode: LabelMode; private _functionBodyOffsets: Array<IFunctionBodyOffset>; private _currentFunctionBodyOffset: number; private _currentSectionId: SectionCode; private _logFirstInstruction: boolean; constructor() { this._lines = []; this._offsets = []; this._buffer = ""; this._indent = null; this._indentLevel = 0; this._addOffsets = false; this._done = false; this._currentPosition = 0; this._nameResolver = new DefaultNameResolver(); this._labelMode = LabelMode.WhenUsed; this._functionBodyOffsets = []; this._currentFunctionBodyOffset = 0; this._currentSectionId = SectionCode.Unknown; this._logFirstInstruction = false; this._reset(); } private _reset(): void { this._types = []; this._funcIndex = 0; this._funcTypes = []; this._importCount = 0; this._globalCount = 0; this._memoryCount = 0; this._tableCount = 0; this._elementCount = 0; this._expression = []; this._backrefLabels = null; this._labelIndex = 0; } public get addOffsets(): boolean { return this._addOffsets; } public set addOffsets(value: boolean) { if (this._currentPosition) throw new Error("Cannot switch addOffsets during processing."); this._addOffsets = value; } public get skipTypes(): boolean { return this._skipTypes; } public set skipTypes(skipTypes: boolean) { if (this._currentPosition) throw new Error("Cannot switch skipTypes during processing."); this._skipTypes = skipTypes; } public get labelMode(): LabelMode { return this._labelMode; } public set labelMode(value: LabelMode) { if (this._currentPosition) throw new Error("Cannot switch labelMode during processing."); this._labelMode = value; } public get exportMetadata(): IExportMetadata { return this._exportMetadata; } public set exportMetadata(exportMetadata: IExportMetadata) { if (this._currentPosition) throw new Error("Cannot switch exportMetadata during processing."); this._exportMetadata = exportMetadata; } public get nameResolver(): INameResolver { return this._nameResolver; } public set nameResolver(resolver: INameResolver) { if (this._currentPosition) throw new Error("Cannot switch nameResolver during processing."); this._nameResolver = resolver; } private appendBuffer(s: string) { this._buffer += s; } private newLine() { if (this.addOffsets) this._offsets.push(this._currentPosition); this._lines.push(this._buffer); this._buffer = ""; } private logStartOfFunctionBodyOffset() { if (this.addOffsets) { this._currentFunctionBodyOffset = this._currentPosition; } } private logEndOfFunctionBodyOffset() { if (this.addOffsets) { this._functionBodyOffsets.push({ start: this._currentFunctionBodyOffset, end: this._currentPosition, }); } } private printFuncType(typeIndex: number): void { var type = this._types[typeIndex]; if (type.form !== Type.func) throw new Error("NYI other function form"); if (type.params.length > 0) { this.appendBuffer(" (param"); for (var i = 0; i < type.params.length; i++) { this.appendBuffer(" "); this.appendBuffer(typeToString(type.params[i])); } this.appendBuffer(")"); } if (type.returns.length > 0) { this.appendBuffer(" (result"); for (var i = 0; i < type.returns.length; i++) { this.appendBuffer(" "); this.appendBuffer(typeToString(type.returns[i])); } this.appendBuffer(")"); } } private printBlockType(type: number): void { if (type === Type.empty_block_type) { return; } if (isTypeIndex(type)) { return this.printFuncType(type); } this.appendBuffer(" (result "); this.appendBuffer(typeToString(type)); this.appendBuffer(")"); } private printString(b: Uint8Array): void { this.appendBuffer('"'); for (var i = 0; i < b.length; i++) { var byte = b[i]; if ( byte < 0x20 || byte >= 0x7f || byte == /* " */ 0x22 || byte == /* \ */ 0x5c ) { this.appendBuffer( "\\" + (byte >> 4).toString(16) + (byte & 15).toString(16) ); } else { this.appendBuffer(String.fromCharCode(byte)); } } this.appendBuffer('"'); } private printExpression(expression: IOperatorInformation[]): void { for (const operator of expression) { this.appendBuffer("("); this.printOperator(operator); this.appendBuffer(")"); } } private useLabel(depth: number): string { if (!this._backrefLabels) { return "" + depth; } var i = this._backrefLabels.length - depth - 1; if (i < 0) { return "" + depth; } var backrefLabel = this._backrefLabels[i]; if (!backrefLabel.useLabel) { backrefLabel.useLabel = true; backrefLabel.label = this._nameResolver.getLabel(this._labelIndex); var line = this._lines[backrefLabel.line]; this._lines[backrefLabel.line] = line.substring(0, backrefLabel.position) + " " + backrefLabel.label + line.substring(backrefLabel.position); this._labelIndex++; } return backrefLabel.label || "" + depth; } private printOperator(operator: IOperatorInformation): void { var code = operator.code; this.appendBuffer(OperatorCodeNames[code]); switch (code) { case OperatorCode.block: case OperatorCode.loop: case OperatorCode.if: if (this._labelMode !== LabelMode.Depth) { const backrefLabel = { line: this._lines.length, position: this._buffer.length, useLabel: false, label: null, }; if (this._labelMode === LabelMode.Always) { backrefLabel.useLabel = true; backrefLabel.label = this._nameResolver.getLabel( this._labelIndex++ ); if (backrefLabel.label) { this.appendBuffer(" "); this.appendBuffer(backrefLabel.label); } } this._backrefLabels.push(backrefLabel); } this.printBlockType(operator.blockType); break; case OperatorCode.end: if (this._labelMode === LabelMode.Depth) { break; } const backrefLabel = this._backrefLabels.pop(); if (backrefLabel.label) { this.appendBuffer(" "); this.appendBuffer(backrefLabel.label); } break; case OperatorCode.br: case OperatorCode.br_if: this.appendBuffer(" "); this.appendBuffer(this.useLabel(operator.brDepth)); break; case OperatorCode.br_table: for (var i = 0; i < operator.brTable.length; i++) { this.appendBuffer(" "); this.appendBuffer(this.useLabel(operator.brTable[i])); } break; case OperatorCode.ref_null: switch (operator.refType) { case Type.funcref: this.appendBuffer(" func"); break; case Type.externref: this.appendBuffer(" extern"); break; default: throw new Error(`Unknown refedtype ${operator.refType}`); } break; case OperatorCode.call: case OperatorCode.return_call: case OperatorCode.ref_func: var funcName = this._nameResolver.getFunctionName( operator.funcIndex, operator.funcIndex < this._importCount, true ); this.appendBuffer(` ${funcName}`); break; case OperatorCode.call_indirect: case OperatorCode.return_call_indirect: this.printFuncType(operator.typeIndex); break; case OperatorCode.local_get: case OperatorCode.local_set: case OperatorCode.local_tee: var paramName = this._nameResolver.getVariableName( this._funcIndex, operator.localIndex, true ); this.appendBuffer(` ${paramName}`); break; case OperatorCode.global_get: case OperatorCode.global_set: var globalName = this._nameResolver.getGlobalName( operator.globalIndex, true ); this.appendBuffer(` ${globalName}`); break; case OperatorCode.i32_load: case OperatorCode.i64_load: case OperatorCode.f32_load: case OperatorCode.f64_load: case OperatorCode.i32_load8_s: case OperatorCode.i32_load8_u: case OperatorCode.i32_load16_s: case OperatorCode.i32_load16_u: case OperatorCode.i64_load8_s: case OperatorCode.i64_load8_u: case OperatorCode.i64_load16_s: case OperatorCode.i64_load16_u: case OperatorCode.i64_load32_s: case OperatorCode.i64_load32_u: case OperatorCode.i32_store: case OperatorCode.i64_store: case OperatorCode.f32_store: case OperatorCode.f64_store: case OperatorCode.i32_store8: case OperatorCode.i32_store16: case OperatorCode.i64_store8: case OperatorCode.i64_store16: case OperatorCode.i64_store32: case OperatorCode.atomic_notify: case OperatorCode.i32_atomic_wait: case OperatorCode.i64_atomic_wait: case OperatorCode.i32_atomic_load: case OperatorCode.i64_atomic_load: case OperatorCode.i32_atomic_load8_u: case OperatorCode.i32_atomic_load16_u: case OperatorCode.i64_atomic_load8_u: case OperatorCode.i64_atomic_load16_u: case OperatorCode.i64_atomic_load32_u: case OperatorCode.i32_atomic_store: case OperatorCode.i64_atomic_store: case OperatorCode.i32_atomic_store8: case OperatorCode.i32_atomic_store16: case OperatorCode.i64_atomic_store8: case OperatorCode.i64_atomic_store16: case OperatorCode.i64_atomic_store32: case OperatorCode.i32_atomic_rmw_add: case OperatorCode.i64_atomic_rmw_add: case OperatorCode.i32_atomic_rmw8_add_u: case OperatorCode.i32_atomic_rmw16_add_u: case OperatorCode.i64_atomic_rmw8_add_u: case OperatorCode.i64_atomic_rmw16_add_u: case OperatorCode.i64_atomic_rmw32_add_u: case OperatorCode.i32_atomic_rmw_sub: case OperatorCode.i64_atomic_rmw_sub: case OperatorCode.i32_atomic_rmw8_sub_u: case OperatorCode.i32_atomic_rmw16_sub_u: case OperatorCode.i64_atomic_rmw8_sub_u: case OperatorCode.i64_atomic_rmw16_sub_u: case OperatorCode.i64_atomic_rmw32_sub_u: case OperatorCode.i32_atomic_rmw_and: case OperatorCode.i64_atomic_rmw_and: case OperatorCode.i32_atomic_rmw8_and_u: case OperatorCode.i32_atomic_rmw16_and_u: case OperatorCode.i64_atomic_rmw8_and_u: case OperatorCode.i64_atomic_rmw16_and_u: case OperatorCode.i64_atomic_rmw32_and_u: case OperatorCode.i32_atomic_rmw_or: case OperatorCode.i64_atomic_rmw_or: case OperatorCode.i32_atomic_rmw8_or_u: case OperatorCode.i32_atomic_rmw16_or_u: case OperatorCode.i64_atomic_rmw8_or_u: case OperatorCode.i64_atomic_rmw16_or_u: case OperatorCode.i64_atomic_rmw32_or_u: case OperatorCode.i32_atomic_rmw_xor: case OperatorCode.i64_atomic_rmw_xor: case OperatorCode.i32_atomic_rmw8_xor_u: case OperatorCode.i32_atomic_rmw16_xor_u: case OperatorCode.i64_atomic_rmw8_xor_u: case OperatorCode.i64_atomic_rmw16_xor_u: case OperatorCode.i64_atomic_rmw32_xor_u: case OperatorCode.i32_atomic_rmw_xchg: case OperatorCode.i64_atomic_rmw_xchg: case OperatorCode.i32_atomic_rmw8_xchg_u: case OperatorCode.i32_atomic_rmw16_xchg_u: case OperatorCode.i64_atomic_rmw8_xchg_u: case OperatorCode.i64_atomic_rmw16_xchg_u: case OperatorCode.i64_atomic_rmw32_xchg_u: case OperatorCode.i32_atomic_rmw_cmpxchg: case OperatorCode.i64_atomic_rmw_cmpxchg: case OperatorCode.i32_atomic_rmw8_cmpxchg_u: case OperatorCode.i32_atomic_rmw16_cmpxchg_u: case OperatorCode.i64_atomic_rmw8_cmpxchg_u: case OperatorCode.i64_atomic_rmw16_cmpxchg_u: case OperatorCode.i64_atomic_rmw32_cmpxchg_u: case OperatorCode.v128_load: case OperatorCode.i16x8_load8x8_s: case OperatorCode.i16x8_load8x8_u: case OperatorCode.i32x4_load16x4_s: case OperatorCode.i32x4_load16x4_u: case OperatorCode.i64x2_load32x2_s: case OperatorCode.i64x2_load32x2_u: case OperatorCode.v8x16_load_splat: case OperatorCode.v16x8_load_splat: case OperatorCode.v32x4_load_splat: case OperatorCode.v64x2_load_splat: case OperatorCode.v128_store: case OperatorCode.v128_load32_zero: case OperatorCode.v128_load64_zero: var memoryAddress = memoryAddressToString( operator.memoryAddress, operator.code ); if (memoryAddress !== null) { this.appendBuffer(" "); this.appendBuffer(memoryAddress); } break; case OperatorCode.current_memory: case OperatorCode.grow_memory: break; case OperatorCode.i32_const: this.appendBuffer(` ${(<number>operator.literal).toString()}`); break; case OperatorCode.i64_const: this.appendBuffer(` ${(<Int64>operator.literal).toString()}`); break; case OperatorCode.f32_const: this.appendBuffer(` ${formatFloat32(<number>operator.literal)}`); break; case OperatorCode.f64_const: this.appendBuffer(` ${formatFloat64(<number>operator.literal)}`); break; case OperatorCode.v128_const: this.appendBuffer(` i32x4 ${formatI32Array(operator.literal, 4)}`); break; case OperatorCode.i8x16_shuffle: this.appendBuffer(` ${formatI8Array(operator.lines, 16)}`); break; case OperatorCode.i8x16_extract_lane_s: case OperatorCode.i8x16_extract_lane_u: case OperatorCode.i8x16_replace_lane: case OperatorCode.i16x8_extract_lane_s: case OperatorCode.i16x8_extract_lane_u: case OperatorCode.i16x8_replace_lane: case OperatorCode.i32x4_extract_lane: case OperatorCode.i32x4_replace_lane: case OperatorCode.f32x4_extract_lane: case OperatorCode.f32x4_replace_lane: case OperatorCode.i64x2_extract_lane: case OperatorCode.i64x2_replace_lane: case OperatorCode.f64x2_extract_lane: case OperatorCode.f64x2_replace_lane: this.appendBuffer(` ${operator.lineIndex}`); break; case OperatorCode.memory_init: case OperatorCode.data_drop: this.appendBuffer(` ${operator.segmentIndex}`); break; case OperatorCode.elem_drop: const elementName = this._nameResolver.getElementName( operator.segmentIndex, true ); this.appendBuffer(` ${elementName}`); break; case OperatorCode.table_set: case OperatorCode.table_get: case OperatorCode.table_fill: { const tableName = this._nameResolver.getTableName( operator.tableIndex, true ); this.appendBuffer(` ${tableName}`); break; } case OperatorCode.table_copy: { // Table index might be omitted and defaults to 0. if (operator.tableIndex !== 0 || operator.destinationIndex !== 0) { const tableName = this._nameResolver.getTableName( operator.tableIndex, true ); const destinationName = this._nameResolver.getTableName( operator.destinationIndex, true ); this.appendBuffer(` ${destinationName} ${tableName}`); } break; } case OperatorCode.table_init: { // Table index might be omitted and defaults to 0. if (operator.tableIndex !== 0) { const tableName = this._nameResolver.getTableName( operator.tableIndex, true ); this.appendBuffer(` ${tableName}`); } const elementName = this._nameResolver.getElementName( operator.segmentIndex, true ); this.appendBuffer(` ${elementName}`); break; } } } private printImportSource(info: IImportEntry): void { this.printString(info.module); this.appendBuffer(" "); this.printString(info.field); } private increaseIndent(): void { this._indent += IndentIncrement; this._indentLevel++; } private decreaseIndent(): void { this._indent = this._indent.slice(0, -IndentIncrement.length); this._indentLevel--; } public disassemble(reader: BinaryReader): string { const done = this.disassembleChunk(reader); if (!done) return null; let lines = this._lines; if (this._addOffsets) { lines = lines.map((line, index) => { var position = formatHex(this._offsets[index], 4); return line + " ;; @" + position; }); } lines.push(""); // we need '\n' after last line const result = lines.join("\n"); this._lines.length = 0; this._offsets.length = 0; this._functionBodyOffsets.length = 0; return result; } public getResult(): IDisassemblerResult { let linesReady = this._lines.length; if (this._backrefLabels && this._labelMode === LabelMode.WhenUsed) { this._backrefLabels.some((backrefLabel) => { if (backrefLabel.useLabel) return false; linesReady = backrefLabel.line; return true; }); } if (linesReady === 0) { return { lines: [], offsets: this._addOffsets ? [] : undefined, done: this._done, functionBodyOffsets: this._addOffsets ? [] : undefined, }; } if (linesReady === this._lines.length) { const result = { lines: this._lines, offsets: this._addOffsets ? this._offsets : undefined, done: this._done, functionBodyOffsets: this._addOffsets ? this._functionBodyOffsets : undefined, }; this._lines = []; if (this._addOffsets) { this._offsets = []; this._functionBodyOffsets = []; } return result; } const result = { lines: this._lines.splice(0, linesReady), offsets: this._addOffsets ? this._offsets.splice(0, linesReady) : undefined, done: false, functionBodyOffsets: this._addOffsets ? this._functionBodyOffsets : undefined, }; if (this._backrefLabels) { this._backrefLabels.forEach((backrefLabel) => { backrefLabel.line -= linesReady; }); } return result; } public disassembleChunk(reader: BinaryReader, offsetInModule = 0): boolean { if (this._done) throw new Error( "Invalid state: disassembly process was already finished." ); while (true) { this._currentPosition = reader.position + offsetInModule; if (!reader.read()) return false; switch (reader.state) { case BinaryReaderState.END_WASM: this.appendBuffer(")"); this.newLine(); this._reset(); if (!reader.hasMoreBytes()) { this._done = true; return true; } break; case BinaryReaderState.ERROR: throw reader.error; case BinaryReaderState.BEGIN_WASM: this.appendBuffer("(module"); this.newLine(); break; case BinaryReaderState.END_SECTION: this._currentSectionId = SectionCode.Unknown; break; case BinaryReaderState.BEGIN_SECTION: var sectionInfo = <ISectionInformation>reader.result; switch (sectionInfo.id) { case SectionCode.Type: case SectionCode.Import: case SectionCode.Export: case SectionCode.Global: case SectionCode.Function: case SectionCode.Start: case SectionCode.Code: case SectionCode.Memory: case SectionCode.Data: case SectionCode.Table: case SectionCode.Element: this._currentSectionId = sectionInfo.id; break; // reading known section; default: reader.skipSection(); break; } break; case BinaryReaderState.MEMORY_SECTION_ENTRY: var memoryInfo = <IMemoryType>reader.result; var memoryIndex = this._memoryCount++; var memoryName = this._nameResolver.getMemoryName(memoryIndex, false); this.appendBuffer(` (memory ${memoryName}`); if (this._exportMetadata !== null) { for (const exportName of this._exportMetadata.getMemoryExportNames( memoryIndex )) { this.appendBuffer(` (export "${exportName}")`); } } this.appendBuffer(` ${limitsToString(memoryInfo.limits)}`); if (memoryInfo.shared) { this.appendBuffer(` shared`); } this.appendBuffer(")"); this.newLine(); break; case BinaryReaderState.TABLE_SECTION_ENTRY: var tableInfo = <ITableType>reader.result; var tableIndex = this._tableCount++; var tableName = this._nameResolver.getTableName(tableIndex, false); this.appendBuffer(` (table ${tableName}`); if (this._exportMetadata !== null) { for (const exportName of this._exportMetadata.getTableExportNames( tableIndex )) { this.appendBuffer(` (export "${exportName}")`); } } this.appendBuffer( ` ${limitsToString(tableInfo.limits)} ${typeToString( tableInfo.elementType )})` ); this.newLine(); break; case BinaryReaderState.EXPORT_SECTION_ENTRY: // Skip printing exports here when we have export metadata // which we can use to print export information inline. if (this._exportMetadata === null) { var exportInfo = <IExportEntry>reader.result; this.appendBuffer(" (export "); this.printString(exportInfo.field); this.appendBuffer(" "); switch (exportInfo.kind) { case ExternalKind.Function: var funcName = this._nameResolver.getFunctionName( exportInfo.index, exportInfo.index < this._importCount, true ); this.appendBuffer(`(func ${funcName})`); break; case ExternalKind.Table: var tableName = this._nameResolver.getTableName( exportInfo.index, true ); this.appendBuffer(`(table ${tableName})`); break; case ExternalKind.Memory: var memoryName = this._nameResolver.getMemoryName( exportInfo.index, true ); this.appendBuffer(`(memory ${memoryName})`); break; case ExternalKind.Global: var globalName = this._nameResolver.getGlobalName( exportInfo.index, true ); this.appendBuffer(`(global ${globalName})`); break; default: throw new Error(`Unsupported export ${exportInfo.kind}`); } this.appendBuffer(")"); this.newLine(); } break; case BinaryReaderState.IMPORT_SECTION_ENTRY: var importInfo = <IImportEntry>reader.result; switch (importInfo.kind) { case ExternalKind.Function: this._importCount++; var funcIndex = this._funcIndex++; var funcName = this._nameResolver.getFunctionName( funcIndex, true, false ); this.appendBuffer(` (func ${funcName}`); if (this._exportMetadata !== null) { for (const exportName of this._exportMetadata.getFunctionExportNames( funcIndex )) { this.appendBuffer(` (export "${exportName}")`); } } this.appendBuffer(` (import `); this.printImportSource(importInfo); this.appendBuffer(")"); this.printFuncType(importInfo.funcTypeIndex); this.appendBuffer(")"); break; case ExternalKind.Global: var globalImportInfo = <IGlobalType>importInfo.type; var globalIndex = this._globalCount++; var globalName = this._nameResolver.getGlobalName( globalIndex, false ); this.appendBuffer(` (global ${globalName}`); if (this._exportMetadata !== null) { for (const exportName of this._exportMetadata.getGlobalExportNames( globalIndex )) { this.appendBuffer(` (export "${exportName}")`); } } this.appendBuffer(` (import `); this.printImportSource(importInfo); this.appendBuffer(`) ${globalTypeToString(globalImportInfo)})`); break; case ExternalKind.Memory: var memoryImportInfo = <IMemoryType>importInfo.type; var memoryIndex = this._memoryCount++; var memoryName = this._nameResolver.getMemoryName( memoryIndex, false ); this.appendBuffer(` (memory ${memoryName}`); if (this._exportMetadata !== null) { for (const exportName of this._exportMetadata.getMemoryExportNames( memoryIndex )) { this.appendBuffer(` (export "${exportName}")`); } } this.appendBuffer(` (import `); this.printImportSource(importInfo); this.appendBuffer(`) ${limitsToString(memoryImportInfo.limits)}`); if (memoryImportInfo.shared) { this.appendBuffer(` shared`); } this.appendBuffer(")"); break; case ExternalKind.Table: var tableImportInfo = <ITableType>importInfo.type; var tableIndex = this._tableCount++; var tableName = this._nameResolver.getTableName( tableIndex, false ); this.appendBuffer(` (table ${tableName}`); if (this._exportMetadata !== null) { for (const exportName of this._exportMetadata.getTableExportNames( tableIndex )) { this.appendBuffer(` (export "${exportName}")`); } } this.appendBuffer(` (import `); this.printImportSource(importInfo); this.appendBuffer( `) ${limitsToString(tableImportInfo.limits)} ${typeToString( tableImportInfo.elementType )})` ); break; default: throw new Error(`NYI other import types: ${importInfo.kind}`); } this.newLine(); break; case BinaryReaderState.BEGIN_ELEMENT_SECTION_ENTRY: var elementSegment = <IElementSegment>reader.result; var elementIndex = this._elementCount++; var elementName = this._nameResolver.getElementName( elementIndex, false ); this.appendBuffer(` (elem ${elementName}`); switch (elementSegment.mode) { case ElementMode.Active: if (elementSegment.tableIndex !== 0) { const tableName = this._nameResolver.getTableName( elementSegment.tableIndex, false ); this.appendBuffer(` (table ${tableName})`); } break; case ElementMode.Passive: break; case ElementMode.Declarative: this.appendBuffer(" declare"); break; } break; case BinaryReaderState.END_ELEMENT_SECTION_ENTRY: this.appendBuffer(")"); this.newLine(); break; case BinaryReaderState.ELEMENT_SECTION_ENTRY_BODY: const elementSegmentBody = <IElementSegmentBody>reader.result; this.appendBuffer(` ${typeToString(elementSegmentBody.elementType)}`); break; case BinaryReaderState.BEGIN_GLOBAL_SECTION_ENTRY: var globalInfo = <IGlobalVariable>reader.result; var globalIndex = this._globalCount++; var globalName = this._nameResolver.getGlobalName(globalIndex, false); this.appendBuffer(` (global ${globalName}`); if (this._exportMetadata !== null) { for (const exportName of this._exportMetadata.getGlobalExportNames( globalIndex )) { this.appendBuffer(` (export "${exportName}")`); } } this.appendBuffer(` ${globalTypeToString(globalInfo.type)}`); break; case BinaryReaderState.END_GLOBAL_SECTION_ENTRY: this.appendBuffer(")"); this.newLine(); break; case BinaryReaderState.TYPE_SECTION_ENTRY: var funcType = <IFunctionType>reader.result; var typeIndex = this._types.length; this._types.push(funcType); if (!this._skipTypes) { var typeName = this._nameResolver.getTypeName(typeIndex, false); this.appendBuffer(` (type ${typeName} (func`); this.printFuncType(typeIndex); this.appendBuffer("))"); this.newLine(); } break; case BinaryReaderState.START_SECTION_ENTRY: var startEntry = <IStartEntry>reader.result; var funcName = this._nameResolver.getFunctionName( startEntry.index, startEntry.index < this._importCount, true ); this.appendBuffer(` (start ${funcName})`); this.newLine(); break; case BinaryReaderState.BEGIN_DATA_SECTION_ENTRY: this.appendBuffer(" (data"); break; case BinaryReaderState.DATA_SECTION_ENTRY_BODY: var body = <IDataSegmentBody>reader.result; this.appendBuffer(" "); this.printString(body.data); break; case BinaryReaderState.END_DATA_SECTION_ENTRY: this.appendBuffer(")"); this.newLine(); break; case BinaryReaderState.BEGIN_INIT_EXPRESSION_BODY: case BinaryReaderState.BEGIN_OFFSET_EXPRESSION_BODY: this._expression = []; break; case BinaryReaderState.INIT_EXPRESSION_OPERATOR: case BinaryReaderState.OFFSET_EXPRESSION_OPERATOR: var operator = <IOperatorInformation>reader.result; if (operator.code !== OperatorCode.end) { this._expression.push(operator); } break; case BinaryReaderState.END_OFFSET_EXPRESSION_BODY: if (this._expression.length > 1) { this.appendBuffer(" (offset "); this.printExpression(this._expression); this.appendBuffer(")"); } else { this.appendBuffer(" "); this.printExpression(this._expression); } this._expression = []; break; case BinaryReaderState.END_INIT_EXPRESSION_BODY: if ( this._expression.length > 1 && this._currentSectionId === SectionCode.Element ) { this.appendBuffer(" (item "); this.printExpression(this._expression); this.appendBuffer(")"); } else { this.appendBuffer(" "); this.printExpression(this._expression); } this._expression = []; break; case BinaryReaderState.FUNCTION_SECTION_ENTRY: this._funcTypes.push((<IFunctionEntry>reader.result).typeIndex); break; case BinaryReaderState.BEGIN_FUNCTION_BODY: var func = <IFunctionInformation>reader.result; var type = this._types[ this._funcTypes[this._funcIndex - this._importCount] ]; this.appendBuffer(" (func "); this.appendBuffer( this._nameResolver.getFunctionName(this._funcIndex, false, false) ); if (this._exportMetadata !== null) { for (const exportName of this._exportMetadata.getFunctionExportNames( this._funcIndex )) { this.appendBuffer(` (export "${exportName}")`); } } for (var i = 0; i < type.params.length; i++) { var paramName = this._nameResolver.getVariableName( this._funcIndex, i, false ); this.appendBuffer( ` (param ${paramName} ${typeToString(type.params[i])})` ); } for (var i = 0; i < type.returns.length; i++) { this.appendBuffer(` (result ${typeToString(type.returns[i])})`); } this.newLine(); var localIndex = type.params.length; if (func.locals.length > 0) { this.appendBuffer(" "); for (var l of func.locals) { for (var i = 0; i < l.count; i++) { var paramName = this._nameResolver.getVariableName( this._funcIndex, localIndex++, false ); this.appendBuffer( ` (local ${paramName} ${typeToString(l.type)})` ); } } this.newLine(); } this._indent = " "; this._indentLevel = 0; this._labelIndex = 0; this._backrefLabels = this._labelMode === LabelMode.Depth ? null : []; this._logFirstInstruction = true; break; case BinaryReaderState.CODE_OPERATOR: if (this._logFirstInstruction) { this.logStartOfFunctionBodyOffset(); this._logFirstInstruction = false; } var operator = <IOperatorInformation>reader.result; if (operator.code == OperatorCode.end && this._indentLevel == 0) { // reached of the function, closing function body this.appendBuffer(` )`); this.newLine(); break; } switch (operator.code) { case OperatorCode.end: case OperatorCode.else: this.decreaseIndent(); break; } this.appendBuffer(this._indent); this.printOperator(operator); this.newLine(); switch (operator.code) { case OperatorCode.if: case OperatorCode.block: case OperatorCode.loop: case OperatorCode.else: this.increaseIndent(); break; } break; case BinaryReaderState.END_FUNCTION_BODY: