1
0
Fork 0
firefox/devtools/client/shared/sourceeditor/wasm.js
Daniel Baumann 5e9a113729
Adding upstream version 140.0.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-25 09:37:52 +02:00

160 lines
4.4 KiB
JavaScript

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
"use strict";
const wasmparser = require("resource://devtools/client/shared/vendor/WasmParser.js");
const wasmdis = require("resource://devtools/client/shared/vendor/WasmDis.js");
const wasmStates = new WeakMap();
function getWasmText(subject, data) {
if (wasmStates.has(subject)) {
const wasmState = wasmStates.get(subject);
return { lines: wasmState.result.lines, done: wasmState.result.done };
}
const parser = new wasmparser.BinaryReader();
parser.setData(data.buffer, 0, data.length);
const dis = new wasmdis.WasmDisassembler();
dis.addOffsets = true;
const done = dis.disassembleChunk(parser);
let result = dis.getResult();
if (result.lines.length === 0) {
result = { lines: ["No luck with wast conversion"], offsets: [0], done };
}
// Cache mappings of WASM offsets to lines
const offsets = result.offsets,
lines = [];
for (let line = 0; line < offsets.length; line++) {
const offset = offsets[line];
lines[offset] = line;
}
wasmStates.set(subject, { offsets, lines, result });
return { lines: result.lines, done: result.done };
}
/**
* Creates wasm formatter function used to generate the hexadecimal number
* displayed in the line gutter
*
* @param {Object} subject - An object which decribes the source, it is comprised of the sourceId
* @returns {Function}
*/
function getWasmLineNumberFormatter(subject) {
const codeOf0 = 48,
codeOfA = 65;
const buffer = [
codeOf0,
codeOf0,
codeOf0,
codeOf0,
codeOf0,
codeOf0,
codeOf0,
codeOf0,
];
let last0 = 7;
return function (line) {
const offset = lineToWasmOffset(subject, line - 1);
if (offset === undefined) {
return "";
}
let i = 7;
for (let n = offset | 0; n !== 0 && i >= 0; n >>= 4, i--) {
const nibble = n & 15;
buffer[i] = nibble < 10 ? codeOf0 + nibble : codeOfA - 10 + nibble;
}
for (let j = i; j > last0; j--) {
buffer[j] = codeOf0;
}
last0 = i;
return String.fromCharCode.apply(null, buffer);
};
}
/**
* Checks if the specified source exists in the cache.
* This is used to determine if the source is a WASM source
*
* @param {Object} subject
* @returns {Boolean}
*/
function isWasm(subject) {
return wasmStates.has(subject);
}
/**
* Converts the source (decimal) line to its WASM offset
*
* @param {Object} subject
* @param {Number} line
* @param {Boolean} findNextOffset
* There are scenarios (e.g are empty lines) where we might want to find the next best offset match.
* Every line will usually have offsets assigned except empty lines (which could be between functions
* or some declarations).
* @returns {Number}
*/
function lineToWasmOffset(subject, line, findNextOffset = false) {
const wasmState = wasmStates.get(subject);
if (!wasmState) {
return undefined;
}
let offset = wasmState.offsets[line];
if (findNextOffset) {
while (offset === undefined && line > 0) {
offset = wasmState.offsets[--line];
}
}
return offset;
}
/**
* Converts the WASM offset to the source line
*
* @param {Object} subject
* @param {Number} offset
* @returns {Number}
*/
function wasmOffsetToLine(subject, offset) {
const wasmState = wasmStates.get(subject);
return wasmState.lines[offset];
}
// A cache of the wasm source text as an array of lines.
// The lines are cached with the value object of the source content
// as the key.
const wasmLines = new WeakMap();
function renderWasmText(subject, content) {
if (wasmLines.has(content)) {
return wasmLines.get(content) || [];
}
// binary does not survive as Uint8Array, converting from string
const { binary } = content.value;
const data = new Uint8Array(binary.length);
for (let i = 0; i < data.length; i++) {
data[i] = binary.charCodeAt(i);
}
const { lines } = getWasmText(subject, data);
const MAX_LINES = 1000000;
if (lines.length > MAX_LINES) {
lines.splice(MAX_LINES, lines.length - MAX_LINES);
lines.push(";; .... text is truncated due to the size");
}
wasmLines.set(content, lines);
return lines;
}
module.exports = {
getWasmText,
getWasmLineNumberFormatter,
isWasm,
lineToWasmOffset,
wasmOffsetToLine,
renderWasmText,
};