chrome-devtools-frontend
Version:
Chrome DevTools UI
1,440 lines (1,417 loc) • 80.9 kB
text/typescript
/* 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,
IEventNameEntry,
IEventType,
IExportEntry,
IFieldNameEntry,
IFunctionEntry,
IFunctionInformation,
IFunctionNameEntry,
IGlobalNameEntry,
IGlobalType,
IGlobalVariable,
IImportEntry,
ILocalNameEntry,
IMemoryAddress,
IMemoryNameEntry,
IMemoryType,
INameEntry,
Int64,
IOperatorInformation,
IResizableLimits,
ISectionInformation,
IStartEntry,
ITableNameEntry,
ITableType,
ITypeEntry,
ITypeNameEntry,
NameType,
OperatorCode,
OperatorCodeNames,
SectionCode,
Type,
TypeKind,
} 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 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 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[];
getEventExportNames(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;
getEventName(index: number, isRef: boolean): string;
getFunctionName(index: number, isImport: boolean, isRef: boolean): string;
getVariableName(funcIndex: number, index: number, isRef: boolean): string;
getFieldName(typeIndex: 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 getEventName(index: number, isRef: boolean): string {
return `$event${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 getFieldName(
typeIndex: number,
index: number,
isRef: boolean
): string {
return "$field" + 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[][];
private readonly _eventExportNames: string[][];
constructor(
functionExportNames: string[][],
globalExportNames: string[][],
memoryExportNames: string[][],
tableExportNames: string[][],
eventExportNames: string[][]
) {
this._functionExportNames = functionExportNames;
this._globalExportNames = globalExportNames;
this._memoryExportNames = memoryExportNames;
this._tableExportNames = tableExportNames;
this._eventExportNames = eventExportNames;
}
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;
}
public getEventExportNames(index: number) {
return this._eventExportNames[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 getEventName(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 getFieldName(
typeIndex: 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<ITypeEntry>;
private _funcIndex: number;
private _funcTypes: Array<number>;
private _importCount: number;
private _globalCount: number;
private _memoryCount: number;
private _eventCount: 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._eventCount = 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 typeIndexToString(typeIndex: number): string {
if (typeIndex >= 0) return this._nameResolver.getTypeName(typeIndex, true);
switch (typeIndex) {
case TypeKind.funcref:
return "func";
case TypeKind.externref:
return "extern";
case TypeKind.anyref:
return "any";
case TypeKind.eqref:
return "eq";
case TypeKind.i31ref:
return "i31";
case TypeKind.dataref:
return "data";
}
}
private typeToString(type: Type): string {
switch (type.kind) {
case TypeKind.i32:
return "i32";
case TypeKind.i64:
return "i64";
case TypeKind.f32:
return "f32";
case TypeKind.f64:
return "f64";
case TypeKind.v128:
return "v128";
case TypeKind.i8:
return "i8";
case TypeKind.i16:
return "i16";
case TypeKind.funcref:
return "funcref";
case TypeKind.externref:
return "externref";
case TypeKind.anyref:
return "anyref";
case TypeKind.eqref:
return "eqref";
case TypeKind.i31ref:
return "i31ref";
case TypeKind.dataref:
return "dataref";
case TypeKind.ref:
return `(ref ${this.typeIndexToString(type.index)})`;
case TypeKind.optref:
return `(ref null ${this.typeIndexToString(type.index)})`;
case TypeKind.rtt:
return `(rtt ${this.typeIndexToString(type.index)})`;
case TypeKind.rtt_d:
return `(rtt ${type.depth} ${this.typeIndexToString(type.index)})`;
default:
throw new Error(`Unexpected type ${JSON.stringify(type)}`);
}
}
private maybeMut(type: string, mutability: boolean): string {
return mutability ? `(mut ${type})` : type;
}
private globalTypeToString(type: IGlobalType): string {
const typeStr = this.typeToString(type.contentType);
return this.maybeMut(typeStr, !!type.mutability);
}
private printFuncType(typeIndex: number): void {
var type = this._types[typeIndex];
if (type.params.length > 0) {
this.appendBuffer(" (param");
for (var i = 0; i < type.params.length; i++) {
this.appendBuffer(" ");
this.appendBuffer(this.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(this.typeToString(type.returns[i]));
}
this.appendBuffer(")");
}
}
private printStructType(typeIndex: number): void {
var type = this._types[typeIndex];
if (type.fields.length === 0) return;
for (var i = 0; i < type.fields.length; i++) {
const fieldType = this.maybeMut(
this.typeToString(type.fields[i]),
type.mutabilities[i]
);
const fieldName = this._nameResolver.getFieldName(typeIndex, i, false);
this.appendBuffer(` (field ${fieldName} ${fieldType})`);
}
}
private printArrayType(typeIndex: number): void {
var type = this._types[typeIndex];
this.appendBuffer(" (field ");
this.appendBuffer(
this.maybeMut(this.typeToString(type.elementType), type.mutability)
);
}
private printBlockType(type: Type): void {
if (type.kind === TypeKind.empty_block_type) {
return;
}
if (type.kind === TypeKind.unspecified) {
return this.printFuncType(type.index);
}
this.appendBuffer(" (result ");
this.appendBuffer(this.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(")");
}
}
// extraDepthOffset is used by "delegate" instructions.
private useLabel(depth: number, extraDepthOffset = 0): string {
if (!this._backrefLabels) {
return "" + depth;
}
var i = this._backrefLabels.length - depth - 1 - extraDepthOffset;
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:
case OperatorCode.try:
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:
case OperatorCode.br_on_null:
case OperatorCode.br_on_non_null:
case OperatorCode.br_on_cast:
case OperatorCode.br_on_cast_fail:
case OperatorCode.br_on_func:
case OperatorCode.br_on_non_func:
case OperatorCode.br_on_data:
case OperatorCode.br_on_non_data:
case OperatorCode.br_on_i31:
case OperatorCode.br_on_non_i31:
this.appendBuffer(" ");
this.appendBuffer(this.useLabel(operator.brDepth));
break;
case OperatorCode.br_on_cast_static:
case OperatorCode.br_on_cast_static_fail: {
const label = this.useLabel(operator.brDepth);
const refType = this._nameResolver.getTypeName(operator.refType, true);
this.appendBuffer(` ${label} ${refType}`);
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.rethrow:
this.appendBuffer(" ");
this.appendBuffer(this.useLabel(operator.relativeDepth));
break;
case OperatorCode.delegate:
this.appendBuffer(" ");
this.appendBuffer(this.useLabel(operator.relativeDepth, 1));
break;
case OperatorCode.catch:
case OperatorCode.throw:
var eventName = this._nameResolver.getEventName(
operator.eventIndex,
true
);
this.appendBuffer(` ${eventName}`);
break;
case OperatorCode.ref_null:
this.appendBuffer(" ");
this.appendBuffer(this.typeIndexToString(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.select_with_type: {
const selectType = this.typeToString(operator.selectType);
this.appendBuffer(` ${selectType}`);
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;
}
case OperatorCode.struct_get:
case OperatorCode.struct_get_s:
case OperatorCode.struct_get_u:
case OperatorCode.struct_set: {
const refType = this._nameResolver.getTypeName(operator.refType, true);
const fieldName = this._nameResolver.getFieldName(
operator.refType,
operator.fieldIndex,
true
);
this.appendBuffer(` ${refType} ${fieldName}`);
break;
}
case OperatorCode.rtt_canon:
case OperatorCode.rtt_sub:
case OperatorCode.rtt_fresh_sub:
case OperatorCode.ref_test_static:
case OperatorCode.ref_cast_static:
case OperatorCode.struct_new_default:
case OperatorCode.struct_new_default_with_rtt:
case OperatorCode.struct_new:
case OperatorCode.struct_new_with_rtt:
case OperatorCode.array_new_default:
case OperatorCode.array_new_default_with_rtt:
case OperatorCode.array_new:
case OperatorCode.array_new_with_rtt:
case OperatorCode.array_get:
case OperatorCode.array_get_s:
case OperatorCode.array_get_u:
case OperatorCode.array_set:
case OperatorCode.array_len: {
const refType = this._nameResolver.getTypeName(operator.refType, true);
this.appendBuffer(` ${refType}`);
break;
}
case OperatorCode.array_copy: {
const dstType = this._nameResolver.getTypeName(operator.refType, true);
const srcType = this._nameResolver.getTypeName(operator.srcType, true);
this.appendBuffer(` ${dstType} ${srcType}`);
break;
}
case OperatorCode.array_init:
case OperatorCode.array_init_static: {
const refType = this._nameResolver.getTypeName(operator.refType, true);
const length = operator.brDepth; // Overloaded field.
this.appendBuffer(` ${refType} ${length}`);
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:
case SectionCode.Event:
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 ${JSON.stringify(exportName)})`);
}
}
this.appendBuffer(` ${limitsToString(memoryInfo.limits)}`);
if (memoryInfo.shared) {
this.appendBuffer(` shared`);
}
this.appendBuffer(")");
this.newLine();
break;
case BinaryReaderState.EVENT_SECTION_ENTRY:
var eventInfo = <IEventType>reader.result;
var eventIndex = this._eventCount++;
var eventName = this._nameResolver.getEventName(eventIndex, false);
this.appendBuffer(` (event ${eventName}`);
if (this._exportMetadata !== null) {
for (const exportName of this._exportMetadata.getEventExportNames(
eventIndex
)) {
this.appendBuffer(` (export ${JSON.stringify(exportName)})`);
}
}
this.printFuncType(eventInfo.typeIndex);
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 ${JSON.stringify(exportName)})`);
}
}
this.appendBuffer(
` ${limitsToString(tableInfo.limits)} ${this.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;
case ExternalKind.Event:
var eventName = this._nameResolver.getEventName(
exportInfo.index,
true
);
this.appendBuffer(`(event ${eventName})`);
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 ${JSON.stringify(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 ${JSON.stringify(exportName)})`);
}
}
this.appendBuffer(` (import `);
this.printImportSource(importInfo);
this.appendBuffer(
`) ${this.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 ${JSON.stringify(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 ${JSON.stringify(exportName)})`);
}
}
this.appendBuffer(` (import `);
this.printImportSource(importInfo);
this.appendBuffer(
`) ${limitsToString(
tableImportInfo.limits
)} ${this.typeToString(tableImportInfo.elementType)})`
);
break;
case ExternalKind.Event:
var eventImportInfo = <IEventType>importInfo.type;
var eventIndex = this._eventCount++;
var eventName = this._nameResolver.getEventName(
eventIndex,
false
);
this.appendBuffer(` (event ${eventName}`);
if (this._exportMetadata !== null) {
for (const exportName of this._exportMetadata.getEventExportNames(
eventIndex
)) {
this.appendBuffer(` (export ${JSON.stringify(exportName)})`);
}
}
this.appendBuffer(` (import `);
this.printImportSource(importInfo);
this.appendBuffer(")");
this.