@tdh-keyboard/pinyin 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +190 -0
- package/data/default.yaml +129 -0
- package/data/luna_pinyin.prism.bin +0 -0
- package/data/luna_pinyin.reverse.bin +0 -0
- package/data/luna_pinyin.schema.yaml +179 -0
- package/data/luna_pinyin.table.bin +0 -0
- package/data/rime-api.data +0 -0
- package/data/rime-api.js +2 -0
- package/data/rime-api.wasm +0 -0
- package/dist/index.d.mts +73 -0
- package/dist/index.d.ts +73 -0
- package/dist/index.js +3416 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +3415 -0
- package/dist/index.mjs.map +1 -0
- package/package.json +41 -0
package/dist/index.js
ADDED
|
@@ -0,0 +1,3416 @@
|
|
|
1
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
2
|
+
//#region data/rime-api.js
|
|
3
|
+
async function createRimeModule(moduleArg = {}) {
|
|
4
|
+
var moduleRtn;
|
|
5
|
+
var Module = moduleArg;
|
|
6
|
+
var ENVIRONMENT_IS_WEB = true;
|
|
7
|
+
var ENVIRONMENT_IS_WORKER = false;
|
|
8
|
+
if (!Module["expectedDataFileDownloads"]) Module["expectedDataFileDownloads"] = 0;
|
|
9
|
+
Module["expectedDataFileDownloads"]++;
|
|
10
|
+
(() => {
|
|
11
|
+
if (typeof ENVIRONMENT_IS_PTHREAD != "undefined" && ENVIRONMENT_IS_PTHREAD || typeof ENVIRONMENT_IS_WASM_WORKER != "undefined" && ENVIRONMENT_IS_WASM_WORKER) return;
|
|
12
|
+
async function loadPackage(metadata) {
|
|
13
|
+
if (typeof window === "object") window["encodeURIComponent"](window.location.pathname.substring(0, window.location.pathname.lastIndexOf("/")) + "/");
|
|
14
|
+
else if (typeof process === "undefined" && typeof location !== "undefined") encodeURIComponent(location.pathname.substring(0, location.pathname.lastIndexOf("/")) + "/");
|
|
15
|
+
var PACKAGE_NAME = "/home/zzx/work/librime/wasm/dist/rime-api.data";
|
|
16
|
+
var REMOTE_PACKAGE_BASE = "rime-api.data";
|
|
17
|
+
var REMOTE_PACKAGE_NAME = Module["locateFile"] ? Module["locateFile"](REMOTE_PACKAGE_BASE, "") : REMOTE_PACKAGE_BASE;
|
|
18
|
+
var REMOTE_PACKAGE_SIZE = metadata["remote_package_size"];
|
|
19
|
+
async function fetchRemotePackage(packageName, packageSize) {
|
|
20
|
+
if (!Module["dataFileDownloads"]) Module["dataFileDownloads"] = {};
|
|
21
|
+
try {
|
|
22
|
+
var response = await fetch(packageName);
|
|
23
|
+
} catch (e) {
|
|
24
|
+
throw new Error(`Network Error: ${packageName}`, { e });
|
|
25
|
+
}
|
|
26
|
+
if (!response.ok) throw new Error(`${response.status}: ${response.url}`);
|
|
27
|
+
const chunks = [];
|
|
28
|
+
const headers = response.headers;
|
|
29
|
+
const total = Number(headers.get("Content-Length") || packageSize);
|
|
30
|
+
let loaded = 0;
|
|
31
|
+
Module["setStatus"] && Module["setStatus"]("Downloading data...");
|
|
32
|
+
const reader = response.body.getReader();
|
|
33
|
+
while (1) {
|
|
34
|
+
var { done, value } = await reader.read();
|
|
35
|
+
if (done) break;
|
|
36
|
+
chunks.push(value);
|
|
37
|
+
loaded += value.length;
|
|
38
|
+
Module["dataFileDownloads"][packageName] = {
|
|
39
|
+
loaded,
|
|
40
|
+
total
|
|
41
|
+
};
|
|
42
|
+
let totalLoaded = 0;
|
|
43
|
+
let totalSize = 0;
|
|
44
|
+
for (const download of Object.values(Module["dataFileDownloads"])) {
|
|
45
|
+
totalLoaded += download.loaded;
|
|
46
|
+
totalSize += download.total;
|
|
47
|
+
}
|
|
48
|
+
Module["setStatus"] && Module["setStatus"](`Downloading data... (${totalLoaded}/${totalSize})`);
|
|
49
|
+
}
|
|
50
|
+
const packageData = new Uint8Array(chunks.map((c) => c.length).reduce((a, b) => a + b, 0));
|
|
51
|
+
let offset = 0;
|
|
52
|
+
for (const chunk of chunks) {
|
|
53
|
+
packageData.set(chunk, offset);
|
|
54
|
+
offset += chunk.length;
|
|
55
|
+
}
|
|
56
|
+
return packageData.buffer;
|
|
57
|
+
}
|
|
58
|
+
var fetchPromise;
|
|
59
|
+
var fetched = Module["getPreloadedPackage"] && Module["getPreloadedPackage"](REMOTE_PACKAGE_NAME, REMOTE_PACKAGE_SIZE);
|
|
60
|
+
if (!fetched) fetchPromise = fetchRemotePackage(REMOTE_PACKAGE_NAME, REMOTE_PACKAGE_SIZE);
|
|
61
|
+
async function runWithFS(Module) {
|
|
62
|
+
function assert(check, msg) {
|
|
63
|
+
if (!check) throw new Error(msg);
|
|
64
|
+
}
|
|
65
|
+
Module["FS_createPath"]("/", "rime", true, true);
|
|
66
|
+
Module["FS_createPath"]("/rime", "opencc", true, true);
|
|
67
|
+
for (var file of metadata["files"]) {
|
|
68
|
+
var name = file["filename"];
|
|
69
|
+
Module["addRunDependency"](`fp ${name}`);
|
|
70
|
+
}
|
|
71
|
+
async function processPackageData(arrayBuffer) {
|
|
72
|
+
assert(arrayBuffer, "Loading data file failed.");
|
|
73
|
+
assert(arrayBuffer.constructor.name === ArrayBuffer.name, "bad input to processPackageData " + arrayBuffer.constructor.name);
|
|
74
|
+
var byteArray = new Uint8Array(arrayBuffer);
|
|
75
|
+
for (var file of metadata["files"]) {
|
|
76
|
+
var name = file["filename"];
|
|
77
|
+
var data = byteArray.subarray(file["start"], file["end"]);
|
|
78
|
+
Module["FS_createDataFile"](name, null, data, true, true, true);
|
|
79
|
+
Module["removeRunDependency"](`fp ${name}`);
|
|
80
|
+
}
|
|
81
|
+
Module["removeRunDependency"]("datafile_/home/zzx/work/librime/wasm/dist/rime-api.data");
|
|
82
|
+
}
|
|
83
|
+
Module["addRunDependency"]("datafile_/home/zzx/work/librime/wasm/dist/rime-api.data");
|
|
84
|
+
if (!Module["preloadResults"]) Module["preloadResults"] = {};
|
|
85
|
+
Module["preloadResults"][PACKAGE_NAME] = { fromCache: false };
|
|
86
|
+
if (!fetched) fetched = await fetchPromise;
|
|
87
|
+
processPackageData(fetched);
|
|
88
|
+
}
|
|
89
|
+
if (Module["calledRun"]) runWithFS(Module);
|
|
90
|
+
else {
|
|
91
|
+
if (!Module["preRun"]) Module["preRun"] = [];
|
|
92
|
+
Module["preRun"].push(runWithFS);
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
loadPackage({
|
|
96
|
+
files: [
|
|
97
|
+
{
|
|
98
|
+
filename: "/rime/opencc/STCharacters.ocd2",
|
|
99
|
+
start: 0,
|
|
100
|
+
end: 45310
|
|
101
|
+
},
|
|
102
|
+
{
|
|
103
|
+
filename: "/rime/opencc/STPhrases.ocd2",
|
|
104
|
+
start: 45310,
|
|
105
|
+
end: 971027
|
|
106
|
+
},
|
|
107
|
+
{
|
|
108
|
+
filename: "/rime/opencc/TSCharacters.ocd2",
|
|
109
|
+
start: 971027,
|
|
110
|
+
end: 1017153
|
|
111
|
+
},
|
|
112
|
+
{
|
|
113
|
+
filename: "/rime/opencc/TSPhrases.ocd2",
|
|
114
|
+
start: 1017153,
|
|
115
|
+
end: 1026935
|
|
116
|
+
},
|
|
117
|
+
{
|
|
118
|
+
filename: "/rime/opencc/s2t.json",
|
|
119
|
+
start: 1026935,
|
|
120
|
+
end: 1027341
|
|
121
|
+
},
|
|
122
|
+
{
|
|
123
|
+
filename: "/rime/opencc/t2s.json",
|
|
124
|
+
start: 1027341,
|
|
125
|
+
end: 1027747
|
|
126
|
+
}
|
|
127
|
+
],
|
|
128
|
+
remote_package_size: 1027747
|
|
129
|
+
});
|
|
130
|
+
})();
|
|
131
|
+
var thisProgram = "./this.program";
|
|
132
|
+
var quit_ = (status, toThrow) => {
|
|
133
|
+
throw toThrow;
|
|
134
|
+
};
|
|
135
|
+
var _scriptName = require("url").pathToFileURL(__filename).href;
|
|
136
|
+
var scriptDirectory = "";
|
|
137
|
+
function locateFile(path) {
|
|
138
|
+
if (Module["locateFile"]) return Module["locateFile"](path, scriptDirectory);
|
|
139
|
+
return scriptDirectory + path;
|
|
140
|
+
}
|
|
141
|
+
var readAsync, readBinary;
|
|
142
|
+
if (ENVIRONMENT_IS_WEB || ENVIRONMENT_IS_WORKER) {
|
|
143
|
+
try {
|
|
144
|
+
scriptDirectory = new URL(".", _scriptName).href;
|
|
145
|
+
} catch {}
|
|
146
|
+
readAsync = async (url) => {
|
|
147
|
+
var response = await fetch(url, { credentials: "same-origin" });
|
|
148
|
+
if (response.ok) return response.arrayBuffer();
|
|
149
|
+
throw new Error(response.status + " : " + response.url);
|
|
150
|
+
};
|
|
151
|
+
}
|
|
152
|
+
var out = console.log.bind(console);
|
|
153
|
+
var err = console.error.bind(console);
|
|
154
|
+
var wasmBinary;
|
|
155
|
+
var ABORT = false;
|
|
156
|
+
var EXITSTATUS;
|
|
157
|
+
var readyPromiseResolve, readyPromiseReject, HEAP8, HEAPU8, HEAP16, HEAP32, HEAPU32, HEAP64;
|
|
158
|
+
var runtimeInitialized = false;
|
|
159
|
+
function updateMemoryViews() {
|
|
160
|
+
var b = wasmMemory.buffer;
|
|
161
|
+
HEAP8 = new Int8Array(b);
|
|
162
|
+
HEAP16 = new Int16Array(b);
|
|
163
|
+
HEAPU8 = new Uint8Array(b);
|
|
164
|
+
new Uint16Array(b);
|
|
165
|
+
HEAP32 = new Int32Array(b);
|
|
166
|
+
HEAPU32 = new Uint32Array(b);
|
|
167
|
+
new Float32Array(b);
|
|
168
|
+
new Float64Array(b);
|
|
169
|
+
HEAP64 = new BigInt64Array(b);
|
|
170
|
+
new BigUint64Array(b);
|
|
171
|
+
}
|
|
172
|
+
function preRun() {
|
|
173
|
+
if (Module["preRun"]) {
|
|
174
|
+
if (typeof Module["preRun"] == "function") Module["preRun"] = [Module["preRun"]];
|
|
175
|
+
while (Module["preRun"].length) addOnPreRun(Module["preRun"].shift());
|
|
176
|
+
}
|
|
177
|
+
callRuntimeCallbacks(onPreRuns);
|
|
178
|
+
}
|
|
179
|
+
function initRuntime() {
|
|
180
|
+
runtimeInitialized = true;
|
|
181
|
+
if (!Module["noFSInit"] && !FS.initialized) FS.init();
|
|
182
|
+
TTY.init();
|
|
183
|
+
wasmExports["Q"]();
|
|
184
|
+
FS.ignorePermissions = false;
|
|
185
|
+
}
|
|
186
|
+
function postRun() {
|
|
187
|
+
if (Module["postRun"]) {
|
|
188
|
+
if (typeof Module["postRun"] == "function") Module["postRun"] = [Module["postRun"]];
|
|
189
|
+
while (Module["postRun"].length) addOnPostRun(Module["postRun"].shift());
|
|
190
|
+
}
|
|
191
|
+
callRuntimeCallbacks(onPostRuns);
|
|
192
|
+
}
|
|
193
|
+
function abort(what) {
|
|
194
|
+
Module["onAbort"]?.(what);
|
|
195
|
+
what = "Aborted(" + what + ")";
|
|
196
|
+
err(what);
|
|
197
|
+
ABORT = true;
|
|
198
|
+
what += ". Build with -sASSERTIONS for more info.";
|
|
199
|
+
if (runtimeInitialized) ___trap();
|
|
200
|
+
var e = new WebAssembly.RuntimeError(what);
|
|
201
|
+
readyPromiseReject?.(e);
|
|
202
|
+
throw e;
|
|
203
|
+
}
|
|
204
|
+
var wasmBinaryFile;
|
|
205
|
+
function findWasmBinary() {
|
|
206
|
+
if (Module["locateFile"]) return locateFile("rime-api.wasm");
|
|
207
|
+
return new URL("rime-api.wasm", require("url").pathToFileURL(__filename).href).href;
|
|
208
|
+
}
|
|
209
|
+
function getBinarySync(file) {
|
|
210
|
+
if (file == wasmBinaryFile && wasmBinary) return new Uint8Array(wasmBinary);
|
|
211
|
+
if (readBinary) return readBinary(file);
|
|
212
|
+
throw "both async and sync fetching of the wasm failed";
|
|
213
|
+
}
|
|
214
|
+
async function getWasmBinary(binaryFile) {
|
|
215
|
+
if (!wasmBinary) try {
|
|
216
|
+
var response = await readAsync(binaryFile);
|
|
217
|
+
return new Uint8Array(response);
|
|
218
|
+
} catch {}
|
|
219
|
+
return getBinarySync(binaryFile);
|
|
220
|
+
}
|
|
221
|
+
async function instantiateArrayBuffer(binaryFile, imports) {
|
|
222
|
+
try {
|
|
223
|
+
var binary = await getWasmBinary(binaryFile);
|
|
224
|
+
return await WebAssembly.instantiate(binary, imports);
|
|
225
|
+
} catch (reason) {
|
|
226
|
+
err(`failed to asynchronously prepare wasm: ${reason}`);
|
|
227
|
+
abort(reason);
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
async function instantiateAsync(binary, binaryFile, imports) {
|
|
231
|
+
if (!binary) try {
|
|
232
|
+
var response = fetch(binaryFile, { credentials: "same-origin" });
|
|
233
|
+
return await WebAssembly.instantiateStreaming(response, imports);
|
|
234
|
+
} catch (reason) {
|
|
235
|
+
err(`wasm streaming compile failed: ${reason}`);
|
|
236
|
+
err("falling back to ArrayBuffer instantiation");
|
|
237
|
+
}
|
|
238
|
+
return instantiateArrayBuffer(binaryFile, imports);
|
|
239
|
+
}
|
|
240
|
+
function getWasmImports() {
|
|
241
|
+
return { a: wasmImports };
|
|
242
|
+
}
|
|
243
|
+
async function createWasm() {
|
|
244
|
+
function receiveInstance(instance, module) {
|
|
245
|
+
wasmExports = instance.exports;
|
|
246
|
+
wasmExports = applySignatureConversions(wasmExports);
|
|
247
|
+
assignWasmExports(wasmExports);
|
|
248
|
+
updateMemoryViews();
|
|
249
|
+
return wasmExports;
|
|
250
|
+
}
|
|
251
|
+
function receiveInstantiationResult(result) {
|
|
252
|
+
return receiveInstance(result["instance"]);
|
|
253
|
+
}
|
|
254
|
+
var info = getWasmImports();
|
|
255
|
+
if (Module["instantiateWasm"]) return new Promise((resolve, reject) => {
|
|
256
|
+
Module["instantiateWasm"](info, (inst, mod) => {
|
|
257
|
+
resolve(receiveInstance(inst, mod));
|
|
258
|
+
});
|
|
259
|
+
});
|
|
260
|
+
wasmBinaryFile ??= findWasmBinary();
|
|
261
|
+
return receiveInstantiationResult(await instantiateAsync(wasmBinary, wasmBinaryFile, info));
|
|
262
|
+
}
|
|
263
|
+
class ExitStatus {
|
|
264
|
+
name = "ExitStatus";
|
|
265
|
+
constructor(status) {
|
|
266
|
+
this.message = `Program terminated with exit(${status})`;
|
|
267
|
+
this.status = status;
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
var callRuntimeCallbacks = (callbacks) => {
|
|
271
|
+
while (callbacks.length > 0) callbacks.shift()(Module);
|
|
272
|
+
};
|
|
273
|
+
var onPostRuns = [];
|
|
274
|
+
var addOnPostRun = (cb) => onPostRuns.push(cb);
|
|
275
|
+
var onPreRuns = [];
|
|
276
|
+
var addOnPreRun = (cb) => onPreRuns.push(cb);
|
|
277
|
+
var noExitRuntime = true;
|
|
278
|
+
var PATH = {
|
|
279
|
+
isAbs: (path) => path.charAt(0) === "/",
|
|
280
|
+
splitPath: (filename) => {
|
|
281
|
+
return /^(\/?|)([\s\S]*?)((?:\.{1,2}|[^\/]+?|)(\.[^.\/]*|))(?:[\/]*)$/.exec(filename).slice(1);
|
|
282
|
+
},
|
|
283
|
+
normalizeArray: (parts, allowAboveRoot) => {
|
|
284
|
+
var up = 0;
|
|
285
|
+
for (var i = parts.length - 1; i >= 0; i--) {
|
|
286
|
+
var last = parts[i];
|
|
287
|
+
if (last === ".") parts.splice(i, 1);
|
|
288
|
+
else if (last === "..") {
|
|
289
|
+
parts.splice(i, 1);
|
|
290
|
+
up++;
|
|
291
|
+
} else if (up) {
|
|
292
|
+
parts.splice(i, 1);
|
|
293
|
+
up--;
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
if (allowAboveRoot) for (; up; up--) parts.unshift("..");
|
|
297
|
+
return parts;
|
|
298
|
+
},
|
|
299
|
+
normalize: (path) => {
|
|
300
|
+
var isAbsolute = PATH.isAbs(path), trailingSlash = path.slice(-1) === "/";
|
|
301
|
+
path = PATH.normalizeArray(path.split("/").filter((p) => !!p), !isAbsolute).join("/");
|
|
302
|
+
if (!path && !isAbsolute) path = ".";
|
|
303
|
+
if (path && trailingSlash) path += "/";
|
|
304
|
+
return (isAbsolute ? "/" : "") + path;
|
|
305
|
+
},
|
|
306
|
+
dirname: (path) => {
|
|
307
|
+
var result = PATH.splitPath(path), root = result[0], dir = result[1];
|
|
308
|
+
if (!root && !dir) return ".";
|
|
309
|
+
if (dir) dir = dir.slice(0, -1);
|
|
310
|
+
return root + dir;
|
|
311
|
+
},
|
|
312
|
+
basename: (path) => path && path.match(/([^\/]+|\/)\/*$/)[1],
|
|
313
|
+
join: (...paths) => PATH.normalize(paths.join("/")),
|
|
314
|
+
join2: (l, r) => PATH.normalize(l + "/" + r)
|
|
315
|
+
};
|
|
316
|
+
var initRandomFill = () => (view) => crypto.getRandomValues(view);
|
|
317
|
+
var randomFill = (view) => {
|
|
318
|
+
(randomFill = initRandomFill())(view);
|
|
319
|
+
};
|
|
320
|
+
var PATH_FS = {
|
|
321
|
+
resolve: (...args) => {
|
|
322
|
+
var resolvedPath = "", resolvedAbsolute = false;
|
|
323
|
+
for (var i = args.length - 1; i >= -1 && !resolvedAbsolute; i--) {
|
|
324
|
+
var path = i >= 0 ? args[i] : FS.cwd();
|
|
325
|
+
if (typeof path != "string") throw new TypeError("Arguments to path.resolve must be strings");
|
|
326
|
+
else if (!path) return "";
|
|
327
|
+
resolvedPath = path + "/" + resolvedPath;
|
|
328
|
+
resolvedAbsolute = PATH.isAbs(path);
|
|
329
|
+
}
|
|
330
|
+
resolvedPath = PATH.normalizeArray(resolvedPath.split("/").filter((p) => !!p), !resolvedAbsolute).join("/");
|
|
331
|
+
return (resolvedAbsolute ? "/" : "") + resolvedPath || ".";
|
|
332
|
+
},
|
|
333
|
+
relative: (from, to) => {
|
|
334
|
+
from = PATH_FS.resolve(from).slice(1);
|
|
335
|
+
to = PATH_FS.resolve(to).slice(1);
|
|
336
|
+
function trim(arr) {
|
|
337
|
+
var start = 0;
|
|
338
|
+
for (; start < arr.length; start++) if (arr[start] !== "") break;
|
|
339
|
+
var end = arr.length - 1;
|
|
340
|
+
for (; end >= 0; end--) if (arr[end] !== "") break;
|
|
341
|
+
if (start > end) return [];
|
|
342
|
+
return arr.slice(start, end - start + 1);
|
|
343
|
+
}
|
|
344
|
+
var fromParts = trim(from.split("/"));
|
|
345
|
+
var toParts = trim(to.split("/"));
|
|
346
|
+
var length = Math.min(fromParts.length, toParts.length);
|
|
347
|
+
var samePartsLength = length;
|
|
348
|
+
for (var i = 0; i < length; i++) if (fromParts[i] !== toParts[i]) {
|
|
349
|
+
samePartsLength = i;
|
|
350
|
+
break;
|
|
351
|
+
}
|
|
352
|
+
var outputParts = [];
|
|
353
|
+
for (var i = samePartsLength; i < fromParts.length; i++) outputParts.push("..");
|
|
354
|
+
outputParts = outputParts.concat(toParts.slice(samePartsLength));
|
|
355
|
+
return outputParts.join("/");
|
|
356
|
+
}
|
|
357
|
+
};
|
|
358
|
+
var UTF8Decoder = new TextDecoder();
|
|
359
|
+
var findStringEnd = (heapOrArray, idx, maxBytesToRead, ignoreNul) => {
|
|
360
|
+
var maxIdx = idx + maxBytesToRead;
|
|
361
|
+
if (ignoreNul) return maxIdx;
|
|
362
|
+
while (heapOrArray[idx] && !(idx >= maxIdx)) ++idx;
|
|
363
|
+
return idx;
|
|
364
|
+
};
|
|
365
|
+
var UTF8ArrayToString = (heapOrArray, idx = 0, maxBytesToRead, ignoreNul) => {
|
|
366
|
+
idx >>>= 0;
|
|
367
|
+
var endPtr = findStringEnd(heapOrArray, idx, maxBytesToRead, ignoreNul);
|
|
368
|
+
return UTF8Decoder.decode(heapOrArray.buffer ? heapOrArray.subarray(idx, endPtr) : new Uint8Array(heapOrArray.slice(idx, endPtr)));
|
|
369
|
+
};
|
|
370
|
+
var FS_stdin_getChar_buffer = [];
|
|
371
|
+
var lengthBytesUTF8 = (str) => {
|
|
372
|
+
var len = 0;
|
|
373
|
+
for (var i = 0; i < str.length; ++i) {
|
|
374
|
+
var c = str.charCodeAt(i);
|
|
375
|
+
if (c <= 127) len++;
|
|
376
|
+
else if (c <= 2047) len += 2;
|
|
377
|
+
else if (c >= 55296 && c <= 57343) {
|
|
378
|
+
len += 4;
|
|
379
|
+
++i;
|
|
380
|
+
} else len += 3;
|
|
381
|
+
}
|
|
382
|
+
return len;
|
|
383
|
+
};
|
|
384
|
+
var stringToUTF8Array = (str, heap, outIdx, maxBytesToWrite) => {
|
|
385
|
+
outIdx >>>= 0;
|
|
386
|
+
if (!(maxBytesToWrite > 0)) return 0;
|
|
387
|
+
var startIdx = outIdx;
|
|
388
|
+
var endIdx = outIdx + maxBytesToWrite - 1;
|
|
389
|
+
for (var i = 0; i < str.length; ++i) {
|
|
390
|
+
var u = str.codePointAt(i);
|
|
391
|
+
if (u <= 127) {
|
|
392
|
+
if (outIdx >= endIdx) break;
|
|
393
|
+
heap[outIdx++ >>> 0] = u;
|
|
394
|
+
} else if (u <= 2047) {
|
|
395
|
+
if (outIdx + 1 >= endIdx) break;
|
|
396
|
+
heap[outIdx++ >>> 0] = 192 | u >> 6;
|
|
397
|
+
heap[outIdx++ >>> 0] = 128 | u & 63;
|
|
398
|
+
} else if (u <= 65535) {
|
|
399
|
+
if (outIdx + 2 >= endIdx) break;
|
|
400
|
+
heap[outIdx++ >>> 0] = 224 | u >> 12;
|
|
401
|
+
heap[outIdx++ >>> 0] = 128 | u >> 6 & 63;
|
|
402
|
+
heap[outIdx++ >>> 0] = 128 | u & 63;
|
|
403
|
+
} else {
|
|
404
|
+
if (outIdx + 3 >= endIdx) break;
|
|
405
|
+
heap[outIdx++ >>> 0] = 240 | u >> 18;
|
|
406
|
+
heap[outIdx++ >>> 0] = 128 | u >> 12 & 63;
|
|
407
|
+
heap[outIdx++ >>> 0] = 128 | u >> 6 & 63;
|
|
408
|
+
heap[outIdx++ >>> 0] = 128 | u & 63;
|
|
409
|
+
i++;
|
|
410
|
+
}
|
|
411
|
+
}
|
|
412
|
+
heap[outIdx >>> 0] = 0;
|
|
413
|
+
return outIdx - startIdx;
|
|
414
|
+
};
|
|
415
|
+
var intArrayFromString = (stringy, dontAddNull, length) => {
|
|
416
|
+
var len = length > 0 ? length : lengthBytesUTF8(stringy) + 1;
|
|
417
|
+
var u8array = new Array(len);
|
|
418
|
+
var numBytesWritten = stringToUTF8Array(stringy, u8array, 0, u8array.length);
|
|
419
|
+
if (dontAddNull) u8array.length = numBytesWritten;
|
|
420
|
+
return u8array;
|
|
421
|
+
};
|
|
422
|
+
var FS_stdin_getChar = () => {
|
|
423
|
+
if (!FS_stdin_getChar_buffer.length) {
|
|
424
|
+
var result = null;
|
|
425
|
+
if (globalThis.window?.prompt) {
|
|
426
|
+
result = window.prompt("Input: ");
|
|
427
|
+
if (result !== null) result += "\n";
|
|
428
|
+
}
|
|
429
|
+
if (!result) return null;
|
|
430
|
+
FS_stdin_getChar_buffer = intArrayFromString(result, true);
|
|
431
|
+
}
|
|
432
|
+
return FS_stdin_getChar_buffer.shift();
|
|
433
|
+
};
|
|
434
|
+
var TTY = {
|
|
435
|
+
ttys: [],
|
|
436
|
+
init() {},
|
|
437
|
+
shutdown() {},
|
|
438
|
+
register(dev, ops) {
|
|
439
|
+
TTY.ttys[dev] = {
|
|
440
|
+
input: [],
|
|
441
|
+
output: [],
|
|
442
|
+
ops
|
|
443
|
+
};
|
|
444
|
+
FS.registerDevice(dev, TTY.stream_ops);
|
|
445
|
+
},
|
|
446
|
+
stream_ops: {
|
|
447
|
+
open(stream) {
|
|
448
|
+
var tty = TTY.ttys[stream.node.rdev];
|
|
449
|
+
if (!tty) throw new FS.ErrnoError(43);
|
|
450
|
+
stream.tty = tty;
|
|
451
|
+
stream.seekable = false;
|
|
452
|
+
},
|
|
453
|
+
close(stream) {
|
|
454
|
+
stream.tty.ops.fsync(stream.tty);
|
|
455
|
+
},
|
|
456
|
+
fsync(stream) {
|
|
457
|
+
stream.tty.ops.fsync(stream.tty);
|
|
458
|
+
},
|
|
459
|
+
read(stream, buffer, offset, length, pos) {
|
|
460
|
+
if (!stream.tty || !stream.tty.ops.get_char) throw new FS.ErrnoError(60);
|
|
461
|
+
var bytesRead = 0;
|
|
462
|
+
for (var i = 0; i < length; i++) {
|
|
463
|
+
var result;
|
|
464
|
+
try {
|
|
465
|
+
result = stream.tty.ops.get_char(stream.tty);
|
|
466
|
+
} catch (e) {
|
|
467
|
+
throw new FS.ErrnoError(29);
|
|
468
|
+
}
|
|
469
|
+
if (result === void 0 && bytesRead === 0) throw new FS.ErrnoError(6);
|
|
470
|
+
if (result === null || result === void 0) break;
|
|
471
|
+
bytesRead++;
|
|
472
|
+
buffer[offset + i] = result;
|
|
473
|
+
}
|
|
474
|
+
if (bytesRead) stream.node.atime = Date.now();
|
|
475
|
+
return bytesRead;
|
|
476
|
+
},
|
|
477
|
+
write(stream, buffer, offset, length, pos) {
|
|
478
|
+
if (!stream.tty || !stream.tty.ops.put_char) throw new FS.ErrnoError(60);
|
|
479
|
+
try {
|
|
480
|
+
for (var i = 0; i < length; i++) stream.tty.ops.put_char(stream.tty, buffer[offset + i]);
|
|
481
|
+
} catch (e) {
|
|
482
|
+
throw new FS.ErrnoError(29);
|
|
483
|
+
}
|
|
484
|
+
if (length) stream.node.mtime = stream.node.ctime = Date.now();
|
|
485
|
+
return i;
|
|
486
|
+
}
|
|
487
|
+
},
|
|
488
|
+
default_tty_ops: {
|
|
489
|
+
get_char(tty) {
|
|
490
|
+
return FS_stdin_getChar();
|
|
491
|
+
},
|
|
492
|
+
put_char(tty, val) {
|
|
493
|
+
if (val === null || val === 10) {
|
|
494
|
+
out(UTF8ArrayToString(tty.output));
|
|
495
|
+
tty.output = [];
|
|
496
|
+
} else if (val != 0) tty.output.push(val);
|
|
497
|
+
},
|
|
498
|
+
fsync(tty) {
|
|
499
|
+
if (tty.output?.length > 0) {
|
|
500
|
+
out(UTF8ArrayToString(tty.output));
|
|
501
|
+
tty.output = [];
|
|
502
|
+
}
|
|
503
|
+
},
|
|
504
|
+
ioctl_tcgets(tty) {
|
|
505
|
+
return {
|
|
506
|
+
c_iflag: 25856,
|
|
507
|
+
c_oflag: 5,
|
|
508
|
+
c_cflag: 191,
|
|
509
|
+
c_lflag: 35387,
|
|
510
|
+
c_cc: [
|
|
511
|
+
3,
|
|
512
|
+
28,
|
|
513
|
+
127,
|
|
514
|
+
21,
|
|
515
|
+
4,
|
|
516
|
+
0,
|
|
517
|
+
1,
|
|
518
|
+
0,
|
|
519
|
+
17,
|
|
520
|
+
19,
|
|
521
|
+
26,
|
|
522
|
+
0,
|
|
523
|
+
18,
|
|
524
|
+
15,
|
|
525
|
+
23,
|
|
526
|
+
22,
|
|
527
|
+
0,
|
|
528
|
+
0,
|
|
529
|
+
0,
|
|
530
|
+
0,
|
|
531
|
+
0,
|
|
532
|
+
0,
|
|
533
|
+
0,
|
|
534
|
+
0,
|
|
535
|
+
0,
|
|
536
|
+
0,
|
|
537
|
+
0,
|
|
538
|
+
0,
|
|
539
|
+
0,
|
|
540
|
+
0,
|
|
541
|
+
0,
|
|
542
|
+
0
|
|
543
|
+
]
|
|
544
|
+
};
|
|
545
|
+
},
|
|
546
|
+
ioctl_tcsets(tty, optional_actions, data) {
|
|
547
|
+
return 0;
|
|
548
|
+
},
|
|
549
|
+
ioctl_tiocgwinsz(tty) {
|
|
550
|
+
return [24, 80];
|
|
551
|
+
}
|
|
552
|
+
},
|
|
553
|
+
default_tty1_ops: {
|
|
554
|
+
put_char(tty, val) {
|
|
555
|
+
if (val === null || val === 10) {
|
|
556
|
+
err(UTF8ArrayToString(tty.output));
|
|
557
|
+
tty.output = [];
|
|
558
|
+
} else if (val != 0) tty.output.push(val);
|
|
559
|
+
},
|
|
560
|
+
fsync(tty) {
|
|
561
|
+
if (tty.output?.length > 0) {
|
|
562
|
+
err(UTF8ArrayToString(tty.output));
|
|
563
|
+
tty.output = [];
|
|
564
|
+
}
|
|
565
|
+
}
|
|
566
|
+
}
|
|
567
|
+
};
|
|
568
|
+
var zeroMemory = (ptr, size) => HEAPU8.fill(0, ptr, ptr + size);
|
|
569
|
+
var alignMemory = (size, alignment) => Math.ceil(size / alignment) * alignment;
|
|
570
|
+
var mmapAlloc = (size) => {
|
|
571
|
+
size = alignMemory(size, 65536);
|
|
572
|
+
var ptr = _emscripten_builtin_memalign(65536, size);
|
|
573
|
+
if (ptr) zeroMemory(ptr, size);
|
|
574
|
+
return ptr;
|
|
575
|
+
};
|
|
576
|
+
var MEMFS = {
|
|
577
|
+
ops_table: null,
|
|
578
|
+
mount(mount) {
|
|
579
|
+
return MEMFS.createNode(null, "/", 16895, 0);
|
|
580
|
+
},
|
|
581
|
+
createNode(parent, name, mode, dev) {
|
|
582
|
+
if (FS.isBlkdev(mode) || FS.isFIFO(mode)) throw new FS.ErrnoError(63);
|
|
583
|
+
MEMFS.ops_table ||= {
|
|
584
|
+
dir: {
|
|
585
|
+
node: {
|
|
586
|
+
getattr: MEMFS.node_ops.getattr,
|
|
587
|
+
setattr: MEMFS.node_ops.setattr,
|
|
588
|
+
lookup: MEMFS.node_ops.lookup,
|
|
589
|
+
mknod: MEMFS.node_ops.mknod,
|
|
590
|
+
rename: MEMFS.node_ops.rename,
|
|
591
|
+
unlink: MEMFS.node_ops.unlink,
|
|
592
|
+
rmdir: MEMFS.node_ops.rmdir,
|
|
593
|
+
readdir: MEMFS.node_ops.readdir,
|
|
594
|
+
symlink: MEMFS.node_ops.symlink
|
|
595
|
+
},
|
|
596
|
+
stream: { llseek: MEMFS.stream_ops.llseek }
|
|
597
|
+
},
|
|
598
|
+
file: {
|
|
599
|
+
node: {
|
|
600
|
+
getattr: MEMFS.node_ops.getattr,
|
|
601
|
+
setattr: MEMFS.node_ops.setattr
|
|
602
|
+
},
|
|
603
|
+
stream: {
|
|
604
|
+
llseek: MEMFS.stream_ops.llseek,
|
|
605
|
+
read: MEMFS.stream_ops.read,
|
|
606
|
+
write: MEMFS.stream_ops.write,
|
|
607
|
+
mmap: MEMFS.stream_ops.mmap,
|
|
608
|
+
msync: MEMFS.stream_ops.msync
|
|
609
|
+
}
|
|
610
|
+
},
|
|
611
|
+
link: {
|
|
612
|
+
node: {
|
|
613
|
+
getattr: MEMFS.node_ops.getattr,
|
|
614
|
+
setattr: MEMFS.node_ops.setattr,
|
|
615
|
+
readlink: MEMFS.node_ops.readlink
|
|
616
|
+
},
|
|
617
|
+
stream: {}
|
|
618
|
+
},
|
|
619
|
+
chrdev: {
|
|
620
|
+
node: {
|
|
621
|
+
getattr: MEMFS.node_ops.getattr,
|
|
622
|
+
setattr: MEMFS.node_ops.setattr
|
|
623
|
+
},
|
|
624
|
+
stream: FS.chrdev_stream_ops
|
|
625
|
+
}
|
|
626
|
+
};
|
|
627
|
+
var node = FS.createNode(parent, name, mode, dev);
|
|
628
|
+
if (FS.isDir(node.mode)) {
|
|
629
|
+
node.node_ops = MEMFS.ops_table.dir.node;
|
|
630
|
+
node.stream_ops = MEMFS.ops_table.dir.stream;
|
|
631
|
+
node.contents = {};
|
|
632
|
+
} else if (FS.isFile(node.mode)) {
|
|
633
|
+
node.node_ops = MEMFS.ops_table.file.node;
|
|
634
|
+
node.stream_ops = MEMFS.ops_table.file.stream;
|
|
635
|
+
node.usedBytes = 0;
|
|
636
|
+
node.contents = null;
|
|
637
|
+
} else if (FS.isLink(node.mode)) {
|
|
638
|
+
node.node_ops = MEMFS.ops_table.link.node;
|
|
639
|
+
node.stream_ops = MEMFS.ops_table.link.stream;
|
|
640
|
+
} else if (FS.isChrdev(node.mode)) {
|
|
641
|
+
node.node_ops = MEMFS.ops_table.chrdev.node;
|
|
642
|
+
node.stream_ops = MEMFS.ops_table.chrdev.stream;
|
|
643
|
+
}
|
|
644
|
+
node.atime = node.mtime = node.ctime = Date.now();
|
|
645
|
+
if (parent) {
|
|
646
|
+
parent.contents[name] = node;
|
|
647
|
+
parent.atime = parent.mtime = parent.ctime = node.atime;
|
|
648
|
+
}
|
|
649
|
+
return node;
|
|
650
|
+
},
|
|
651
|
+
getFileDataAsTypedArray(node) {
|
|
652
|
+
if (!node.contents) return new Uint8Array(0);
|
|
653
|
+
if (node.contents.subarray) return node.contents.subarray(0, node.usedBytes);
|
|
654
|
+
return new Uint8Array(node.contents);
|
|
655
|
+
},
|
|
656
|
+
expandFileStorage(node, newCapacity) {
|
|
657
|
+
var prevCapacity = node.contents ? node.contents.length : 0;
|
|
658
|
+
if (prevCapacity >= newCapacity) return;
|
|
659
|
+
newCapacity = Math.max(newCapacity, prevCapacity * (prevCapacity < 1024 * 1024 ? 2 : 1.125) >>> 0);
|
|
660
|
+
if (prevCapacity != 0) newCapacity = Math.max(newCapacity, 256);
|
|
661
|
+
var oldContents = node.contents;
|
|
662
|
+
node.contents = new Uint8Array(newCapacity);
|
|
663
|
+
if (node.usedBytes > 0) node.contents.set(oldContents.subarray(0, node.usedBytes), 0);
|
|
664
|
+
},
|
|
665
|
+
resizeFileStorage(node, newSize) {
|
|
666
|
+
if (node.usedBytes == newSize) return;
|
|
667
|
+
if (newSize == 0) {
|
|
668
|
+
node.contents = null;
|
|
669
|
+
node.usedBytes = 0;
|
|
670
|
+
} else {
|
|
671
|
+
var oldContents = node.contents;
|
|
672
|
+
node.contents = new Uint8Array(newSize);
|
|
673
|
+
if (oldContents) node.contents.set(oldContents.subarray(0, Math.min(newSize, node.usedBytes)));
|
|
674
|
+
node.usedBytes = newSize;
|
|
675
|
+
}
|
|
676
|
+
},
|
|
677
|
+
node_ops: {
|
|
678
|
+
getattr(node) {
|
|
679
|
+
var attr = {};
|
|
680
|
+
attr.dev = FS.isChrdev(node.mode) ? node.id : 1;
|
|
681
|
+
attr.ino = node.id;
|
|
682
|
+
attr.mode = node.mode;
|
|
683
|
+
attr.nlink = 1;
|
|
684
|
+
attr.uid = 0;
|
|
685
|
+
attr.gid = 0;
|
|
686
|
+
attr.rdev = node.rdev;
|
|
687
|
+
if (FS.isDir(node.mode)) attr.size = 4096;
|
|
688
|
+
else if (FS.isFile(node.mode)) attr.size = node.usedBytes;
|
|
689
|
+
else if (FS.isLink(node.mode)) attr.size = node.link.length;
|
|
690
|
+
else attr.size = 0;
|
|
691
|
+
attr.atime = new Date(node.atime);
|
|
692
|
+
attr.mtime = new Date(node.mtime);
|
|
693
|
+
attr.ctime = new Date(node.ctime);
|
|
694
|
+
attr.blksize = 4096;
|
|
695
|
+
attr.blocks = Math.ceil(attr.size / attr.blksize);
|
|
696
|
+
return attr;
|
|
697
|
+
},
|
|
698
|
+
setattr(node, attr) {
|
|
699
|
+
for (const key of [
|
|
700
|
+
"mode",
|
|
701
|
+
"atime",
|
|
702
|
+
"mtime",
|
|
703
|
+
"ctime"
|
|
704
|
+
]) if (attr[key] != null) node[key] = attr[key];
|
|
705
|
+
if (attr.size !== void 0) MEMFS.resizeFileStorage(node, attr.size);
|
|
706
|
+
},
|
|
707
|
+
lookup(parent, name) {
|
|
708
|
+
if (!MEMFS.doesNotExistError) {
|
|
709
|
+
MEMFS.doesNotExistError = new FS.ErrnoError(44);
|
|
710
|
+
MEMFS.doesNotExistError.stack = "<generic error, no stack>";
|
|
711
|
+
}
|
|
712
|
+
throw MEMFS.doesNotExistError;
|
|
713
|
+
},
|
|
714
|
+
mknod(parent, name, mode, dev) {
|
|
715
|
+
return MEMFS.createNode(parent, name, mode, dev);
|
|
716
|
+
},
|
|
717
|
+
rename(old_node, new_dir, new_name) {
|
|
718
|
+
var new_node;
|
|
719
|
+
try {
|
|
720
|
+
new_node = FS.lookupNode(new_dir, new_name);
|
|
721
|
+
} catch (e) {}
|
|
722
|
+
if (new_node) {
|
|
723
|
+
if (FS.isDir(old_node.mode)) for (var i in new_node.contents) throw new FS.ErrnoError(55);
|
|
724
|
+
FS.hashRemoveNode(new_node);
|
|
725
|
+
}
|
|
726
|
+
delete old_node.parent.contents[old_node.name];
|
|
727
|
+
new_dir.contents[new_name] = old_node;
|
|
728
|
+
old_node.name = new_name;
|
|
729
|
+
new_dir.ctime = new_dir.mtime = old_node.parent.ctime = old_node.parent.mtime = Date.now();
|
|
730
|
+
},
|
|
731
|
+
unlink(parent, name) {
|
|
732
|
+
delete parent.contents[name];
|
|
733
|
+
parent.ctime = parent.mtime = Date.now();
|
|
734
|
+
},
|
|
735
|
+
rmdir(parent, name) {
|
|
736
|
+
for (var i in FS.lookupNode(parent, name).contents) throw new FS.ErrnoError(55);
|
|
737
|
+
delete parent.contents[name];
|
|
738
|
+
parent.ctime = parent.mtime = Date.now();
|
|
739
|
+
},
|
|
740
|
+
readdir(node) {
|
|
741
|
+
return [
|
|
742
|
+
".",
|
|
743
|
+
"..",
|
|
744
|
+
...Object.keys(node.contents)
|
|
745
|
+
];
|
|
746
|
+
},
|
|
747
|
+
symlink(parent, newname, oldpath) {
|
|
748
|
+
var node = MEMFS.createNode(parent, newname, 41471, 0);
|
|
749
|
+
node.link = oldpath;
|
|
750
|
+
return node;
|
|
751
|
+
},
|
|
752
|
+
readlink(node) {
|
|
753
|
+
if (!FS.isLink(node.mode)) throw new FS.ErrnoError(28);
|
|
754
|
+
return node.link;
|
|
755
|
+
}
|
|
756
|
+
},
|
|
757
|
+
stream_ops: {
|
|
758
|
+
read(stream, buffer, offset, length, position) {
|
|
759
|
+
var contents = stream.node.contents;
|
|
760
|
+
if (position >= stream.node.usedBytes) return 0;
|
|
761
|
+
var size = Math.min(stream.node.usedBytes - position, length);
|
|
762
|
+
if (size > 8 && contents.subarray) buffer.set(contents.subarray(position, position + size), offset);
|
|
763
|
+
else for (var i = 0; i < size; i++) buffer[offset + i] = contents[position + i];
|
|
764
|
+
return size;
|
|
765
|
+
},
|
|
766
|
+
write(stream, buffer, offset, length, position, canOwn) {
|
|
767
|
+
if (buffer.buffer === HEAP8.buffer) canOwn = false;
|
|
768
|
+
if (!length) return 0;
|
|
769
|
+
var node = stream.node;
|
|
770
|
+
node.mtime = node.ctime = Date.now();
|
|
771
|
+
if (buffer.subarray && (!node.contents || node.contents.subarray)) {
|
|
772
|
+
if (canOwn) {
|
|
773
|
+
node.contents = buffer.subarray(offset, offset + length);
|
|
774
|
+
node.usedBytes = length;
|
|
775
|
+
return length;
|
|
776
|
+
} else if (node.usedBytes === 0 && position === 0) {
|
|
777
|
+
node.contents = buffer.slice(offset, offset + length);
|
|
778
|
+
node.usedBytes = length;
|
|
779
|
+
return length;
|
|
780
|
+
} else if (position + length <= node.usedBytes) {
|
|
781
|
+
node.contents.set(buffer.subarray(offset, offset + length), position);
|
|
782
|
+
return length;
|
|
783
|
+
}
|
|
784
|
+
}
|
|
785
|
+
MEMFS.expandFileStorage(node, position + length);
|
|
786
|
+
if (node.contents.subarray && buffer.subarray) node.contents.set(buffer.subarray(offset, offset + length), position);
|
|
787
|
+
else for (var i = 0; i < length; i++) node.contents[position + i] = buffer[offset + i];
|
|
788
|
+
node.usedBytes = Math.max(node.usedBytes, position + length);
|
|
789
|
+
return length;
|
|
790
|
+
},
|
|
791
|
+
llseek(stream, offset, whence) {
|
|
792
|
+
var position = offset;
|
|
793
|
+
if (whence === 1) position += stream.position;
|
|
794
|
+
else if (whence === 2) {
|
|
795
|
+
if (FS.isFile(stream.node.mode)) position += stream.node.usedBytes;
|
|
796
|
+
}
|
|
797
|
+
if (position < 0) throw new FS.ErrnoError(28);
|
|
798
|
+
return position;
|
|
799
|
+
},
|
|
800
|
+
mmap(stream, length, position, prot, flags) {
|
|
801
|
+
if (!FS.isFile(stream.node.mode)) throw new FS.ErrnoError(43);
|
|
802
|
+
var ptr;
|
|
803
|
+
var allocated;
|
|
804
|
+
var contents = stream.node.contents;
|
|
805
|
+
if (!(flags & 2) && contents && contents.buffer === HEAP8.buffer) {
|
|
806
|
+
allocated = false;
|
|
807
|
+
ptr = contents.byteOffset;
|
|
808
|
+
} else {
|
|
809
|
+
allocated = true;
|
|
810
|
+
ptr = mmapAlloc(length);
|
|
811
|
+
if (!ptr) throw new FS.ErrnoError(48);
|
|
812
|
+
if (contents) {
|
|
813
|
+
if (position > 0 || position + length < contents.length) if (contents.subarray) contents = contents.subarray(position, position + length);
|
|
814
|
+
else contents = Array.prototype.slice.call(contents, position, position + length);
|
|
815
|
+
HEAP8.set(contents, ptr >>> 0);
|
|
816
|
+
}
|
|
817
|
+
}
|
|
818
|
+
return {
|
|
819
|
+
ptr,
|
|
820
|
+
allocated
|
|
821
|
+
};
|
|
822
|
+
},
|
|
823
|
+
msync(stream, buffer, offset, length, mmapFlags) {
|
|
824
|
+
MEMFS.stream_ops.write(stream, buffer, 0, length, offset, false);
|
|
825
|
+
return 0;
|
|
826
|
+
}
|
|
827
|
+
}
|
|
828
|
+
};
|
|
829
|
+
var FS_modeStringToFlags = (str) => {
|
|
830
|
+
var flags = {
|
|
831
|
+
r: 0,
|
|
832
|
+
"r+": 2,
|
|
833
|
+
w: 577,
|
|
834
|
+
"w+": 578,
|
|
835
|
+
a: 1089,
|
|
836
|
+
"a+": 1090
|
|
837
|
+
}[str];
|
|
838
|
+
if (typeof flags == "undefined") throw new Error(`Unknown file open mode: ${str}`);
|
|
839
|
+
return flags;
|
|
840
|
+
};
|
|
841
|
+
var FS_getMode = (canRead, canWrite) => {
|
|
842
|
+
var mode = 0;
|
|
843
|
+
if (canRead) mode |= 365;
|
|
844
|
+
if (canWrite) mode |= 146;
|
|
845
|
+
return mode;
|
|
846
|
+
};
|
|
847
|
+
var IDBFS = {
|
|
848
|
+
dbs: {},
|
|
849
|
+
indexedDB: () => indexedDB,
|
|
850
|
+
DB_VERSION: 21,
|
|
851
|
+
DB_STORE_NAME: "FILE_DATA",
|
|
852
|
+
queuePersist: (mount) => {
|
|
853
|
+
function onPersistComplete() {
|
|
854
|
+
if (mount.idbPersistState === "again") startPersist();
|
|
855
|
+
else mount.idbPersistState = 0;
|
|
856
|
+
}
|
|
857
|
+
function startPersist() {
|
|
858
|
+
mount.idbPersistState = "idb";
|
|
859
|
+
IDBFS.syncfs(mount, false, onPersistComplete);
|
|
860
|
+
}
|
|
861
|
+
if (!mount.idbPersistState) mount.idbPersistState = setTimeout(startPersist, 0);
|
|
862
|
+
else if (mount.idbPersistState === "idb") mount.idbPersistState = "again";
|
|
863
|
+
},
|
|
864
|
+
mount: (mount) => {
|
|
865
|
+
var mnt = MEMFS.mount(mount);
|
|
866
|
+
if (mount?.opts?.autoPersist) {
|
|
867
|
+
mount.idbPersistState = 0;
|
|
868
|
+
var memfs_node_ops = mnt.node_ops;
|
|
869
|
+
mnt.node_ops = { ...mnt.node_ops };
|
|
870
|
+
mnt.node_ops.mknod = (parent, name, mode, dev) => {
|
|
871
|
+
var node = memfs_node_ops.mknod(parent, name, mode, dev);
|
|
872
|
+
node.node_ops = mnt.node_ops;
|
|
873
|
+
node.idbfs_mount = mnt.mount;
|
|
874
|
+
node.memfs_stream_ops = node.stream_ops;
|
|
875
|
+
node.stream_ops = { ...node.stream_ops };
|
|
876
|
+
node.stream_ops.write = (stream, buffer, offset, length, position, canOwn) => {
|
|
877
|
+
stream.node.isModified = true;
|
|
878
|
+
return node.memfs_stream_ops.write(stream, buffer, offset, length, position, canOwn);
|
|
879
|
+
};
|
|
880
|
+
node.stream_ops.close = (stream) => {
|
|
881
|
+
var n = stream.node;
|
|
882
|
+
if (n.isModified) {
|
|
883
|
+
IDBFS.queuePersist(n.idbfs_mount);
|
|
884
|
+
n.isModified = false;
|
|
885
|
+
}
|
|
886
|
+
if (n.memfs_stream_ops.close) return n.memfs_stream_ops.close(stream);
|
|
887
|
+
};
|
|
888
|
+
IDBFS.queuePersist(mnt.mount);
|
|
889
|
+
return node;
|
|
890
|
+
};
|
|
891
|
+
mnt.node_ops.rmdir = (...args) => (IDBFS.queuePersist(mnt.mount), memfs_node_ops.rmdir(...args));
|
|
892
|
+
mnt.node_ops.symlink = (...args) => (IDBFS.queuePersist(mnt.mount), memfs_node_ops.symlink(...args));
|
|
893
|
+
mnt.node_ops.unlink = (...args) => (IDBFS.queuePersist(mnt.mount), memfs_node_ops.unlink(...args));
|
|
894
|
+
mnt.node_ops.rename = (...args) => (IDBFS.queuePersist(mnt.mount), memfs_node_ops.rename(...args));
|
|
895
|
+
}
|
|
896
|
+
return mnt;
|
|
897
|
+
},
|
|
898
|
+
syncfs: (mount, populate, callback) => {
|
|
899
|
+
IDBFS.getLocalSet(mount, (err, local) => {
|
|
900
|
+
if (err) return callback(err);
|
|
901
|
+
IDBFS.getRemoteSet(mount, (err, remote) => {
|
|
902
|
+
if (err) return callback(err);
|
|
903
|
+
var src = populate ? remote : local;
|
|
904
|
+
var dst = populate ? local : remote;
|
|
905
|
+
IDBFS.reconcile(src, dst, callback);
|
|
906
|
+
});
|
|
907
|
+
});
|
|
908
|
+
},
|
|
909
|
+
quit: () => {
|
|
910
|
+
for (var value of Object.values(IDBFS.dbs)) value.close();
|
|
911
|
+
IDBFS.dbs = {};
|
|
912
|
+
},
|
|
913
|
+
getDB: (name, callback) => {
|
|
914
|
+
var db = IDBFS.dbs[name];
|
|
915
|
+
if (db) return callback(null, db);
|
|
916
|
+
var req;
|
|
917
|
+
try {
|
|
918
|
+
req = IDBFS.indexedDB().open(name, IDBFS.DB_VERSION);
|
|
919
|
+
} catch (e) {
|
|
920
|
+
return callback(e);
|
|
921
|
+
}
|
|
922
|
+
if (!req) return callback("Unable to connect to IndexedDB");
|
|
923
|
+
req.onupgradeneeded = (e) => {
|
|
924
|
+
var db = e.target.result;
|
|
925
|
+
var transaction = e.target.transaction;
|
|
926
|
+
var fileStore;
|
|
927
|
+
if (db.objectStoreNames.contains(IDBFS.DB_STORE_NAME)) fileStore = transaction.objectStore(IDBFS.DB_STORE_NAME);
|
|
928
|
+
else fileStore = db.createObjectStore(IDBFS.DB_STORE_NAME);
|
|
929
|
+
if (!fileStore.indexNames.contains("timestamp")) fileStore.createIndex("timestamp", "timestamp", { unique: false });
|
|
930
|
+
};
|
|
931
|
+
req.onsuccess = () => {
|
|
932
|
+
db = req.result;
|
|
933
|
+
IDBFS.dbs[name] = db;
|
|
934
|
+
callback(null, db);
|
|
935
|
+
};
|
|
936
|
+
req.onerror = (e) => {
|
|
937
|
+
callback(e.target.error);
|
|
938
|
+
e.preventDefault();
|
|
939
|
+
};
|
|
940
|
+
},
|
|
941
|
+
getLocalSet: (mount, callback) => {
|
|
942
|
+
var entries = {};
|
|
943
|
+
function isRealDir(p) {
|
|
944
|
+
return p !== "." && p !== "..";
|
|
945
|
+
}
|
|
946
|
+
function toAbsolute(root) {
|
|
947
|
+
return (p) => PATH.join2(root, p);
|
|
948
|
+
}
|
|
949
|
+
var check = FS.readdir(mount.mountpoint).filter(isRealDir).map(toAbsolute(mount.mountpoint));
|
|
950
|
+
while (check.length) {
|
|
951
|
+
var path = check.pop();
|
|
952
|
+
var stat;
|
|
953
|
+
try {
|
|
954
|
+
stat = FS.stat(path);
|
|
955
|
+
} catch (e) {
|
|
956
|
+
return callback(e);
|
|
957
|
+
}
|
|
958
|
+
if (FS.isDir(stat.mode)) check.push(...FS.readdir(path).filter(isRealDir).map(toAbsolute(path)));
|
|
959
|
+
entries[path] = { timestamp: stat.mtime };
|
|
960
|
+
}
|
|
961
|
+
return callback(null, {
|
|
962
|
+
type: "local",
|
|
963
|
+
entries
|
|
964
|
+
});
|
|
965
|
+
},
|
|
966
|
+
getRemoteSet: (mount, callback) => {
|
|
967
|
+
var entries = {};
|
|
968
|
+
IDBFS.getDB(mount.mountpoint, (err, db) => {
|
|
969
|
+
if (err) return callback(err);
|
|
970
|
+
try {
|
|
971
|
+
var transaction = db.transaction([IDBFS.DB_STORE_NAME], "readonly");
|
|
972
|
+
transaction.onerror = (e) => {
|
|
973
|
+
callback(e.target.error);
|
|
974
|
+
e.preventDefault();
|
|
975
|
+
};
|
|
976
|
+
var index = transaction.objectStore(IDBFS.DB_STORE_NAME).index("timestamp");
|
|
977
|
+
index.openKeyCursor().onsuccess = (event) => {
|
|
978
|
+
var cursor = event.target.result;
|
|
979
|
+
if (!cursor) return callback(null, {
|
|
980
|
+
type: "remote",
|
|
981
|
+
db,
|
|
982
|
+
entries
|
|
983
|
+
});
|
|
984
|
+
entries[cursor.primaryKey] = { timestamp: cursor.key };
|
|
985
|
+
cursor.continue();
|
|
986
|
+
};
|
|
987
|
+
} catch (e) {
|
|
988
|
+
return callback(e);
|
|
989
|
+
}
|
|
990
|
+
});
|
|
991
|
+
},
|
|
992
|
+
loadLocalEntry: (path, callback) => {
|
|
993
|
+
var stat, node;
|
|
994
|
+
try {
|
|
995
|
+
node = FS.lookupPath(path).node;
|
|
996
|
+
stat = FS.stat(path);
|
|
997
|
+
} catch (e) {
|
|
998
|
+
return callback(e);
|
|
999
|
+
}
|
|
1000
|
+
if (FS.isDir(stat.mode)) return callback(null, {
|
|
1001
|
+
timestamp: stat.mtime,
|
|
1002
|
+
mode: stat.mode
|
|
1003
|
+
});
|
|
1004
|
+
else if (FS.isFile(stat.mode)) {
|
|
1005
|
+
node.contents = MEMFS.getFileDataAsTypedArray(node);
|
|
1006
|
+
return callback(null, {
|
|
1007
|
+
timestamp: stat.mtime,
|
|
1008
|
+
mode: stat.mode,
|
|
1009
|
+
contents: node.contents
|
|
1010
|
+
});
|
|
1011
|
+
} else return callback(/* @__PURE__ */ new Error("node type not supported"));
|
|
1012
|
+
},
|
|
1013
|
+
storeLocalEntry: (path, entry, callback) => {
|
|
1014
|
+
try {
|
|
1015
|
+
if (FS.isDir(entry["mode"])) FS.mkdirTree(path, entry["mode"]);
|
|
1016
|
+
else if (FS.isFile(entry["mode"])) FS.writeFile(path, entry["contents"], { canOwn: true });
|
|
1017
|
+
else return callback(/* @__PURE__ */ new Error("node type not supported"));
|
|
1018
|
+
FS.chmod(path, entry["mode"]);
|
|
1019
|
+
FS.utime(path, entry["timestamp"], entry["timestamp"]);
|
|
1020
|
+
} catch (e) {
|
|
1021
|
+
return callback(e);
|
|
1022
|
+
}
|
|
1023
|
+
callback(null);
|
|
1024
|
+
},
|
|
1025
|
+
removeLocalEntry: (path, callback) => {
|
|
1026
|
+
try {
|
|
1027
|
+
var stat = FS.stat(path);
|
|
1028
|
+
if (FS.isDir(stat.mode)) FS.rmdir(path);
|
|
1029
|
+
else if (FS.isFile(stat.mode)) FS.unlink(path);
|
|
1030
|
+
} catch (e) {
|
|
1031
|
+
return callback(e);
|
|
1032
|
+
}
|
|
1033
|
+
callback(null);
|
|
1034
|
+
},
|
|
1035
|
+
loadRemoteEntry: (store, path, callback) => {
|
|
1036
|
+
var req = store.get(path);
|
|
1037
|
+
req.onsuccess = (event) => callback(null, event.target.result);
|
|
1038
|
+
req.onerror = (e) => {
|
|
1039
|
+
callback(e.target.error);
|
|
1040
|
+
e.preventDefault();
|
|
1041
|
+
};
|
|
1042
|
+
},
|
|
1043
|
+
storeRemoteEntry: (store, path, entry, callback) => {
|
|
1044
|
+
try {
|
|
1045
|
+
var req = store.put(entry, path);
|
|
1046
|
+
} catch (e) {
|
|
1047
|
+
callback(e);
|
|
1048
|
+
return;
|
|
1049
|
+
}
|
|
1050
|
+
req.onsuccess = (event) => callback();
|
|
1051
|
+
req.onerror = (e) => {
|
|
1052
|
+
callback(e.target.error);
|
|
1053
|
+
e.preventDefault();
|
|
1054
|
+
};
|
|
1055
|
+
},
|
|
1056
|
+
removeRemoteEntry: (store, path, callback) => {
|
|
1057
|
+
var req = store.delete(path);
|
|
1058
|
+
req.onsuccess = (event) => callback();
|
|
1059
|
+
req.onerror = (e) => {
|
|
1060
|
+
callback(e.target.error);
|
|
1061
|
+
e.preventDefault();
|
|
1062
|
+
};
|
|
1063
|
+
},
|
|
1064
|
+
reconcile: (src, dst, callback) => {
|
|
1065
|
+
var total = 0;
|
|
1066
|
+
var create = [];
|
|
1067
|
+
for (var [key, e] of Object.entries(src.entries)) {
|
|
1068
|
+
var e2 = dst.entries[key];
|
|
1069
|
+
if (!e2 || e["timestamp"].getTime() != e2["timestamp"].getTime()) {
|
|
1070
|
+
create.push(key);
|
|
1071
|
+
total++;
|
|
1072
|
+
}
|
|
1073
|
+
}
|
|
1074
|
+
var remove = [];
|
|
1075
|
+
for (var key of Object.keys(dst.entries)) if (!src.entries[key]) {
|
|
1076
|
+
remove.push(key);
|
|
1077
|
+
total++;
|
|
1078
|
+
}
|
|
1079
|
+
if (!total) return callback(null);
|
|
1080
|
+
var errored = false;
|
|
1081
|
+
var transaction = (src.type === "remote" ? src.db : dst.db).transaction([IDBFS.DB_STORE_NAME], "readwrite");
|
|
1082
|
+
var store = transaction.objectStore(IDBFS.DB_STORE_NAME);
|
|
1083
|
+
function done(err) {
|
|
1084
|
+
if (err && !errored) {
|
|
1085
|
+
errored = true;
|
|
1086
|
+
return callback(err);
|
|
1087
|
+
}
|
|
1088
|
+
}
|
|
1089
|
+
transaction.onerror = transaction.onabort = (e) => {
|
|
1090
|
+
done(e.target.error);
|
|
1091
|
+
e.preventDefault();
|
|
1092
|
+
};
|
|
1093
|
+
transaction.oncomplete = (e) => {
|
|
1094
|
+
if (!errored) callback(null);
|
|
1095
|
+
};
|
|
1096
|
+
for (const path of create.sort()) if (dst.type === "local") IDBFS.loadRemoteEntry(store, path, (err, entry) => {
|
|
1097
|
+
if (err) return done(err);
|
|
1098
|
+
IDBFS.storeLocalEntry(path, entry, done);
|
|
1099
|
+
});
|
|
1100
|
+
else IDBFS.loadLocalEntry(path, (err, entry) => {
|
|
1101
|
+
if (err) return done(err);
|
|
1102
|
+
IDBFS.storeRemoteEntry(store, path, entry, done);
|
|
1103
|
+
});
|
|
1104
|
+
for (var path of remove.sort().reverse()) if (dst.type === "local") IDBFS.removeLocalEntry(path, done);
|
|
1105
|
+
else IDBFS.removeRemoteEntry(store, path, done);
|
|
1106
|
+
}
|
|
1107
|
+
};
|
|
1108
|
+
var asyncLoad = async (url) => {
|
|
1109
|
+
var arrayBuffer = await readAsync(url);
|
|
1110
|
+
return new Uint8Array(arrayBuffer);
|
|
1111
|
+
};
|
|
1112
|
+
var FS_createDataFile = (...args) => FS.createDataFile(...args);
|
|
1113
|
+
var getUniqueRunDependency = (id) => id;
|
|
1114
|
+
var runDependencies = 0;
|
|
1115
|
+
var dependenciesFulfilled = null;
|
|
1116
|
+
var removeRunDependency = (id) => {
|
|
1117
|
+
runDependencies--;
|
|
1118
|
+
Module["monitorRunDependencies"]?.(runDependencies);
|
|
1119
|
+
if (runDependencies == 0) {
|
|
1120
|
+
if (dependenciesFulfilled) {
|
|
1121
|
+
var callback = dependenciesFulfilled;
|
|
1122
|
+
dependenciesFulfilled = null;
|
|
1123
|
+
callback();
|
|
1124
|
+
}
|
|
1125
|
+
}
|
|
1126
|
+
};
|
|
1127
|
+
var addRunDependency = (id) => {
|
|
1128
|
+
runDependencies++;
|
|
1129
|
+
Module["monitorRunDependencies"]?.(runDependencies);
|
|
1130
|
+
};
|
|
1131
|
+
var preloadPlugins = [];
|
|
1132
|
+
var FS_handledByPreloadPlugin = async (byteArray, fullname) => {
|
|
1133
|
+
if (typeof Browser != "undefined") Browser.init();
|
|
1134
|
+
for (var plugin of preloadPlugins) if (plugin["canHandle"](fullname)) return plugin["handle"](byteArray, fullname);
|
|
1135
|
+
return byteArray;
|
|
1136
|
+
};
|
|
1137
|
+
var FS_preloadFile = async (parent, name, url, canRead, canWrite, dontCreateFile, canOwn, preFinish) => {
|
|
1138
|
+
var fullname = name ? PATH_FS.resolve(PATH.join2(parent, name)) : parent;
|
|
1139
|
+
var dep = getUniqueRunDependency(`cp ${fullname}`);
|
|
1140
|
+
addRunDependency(dep);
|
|
1141
|
+
try {
|
|
1142
|
+
var byteArray = url;
|
|
1143
|
+
if (typeof url == "string") byteArray = await asyncLoad(url);
|
|
1144
|
+
byteArray = await FS_handledByPreloadPlugin(byteArray, fullname);
|
|
1145
|
+
preFinish?.();
|
|
1146
|
+
if (!dontCreateFile) FS_createDataFile(parent, name, byteArray, canRead, canWrite, canOwn);
|
|
1147
|
+
} finally {
|
|
1148
|
+
removeRunDependency(dep);
|
|
1149
|
+
}
|
|
1150
|
+
};
|
|
1151
|
+
var FS_createPreloadedFile = (parent, name, url, canRead, canWrite, onload, onerror, dontCreateFile, canOwn, preFinish) => {
|
|
1152
|
+
FS_preloadFile(parent, name, url, canRead, canWrite, dontCreateFile, canOwn, preFinish).then(onload).catch(onerror);
|
|
1153
|
+
};
|
|
1154
|
+
var FS = {
|
|
1155
|
+
root: null,
|
|
1156
|
+
mounts: [],
|
|
1157
|
+
devices: {},
|
|
1158
|
+
streams: [],
|
|
1159
|
+
nextInode: 1,
|
|
1160
|
+
nameTable: null,
|
|
1161
|
+
currentPath: "/",
|
|
1162
|
+
initialized: false,
|
|
1163
|
+
ignorePermissions: true,
|
|
1164
|
+
filesystems: null,
|
|
1165
|
+
syncFSRequests: 0,
|
|
1166
|
+
ErrnoError: class {
|
|
1167
|
+
name = "ErrnoError";
|
|
1168
|
+
constructor(errno) {
|
|
1169
|
+
this.errno = errno;
|
|
1170
|
+
}
|
|
1171
|
+
},
|
|
1172
|
+
FSStream: class {
|
|
1173
|
+
shared = {};
|
|
1174
|
+
get object() {
|
|
1175
|
+
return this.node;
|
|
1176
|
+
}
|
|
1177
|
+
set object(val) {
|
|
1178
|
+
this.node = val;
|
|
1179
|
+
}
|
|
1180
|
+
get isRead() {
|
|
1181
|
+
return (this.flags & 2097155) !== 1;
|
|
1182
|
+
}
|
|
1183
|
+
get isWrite() {
|
|
1184
|
+
return (this.flags & 2097155) !== 0;
|
|
1185
|
+
}
|
|
1186
|
+
get isAppend() {
|
|
1187
|
+
return this.flags & 1024;
|
|
1188
|
+
}
|
|
1189
|
+
get flags() {
|
|
1190
|
+
return this.shared.flags;
|
|
1191
|
+
}
|
|
1192
|
+
set flags(val) {
|
|
1193
|
+
this.shared.flags = val;
|
|
1194
|
+
}
|
|
1195
|
+
get position() {
|
|
1196
|
+
return this.shared.position;
|
|
1197
|
+
}
|
|
1198
|
+
set position(val) {
|
|
1199
|
+
this.shared.position = val;
|
|
1200
|
+
}
|
|
1201
|
+
},
|
|
1202
|
+
FSNode: class {
|
|
1203
|
+
node_ops = {};
|
|
1204
|
+
stream_ops = {};
|
|
1205
|
+
readMode = 365;
|
|
1206
|
+
writeMode = 146;
|
|
1207
|
+
mounted = null;
|
|
1208
|
+
constructor(parent, name, mode, rdev) {
|
|
1209
|
+
if (!parent) parent = this;
|
|
1210
|
+
this.parent = parent;
|
|
1211
|
+
this.mount = parent.mount;
|
|
1212
|
+
this.id = FS.nextInode++;
|
|
1213
|
+
this.name = name;
|
|
1214
|
+
this.mode = mode;
|
|
1215
|
+
this.rdev = rdev;
|
|
1216
|
+
this.atime = this.mtime = this.ctime = Date.now();
|
|
1217
|
+
}
|
|
1218
|
+
get read() {
|
|
1219
|
+
return (this.mode & this.readMode) === this.readMode;
|
|
1220
|
+
}
|
|
1221
|
+
set read(val) {
|
|
1222
|
+
val ? this.mode |= this.readMode : this.mode &= ~this.readMode;
|
|
1223
|
+
}
|
|
1224
|
+
get write() {
|
|
1225
|
+
return (this.mode & this.writeMode) === this.writeMode;
|
|
1226
|
+
}
|
|
1227
|
+
set write(val) {
|
|
1228
|
+
val ? this.mode |= this.writeMode : this.mode &= ~this.writeMode;
|
|
1229
|
+
}
|
|
1230
|
+
get isFolder() {
|
|
1231
|
+
return FS.isDir(this.mode);
|
|
1232
|
+
}
|
|
1233
|
+
get isDevice() {
|
|
1234
|
+
return FS.isChrdev(this.mode);
|
|
1235
|
+
}
|
|
1236
|
+
},
|
|
1237
|
+
lookupPath(path, opts = {}) {
|
|
1238
|
+
if (!path) throw new FS.ErrnoError(44);
|
|
1239
|
+
opts.follow_mount ??= true;
|
|
1240
|
+
if (!PATH.isAbs(path)) path = FS.cwd() + "/" + path;
|
|
1241
|
+
linkloop: for (var nlinks = 0; nlinks < 40; nlinks++) {
|
|
1242
|
+
var parts = path.split("/").filter((p) => !!p);
|
|
1243
|
+
var current = FS.root;
|
|
1244
|
+
var current_path = "/";
|
|
1245
|
+
for (var i = 0; i < parts.length; i++) {
|
|
1246
|
+
var islast = i === parts.length - 1;
|
|
1247
|
+
if (islast && opts.parent) break;
|
|
1248
|
+
if (parts[i] === ".") continue;
|
|
1249
|
+
if (parts[i] === "..") {
|
|
1250
|
+
current_path = PATH.dirname(current_path);
|
|
1251
|
+
if (FS.isRoot(current)) {
|
|
1252
|
+
path = current_path + "/" + parts.slice(i + 1).join("/");
|
|
1253
|
+
nlinks--;
|
|
1254
|
+
continue linkloop;
|
|
1255
|
+
} else current = current.parent;
|
|
1256
|
+
continue;
|
|
1257
|
+
}
|
|
1258
|
+
current_path = PATH.join2(current_path, parts[i]);
|
|
1259
|
+
try {
|
|
1260
|
+
current = FS.lookupNode(current, parts[i]);
|
|
1261
|
+
} catch (e) {
|
|
1262
|
+
if (e?.errno === 44 && islast && opts.noent_okay) return { path: current_path };
|
|
1263
|
+
throw e;
|
|
1264
|
+
}
|
|
1265
|
+
if (FS.isMountpoint(current) && (!islast || opts.follow_mount)) current = current.mounted.root;
|
|
1266
|
+
if (FS.isLink(current.mode) && (!islast || opts.follow)) {
|
|
1267
|
+
if (!current.node_ops.readlink) throw new FS.ErrnoError(52);
|
|
1268
|
+
var link = current.node_ops.readlink(current);
|
|
1269
|
+
if (!PATH.isAbs(link)) link = PATH.dirname(current_path) + "/" + link;
|
|
1270
|
+
path = link + "/" + parts.slice(i + 1).join("/");
|
|
1271
|
+
continue linkloop;
|
|
1272
|
+
}
|
|
1273
|
+
}
|
|
1274
|
+
return {
|
|
1275
|
+
path: current_path,
|
|
1276
|
+
node: current
|
|
1277
|
+
};
|
|
1278
|
+
}
|
|
1279
|
+
throw new FS.ErrnoError(32);
|
|
1280
|
+
},
|
|
1281
|
+
getPath(node) {
|
|
1282
|
+
var path;
|
|
1283
|
+
while (true) {
|
|
1284
|
+
if (FS.isRoot(node)) {
|
|
1285
|
+
var mount = node.mount.mountpoint;
|
|
1286
|
+
if (!path) return mount;
|
|
1287
|
+
return mount[mount.length - 1] !== "/" ? `${mount}/${path}` : mount + path;
|
|
1288
|
+
}
|
|
1289
|
+
path = path ? `${node.name}/${path}` : node.name;
|
|
1290
|
+
node = node.parent;
|
|
1291
|
+
}
|
|
1292
|
+
},
|
|
1293
|
+
hashName(parentid, name) {
|
|
1294
|
+
var hash = 0;
|
|
1295
|
+
for (var i = 0; i < name.length; i++) hash = (hash << 5) - hash + name.charCodeAt(i) | 0;
|
|
1296
|
+
return (parentid + hash >>> 0) % FS.nameTable.length;
|
|
1297
|
+
},
|
|
1298
|
+
hashAddNode(node) {
|
|
1299
|
+
var hash = FS.hashName(node.parent.id, node.name);
|
|
1300
|
+
node.name_next = FS.nameTable[hash];
|
|
1301
|
+
FS.nameTable[hash] = node;
|
|
1302
|
+
},
|
|
1303
|
+
hashRemoveNode(node) {
|
|
1304
|
+
var hash = FS.hashName(node.parent.id, node.name);
|
|
1305
|
+
if (FS.nameTable[hash] === node) FS.nameTable[hash] = node.name_next;
|
|
1306
|
+
else {
|
|
1307
|
+
var current = FS.nameTable[hash];
|
|
1308
|
+
while (current) {
|
|
1309
|
+
if (current.name_next === node) {
|
|
1310
|
+
current.name_next = node.name_next;
|
|
1311
|
+
break;
|
|
1312
|
+
}
|
|
1313
|
+
current = current.name_next;
|
|
1314
|
+
}
|
|
1315
|
+
}
|
|
1316
|
+
},
|
|
1317
|
+
lookupNode(parent, name) {
|
|
1318
|
+
var errCode = FS.mayLookup(parent);
|
|
1319
|
+
if (errCode) throw new FS.ErrnoError(errCode);
|
|
1320
|
+
var hash = FS.hashName(parent.id, name);
|
|
1321
|
+
for (var node = FS.nameTable[hash]; node; node = node.name_next) {
|
|
1322
|
+
var nodeName = node.name;
|
|
1323
|
+
if (node.parent.id === parent.id && nodeName === name) return node;
|
|
1324
|
+
}
|
|
1325
|
+
return FS.lookup(parent, name);
|
|
1326
|
+
},
|
|
1327
|
+
createNode(parent, name, mode, rdev) {
|
|
1328
|
+
var node = new FS.FSNode(parent, name, mode, rdev);
|
|
1329
|
+
FS.hashAddNode(node);
|
|
1330
|
+
return node;
|
|
1331
|
+
},
|
|
1332
|
+
destroyNode(node) {
|
|
1333
|
+
FS.hashRemoveNode(node);
|
|
1334
|
+
},
|
|
1335
|
+
isRoot(node) {
|
|
1336
|
+
return node === node.parent;
|
|
1337
|
+
},
|
|
1338
|
+
isMountpoint(node) {
|
|
1339
|
+
return !!node.mounted;
|
|
1340
|
+
},
|
|
1341
|
+
isFile(mode) {
|
|
1342
|
+
return (mode & 61440) === 32768;
|
|
1343
|
+
},
|
|
1344
|
+
isDir(mode) {
|
|
1345
|
+
return (mode & 61440) === 16384;
|
|
1346
|
+
},
|
|
1347
|
+
isLink(mode) {
|
|
1348
|
+
return (mode & 61440) === 40960;
|
|
1349
|
+
},
|
|
1350
|
+
isChrdev(mode) {
|
|
1351
|
+
return (mode & 61440) === 8192;
|
|
1352
|
+
},
|
|
1353
|
+
isBlkdev(mode) {
|
|
1354
|
+
return (mode & 61440) === 24576;
|
|
1355
|
+
},
|
|
1356
|
+
isFIFO(mode) {
|
|
1357
|
+
return (mode & 61440) === 4096;
|
|
1358
|
+
},
|
|
1359
|
+
isSocket(mode) {
|
|
1360
|
+
return (mode & 49152) === 49152;
|
|
1361
|
+
},
|
|
1362
|
+
flagsToPermissionString(flag) {
|
|
1363
|
+
var perms = [
|
|
1364
|
+
"r",
|
|
1365
|
+
"w",
|
|
1366
|
+
"rw"
|
|
1367
|
+
][flag & 3];
|
|
1368
|
+
if (flag & 512) perms += "w";
|
|
1369
|
+
return perms;
|
|
1370
|
+
},
|
|
1371
|
+
nodePermissions(node, perms) {
|
|
1372
|
+
if (FS.ignorePermissions) return 0;
|
|
1373
|
+
if (perms.includes("r") && !(node.mode & 292)) return 2;
|
|
1374
|
+
if (perms.includes("w") && !(node.mode & 146)) return 2;
|
|
1375
|
+
if (perms.includes("x") && !(node.mode & 73)) return 2;
|
|
1376
|
+
return 0;
|
|
1377
|
+
},
|
|
1378
|
+
mayLookup(dir) {
|
|
1379
|
+
if (!FS.isDir(dir.mode)) return 54;
|
|
1380
|
+
var errCode = FS.nodePermissions(dir, "x");
|
|
1381
|
+
if (errCode) return errCode;
|
|
1382
|
+
if (!dir.node_ops.lookup) return 2;
|
|
1383
|
+
return 0;
|
|
1384
|
+
},
|
|
1385
|
+
mayCreate(dir, name) {
|
|
1386
|
+
if (!FS.isDir(dir.mode)) return 54;
|
|
1387
|
+
try {
|
|
1388
|
+
FS.lookupNode(dir, name);
|
|
1389
|
+
return 20;
|
|
1390
|
+
} catch (e) {}
|
|
1391
|
+
return FS.nodePermissions(dir, "wx");
|
|
1392
|
+
},
|
|
1393
|
+
mayDelete(dir, name, isdir) {
|
|
1394
|
+
var node;
|
|
1395
|
+
try {
|
|
1396
|
+
node = FS.lookupNode(dir, name);
|
|
1397
|
+
} catch (e) {
|
|
1398
|
+
return e.errno;
|
|
1399
|
+
}
|
|
1400
|
+
var errCode = FS.nodePermissions(dir, "wx");
|
|
1401
|
+
if (errCode) return errCode;
|
|
1402
|
+
if (isdir) {
|
|
1403
|
+
if (!FS.isDir(node.mode)) return 54;
|
|
1404
|
+
if (FS.isRoot(node) || FS.getPath(node) === FS.cwd()) return 10;
|
|
1405
|
+
} else if (FS.isDir(node.mode)) return 31;
|
|
1406
|
+
return 0;
|
|
1407
|
+
},
|
|
1408
|
+
mayOpen(node, flags) {
|
|
1409
|
+
if (!node) return 44;
|
|
1410
|
+
if (FS.isLink(node.mode)) return 32;
|
|
1411
|
+
var mode = FS.flagsToPermissionString(flags);
|
|
1412
|
+
if (FS.isDir(node.mode)) {
|
|
1413
|
+
if (mode !== "r" || flags & 576) return 31;
|
|
1414
|
+
}
|
|
1415
|
+
return FS.nodePermissions(node, mode);
|
|
1416
|
+
},
|
|
1417
|
+
checkOpExists(op, err) {
|
|
1418
|
+
if (!op) throw new FS.ErrnoError(err);
|
|
1419
|
+
return op;
|
|
1420
|
+
},
|
|
1421
|
+
MAX_OPEN_FDS: 4096,
|
|
1422
|
+
nextfd() {
|
|
1423
|
+
for (var fd = 0; fd <= FS.MAX_OPEN_FDS; fd++) if (!FS.streams[fd]) return fd;
|
|
1424
|
+
throw new FS.ErrnoError(33);
|
|
1425
|
+
},
|
|
1426
|
+
getStreamChecked(fd) {
|
|
1427
|
+
var stream = FS.getStream(fd);
|
|
1428
|
+
if (!stream) throw new FS.ErrnoError(8);
|
|
1429
|
+
return stream;
|
|
1430
|
+
},
|
|
1431
|
+
getStream: (fd) => FS.streams[fd],
|
|
1432
|
+
createStream(stream, fd = -1) {
|
|
1433
|
+
stream = Object.assign(new FS.FSStream(), stream);
|
|
1434
|
+
if (fd == -1) fd = FS.nextfd();
|
|
1435
|
+
stream.fd = fd;
|
|
1436
|
+
FS.streams[fd] = stream;
|
|
1437
|
+
return stream;
|
|
1438
|
+
},
|
|
1439
|
+
closeStream(fd) {
|
|
1440
|
+
FS.streams[fd] = null;
|
|
1441
|
+
},
|
|
1442
|
+
dupStream(origStream, fd = -1) {
|
|
1443
|
+
var stream = FS.createStream(origStream, fd);
|
|
1444
|
+
stream.stream_ops?.dup?.(stream);
|
|
1445
|
+
return stream;
|
|
1446
|
+
},
|
|
1447
|
+
doSetAttr(stream, node, attr) {
|
|
1448
|
+
var setattr = stream?.stream_ops.setattr;
|
|
1449
|
+
var arg = setattr ? stream : node;
|
|
1450
|
+
setattr ??= node.node_ops.setattr;
|
|
1451
|
+
FS.checkOpExists(setattr, 63);
|
|
1452
|
+
setattr(arg, attr);
|
|
1453
|
+
},
|
|
1454
|
+
chrdev_stream_ops: {
|
|
1455
|
+
open(stream) {
|
|
1456
|
+
stream.stream_ops = FS.getDevice(stream.node.rdev).stream_ops;
|
|
1457
|
+
stream.stream_ops.open?.(stream);
|
|
1458
|
+
},
|
|
1459
|
+
llseek() {
|
|
1460
|
+
throw new FS.ErrnoError(70);
|
|
1461
|
+
}
|
|
1462
|
+
},
|
|
1463
|
+
major: (dev) => dev >> 8,
|
|
1464
|
+
minor: (dev) => dev & 255,
|
|
1465
|
+
makedev: (ma, mi) => ma << 8 | mi,
|
|
1466
|
+
registerDevice(dev, ops) {
|
|
1467
|
+
FS.devices[dev] = { stream_ops: ops };
|
|
1468
|
+
},
|
|
1469
|
+
getDevice: (dev) => FS.devices[dev],
|
|
1470
|
+
getMounts(mount) {
|
|
1471
|
+
var mounts = [];
|
|
1472
|
+
var check = [mount];
|
|
1473
|
+
while (check.length) {
|
|
1474
|
+
var m = check.pop();
|
|
1475
|
+
mounts.push(m);
|
|
1476
|
+
check.push(...m.mounts);
|
|
1477
|
+
}
|
|
1478
|
+
return mounts;
|
|
1479
|
+
},
|
|
1480
|
+
syncfs(populate, callback) {
|
|
1481
|
+
if (typeof populate == "function") {
|
|
1482
|
+
callback = populate;
|
|
1483
|
+
populate = false;
|
|
1484
|
+
}
|
|
1485
|
+
FS.syncFSRequests++;
|
|
1486
|
+
if (FS.syncFSRequests > 1) err(`warning: ${FS.syncFSRequests} FS.syncfs operations in flight at once, probably just doing extra work`);
|
|
1487
|
+
var mounts = FS.getMounts(FS.root.mount);
|
|
1488
|
+
var completed = 0;
|
|
1489
|
+
function doCallback(errCode) {
|
|
1490
|
+
FS.syncFSRequests--;
|
|
1491
|
+
return callback(errCode);
|
|
1492
|
+
}
|
|
1493
|
+
function done(errCode) {
|
|
1494
|
+
if (errCode) {
|
|
1495
|
+
if (!done.errored) {
|
|
1496
|
+
done.errored = true;
|
|
1497
|
+
return doCallback(errCode);
|
|
1498
|
+
}
|
|
1499
|
+
return;
|
|
1500
|
+
}
|
|
1501
|
+
if (++completed >= mounts.length) doCallback(null);
|
|
1502
|
+
}
|
|
1503
|
+
for (var mount of mounts) if (mount.type.syncfs) mount.type.syncfs(mount, populate, done);
|
|
1504
|
+
else done(null);
|
|
1505
|
+
},
|
|
1506
|
+
mount(type, opts, mountpoint) {
|
|
1507
|
+
var root = mountpoint === "/";
|
|
1508
|
+
var pseudo = !mountpoint;
|
|
1509
|
+
var node;
|
|
1510
|
+
if (root && FS.root) throw new FS.ErrnoError(10);
|
|
1511
|
+
else if (!root && !pseudo) {
|
|
1512
|
+
var lookup = FS.lookupPath(mountpoint, { follow_mount: false });
|
|
1513
|
+
mountpoint = lookup.path;
|
|
1514
|
+
node = lookup.node;
|
|
1515
|
+
if (FS.isMountpoint(node)) throw new FS.ErrnoError(10);
|
|
1516
|
+
if (!FS.isDir(node.mode)) throw new FS.ErrnoError(54);
|
|
1517
|
+
}
|
|
1518
|
+
var mount = {
|
|
1519
|
+
type,
|
|
1520
|
+
opts,
|
|
1521
|
+
mountpoint,
|
|
1522
|
+
mounts: []
|
|
1523
|
+
};
|
|
1524
|
+
var mountRoot = type.mount(mount);
|
|
1525
|
+
mountRoot.mount = mount;
|
|
1526
|
+
mount.root = mountRoot;
|
|
1527
|
+
if (root) FS.root = mountRoot;
|
|
1528
|
+
else if (node) {
|
|
1529
|
+
node.mounted = mount;
|
|
1530
|
+
if (node.mount) node.mount.mounts.push(mount);
|
|
1531
|
+
}
|
|
1532
|
+
return mountRoot;
|
|
1533
|
+
},
|
|
1534
|
+
unmount(mountpoint) {
|
|
1535
|
+
var lookup = FS.lookupPath(mountpoint, { follow_mount: false });
|
|
1536
|
+
if (!FS.isMountpoint(lookup.node)) throw new FS.ErrnoError(28);
|
|
1537
|
+
var node = lookup.node;
|
|
1538
|
+
var mount = node.mounted;
|
|
1539
|
+
var mounts = FS.getMounts(mount);
|
|
1540
|
+
for (var [hash, current] of Object.entries(FS.nameTable)) while (current) {
|
|
1541
|
+
var next = current.name_next;
|
|
1542
|
+
if (mounts.includes(current.mount)) FS.destroyNode(current);
|
|
1543
|
+
current = next;
|
|
1544
|
+
}
|
|
1545
|
+
node.mounted = null;
|
|
1546
|
+
var idx = node.mount.mounts.indexOf(mount);
|
|
1547
|
+
node.mount.mounts.splice(idx, 1);
|
|
1548
|
+
},
|
|
1549
|
+
lookup(parent, name) {
|
|
1550
|
+
return parent.node_ops.lookup(parent, name);
|
|
1551
|
+
},
|
|
1552
|
+
mknod(path, mode, dev) {
|
|
1553
|
+
var parent = FS.lookupPath(path, { parent: true }).node;
|
|
1554
|
+
var name = PATH.basename(path);
|
|
1555
|
+
if (!name) throw new FS.ErrnoError(28);
|
|
1556
|
+
if (name === "." || name === "..") throw new FS.ErrnoError(20);
|
|
1557
|
+
var errCode = FS.mayCreate(parent, name);
|
|
1558
|
+
if (errCode) throw new FS.ErrnoError(errCode);
|
|
1559
|
+
if (!parent.node_ops.mknod) throw new FS.ErrnoError(63);
|
|
1560
|
+
return parent.node_ops.mknod(parent, name, mode, dev);
|
|
1561
|
+
},
|
|
1562
|
+
statfs(path) {
|
|
1563
|
+
return FS.statfsNode(FS.lookupPath(path, { follow: true }).node);
|
|
1564
|
+
},
|
|
1565
|
+
statfsStream(stream) {
|
|
1566
|
+
return FS.statfsNode(stream.node);
|
|
1567
|
+
},
|
|
1568
|
+
statfsNode(node) {
|
|
1569
|
+
var rtn = {
|
|
1570
|
+
bsize: 4096,
|
|
1571
|
+
frsize: 4096,
|
|
1572
|
+
blocks: 1e6,
|
|
1573
|
+
bfree: 5e5,
|
|
1574
|
+
bavail: 5e5,
|
|
1575
|
+
files: FS.nextInode,
|
|
1576
|
+
ffree: FS.nextInode - 1,
|
|
1577
|
+
fsid: 42,
|
|
1578
|
+
flags: 2,
|
|
1579
|
+
namelen: 255
|
|
1580
|
+
};
|
|
1581
|
+
if (node.node_ops.statfs) Object.assign(rtn, node.node_ops.statfs(node.mount.opts.root));
|
|
1582
|
+
return rtn;
|
|
1583
|
+
},
|
|
1584
|
+
create(path, mode = 438) {
|
|
1585
|
+
mode &= 4095;
|
|
1586
|
+
mode |= 32768;
|
|
1587
|
+
return FS.mknod(path, mode, 0);
|
|
1588
|
+
},
|
|
1589
|
+
mkdir(path, mode = 511) {
|
|
1590
|
+
mode &= 1023;
|
|
1591
|
+
mode |= 16384;
|
|
1592
|
+
return FS.mknod(path, mode, 0);
|
|
1593
|
+
},
|
|
1594
|
+
mkdirTree(path, mode) {
|
|
1595
|
+
var dirs = path.split("/");
|
|
1596
|
+
var d = "";
|
|
1597
|
+
for (var dir of dirs) {
|
|
1598
|
+
if (!dir) continue;
|
|
1599
|
+
if (d || PATH.isAbs(path)) d += "/";
|
|
1600
|
+
d += dir;
|
|
1601
|
+
try {
|
|
1602
|
+
FS.mkdir(d, mode);
|
|
1603
|
+
} catch (e) {
|
|
1604
|
+
if (e.errno != 20) throw e;
|
|
1605
|
+
}
|
|
1606
|
+
}
|
|
1607
|
+
},
|
|
1608
|
+
mkdev(path, mode, dev) {
|
|
1609
|
+
if (typeof dev == "undefined") {
|
|
1610
|
+
dev = mode;
|
|
1611
|
+
mode = 438;
|
|
1612
|
+
}
|
|
1613
|
+
mode |= 8192;
|
|
1614
|
+
return FS.mknod(path, mode, dev);
|
|
1615
|
+
},
|
|
1616
|
+
symlink(oldpath, newpath) {
|
|
1617
|
+
if (!PATH_FS.resolve(oldpath)) throw new FS.ErrnoError(44);
|
|
1618
|
+
var parent = FS.lookupPath(newpath, { parent: true }).node;
|
|
1619
|
+
if (!parent) throw new FS.ErrnoError(44);
|
|
1620
|
+
var newname = PATH.basename(newpath);
|
|
1621
|
+
var errCode = FS.mayCreate(parent, newname);
|
|
1622
|
+
if (errCode) throw new FS.ErrnoError(errCode);
|
|
1623
|
+
if (!parent.node_ops.symlink) throw new FS.ErrnoError(63);
|
|
1624
|
+
return parent.node_ops.symlink(parent, newname, oldpath);
|
|
1625
|
+
},
|
|
1626
|
+
rename(old_path, new_path) {
|
|
1627
|
+
var old_dirname = PATH.dirname(old_path);
|
|
1628
|
+
var new_dirname = PATH.dirname(new_path);
|
|
1629
|
+
var old_name = PATH.basename(old_path);
|
|
1630
|
+
var new_name = PATH.basename(new_path);
|
|
1631
|
+
var lookup = FS.lookupPath(old_path, { parent: true }), old_dir = lookup.node, new_dir;
|
|
1632
|
+
lookup = FS.lookupPath(new_path, { parent: true });
|
|
1633
|
+
new_dir = lookup.node;
|
|
1634
|
+
if (!old_dir || !new_dir) throw new FS.ErrnoError(44);
|
|
1635
|
+
if (old_dir.mount !== new_dir.mount) throw new FS.ErrnoError(75);
|
|
1636
|
+
var old_node = FS.lookupNode(old_dir, old_name);
|
|
1637
|
+
var relative = PATH_FS.relative(old_path, new_dirname);
|
|
1638
|
+
if (relative.charAt(0) !== ".") throw new FS.ErrnoError(28);
|
|
1639
|
+
relative = PATH_FS.relative(new_path, old_dirname);
|
|
1640
|
+
if (relative.charAt(0) !== ".") throw new FS.ErrnoError(55);
|
|
1641
|
+
var new_node;
|
|
1642
|
+
try {
|
|
1643
|
+
new_node = FS.lookupNode(new_dir, new_name);
|
|
1644
|
+
} catch (e) {}
|
|
1645
|
+
if (old_node === new_node) return;
|
|
1646
|
+
var isdir = FS.isDir(old_node.mode);
|
|
1647
|
+
var errCode = FS.mayDelete(old_dir, old_name, isdir);
|
|
1648
|
+
if (errCode) throw new FS.ErrnoError(errCode);
|
|
1649
|
+
errCode = new_node ? FS.mayDelete(new_dir, new_name, isdir) : FS.mayCreate(new_dir, new_name);
|
|
1650
|
+
if (errCode) throw new FS.ErrnoError(errCode);
|
|
1651
|
+
if (!old_dir.node_ops.rename) throw new FS.ErrnoError(63);
|
|
1652
|
+
if (FS.isMountpoint(old_node) || new_node && FS.isMountpoint(new_node)) throw new FS.ErrnoError(10);
|
|
1653
|
+
if (new_dir !== old_dir) {
|
|
1654
|
+
errCode = FS.nodePermissions(old_dir, "w");
|
|
1655
|
+
if (errCode) throw new FS.ErrnoError(errCode);
|
|
1656
|
+
}
|
|
1657
|
+
FS.hashRemoveNode(old_node);
|
|
1658
|
+
try {
|
|
1659
|
+
old_dir.node_ops.rename(old_node, new_dir, new_name);
|
|
1660
|
+
old_node.parent = new_dir;
|
|
1661
|
+
} catch (e) {
|
|
1662
|
+
throw e;
|
|
1663
|
+
} finally {
|
|
1664
|
+
FS.hashAddNode(old_node);
|
|
1665
|
+
}
|
|
1666
|
+
},
|
|
1667
|
+
rmdir(path) {
|
|
1668
|
+
var parent = FS.lookupPath(path, { parent: true }).node;
|
|
1669
|
+
var name = PATH.basename(path);
|
|
1670
|
+
var node = FS.lookupNode(parent, name);
|
|
1671
|
+
var errCode = FS.mayDelete(parent, name, true);
|
|
1672
|
+
if (errCode) throw new FS.ErrnoError(errCode);
|
|
1673
|
+
if (!parent.node_ops.rmdir) throw new FS.ErrnoError(63);
|
|
1674
|
+
if (FS.isMountpoint(node)) throw new FS.ErrnoError(10);
|
|
1675
|
+
parent.node_ops.rmdir(parent, name);
|
|
1676
|
+
FS.destroyNode(node);
|
|
1677
|
+
},
|
|
1678
|
+
readdir(path) {
|
|
1679
|
+
var node = FS.lookupPath(path, { follow: true }).node;
|
|
1680
|
+
return FS.checkOpExists(node.node_ops.readdir, 54)(node);
|
|
1681
|
+
},
|
|
1682
|
+
unlink(path) {
|
|
1683
|
+
var parent = FS.lookupPath(path, { parent: true }).node;
|
|
1684
|
+
if (!parent) throw new FS.ErrnoError(44);
|
|
1685
|
+
var name = PATH.basename(path);
|
|
1686
|
+
var node = FS.lookupNode(parent, name);
|
|
1687
|
+
var errCode = FS.mayDelete(parent, name, false);
|
|
1688
|
+
if (errCode) throw new FS.ErrnoError(errCode);
|
|
1689
|
+
if (!parent.node_ops.unlink) throw new FS.ErrnoError(63);
|
|
1690
|
+
if (FS.isMountpoint(node)) throw new FS.ErrnoError(10);
|
|
1691
|
+
parent.node_ops.unlink(parent, name);
|
|
1692
|
+
FS.destroyNode(node);
|
|
1693
|
+
},
|
|
1694
|
+
readlink(path) {
|
|
1695
|
+
var link = FS.lookupPath(path).node;
|
|
1696
|
+
if (!link) throw new FS.ErrnoError(44);
|
|
1697
|
+
if (!link.node_ops.readlink) throw new FS.ErrnoError(28);
|
|
1698
|
+
return link.node_ops.readlink(link);
|
|
1699
|
+
},
|
|
1700
|
+
stat(path, dontFollow) {
|
|
1701
|
+
var node = FS.lookupPath(path, { follow: !dontFollow }).node;
|
|
1702
|
+
return FS.checkOpExists(node.node_ops.getattr, 63)(node);
|
|
1703
|
+
},
|
|
1704
|
+
fstat(fd) {
|
|
1705
|
+
var stream = FS.getStreamChecked(fd);
|
|
1706
|
+
var node = stream.node;
|
|
1707
|
+
var getattr = stream.stream_ops.getattr;
|
|
1708
|
+
var arg = getattr ? stream : node;
|
|
1709
|
+
getattr ??= node.node_ops.getattr;
|
|
1710
|
+
FS.checkOpExists(getattr, 63);
|
|
1711
|
+
return getattr(arg);
|
|
1712
|
+
},
|
|
1713
|
+
lstat(path) {
|
|
1714
|
+
return FS.stat(path, true);
|
|
1715
|
+
},
|
|
1716
|
+
doChmod(stream, node, mode, dontFollow) {
|
|
1717
|
+
FS.doSetAttr(stream, node, {
|
|
1718
|
+
mode: mode & 4095 | node.mode & -4096,
|
|
1719
|
+
ctime: Date.now(),
|
|
1720
|
+
dontFollow
|
|
1721
|
+
});
|
|
1722
|
+
},
|
|
1723
|
+
chmod(path, mode, dontFollow) {
|
|
1724
|
+
var node;
|
|
1725
|
+
if (typeof path == "string") node = FS.lookupPath(path, { follow: !dontFollow }).node;
|
|
1726
|
+
else node = path;
|
|
1727
|
+
FS.doChmod(null, node, mode, dontFollow);
|
|
1728
|
+
},
|
|
1729
|
+
lchmod(path, mode) {
|
|
1730
|
+
FS.chmod(path, mode, true);
|
|
1731
|
+
},
|
|
1732
|
+
fchmod(fd, mode) {
|
|
1733
|
+
var stream = FS.getStreamChecked(fd);
|
|
1734
|
+
FS.doChmod(stream, stream.node, mode, false);
|
|
1735
|
+
},
|
|
1736
|
+
doChown(stream, node, dontFollow) {
|
|
1737
|
+
FS.doSetAttr(stream, node, {
|
|
1738
|
+
timestamp: Date.now(),
|
|
1739
|
+
dontFollow
|
|
1740
|
+
});
|
|
1741
|
+
},
|
|
1742
|
+
chown(path, uid, gid, dontFollow) {
|
|
1743
|
+
var node;
|
|
1744
|
+
if (typeof path == "string") node = FS.lookupPath(path, { follow: !dontFollow }).node;
|
|
1745
|
+
else node = path;
|
|
1746
|
+
FS.doChown(null, node, dontFollow);
|
|
1747
|
+
},
|
|
1748
|
+
lchown(path, uid, gid) {
|
|
1749
|
+
FS.chown(path, uid, gid, true);
|
|
1750
|
+
},
|
|
1751
|
+
fchown(fd, uid, gid) {
|
|
1752
|
+
var stream = FS.getStreamChecked(fd);
|
|
1753
|
+
FS.doChown(stream, stream.node, false);
|
|
1754
|
+
},
|
|
1755
|
+
doTruncate(stream, node, len) {
|
|
1756
|
+
if (FS.isDir(node.mode)) throw new FS.ErrnoError(31);
|
|
1757
|
+
if (!FS.isFile(node.mode)) throw new FS.ErrnoError(28);
|
|
1758
|
+
var errCode = FS.nodePermissions(node, "w");
|
|
1759
|
+
if (errCode) throw new FS.ErrnoError(errCode);
|
|
1760
|
+
FS.doSetAttr(stream, node, {
|
|
1761
|
+
size: len,
|
|
1762
|
+
timestamp: Date.now()
|
|
1763
|
+
});
|
|
1764
|
+
},
|
|
1765
|
+
truncate(path, len) {
|
|
1766
|
+
if (len < 0) throw new FS.ErrnoError(28);
|
|
1767
|
+
var node;
|
|
1768
|
+
if (typeof path == "string") node = FS.lookupPath(path, { follow: true }).node;
|
|
1769
|
+
else node = path;
|
|
1770
|
+
FS.doTruncate(null, node, len);
|
|
1771
|
+
},
|
|
1772
|
+
ftruncate(fd, len) {
|
|
1773
|
+
var stream = FS.getStreamChecked(fd);
|
|
1774
|
+
if (len < 0 || (stream.flags & 2097155) === 0) throw new FS.ErrnoError(28);
|
|
1775
|
+
FS.doTruncate(stream, stream.node, len);
|
|
1776
|
+
},
|
|
1777
|
+
utime(path, atime, mtime) {
|
|
1778
|
+
var node = FS.lookupPath(path, { follow: true }).node;
|
|
1779
|
+
FS.checkOpExists(node.node_ops.setattr, 63)(node, {
|
|
1780
|
+
atime,
|
|
1781
|
+
mtime
|
|
1782
|
+
});
|
|
1783
|
+
},
|
|
1784
|
+
open(path, flags, mode = 438) {
|
|
1785
|
+
if (path === "") throw new FS.ErrnoError(44);
|
|
1786
|
+
flags = typeof flags == "string" ? FS_modeStringToFlags(flags) : flags;
|
|
1787
|
+
if (flags & 64) mode = mode & 4095 | 32768;
|
|
1788
|
+
else mode = 0;
|
|
1789
|
+
var node;
|
|
1790
|
+
var isDirPath;
|
|
1791
|
+
if (typeof path == "object") node = path;
|
|
1792
|
+
else {
|
|
1793
|
+
isDirPath = path.endsWith("/");
|
|
1794
|
+
var lookup = FS.lookupPath(path, {
|
|
1795
|
+
follow: !(flags & 131072),
|
|
1796
|
+
noent_okay: true
|
|
1797
|
+
});
|
|
1798
|
+
node = lookup.node;
|
|
1799
|
+
path = lookup.path;
|
|
1800
|
+
}
|
|
1801
|
+
var created = false;
|
|
1802
|
+
if (flags & 64) if (node) {
|
|
1803
|
+
if (flags & 128) throw new FS.ErrnoError(20);
|
|
1804
|
+
} else if (isDirPath) throw new FS.ErrnoError(31);
|
|
1805
|
+
else {
|
|
1806
|
+
node = FS.mknod(path, mode | 511, 0);
|
|
1807
|
+
created = true;
|
|
1808
|
+
}
|
|
1809
|
+
if (!node) throw new FS.ErrnoError(44);
|
|
1810
|
+
if (FS.isChrdev(node.mode)) flags &= -513;
|
|
1811
|
+
if (flags & 65536 && !FS.isDir(node.mode)) throw new FS.ErrnoError(54);
|
|
1812
|
+
if (!created) {
|
|
1813
|
+
var errCode = FS.mayOpen(node, flags);
|
|
1814
|
+
if (errCode) throw new FS.ErrnoError(errCode);
|
|
1815
|
+
}
|
|
1816
|
+
if (flags & 512 && !created) FS.truncate(node, 0);
|
|
1817
|
+
flags &= -131713;
|
|
1818
|
+
var stream = FS.createStream({
|
|
1819
|
+
node,
|
|
1820
|
+
path: FS.getPath(node),
|
|
1821
|
+
flags,
|
|
1822
|
+
seekable: true,
|
|
1823
|
+
position: 0,
|
|
1824
|
+
stream_ops: node.stream_ops,
|
|
1825
|
+
ungotten: [],
|
|
1826
|
+
error: false
|
|
1827
|
+
});
|
|
1828
|
+
if (stream.stream_ops.open) stream.stream_ops.open(stream);
|
|
1829
|
+
if (created) FS.chmod(node, mode & 511);
|
|
1830
|
+
return stream;
|
|
1831
|
+
},
|
|
1832
|
+
close(stream) {
|
|
1833
|
+
if (FS.isClosed(stream)) throw new FS.ErrnoError(8);
|
|
1834
|
+
if (stream.getdents) stream.getdents = null;
|
|
1835
|
+
try {
|
|
1836
|
+
if (stream.stream_ops.close) stream.stream_ops.close(stream);
|
|
1837
|
+
} catch (e) {
|
|
1838
|
+
throw e;
|
|
1839
|
+
} finally {
|
|
1840
|
+
FS.closeStream(stream.fd);
|
|
1841
|
+
}
|
|
1842
|
+
stream.fd = null;
|
|
1843
|
+
},
|
|
1844
|
+
isClosed(stream) {
|
|
1845
|
+
return stream.fd === null;
|
|
1846
|
+
},
|
|
1847
|
+
llseek(stream, offset, whence) {
|
|
1848
|
+
if (FS.isClosed(stream)) throw new FS.ErrnoError(8);
|
|
1849
|
+
if (!stream.seekable || !stream.stream_ops.llseek) throw new FS.ErrnoError(70);
|
|
1850
|
+
if (whence != 0 && whence != 1 && whence != 2) throw new FS.ErrnoError(28);
|
|
1851
|
+
stream.position = stream.stream_ops.llseek(stream, offset, whence);
|
|
1852
|
+
stream.ungotten = [];
|
|
1853
|
+
return stream.position;
|
|
1854
|
+
},
|
|
1855
|
+
read(stream, buffer, offset, length, position) {
|
|
1856
|
+
if (length < 0 || position < 0) throw new FS.ErrnoError(28);
|
|
1857
|
+
if (FS.isClosed(stream)) throw new FS.ErrnoError(8);
|
|
1858
|
+
if ((stream.flags & 2097155) === 1) throw new FS.ErrnoError(8);
|
|
1859
|
+
if (FS.isDir(stream.node.mode)) throw new FS.ErrnoError(31);
|
|
1860
|
+
if (!stream.stream_ops.read) throw new FS.ErrnoError(28);
|
|
1861
|
+
var seeking = typeof position != "undefined";
|
|
1862
|
+
if (!seeking) position = stream.position;
|
|
1863
|
+
else if (!stream.seekable) throw new FS.ErrnoError(70);
|
|
1864
|
+
var bytesRead = stream.stream_ops.read(stream, buffer, offset, length, position);
|
|
1865
|
+
if (!seeking) stream.position += bytesRead;
|
|
1866
|
+
return bytesRead;
|
|
1867
|
+
},
|
|
1868
|
+
write(stream, buffer, offset, length, position, canOwn) {
|
|
1869
|
+
if (length < 0 || position < 0) throw new FS.ErrnoError(28);
|
|
1870
|
+
if (FS.isClosed(stream)) throw new FS.ErrnoError(8);
|
|
1871
|
+
if ((stream.flags & 2097155) === 0) throw new FS.ErrnoError(8);
|
|
1872
|
+
if (FS.isDir(stream.node.mode)) throw new FS.ErrnoError(31);
|
|
1873
|
+
if (!stream.stream_ops.write) throw new FS.ErrnoError(28);
|
|
1874
|
+
if (stream.seekable && stream.flags & 1024) FS.llseek(stream, 0, 2);
|
|
1875
|
+
var seeking = typeof position != "undefined";
|
|
1876
|
+
if (!seeking) position = stream.position;
|
|
1877
|
+
else if (!stream.seekable) throw new FS.ErrnoError(70);
|
|
1878
|
+
var bytesWritten = stream.stream_ops.write(stream, buffer, offset, length, position, canOwn);
|
|
1879
|
+
if (!seeking) stream.position += bytesWritten;
|
|
1880
|
+
return bytesWritten;
|
|
1881
|
+
},
|
|
1882
|
+
mmap(stream, length, position, prot, flags) {
|
|
1883
|
+
if ((prot & 2) !== 0 && (flags & 2) === 0 && (stream.flags & 2097155) !== 2) throw new FS.ErrnoError(2);
|
|
1884
|
+
if ((stream.flags & 2097155) === 1) throw new FS.ErrnoError(2);
|
|
1885
|
+
if (!stream.stream_ops.mmap) throw new FS.ErrnoError(43);
|
|
1886
|
+
if (!length) throw new FS.ErrnoError(28);
|
|
1887
|
+
return stream.stream_ops.mmap(stream, length, position, prot, flags);
|
|
1888
|
+
},
|
|
1889
|
+
msync(stream, buffer, offset, length, mmapFlags) {
|
|
1890
|
+
if (!stream.stream_ops.msync) return 0;
|
|
1891
|
+
return stream.stream_ops.msync(stream, buffer, offset, length, mmapFlags);
|
|
1892
|
+
},
|
|
1893
|
+
ioctl(stream, cmd, arg) {
|
|
1894
|
+
if (!stream.stream_ops.ioctl) throw new FS.ErrnoError(59);
|
|
1895
|
+
return stream.stream_ops.ioctl(stream, cmd, arg);
|
|
1896
|
+
},
|
|
1897
|
+
readFile(path, opts = {}) {
|
|
1898
|
+
opts.flags = opts.flags || 0;
|
|
1899
|
+
opts.encoding = opts.encoding || "binary";
|
|
1900
|
+
if (opts.encoding !== "utf8" && opts.encoding !== "binary") abort(`Invalid encoding type "${opts.encoding}"`);
|
|
1901
|
+
var stream = FS.open(path, opts.flags);
|
|
1902
|
+
var length = FS.stat(path).size;
|
|
1903
|
+
var buf = new Uint8Array(length);
|
|
1904
|
+
FS.read(stream, buf, 0, length, 0);
|
|
1905
|
+
if (opts.encoding === "utf8") buf = UTF8ArrayToString(buf);
|
|
1906
|
+
FS.close(stream);
|
|
1907
|
+
return buf;
|
|
1908
|
+
},
|
|
1909
|
+
writeFile(path, data, opts = {}) {
|
|
1910
|
+
opts.flags = opts.flags || 577;
|
|
1911
|
+
var stream = FS.open(path, opts.flags, opts.mode);
|
|
1912
|
+
if (typeof data == "string") data = new Uint8Array(intArrayFromString(data, true));
|
|
1913
|
+
if (ArrayBuffer.isView(data)) FS.write(stream, data, 0, data.byteLength, void 0, opts.canOwn);
|
|
1914
|
+
else abort("Unsupported data type");
|
|
1915
|
+
FS.close(stream);
|
|
1916
|
+
},
|
|
1917
|
+
cwd: () => FS.currentPath,
|
|
1918
|
+
chdir(path) {
|
|
1919
|
+
var lookup = FS.lookupPath(path, { follow: true });
|
|
1920
|
+
if (lookup.node === null) throw new FS.ErrnoError(44);
|
|
1921
|
+
if (!FS.isDir(lookup.node.mode)) throw new FS.ErrnoError(54);
|
|
1922
|
+
var errCode = FS.nodePermissions(lookup.node, "x");
|
|
1923
|
+
if (errCode) throw new FS.ErrnoError(errCode);
|
|
1924
|
+
FS.currentPath = lookup.path;
|
|
1925
|
+
},
|
|
1926
|
+
createDefaultDirectories() {
|
|
1927
|
+
FS.mkdir("/tmp");
|
|
1928
|
+
FS.mkdir("/home");
|
|
1929
|
+
FS.mkdir("/home/web_user");
|
|
1930
|
+
},
|
|
1931
|
+
createDefaultDevices() {
|
|
1932
|
+
FS.mkdir("/dev");
|
|
1933
|
+
FS.registerDevice(FS.makedev(1, 3), {
|
|
1934
|
+
read: () => 0,
|
|
1935
|
+
write: (stream, buffer, offset, length, pos) => length,
|
|
1936
|
+
llseek: () => 0
|
|
1937
|
+
});
|
|
1938
|
+
FS.mkdev("/dev/null", FS.makedev(1, 3));
|
|
1939
|
+
TTY.register(FS.makedev(5, 0), TTY.default_tty_ops);
|
|
1940
|
+
TTY.register(FS.makedev(6, 0), TTY.default_tty1_ops);
|
|
1941
|
+
FS.mkdev("/dev/tty", FS.makedev(5, 0));
|
|
1942
|
+
FS.mkdev("/dev/tty1", FS.makedev(6, 0));
|
|
1943
|
+
var randomBuffer = new Uint8Array(1024), randomLeft = 0;
|
|
1944
|
+
var randomByte = () => {
|
|
1945
|
+
if (randomLeft === 0) {
|
|
1946
|
+
randomFill(randomBuffer);
|
|
1947
|
+
randomLeft = randomBuffer.byteLength;
|
|
1948
|
+
}
|
|
1949
|
+
return randomBuffer[--randomLeft];
|
|
1950
|
+
};
|
|
1951
|
+
FS.createDevice("/dev", "random", randomByte);
|
|
1952
|
+
FS.createDevice("/dev", "urandom", randomByte);
|
|
1953
|
+
FS.mkdir("/dev/shm");
|
|
1954
|
+
FS.mkdir("/dev/shm/tmp");
|
|
1955
|
+
},
|
|
1956
|
+
createSpecialDirectories() {
|
|
1957
|
+
FS.mkdir("/proc");
|
|
1958
|
+
var proc_self = FS.mkdir("/proc/self");
|
|
1959
|
+
FS.mkdir("/proc/self/fd");
|
|
1960
|
+
FS.mount({ mount() {
|
|
1961
|
+
var node = FS.createNode(proc_self, "fd", 16895, 73);
|
|
1962
|
+
node.stream_ops = { llseek: MEMFS.stream_ops.llseek };
|
|
1963
|
+
node.node_ops = {
|
|
1964
|
+
lookup(parent, name) {
|
|
1965
|
+
var fd = +name;
|
|
1966
|
+
var stream = FS.getStreamChecked(fd);
|
|
1967
|
+
var ret = {
|
|
1968
|
+
parent: null,
|
|
1969
|
+
mount: { mountpoint: "fake" },
|
|
1970
|
+
node_ops: { readlink: () => stream.path },
|
|
1971
|
+
id: fd + 1
|
|
1972
|
+
};
|
|
1973
|
+
ret.parent = ret;
|
|
1974
|
+
return ret;
|
|
1975
|
+
},
|
|
1976
|
+
readdir() {
|
|
1977
|
+
return Array.from(FS.streams.entries()).filter(([k, v]) => v).map(([k, v]) => k.toString());
|
|
1978
|
+
}
|
|
1979
|
+
};
|
|
1980
|
+
return node;
|
|
1981
|
+
} }, {}, "/proc/self/fd");
|
|
1982
|
+
},
|
|
1983
|
+
createStandardStreams(input, output, error) {
|
|
1984
|
+
if (input) FS.createDevice("/dev", "stdin", input);
|
|
1985
|
+
else FS.symlink("/dev/tty", "/dev/stdin");
|
|
1986
|
+
if (output) FS.createDevice("/dev", "stdout", null, output);
|
|
1987
|
+
else FS.symlink("/dev/tty", "/dev/stdout");
|
|
1988
|
+
if (error) FS.createDevice("/dev", "stderr", null, error);
|
|
1989
|
+
else FS.symlink("/dev/tty1", "/dev/stderr");
|
|
1990
|
+
FS.open("/dev/stdin", 0);
|
|
1991
|
+
FS.open("/dev/stdout", 1);
|
|
1992
|
+
FS.open("/dev/stderr", 1);
|
|
1993
|
+
},
|
|
1994
|
+
staticInit() {
|
|
1995
|
+
FS.nameTable = new Array(4096);
|
|
1996
|
+
FS.mount(MEMFS, {}, "/");
|
|
1997
|
+
FS.createDefaultDirectories();
|
|
1998
|
+
FS.createDefaultDevices();
|
|
1999
|
+
FS.createSpecialDirectories();
|
|
2000
|
+
FS.filesystems = {
|
|
2001
|
+
MEMFS,
|
|
2002
|
+
IDBFS
|
|
2003
|
+
};
|
|
2004
|
+
},
|
|
2005
|
+
init(input, output, error) {
|
|
2006
|
+
FS.initialized = true;
|
|
2007
|
+
input ??= Module["stdin"];
|
|
2008
|
+
output ??= Module["stdout"];
|
|
2009
|
+
error ??= Module["stderr"];
|
|
2010
|
+
FS.createStandardStreams(input, output, error);
|
|
2011
|
+
},
|
|
2012
|
+
quit() {
|
|
2013
|
+
FS.initialized = false;
|
|
2014
|
+
for (var stream of FS.streams) if (stream) FS.close(stream);
|
|
2015
|
+
},
|
|
2016
|
+
findObject(path, dontResolveLastLink) {
|
|
2017
|
+
var ret = FS.analyzePath(path, dontResolveLastLink);
|
|
2018
|
+
if (!ret.exists) return null;
|
|
2019
|
+
return ret.object;
|
|
2020
|
+
},
|
|
2021
|
+
analyzePath(path, dontResolveLastLink) {
|
|
2022
|
+
try {
|
|
2023
|
+
var lookup = FS.lookupPath(path, { follow: !dontResolveLastLink });
|
|
2024
|
+
path = lookup.path;
|
|
2025
|
+
} catch (e) {}
|
|
2026
|
+
var ret = {
|
|
2027
|
+
isRoot: false,
|
|
2028
|
+
exists: false,
|
|
2029
|
+
error: 0,
|
|
2030
|
+
name: null,
|
|
2031
|
+
path: null,
|
|
2032
|
+
object: null,
|
|
2033
|
+
parentExists: false,
|
|
2034
|
+
parentPath: null,
|
|
2035
|
+
parentObject: null
|
|
2036
|
+
};
|
|
2037
|
+
try {
|
|
2038
|
+
var lookup = FS.lookupPath(path, { parent: true });
|
|
2039
|
+
ret.parentExists = true;
|
|
2040
|
+
ret.parentPath = lookup.path;
|
|
2041
|
+
ret.parentObject = lookup.node;
|
|
2042
|
+
ret.name = PATH.basename(path);
|
|
2043
|
+
lookup = FS.lookupPath(path, { follow: !dontResolveLastLink });
|
|
2044
|
+
ret.exists = true;
|
|
2045
|
+
ret.path = lookup.path;
|
|
2046
|
+
ret.object = lookup.node;
|
|
2047
|
+
ret.name = lookup.node.name;
|
|
2048
|
+
ret.isRoot = lookup.path === "/";
|
|
2049
|
+
} catch (e) {
|
|
2050
|
+
ret.error = e.errno;
|
|
2051
|
+
}
|
|
2052
|
+
return ret;
|
|
2053
|
+
},
|
|
2054
|
+
createPath(parent, path, canRead, canWrite) {
|
|
2055
|
+
parent = typeof parent == "string" ? parent : FS.getPath(parent);
|
|
2056
|
+
var parts = path.split("/").reverse();
|
|
2057
|
+
while (parts.length) {
|
|
2058
|
+
var part = parts.pop();
|
|
2059
|
+
if (!part) continue;
|
|
2060
|
+
var current = PATH.join2(parent, part);
|
|
2061
|
+
try {
|
|
2062
|
+
FS.mkdir(current);
|
|
2063
|
+
} catch (e) {
|
|
2064
|
+
if (e.errno != 20) throw e;
|
|
2065
|
+
}
|
|
2066
|
+
parent = current;
|
|
2067
|
+
}
|
|
2068
|
+
return current;
|
|
2069
|
+
},
|
|
2070
|
+
createFile(parent, name, properties, canRead, canWrite) {
|
|
2071
|
+
var path = PATH.join2(typeof parent == "string" ? parent : FS.getPath(parent), name);
|
|
2072
|
+
var mode = FS_getMode(canRead, canWrite);
|
|
2073
|
+
return FS.create(path, mode);
|
|
2074
|
+
},
|
|
2075
|
+
createDataFile(parent, name, data, canRead, canWrite, canOwn) {
|
|
2076
|
+
var path = name;
|
|
2077
|
+
if (parent) {
|
|
2078
|
+
parent = typeof parent == "string" ? parent : FS.getPath(parent);
|
|
2079
|
+
path = name ? PATH.join2(parent, name) : parent;
|
|
2080
|
+
}
|
|
2081
|
+
var mode = FS_getMode(canRead, canWrite);
|
|
2082
|
+
var node = FS.create(path, mode);
|
|
2083
|
+
if (data) {
|
|
2084
|
+
if (typeof data == "string") {
|
|
2085
|
+
var arr = new Array(data.length);
|
|
2086
|
+
for (var i = 0, len = data.length; i < len; ++i) arr[i] = data.charCodeAt(i);
|
|
2087
|
+
data = arr;
|
|
2088
|
+
}
|
|
2089
|
+
FS.chmod(node, mode | 146);
|
|
2090
|
+
var stream = FS.open(node, 577);
|
|
2091
|
+
FS.write(stream, data, 0, data.length, 0, canOwn);
|
|
2092
|
+
FS.close(stream);
|
|
2093
|
+
FS.chmod(node, mode);
|
|
2094
|
+
}
|
|
2095
|
+
},
|
|
2096
|
+
createDevice(parent, name, input, output) {
|
|
2097
|
+
var path = PATH.join2(typeof parent == "string" ? parent : FS.getPath(parent), name);
|
|
2098
|
+
var mode = FS_getMode(!!input, !!output);
|
|
2099
|
+
FS.createDevice.major ??= 64;
|
|
2100
|
+
var dev = FS.makedev(FS.createDevice.major++, 0);
|
|
2101
|
+
FS.registerDevice(dev, {
|
|
2102
|
+
open(stream) {
|
|
2103
|
+
stream.seekable = false;
|
|
2104
|
+
},
|
|
2105
|
+
close(stream) {
|
|
2106
|
+
if (output?.buffer?.length) output(10);
|
|
2107
|
+
},
|
|
2108
|
+
read(stream, buffer, offset, length, pos) {
|
|
2109
|
+
var bytesRead = 0;
|
|
2110
|
+
for (var i = 0; i < length; i++) {
|
|
2111
|
+
var result;
|
|
2112
|
+
try {
|
|
2113
|
+
result = input();
|
|
2114
|
+
} catch (e) {
|
|
2115
|
+
throw new FS.ErrnoError(29);
|
|
2116
|
+
}
|
|
2117
|
+
if (result === void 0 && bytesRead === 0) throw new FS.ErrnoError(6);
|
|
2118
|
+
if (result === null || result === void 0) break;
|
|
2119
|
+
bytesRead++;
|
|
2120
|
+
buffer[offset + i] = result;
|
|
2121
|
+
}
|
|
2122
|
+
if (bytesRead) stream.node.atime = Date.now();
|
|
2123
|
+
return bytesRead;
|
|
2124
|
+
},
|
|
2125
|
+
write(stream, buffer, offset, length, pos) {
|
|
2126
|
+
for (var i = 0; i < length; i++) try {
|
|
2127
|
+
output(buffer[offset + i]);
|
|
2128
|
+
} catch (e) {
|
|
2129
|
+
throw new FS.ErrnoError(29);
|
|
2130
|
+
}
|
|
2131
|
+
if (length) stream.node.mtime = stream.node.ctime = Date.now();
|
|
2132
|
+
return i;
|
|
2133
|
+
}
|
|
2134
|
+
});
|
|
2135
|
+
return FS.mkdev(path, mode, dev);
|
|
2136
|
+
},
|
|
2137
|
+
forceLoadFile(obj) {
|
|
2138
|
+
if (obj.isDevice || obj.isFolder || obj.link || obj.contents) return true;
|
|
2139
|
+
if (globalThis.XMLHttpRequest) abort("Lazy loading should have been performed (contents set) in createLazyFile, but it was not. Lazy loading only works in web workers. Use --embed-file or --preload-file in emcc on the main thread.");
|
|
2140
|
+
else try {
|
|
2141
|
+
obj.contents = readBinary(obj.url);
|
|
2142
|
+
} catch (e) {
|
|
2143
|
+
throw new FS.ErrnoError(29);
|
|
2144
|
+
}
|
|
2145
|
+
},
|
|
2146
|
+
createLazyFile(parent, name, url, canRead, canWrite) {
|
|
2147
|
+
class LazyUint8Array {
|
|
2148
|
+
lengthKnown = false;
|
|
2149
|
+
chunks = [];
|
|
2150
|
+
get(idx) {
|
|
2151
|
+
if (idx > this.length - 1 || idx < 0) return;
|
|
2152
|
+
var chunkOffset = idx % this.chunkSize;
|
|
2153
|
+
var chunkNum = idx / this.chunkSize | 0;
|
|
2154
|
+
return this.getter(chunkNum)[chunkOffset];
|
|
2155
|
+
}
|
|
2156
|
+
setDataGetter(getter) {
|
|
2157
|
+
this.getter = getter;
|
|
2158
|
+
}
|
|
2159
|
+
cacheLength() {
|
|
2160
|
+
var xhr = new XMLHttpRequest();
|
|
2161
|
+
xhr.open("HEAD", url, false);
|
|
2162
|
+
xhr.send(null);
|
|
2163
|
+
if (!(xhr.status >= 200 && xhr.status < 300 || xhr.status === 304)) abort("Couldn't load " + url + ". Status: " + xhr.status);
|
|
2164
|
+
var datalength = Number(xhr.getResponseHeader("Content-length"));
|
|
2165
|
+
var header;
|
|
2166
|
+
var hasByteServing = (header = xhr.getResponseHeader("Accept-Ranges")) && header === "bytes";
|
|
2167
|
+
var usesGzip = (header = xhr.getResponseHeader("Content-Encoding")) && header === "gzip";
|
|
2168
|
+
var chunkSize = 1024 * 1024;
|
|
2169
|
+
if (!hasByteServing) chunkSize = datalength;
|
|
2170
|
+
var doXHR = (from, to) => {
|
|
2171
|
+
if (from > to) abort("invalid range (" + from + ", " + to + ") or no bytes requested!");
|
|
2172
|
+
if (to > datalength - 1) abort("only " + datalength + " bytes available! programmer error!");
|
|
2173
|
+
var xhr = new XMLHttpRequest();
|
|
2174
|
+
xhr.open("GET", url, false);
|
|
2175
|
+
if (datalength !== chunkSize) xhr.setRequestHeader("Range", "bytes=" + from + "-" + to);
|
|
2176
|
+
xhr.responseType = "arraybuffer";
|
|
2177
|
+
if (xhr.overrideMimeType) xhr.overrideMimeType("text/plain; charset=x-user-defined");
|
|
2178
|
+
xhr.send(null);
|
|
2179
|
+
if (!(xhr.status >= 200 && xhr.status < 300 || xhr.status === 304)) abort("Couldn't load " + url + ". Status: " + xhr.status);
|
|
2180
|
+
if (xhr.response !== void 0) return new Uint8Array(xhr.response || []);
|
|
2181
|
+
return intArrayFromString(xhr.responseText || "", true);
|
|
2182
|
+
};
|
|
2183
|
+
var lazyArray = this;
|
|
2184
|
+
lazyArray.setDataGetter((chunkNum) => {
|
|
2185
|
+
var start = chunkNum * chunkSize;
|
|
2186
|
+
var end = (chunkNum + 1) * chunkSize - 1;
|
|
2187
|
+
end = Math.min(end, datalength - 1);
|
|
2188
|
+
if (typeof lazyArray.chunks[chunkNum] == "undefined") lazyArray.chunks[chunkNum] = doXHR(start, end);
|
|
2189
|
+
if (typeof lazyArray.chunks[chunkNum] == "undefined") abort("doXHR failed!");
|
|
2190
|
+
return lazyArray.chunks[chunkNum];
|
|
2191
|
+
});
|
|
2192
|
+
if (usesGzip || !datalength) {
|
|
2193
|
+
chunkSize = datalength = 1;
|
|
2194
|
+
datalength = this.getter(0).length;
|
|
2195
|
+
chunkSize = datalength;
|
|
2196
|
+
out("LazyFiles on gzip forces download of the whole file when length is accessed");
|
|
2197
|
+
}
|
|
2198
|
+
this._length = datalength;
|
|
2199
|
+
this._chunkSize = chunkSize;
|
|
2200
|
+
this.lengthKnown = true;
|
|
2201
|
+
}
|
|
2202
|
+
get length() {
|
|
2203
|
+
if (!this.lengthKnown) this.cacheLength();
|
|
2204
|
+
return this._length;
|
|
2205
|
+
}
|
|
2206
|
+
get chunkSize() {
|
|
2207
|
+
if (!this.lengthKnown) this.cacheLength();
|
|
2208
|
+
return this._chunkSize;
|
|
2209
|
+
}
|
|
2210
|
+
}
|
|
2211
|
+
if (globalThis.XMLHttpRequest) {
|
|
2212
|
+
if (!ENVIRONMENT_IS_WORKER) abort("Cannot do synchronous binary XHRs outside webworkers in modern browsers. Use --embed-file or --preload-file in emcc");
|
|
2213
|
+
var properties = {
|
|
2214
|
+
isDevice: false,
|
|
2215
|
+
contents: new LazyUint8Array()
|
|
2216
|
+
};
|
|
2217
|
+
} else var properties = {
|
|
2218
|
+
isDevice: false,
|
|
2219
|
+
url
|
|
2220
|
+
};
|
|
2221
|
+
var node = FS.createFile(parent, name, properties, canRead, canWrite);
|
|
2222
|
+
if (properties.contents) node.contents = properties.contents;
|
|
2223
|
+
else if (properties.url) {
|
|
2224
|
+
node.contents = null;
|
|
2225
|
+
node.url = properties.url;
|
|
2226
|
+
}
|
|
2227
|
+
Object.defineProperties(node, { usedBytes: { get: function() {
|
|
2228
|
+
return this.contents.length;
|
|
2229
|
+
} } });
|
|
2230
|
+
var stream_ops = {};
|
|
2231
|
+
for (const [key, fn] of Object.entries(node.stream_ops)) stream_ops[key] = (...args) => {
|
|
2232
|
+
FS.forceLoadFile(node);
|
|
2233
|
+
return fn(...args);
|
|
2234
|
+
};
|
|
2235
|
+
function writeChunks(stream, buffer, offset, length, position) {
|
|
2236
|
+
var contents = stream.node.contents;
|
|
2237
|
+
if (position >= contents.length) return 0;
|
|
2238
|
+
var size = Math.min(contents.length - position, length);
|
|
2239
|
+
if (contents.slice) for (var i = 0; i < size; i++) buffer[offset + i] = contents[position + i];
|
|
2240
|
+
else for (var i = 0; i < size; i++) buffer[offset + i] = contents.get(position + i);
|
|
2241
|
+
return size;
|
|
2242
|
+
}
|
|
2243
|
+
stream_ops.read = (stream, buffer, offset, length, position) => {
|
|
2244
|
+
FS.forceLoadFile(node);
|
|
2245
|
+
return writeChunks(stream, buffer, offset, length, position);
|
|
2246
|
+
};
|
|
2247
|
+
stream_ops.mmap = (stream, length, position, prot, flags) => {
|
|
2248
|
+
FS.forceLoadFile(node);
|
|
2249
|
+
var ptr = mmapAlloc(length);
|
|
2250
|
+
if (!ptr) throw new FS.ErrnoError(48);
|
|
2251
|
+
writeChunks(stream, HEAP8, ptr, length, position);
|
|
2252
|
+
return {
|
|
2253
|
+
ptr,
|
|
2254
|
+
allocated: true
|
|
2255
|
+
};
|
|
2256
|
+
};
|
|
2257
|
+
node.stream_ops = stream_ops;
|
|
2258
|
+
return node;
|
|
2259
|
+
}
|
|
2260
|
+
};
|
|
2261
|
+
var UTF8ToString = (ptr, maxBytesToRead, ignoreNul) => {
|
|
2262
|
+
ptr >>>= 0;
|
|
2263
|
+
if (!ptr) return "";
|
|
2264
|
+
var end = findStringEnd(HEAPU8, ptr, maxBytesToRead, ignoreNul);
|
|
2265
|
+
return UTF8Decoder.decode(HEAPU8.subarray(ptr >>> 0, end >>> 0));
|
|
2266
|
+
};
|
|
2267
|
+
var SYSCALLS = {
|
|
2268
|
+
calculateAt(dirfd, path, allowEmpty) {
|
|
2269
|
+
if (PATH.isAbs(path)) return path;
|
|
2270
|
+
var dir;
|
|
2271
|
+
if (dirfd === -100) dir = FS.cwd();
|
|
2272
|
+
else dir = SYSCALLS.getStreamFromFD(dirfd).path;
|
|
2273
|
+
if (path.length == 0) {
|
|
2274
|
+
if (!allowEmpty) throw new FS.ErrnoError(44);
|
|
2275
|
+
return dir;
|
|
2276
|
+
}
|
|
2277
|
+
return dir + "/" + path;
|
|
2278
|
+
},
|
|
2279
|
+
writeStat(buf, stat) {
|
|
2280
|
+
HEAPU32[buf >>> 2 >>> 0] = stat.dev;
|
|
2281
|
+
HEAPU32[buf + 4 >>> 2 >>> 0] = stat.mode;
|
|
2282
|
+
HEAPU32[buf + 8 >>> 2 >>> 0] = stat.nlink;
|
|
2283
|
+
HEAPU32[buf + 12 >>> 2 >>> 0] = stat.uid;
|
|
2284
|
+
HEAPU32[buf + 16 >>> 2 >>> 0] = stat.gid;
|
|
2285
|
+
HEAPU32[buf + 20 >>> 2 >>> 0] = stat.rdev;
|
|
2286
|
+
HEAP64[buf + 24 >>> 3 >>> 0] = BigInt(stat.size);
|
|
2287
|
+
HEAP32[buf + 32 >>> 2 >>> 0] = 4096;
|
|
2288
|
+
HEAP32[buf + 36 >>> 2 >>> 0] = stat.blocks;
|
|
2289
|
+
var atime = stat.atime.getTime();
|
|
2290
|
+
var mtime = stat.mtime.getTime();
|
|
2291
|
+
var ctime = stat.ctime.getTime();
|
|
2292
|
+
HEAP64[buf + 40 >>> 3 >>> 0] = BigInt(Math.floor(atime / 1e3));
|
|
2293
|
+
HEAPU32[buf + 48 >>> 2 >>> 0] = atime % 1e3 * 1e3 * 1e3;
|
|
2294
|
+
HEAP64[buf + 56 >>> 3 >>> 0] = BigInt(Math.floor(mtime / 1e3));
|
|
2295
|
+
HEAPU32[buf + 64 >>> 2 >>> 0] = mtime % 1e3 * 1e3 * 1e3;
|
|
2296
|
+
HEAP64[buf + 72 >>> 3 >>> 0] = BigInt(Math.floor(ctime / 1e3));
|
|
2297
|
+
HEAPU32[buf + 80 >>> 2 >>> 0] = ctime % 1e3 * 1e3 * 1e3;
|
|
2298
|
+
HEAP64[buf + 88 >>> 3 >>> 0] = BigInt(stat.ino);
|
|
2299
|
+
return 0;
|
|
2300
|
+
},
|
|
2301
|
+
writeStatFs(buf, stats) {
|
|
2302
|
+
HEAPU32[buf + 4 >>> 2 >>> 0] = stats.bsize;
|
|
2303
|
+
HEAPU32[buf + 60 >>> 2 >>> 0] = stats.bsize;
|
|
2304
|
+
HEAP64[buf + 8 >>> 3 >>> 0] = BigInt(stats.blocks);
|
|
2305
|
+
HEAP64[buf + 16 >>> 3 >>> 0] = BigInt(stats.bfree);
|
|
2306
|
+
HEAP64[buf + 24 >>> 3 >>> 0] = BigInt(stats.bavail);
|
|
2307
|
+
HEAP64[buf + 32 >>> 3 >>> 0] = BigInt(stats.files);
|
|
2308
|
+
HEAP64[buf + 40 >>> 3 >>> 0] = BigInt(stats.ffree);
|
|
2309
|
+
HEAPU32[buf + 48 >>> 2 >>> 0] = stats.fsid;
|
|
2310
|
+
HEAPU32[buf + 64 >>> 2 >>> 0] = stats.flags;
|
|
2311
|
+
HEAPU32[buf + 56 >>> 2 >>> 0] = stats.namelen;
|
|
2312
|
+
},
|
|
2313
|
+
doMsync(addr, stream, len, flags, offset) {
|
|
2314
|
+
if (!FS.isFile(stream.node.mode)) throw new FS.ErrnoError(43);
|
|
2315
|
+
if (flags & 2) return 0;
|
|
2316
|
+
var buffer = HEAPU8.slice(addr, addr + len);
|
|
2317
|
+
FS.msync(stream, buffer, offset, len, flags);
|
|
2318
|
+
},
|
|
2319
|
+
getStreamFromFD(fd) {
|
|
2320
|
+
return FS.getStreamChecked(fd);
|
|
2321
|
+
},
|
|
2322
|
+
varargs: void 0,
|
|
2323
|
+
getStr(ptr) {
|
|
2324
|
+
return UTF8ToString(ptr);
|
|
2325
|
+
}
|
|
2326
|
+
};
|
|
2327
|
+
var INT53_MAX = 9007199254740992;
|
|
2328
|
+
var INT53_MIN = -9007199254740992;
|
|
2329
|
+
var bigintToI53Checked = (num) => num < INT53_MIN || num > INT53_MAX ? NaN : Number(num);
|
|
2330
|
+
function ___syscall_chmod(path, mode) {
|
|
2331
|
+
path >>>= 0;
|
|
2332
|
+
try {
|
|
2333
|
+
path = SYSCALLS.getStr(path);
|
|
2334
|
+
FS.chmod(path, mode);
|
|
2335
|
+
return 0;
|
|
2336
|
+
} catch (e) {
|
|
2337
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2338
|
+
return -e.errno;
|
|
2339
|
+
}
|
|
2340
|
+
}
|
|
2341
|
+
function ___syscall_faccessat(dirfd, path, amode, flags) {
|
|
2342
|
+
path >>>= 0;
|
|
2343
|
+
try {
|
|
2344
|
+
path = SYSCALLS.getStr(path);
|
|
2345
|
+
path = SYSCALLS.calculateAt(dirfd, path);
|
|
2346
|
+
if (amode & -8) return -28;
|
|
2347
|
+
var node = FS.lookupPath(path, { follow: true }).node;
|
|
2348
|
+
if (!node) return -44;
|
|
2349
|
+
var perms = "";
|
|
2350
|
+
if (amode & 4) perms += "r";
|
|
2351
|
+
if (amode & 2) perms += "w";
|
|
2352
|
+
if (amode & 1) perms += "x";
|
|
2353
|
+
if (perms && FS.nodePermissions(node, perms)) return -2;
|
|
2354
|
+
return 0;
|
|
2355
|
+
} catch (e) {
|
|
2356
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2357
|
+
return -e.errno;
|
|
2358
|
+
}
|
|
2359
|
+
}
|
|
2360
|
+
function ___syscall_fchmod(fd, mode) {
|
|
2361
|
+
try {
|
|
2362
|
+
FS.fchmod(fd, mode);
|
|
2363
|
+
return 0;
|
|
2364
|
+
} catch (e) {
|
|
2365
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2366
|
+
return -e.errno;
|
|
2367
|
+
}
|
|
2368
|
+
}
|
|
2369
|
+
var syscallGetVarargI = () => {
|
|
2370
|
+
var ret = HEAP32[+SYSCALLS.varargs >>> 2 >>> 0];
|
|
2371
|
+
SYSCALLS.varargs += 4;
|
|
2372
|
+
return ret;
|
|
2373
|
+
};
|
|
2374
|
+
var syscallGetVarargP = syscallGetVarargI;
|
|
2375
|
+
function ___syscall_fcntl64(fd, cmd, varargs) {
|
|
2376
|
+
varargs >>>= 0;
|
|
2377
|
+
SYSCALLS.varargs = varargs;
|
|
2378
|
+
try {
|
|
2379
|
+
var stream = SYSCALLS.getStreamFromFD(fd);
|
|
2380
|
+
switch (cmd) {
|
|
2381
|
+
case 0:
|
|
2382
|
+
var arg = syscallGetVarargI();
|
|
2383
|
+
if (arg < 0) return -28;
|
|
2384
|
+
while (FS.streams[arg]) arg++;
|
|
2385
|
+
return FS.dupStream(stream, arg).fd;
|
|
2386
|
+
case 1:
|
|
2387
|
+
case 2: return 0;
|
|
2388
|
+
case 3: return stream.flags;
|
|
2389
|
+
case 4:
|
|
2390
|
+
var arg = syscallGetVarargI();
|
|
2391
|
+
stream.flags |= arg;
|
|
2392
|
+
return 0;
|
|
2393
|
+
case 12:
|
|
2394
|
+
var arg = syscallGetVarargP();
|
|
2395
|
+
var offset = 0;
|
|
2396
|
+
HEAP16[arg + offset >>> 1 >>> 0] = 2;
|
|
2397
|
+
return 0;
|
|
2398
|
+
case 13:
|
|
2399
|
+
case 14: return 0;
|
|
2400
|
+
}
|
|
2401
|
+
return -28;
|
|
2402
|
+
} catch (e) {
|
|
2403
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2404
|
+
return -e.errno;
|
|
2405
|
+
}
|
|
2406
|
+
}
|
|
2407
|
+
function ___syscall_fdatasync(fd) {
|
|
2408
|
+
try {
|
|
2409
|
+
SYSCALLS.getStreamFromFD(fd);
|
|
2410
|
+
return 0;
|
|
2411
|
+
} catch (e) {
|
|
2412
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2413
|
+
return -e.errno;
|
|
2414
|
+
}
|
|
2415
|
+
}
|
|
2416
|
+
function ___syscall_fstat64(fd, buf) {
|
|
2417
|
+
buf >>>= 0;
|
|
2418
|
+
try {
|
|
2419
|
+
return SYSCALLS.writeStat(buf, FS.fstat(fd));
|
|
2420
|
+
} catch (e) {
|
|
2421
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2422
|
+
return -e.errno;
|
|
2423
|
+
}
|
|
2424
|
+
}
|
|
2425
|
+
function ___syscall_ftruncate64(fd, length) {
|
|
2426
|
+
length = bigintToI53Checked(length);
|
|
2427
|
+
try {
|
|
2428
|
+
if (isNaN(length)) return -61;
|
|
2429
|
+
FS.ftruncate(fd, length);
|
|
2430
|
+
return 0;
|
|
2431
|
+
} catch (e) {
|
|
2432
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2433
|
+
return -e.errno;
|
|
2434
|
+
}
|
|
2435
|
+
}
|
|
2436
|
+
var stringToUTF8 = (str, outPtr, maxBytesToWrite) => stringToUTF8Array(str, HEAPU8, outPtr, maxBytesToWrite);
|
|
2437
|
+
function ___syscall_getcwd(buf, size) {
|
|
2438
|
+
buf >>>= 0;
|
|
2439
|
+
size >>>= 0;
|
|
2440
|
+
try {
|
|
2441
|
+
if (size === 0) return -28;
|
|
2442
|
+
var cwd = FS.cwd();
|
|
2443
|
+
var cwdLengthInBytes = lengthBytesUTF8(cwd) + 1;
|
|
2444
|
+
if (size < cwdLengthInBytes) return -68;
|
|
2445
|
+
stringToUTF8(cwd, buf, size);
|
|
2446
|
+
return cwdLengthInBytes;
|
|
2447
|
+
} catch (e) {
|
|
2448
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2449
|
+
return -e.errno;
|
|
2450
|
+
}
|
|
2451
|
+
}
|
|
2452
|
+
function ___syscall_getdents64(fd, dirp, count) {
|
|
2453
|
+
dirp >>>= 0;
|
|
2454
|
+
count >>>= 0;
|
|
2455
|
+
try {
|
|
2456
|
+
var stream = SYSCALLS.getStreamFromFD(fd);
|
|
2457
|
+
stream.getdents ||= FS.readdir(stream.path);
|
|
2458
|
+
var struct_size = 280;
|
|
2459
|
+
var pos = 0;
|
|
2460
|
+
var off = FS.llseek(stream, 0, 1);
|
|
2461
|
+
var startIdx = Math.floor(off / struct_size);
|
|
2462
|
+
var endIdx = Math.min(stream.getdents.length, startIdx + Math.floor(count / struct_size));
|
|
2463
|
+
for (var idx = startIdx; idx < endIdx; idx++) {
|
|
2464
|
+
var id;
|
|
2465
|
+
var type;
|
|
2466
|
+
var name = stream.getdents[idx];
|
|
2467
|
+
if (name === ".") {
|
|
2468
|
+
id = stream.node.id;
|
|
2469
|
+
type = 4;
|
|
2470
|
+
} else if (name === "..") {
|
|
2471
|
+
id = FS.lookupPath(stream.path, { parent: true }).node.id;
|
|
2472
|
+
type = 4;
|
|
2473
|
+
} else {
|
|
2474
|
+
var child;
|
|
2475
|
+
try {
|
|
2476
|
+
child = FS.lookupNode(stream.node, name);
|
|
2477
|
+
} catch (e) {
|
|
2478
|
+
if (e?.errno === 28) continue;
|
|
2479
|
+
throw e;
|
|
2480
|
+
}
|
|
2481
|
+
id = child.id;
|
|
2482
|
+
type = FS.isChrdev(child.mode) ? 2 : FS.isDir(child.mode) ? 4 : FS.isLink(child.mode) ? 10 : 8;
|
|
2483
|
+
}
|
|
2484
|
+
HEAP64[dirp + pos >>> 3 >>> 0] = BigInt(id);
|
|
2485
|
+
HEAP64[dirp + pos + 8 >>> 3 >>> 0] = BigInt((idx + 1) * struct_size);
|
|
2486
|
+
HEAP16[dirp + pos + 16 >>> 1 >>> 0] = 280;
|
|
2487
|
+
HEAP8[dirp + pos + 18 >>> 0] = type;
|
|
2488
|
+
stringToUTF8(name, dirp + pos + 19, 256);
|
|
2489
|
+
pos += struct_size;
|
|
2490
|
+
}
|
|
2491
|
+
FS.llseek(stream, idx * struct_size, 0);
|
|
2492
|
+
return pos;
|
|
2493
|
+
} catch (e) {
|
|
2494
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2495
|
+
return -e.errno;
|
|
2496
|
+
}
|
|
2497
|
+
}
|
|
2498
|
+
function ___syscall_ioctl(fd, op, varargs) {
|
|
2499
|
+
varargs >>>= 0;
|
|
2500
|
+
SYSCALLS.varargs = varargs;
|
|
2501
|
+
try {
|
|
2502
|
+
var stream = SYSCALLS.getStreamFromFD(fd);
|
|
2503
|
+
switch (op) {
|
|
2504
|
+
case 21509:
|
|
2505
|
+
if (!stream.tty) return -59;
|
|
2506
|
+
return 0;
|
|
2507
|
+
case 21505:
|
|
2508
|
+
if (!stream.tty) return -59;
|
|
2509
|
+
if (stream.tty.ops.ioctl_tcgets) {
|
|
2510
|
+
var termios = stream.tty.ops.ioctl_tcgets(stream);
|
|
2511
|
+
var argp = syscallGetVarargP();
|
|
2512
|
+
HEAP32[argp >>> 2 >>> 0] = termios.c_iflag || 0;
|
|
2513
|
+
HEAP32[argp + 4 >>> 2 >>> 0] = termios.c_oflag || 0;
|
|
2514
|
+
HEAP32[argp + 8 >>> 2 >>> 0] = termios.c_cflag || 0;
|
|
2515
|
+
HEAP32[argp + 12 >>> 2 >>> 0] = termios.c_lflag || 0;
|
|
2516
|
+
for (var i = 0; i < 32; i++) HEAP8[argp + i + 17 >>> 0] = termios.c_cc[i] || 0;
|
|
2517
|
+
return 0;
|
|
2518
|
+
}
|
|
2519
|
+
return 0;
|
|
2520
|
+
case 21510:
|
|
2521
|
+
case 21511:
|
|
2522
|
+
case 21512:
|
|
2523
|
+
if (!stream.tty) return -59;
|
|
2524
|
+
return 0;
|
|
2525
|
+
case 21506:
|
|
2526
|
+
case 21507:
|
|
2527
|
+
case 21508:
|
|
2528
|
+
if (!stream.tty) return -59;
|
|
2529
|
+
if (stream.tty.ops.ioctl_tcsets) {
|
|
2530
|
+
var argp = syscallGetVarargP();
|
|
2531
|
+
var c_iflag = HEAP32[argp >>> 2 >>> 0];
|
|
2532
|
+
var c_oflag = HEAP32[argp + 4 >>> 2 >>> 0];
|
|
2533
|
+
var c_cflag = HEAP32[argp + 8 >>> 2 >>> 0];
|
|
2534
|
+
var c_lflag = HEAP32[argp + 12 >>> 2 >>> 0];
|
|
2535
|
+
var c_cc = [];
|
|
2536
|
+
for (var i = 0; i < 32; i++) c_cc.push(HEAP8[argp + i + 17 >>> 0]);
|
|
2537
|
+
return stream.tty.ops.ioctl_tcsets(stream.tty, op, {
|
|
2538
|
+
c_iflag,
|
|
2539
|
+
c_oflag,
|
|
2540
|
+
c_cflag,
|
|
2541
|
+
c_lflag,
|
|
2542
|
+
c_cc
|
|
2543
|
+
});
|
|
2544
|
+
}
|
|
2545
|
+
return 0;
|
|
2546
|
+
case 21519:
|
|
2547
|
+
if (!stream.tty) return -59;
|
|
2548
|
+
var argp = syscallGetVarargP();
|
|
2549
|
+
HEAP32[argp >>> 2 >>> 0] = 0;
|
|
2550
|
+
return 0;
|
|
2551
|
+
case 21520:
|
|
2552
|
+
if (!stream.tty) return -59;
|
|
2553
|
+
return -28;
|
|
2554
|
+
case 21537:
|
|
2555
|
+
case 21531:
|
|
2556
|
+
var argp = syscallGetVarargP();
|
|
2557
|
+
return FS.ioctl(stream, op, argp);
|
|
2558
|
+
case 21523:
|
|
2559
|
+
if (!stream.tty) return -59;
|
|
2560
|
+
if (stream.tty.ops.ioctl_tiocgwinsz) {
|
|
2561
|
+
var winsize = stream.tty.ops.ioctl_tiocgwinsz(stream.tty);
|
|
2562
|
+
var argp = syscallGetVarargP();
|
|
2563
|
+
HEAP16[argp >>> 1 >>> 0] = winsize[0];
|
|
2564
|
+
HEAP16[argp + 2 >>> 1 >>> 0] = winsize[1];
|
|
2565
|
+
}
|
|
2566
|
+
return 0;
|
|
2567
|
+
case 21524:
|
|
2568
|
+
if (!stream.tty) return -59;
|
|
2569
|
+
return 0;
|
|
2570
|
+
case 21515:
|
|
2571
|
+
if (!stream.tty) return -59;
|
|
2572
|
+
return 0;
|
|
2573
|
+
default: return -28;
|
|
2574
|
+
}
|
|
2575
|
+
} catch (e) {
|
|
2576
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2577
|
+
return -e.errno;
|
|
2578
|
+
}
|
|
2579
|
+
}
|
|
2580
|
+
function ___syscall_lstat64(path, buf) {
|
|
2581
|
+
path >>>= 0;
|
|
2582
|
+
buf >>>= 0;
|
|
2583
|
+
try {
|
|
2584
|
+
path = SYSCALLS.getStr(path);
|
|
2585
|
+
return SYSCALLS.writeStat(buf, FS.lstat(path));
|
|
2586
|
+
} catch (e) {
|
|
2587
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2588
|
+
return -e.errno;
|
|
2589
|
+
}
|
|
2590
|
+
}
|
|
2591
|
+
function ___syscall_mkdirat(dirfd, path, mode) {
|
|
2592
|
+
path >>>= 0;
|
|
2593
|
+
try {
|
|
2594
|
+
path = SYSCALLS.getStr(path);
|
|
2595
|
+
path = SYSCALLS.calculateAt(dirfd, path);
|
|
2596
|
+
FS.mkdir(path, mode, 0);
|
|
2597
|
+
return 0;
|
|
2598
|
+
} catch (e) {
|
|
2599
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2600
|
+
return -e.errno;
|
|
2601
|
+
}
|
|
2602
|
+
}
|
|
2603
|
+
function ___syscall_newfstatat(dirfd, path, buf, flags) {
|
|
2604
|
+
path >>>= 0;
|
|
2605
|
+
buf >>>= 0;
|
|
2606
|
+
try {
|
|
2607
|
+
path = SYSCALLS.getStr(path);
|
|
2608
|
+
var nofollow = flags & 256;
|
|
2609
|
+
var allowEmpty = flags & 4096;
|
|
2610
|
+
flags = flags & -6401;
|
|
2611
|
+
path = SYSCALLS.calculateAt(dirfd, path, allowEmpty);
|
|
2612
|
+
return SYSCALLS.writeStat(buf, nofollow ? FS.lstat(path) : FS.stat(path));
|
|
2613
|
+
} catch (e) {
|
|
2614
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2615
|
+
return -e.errno;
|
|
2616
|
+
}
|
|
2617
|
+
}
|
|
2618
|
+
function ___syscall_openat(dirfd, path, flags, varargs) {
|
|
2619
|
+
path >>>= 0;
|
|
2620
|
+
varargs >>>= 0;
|
|
2621
|
+
SYSCALLS.varargs = varargs;
|
|
2622
|
+
try {
|
|
2623
|
+
path = SYSCALLS.getStr(path);
|
|
2624
|
+
path = SYSCALLS.calculateAt(dirfd, path);
|
|
2625
|
+
var mode = varargs ? syscallGetVarargI() : 0;
|
|
2626
|
+
return FS.open(path, flags, mode).fd;
|
|
2627
|
+
} catch (e) {
|
|
2628
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2629
|
+
return -e.errno;
|
|
2630
|
+
}
|
|
2631
|
+
}
|
|
2632
|
+
function ___syscall_readlinkat(dirfd, path, buf, bufsize) {
|
|
2633
|
+
path >>>= 0;
|
|
2634
|
+
buf >>>= 0;
|
|
2635
|
+
bufsize >>>= 0;
|
|
2636
|
+
try {
|
|
2637
|
+
path = SYSCALLS.getStr(path);
|
|
2638
|
+
path = SYSCALLS.calculateAt(dirfd, path);
|
|
2639
|
+
if (bufsize <= 0) return -28;
|
|
2640
|
+
var ret = FS.readlink(path);
|
|
2641
|
+
var len = Math.min(bufsize, lengthBytesUTF8(ret));
|
|
2642
|
+
var endChar = HEAP8[buf + len >>> 0];
|
|
2643
|
+
stringToUTF8(ret, buf, bufsize + 1);
|
|
2644
|
+
HEAP8[buf + len >>> 0] = endChar;
|
|
2645
|
+
return len;
|
|
2646
|
+
} catch (e) {
|
|
2647
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2648
|
+
return -e.errno;
|
|
2649
|
+
}
|
|
2650
|
+
}
|
|
2651
|
+
function ___syscall_renameat(olddirfd, oldpath, newdirfd, newpath) {
|
|
2652
|
+
oldpath >>>= 0;
|
|
2653
|
+
newpath >>>= 0;
|
|
2654
|
+
try {
|
|
2655
|
+
oldpath = SYSCALLS.getStr(oldpath);
|
|
2656
|
+
newpath = SYSCALLS.getStr(newpath);
|
|
2657
|
+
oldpath = SYSCALLS.calculateAt(olddirfd, oldpath);
|
|
2658
|
+
newpath = SYSCALLS.calculateAt(newdirfd, newpath);
|
|
2659
|
+
FS.rename(oldpath, newpath);
|
|
2660
|
+
return 0;
|
|
2661
|
+
} catch (e) {
|
|
2662
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2663
|
+
return -e.errno;
|
|
2664
|
+
}
|
|
2665
|
+
}
|
|
2666
|
+
function ___syscall_rmdir(path) {
|
|
2667
|
+
path >>>= 0;
|
|
2668
|
+
try {
|
|
2669
|
+
path = SYSCALLS.getStr(path);
|
|
2670
|
+
FS.rmdir(path);
|
|
2671
|
+
return 0;
|
|
2672
|
+
} catch (e) {
|
|
2673
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2674
|
+
return -e.errno;
|
|
2675
|
+
}
|
|
2676
|
+
}
|
|
2677
|
+
function ___syscall_stat64(path, buf) {
|
|
2678
|
+
path >>>= 0;
|
|
2679
|
+
buf >>>= 0;
|
|
2680
|
+
try {
|
|
2681
|
+
path = SYSCALLS.getStr(path);
|
|
2682
|
+
return SYSCALLS.writeStat(buf, FS.stat(path));
|
|
2683
|
+
} catch (e) {
|
|
2684
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2685
|
+
return -e.errno;
|
|
2686
|
+
}
|
|
2687
|
+
}
|
|
2688
|
+
function ___syscall_truncate64(path, length) {
|
|
2689
|
+
path >>>= 0;
|
|
2690
|
+
length = bigintToI53Checked(length);
|
|
2691
|
+
try {
|
|
2692
|
+
if (isNaN(length)) return -61;
|
|
2693
|
+
path = SYSCALLS.getStr(path);
|
|
2694
|
+
FS.truncate(path, length);
|
|
2695
|
+
return 0;
|
|
2696
|
+
} catch (e) {
|
|
2697
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2698
|
+
return -e.errno;
|
|
2699
|
+
}
|
|
2700
|
+
}
|
|
2701
|
+
function ___syscall_unlinkat(dirfd, path, flags) {
|
|
2702
|
+
path >>>= 0;
|
|
2703
|
+
try {
|
|
2704
|
+
path = SYSCALLS.getStr(path);
|
|
2705
|
+
path = SYSCALLS.calculateAt(dirfd, path);
|
|
2706
|
+
if (!flags) FS.unlink(path);
|
|
2707
|
+
else if (flags === 512) FS.rmdir(path);
|
|
2708
|
+
else return -28;
|
|
2709
|
+
return 0;
|
|
2710
|
+
} catch (e) {
|
|
2711
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2712
|
+
return -e.errno;
|
|
2713
|
+
}
|
|
2714
|
+
}
|
|
2715
|
+
var __abort_js = () => abort("");
|
|
2716
|
+
var runtimeKeepaliveCounter = 0;
|
|
2717
|
+
var __emscripten_runtime_keepalive_clear = () => {
|
|
2718
|
+
noExitRuntime = false;
|
|
2719
|
+
runtimeKeepaliveCounter = 0;
|
|
2720
|
+
};
|
|
2721
|
+
var isLeapYear = (year) => year % 4 === 0 && (year % 100 !== 0 || year % 400 === 0);
|
|
2722
|
+
var MONTH_DAYS_LEAP_CUMULATIVE = [
|
|
2723
|
+
0,
|
|
2724
|
+
31,
|
|
2725
|
+
60,
|
|
2726
|
+
91,
|
|
2727
|
+
121,
|
|
2728
|
+
152,
|
|
2729
|
+
182,
|
|
2730
|
+
213,
|
|
2731
|
+
244,
|
|
2732
|
+
274,
|
|
2733
|
+
305,
|
|
2734
|
+
335
|
|
2735
|
+
];
|
|
2736
|
+
var MONTH_DAYS_REGULAR_CUMULATIVE = [
|
|
2737
|
+
0,
|
|
2738
|
+
31,
|
|
2739
|
+
59,
|
|
2740
|
+
90,
|
|
2741
|
+
120,
|
|
2742
|
+
151,
|
|
2743
|
+
181,
|
|
2744
|
+
212,
|
|
2745
|
+
243,
|
|
2746
|
+
273,
|
|
2747
|
+
304,
|
|
2748
|
+
334
|
|
2749
|
+
];
|
|
2750
|
+
var ydayFromDate = (date) => {
|
|
2751
|
+
return (isLeapYear(date.getFullYear()) ? MONTH_DAYS_LEAP_CUMULATIVE : MONTH_DAYS_REGULAR_CUMULATIVE)[date.getMonth()] + date.getDate() - 1;
|
|
2752
|
+
};
|
|
2753
|
+
function __localtime_js(time, tmPtr) {
|
|
2754
|
+
time = bigintToI53Checked(time);
|
|
2755
|
+
tmPtr >>>= 0;
|
|
2756
|
+
var date = /* @__PURE__ */ new Date(time * 1e3);
|
|
2757
|
+
HEAP32[tmPtr >>> 2 >>> 0] = date.getSeconds();
|
|
2758
|
+
HEAP32[tmPtr + 4 >>> 2 >>> 0] = date.getMinutes();
|
|
2759
|
+
HEAP32[tmPtr + 8 >>> 2 >>> 0] = date.getHours();
|
|
2760
|
+
HEAP32[tmPtr + 12 >>> 2 >>> 0] = date.getDate();
|
|
2761
|
+
HEAP32[tmPtr + 16 >>> 2 >>> 0] = date.getMonth();
|
|
2762
|
+
HEAP32[tmPtr + 20 >>> 2 >>> 0] = date.getFullYear() - 1900;
|
|
2763
|
+
HEAP32[tmPtr + 24 >>> 2 >>> 0] = date.getDay();
|
|
2764
|
+
var yday = ydayFromDate(date) | 0;
|
|
2765
|
+
HEAP32[tmPtr + 28 >>> 2 >>> 0] = yday;
|
|
2766
|
+
HEAP32[tmPtr + 36 >>> 2 >>> 0] = -(date.getTimezoneOffset() * 60);
|
|
2767
|
+
var start = new Date(date.getFullYear(), 0, 1);
|
|
2768
|
+
var summerOffset = new Date(date.getFullYear(), 6, 1).getTimezoneOffset();
|
|
2769
|
+
var winterOffset = start.getTimezoneOffset();
|
|
2770
|
+
var dst = (summerOffset != winterOffset && date.getTimezoneOffset() == Math.min(winterOffset, summerOffset)) | 0;
|
|
2771
|
+
HEAP32[tmPtr + 32 >>> 2 >>> 0] = dst;
|
|
2772
|
+
}
|
|
2773
|
+
function __mmap_js(len, prot, flags, fd, offset, allocated, addr) {
|
|
2774
|
+
len >>>= 0;
|
|
2775
|
+
offset = bigintToI53Checked(offset);
|
|
2776
|
+
allocated >>>= 0;
|
|
2777
|
+
addr >>>= 0;
|
|
2778
|
+
try {
|
|
2779
|
+
var stream = SYSCALLS.getStreamFromFD(fd);
|
|
2780
|
+
var res = FS.mmap(stream, len, offset, prot, flags);
|
|
2781
|
+
var ptr = res.ptr;
|
|
2782
|
+
HEAP32[allocated >>> 2 >>> 0] = res.allocated;
|
|
2783
|
+
HEAPU32[addr >>> 2 >>> 0] = ptr;
|
|
2784
|
+
return 0;
|
|
2785
|
+
} catch (e) {
|
|
2786
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2787
|
+
return -e.errno;
|
|
2788
|
+
}
|
|
2789
|
+
}
|
|
2790
|
+
function __munmap_js(addr, len, prot, flags, fd, offset) {
|
|
2791
|
+
addr >>>= 0;
|
|
2792
|
+
len >>>= 0;
|
|
2793
|
+
offset = bigintToI53Checked(offset);
|
|
2794
|
+
try {
|
|
2795
|
+
var stream = SYSCALLS.getStreamFromFD(fd);
|
|
2796
|
+
if (prot & 2) SYSCALLS.doMsync(addr, stream, len, flags, offset);
|
|
2797
|
+
} catch (e) {
|
|
2798
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2799
|
+
return -e.errno;
|
|
2800
|
+
}
|
|
2801
|
+
}
|
|
2802
|
+
var timers = {};
|
|
2803
|
+
var handleException = (e) => {
|
|
2804
|
+
if (e instanceof ExitStatus || e == "unwind") return EXITSTATUS;
|
|
2805
|
+
quit_(1, e);
|
|
2806
|
+
};
|
|
2807
|
+
var keepRuntimeAlive = () => noExitRuntime || runtimeKeepaliveCounter > 0;
|
|
2808
|
+
var _proc_exit = (code) => {
|
|
2809
|
+
EXITSTATUS = code;
|
|
2810
|
+
if (!keepRuntimeAlive()) {
|
|
2811
|
+
Module["onExit"]?.(code);
|
|
2812
|
+
ABORT = true;
|
|
2813
|
+
}
|
|
2814
|
+
quit_(code, new ExitStatus(code));
|
|
2815
|
+
};
|
|
2816
|
+
var exitJS = (status, implicit) => {
|
|
2817
|
+
EXITSTATUS = status;
|
|
2818
|
+
_proc_exit(status);
|
|
2819
|
+
};
|
|
2820
|
+
var _exit = exitJS;
|
|
2821
|
+
var maybeExit = () => {
|
|
2822
|
+
if (!keepRuntimeAlive()) try {
|
|
2823
|
+
_exit(EXITSTATUS);
|
|
2824
|
+
} catch (e) {
|
|
2825
|
+
handleException(e);
|
|
2826
|
+
}
|
|
2827
|
+
};
|
|
2828
|
+
var callUserCallback = (func) => {
|
|
2829
|
+
if (ABORT) return;
|
|
2830
|
+
try {
|
|
2831
|
+
return func();
|
|
2832
|
+
} catch (e) {
|
|
2833
|
+
handleException(e);
|
|
2834
|
+
} finally {
|
|
2835
|
+
maybeExit();
|
|
2836
|
+
}
|
|
2837
|
+
};
|
|
2838
|
+
var _emscripten_get_now = () => performance.now();
|
|
2839
|
+
var __setitimer_js = (which, timeout_ms) => {
|
|
2840
|
+
if (timers[which]) {
|
|
2841
|
+
clearTimeout(timers[which].id);
|
|
2842
|
+
delete timers[which];
|
|
2843
|
+
}
|
|
2844
|
+
if (!timeout_ms) return 0;
|
|
2845
|
+
timers[which] = {
|
|
2846
|
+
id: setTimeout(() => {
|
|
2847
|
+
delete timers[which];
|
|
2848
|
+
callUserCallback(() => __emscripten_timeout(which, _emscripten_get_now()));
|
|
2849
|
+
}, timeout_ms),
|
|
2850
|
+
timeout_ms
|
|
2851
|
+
};
|
|
2852
|
+
return 0;
|
|
2853
|
+
};
|
|
2854
|
+
var __tzset_js = function(timezone, daylight, std_name, dst_name) {
|
|
2855
|
+
timezone >>>= 0;
|
|
2856
|
+
daylight >>>= 0;
|
|
2857
|
+
std_name >>>= 0;
|
|
2858
|
+
dst_name >>>= 0;
|
|
2859
|
+
var currentYear = (/* @__PURE__ */ new Date()).getFullYear();
|
|
2860
|
+
var winter = new Date(currentYear, 0, 1);
|
|
2861
|
+
var summer = new Date(currentYear, 6, 1);
|
|
2862
|
+
var winterOffset = winter.getTimezoneOffset();
|
|
2863
|
+
var summerOffset = summer.getTimezoneOffset();
|
|
2864
|
+
var stdTimezoneOffset = Math.max(winterOffset, summerOffset);
|
|
2865
|
+
HEAPU32[timezone >>> 2 >>> 0] = stdTimezoneOffset * 60;
|
|
2866
|
+
HEAP32[daylight >>> 2 >>> 0] = Number(winterOffset != summerOffset);
|
|
2867
|
+
var extractZone = (timezoneOffset) => {
|
|
2868
|
+
var sign = timezoneOffset >= 0 ? "-" : "+";
|
|
2869
|
+
var absOffset = Math.abs(timezoneOffset);
|
|
2870
|
+
return `UTC${sign}${String(Math.floor(absOffset / 60)).padStart(2, "0")}${String(absOffset % 60).padStart(2, "0")}`;
|
|
2871
|
+
};
|
|
2872
|
+
var winterName = extractZone(winterOffset);
|
|
2873
|
+
var summerName = extractZone(summerOffset);
|
|
2874
|
+
if (summerOffset < winterOffset) {
|
|
2875
|
+
stringToUTF8(winterName, std_name, 17);
|
|
2876
|
+
stringToUTF8(summerName, dst_name, 17);
|
|
2877
|
+
} else {
|
|
2878
|
+
stringToUTF8(winterName, dst_name, 17);
|
|
2879
|
+
stringToUTF8(summerName, std_name, 17);
|
|
2880
|
+
}
|
|
2881
|
+
};
|
|
2882
|
+
var _emscripten_date_now = () => Date.now();
|
|
2883
|
+
var nowIsMonotonic = 1;
|
|
2884
|
+
var checkWasiClock = (clock_id) => clock_id >= 0 && clock_id <= 3;
|
|
2885
|
+
function _clock_time_get(clk_id, ignored_precision, ptime) {
|
|
2886
|
+
ignored_precision = bigintToI53Checked(ignored_precision);
|
|
2887
|
+
ptime >>>= 0;
|
|
2888
|
+
if (!checkWasiClock(clk_id)) return 28;
|
|
2889
|
+
var now;
|
|
2890
|
+
if (clk_id === 0) now = _emscripten_date_now();
|
|
2891
|
+
else if (nowIsMonotonic) now = _emscripten_get_now();
|
|
2892
|
+
else return 52;
|
|
2893
|
+
var nsec = Math.round(now * 1e3 * 1e3);
|
|
2894
|
+
HEAP64[ptime >>> 3 >>> 0] = BigInt(nsec);
|
|
2895
|
+
return 0;
|
|
2896
|
+
}
|
|
2897
|
+
var getHeapMax = () => 4294901760;
|
|
2898
|
+
var growMemory = (size) => {
|
|
2899
|
+
var pages = (size - wasmMemory.buffer.byteLength + 65535) / 65536 | 0;
|
|
2900
|
+
try {
|
|
2901
|
+
wasmMemory.grow(pages);
|
|
2902
|
+
updateMemoryViews();
|
|
2903
|
+
return 1;
|
|
2904
|
+
} catch (e) {}
|
|
2905
|
+
};
|
|
2906
|
+
function _emscripten_resize_heap(requestedSize) {
|
|
2907
|
+
requestedSize >>>= 0;
|
|
2908
|
+
var oldSize = HEAPU8.length;
|
|
2909
|
+
var maxHeapSize = getHeapMax();
|
|
2910
|
+
if (requestedSize > maxHeapSize) return false;
|
|
2911
|
+
for (var cutDown = 1; cutDown <= 4; cutDown *= 2) {
|
|
2912
|
+
var overGrownHeapSize = oldSize * (1 + .2 / cutDown);
|
|
2913
|
+
overGrownHeapSize = Math.min(overGrownHeapSize, requestedSize + 100663296);
|
|
2914
|
+
if (growMemory(Math.min(maxHeapSize, alignMemory(Math.max(requestedSize, overGrownHeapSize), 65536)))) return true;
|
|
2915
|
+
}
|
|
2916
|
+
return false;
|
|
2917
|
+
}
|
|
2918
|
+
var ENV = {};
|
|
2919
|
+
var getExecutableName = () => thisProgram || "./this.program";
|
|
2920
|
+
var getEnvStrings = () => {
|
|
2921
|
+
if (!getEnvStrings.strings) {
|
|
2922
|
+
var env = {
|
|
2923
|
+
USER: "web_user",
|
|
2924
|
+
LOGNAME: "web_user",
|
|
2925
|
+
PATH: "/",
|
|
2926
|
+
PWD: "/",
|
|
2927
|
+
HOME: "/home/web_user",
|
|
2928
|
+
LANG: (globalThis.navigator?.language ?? "C").replace("-", "_") + ".UTF-8",
|
|
2929
|
+
_: getExecutableName()
|
|
2930
|
+
};
|
|
2931
|
+
for (var x in ENV) if (ENV[x] === void 0) delete env[x];
|
|
2932
|
+
else env[x] = ENV[x];
|
|
2933
|
+
var strings = [];
|
|
2934
|
+
for (var x in env) strings.push(`${x}=${env[x]}`);
|
|
2935
|
+
getEnvStrings.strings = strings;
|
|
2936
|
+
}
|
|
2937
|
+
return getEnvStrings.strings;
|
|
2938
|
+
};
|
|
2939
|
+
function _environ_get(__environ, environ_buf) {
|
|
2940
|
+
__environ >>>= 0;
|
|
2941
|
+
environ_buf >>>= 0;
|
|
2942
|
+
var bufSize = 0;
|
|
2943
|
+
var envp = 0;
|
|
2944
|
+
for (var string of getEnvStrings()) {
|
|
2945
|
+
var ptr = environ_buf + bufSize;
|
|
2946
|
+
HEAPU32[__environ + envp >>> 2 >>> 0] = ptr;
|
|
2947
|
+
bufSize += stringToUTF8(string, ptr, Infinity) + 1;
|
|
2948
|
+
envp += 4;
|
|
2949
|
+
}
|
|
2950
|
+
return 0;
|
|
2951
|
+
}
|
|
2952
|
+
function _environ_sizes_get(penviron_count, penviron_buf_size) {
|
|
2953
|
+
penviron_count >>>= 0;
|
|
2954
|
+
penviron_buf_size >>>= 0;
|
|
2955
|
+
var strings = getEnvStrings();
|
|
2956
|
+
HEAPU32[penviron_count >>> 2 >>> 0] = strings.length;
|
|
2957
|
+
var bufSize = 0;
|
|
2958
|
+
for (var string of strings) bufSize += lengthBytesUTF8(string) + 1;
|
|
2959
|
+
HEAPU32[penviron_buf_size >>> 2 >>> 0] = bufSize;
|
|
2960
|
+
return 0;
|
|
2961
|
+
}
|
|
2962
|
+
function _fd_close(fd) {
|
|
2963
|
+
try {
|
|
2964
|
+
var stream = SYSCALLS.getStreamFromFD(fd);
|
|
2965
|
+
FS.close(stream);
|
|
2966
|
+
return 0;
|
|
2967
|
+
} catch (e) {
|
|
2968
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2969
|
+
return e.errno;
|
|
2970
|
+
}
|
|
2971
|
+
}
|
|
2972
|
+
function _fd_fdstat_get(fd, pbuf) {
|
|
2973
|
+
pbuf >>>= 0;
|
|
2974
|
+
try {
|
|
2975
|
+
var rightsBase = 0;
|
|
2976
|
+
var rightsInheriting = 0;
|
|
2977
|
+
var flags = 0;
|
|
2978
|
+
var stream = SYSCALLS.getStreamFromFD(fd);
|
|
2979
|
+
var type = stream.tty ? 2 : FS.isDir(stream.mode) ? 3 : FS.isLink(stream.mode) ? 7 : 4;
|
|
2980
|
+
HEAP8[pbuf >>> 0] = type;
|
|
2981
|
+
HEAP16[pbuf + 2 >>> 1 >>> 0] = flags;
|
|
2982
|
+
HEAP64[pbuf + 8 >>> 3 >>> 0] = BigInt(rightsBase);
|
|
2983
|
+
HEAP64[pbuf + 16 >>> 3 >>> 0] = BigInt(rightsInheriting);
|
|
2984
|
+
return 0;
|
|
2985
|
+
} catch (e) {
|
|
2986
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
2987
|
+
return e.errno;
|
|
2988
|
+
}
|
|
2989
|
+
}
|
|
2990
|
+
var doReadv = (stream, iov, iovcnt, offset) => {
|
|
2991
|
+
var ret = 0;
|
|
2992
|
+
for (var i = 0; i < iovcnt; i++) {
|
|
2993
|
+
var ptr = HEAPU32[iov >>> 2 >>> 0];
|
|
2994
|
+
var len = HEAPU32[iov + 4 >>> 2 >>> 0];
|
|
2995
|
+
iov += 8;
|
|
2996
|
+
var curr = FS.read(stream, HEAP8, ptr, len, offset);
|
|
2997
|
+
if (curr < 0) return -1;
|
|
2998
|
+
ret += curr;
|
|
2999
|
+
if (curr < len) break;
|
|
3000
|
+
if (typeof offset != "undefined") offset += curr;
|
|
3001
|
+
}
|
|
3002
|
+
return ret;
|
|
3003
|
+
};
|
|
3004
|
+
function _fd_pread(fd, iov, iovcnt, offset, pnum) {
|
|
3005
|
+
iov >>>= 0;
|
|
3006
|
+
iovcnt >>>= 0;
|
|
3007
|
+
offset = bigintToI53Checked(offset);
|
|
3008
|
+
pnum >>>= 0;
|
|
3009
|
+
try {
|
|
3010
|
+
if (isNaN(offset)) return 61;
|
|
3011
|
+
var num = doReadv(SYSCALLS.getStreamFromFD(fd), iov, iovcnt, offset);
|
|
3012
|
+
HEAPU32[pnum >>> 2 >>> 0] = num;
|
|
3013
|
+
return 0;
|
|
3014
|
+
} catch (e) {
|
|
3015
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
3016
|
+
return e.errno;
|
|
3017
|
+
}
|
|
3018
|
+
}
|
|
3019
|
+
function _fd_read(fd, iov, iovcnt, pnum) {
|
|
3020
|
+
iov >>>= 0;
|
|
3021
|
+
iovcnt >>>= 0;
|
|
3022
|
+
pnum >>>= 0;
|
|
3023
|
+
try {
|
|
3024
|
+
var num = doReadv(SYSCALLS.getStreamFromFD(fd), iov, iovcnt);
|
|
3025
|
+
HEAPU32[pnum >>> 2 >>> 0] = num;
|
|
3026
|
+
return 0;
|
|
3027
|
+
} catch (e) {
|
|
3028
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
3029
|
+
return e.errno;
|
|
3030
|
+
}
|
|
3031
|
+
}
|
|
3032
|
+
function _fd_seek(fd, offset, whence, newOffset) {
|
|
3033
|
+
offset = bigintToI53Checked(offset);
|
|
3034
|
+
newOffset >>>= 0;
|
|
3035
|
+
try {
|
|
3036
|
+
if (isNaN(offset)) return 61;
|
|
3037
|
+
var stream = SYSCALLS.getStreamFromFD(fd);
|
|
3038
|
+
FS.llseek(stream, offset, whence);
|
|
3039
|
+
HEAP64[newOffset >>> 3 >>> 0] = BigInt(stream.position);
|
|
3040
|
+
if (stream.getdents && offset === 0 && whence === 0) stream.getdents = null;
|
|
3041
|
+
return 0;
|
|
3042
|
+
} catch (e) {
|
|
3043
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
3044
|
+
return e.errno;
|
|
3045
|
+
}
|
|
3046
|
+
}
|
|
3047
|
+
var doWritev = (stream, iov, iovcnt, offset) => {
|
|
3048
|
+
var ret = 0;
|
|
3049
|
+
for (var i = 0; i < iovcnt; i++) {
|
|
3050
|
+
var ptr = HEAPU32[iov >>> 2 >>> 0];
|
|
3051
|
+
var len = HEAPU32[iov + 4 >>> 2 >>> 0];
|
|
3052
|
+
iov += 8;
|
|
3053
|
+
var curr = FS.write(stream, HEAP8, ptr, len, offset);
|
|
3054
|
+
if (curr < 0) return -1;
|
|
3055
|
+
ret += curr;
|
|
3056
|
+
if (curr < len) break;
|
|
3057
|
+
if (typeof offset != "undefined") offset += curr;
|
|
3058
|
+
}
|
|
3059
|
+
return ret;
|
|
3060
|
+
};
|
|
3061
|
+
function _fd_write(fd, iov, iovcnt, pnum) {
|
|
3062
|
+
iov >>>= 0;
|
|
3063
|
+
iovcnt >>>= 0;
|
|
3064
|
+
pnum >>>= 0;
|
|
3065
|
+
try {
|
|
3066
|
+
var num = doWritev(SYSCALLS.getStreamFromFD(fd), iov, iovcnt);
|
|
3067
|
+
HEAPU32[pnum >>> 2 >>> 0] = num;
|
|
3068
|
+
return 0;
|
|
3069
|
+
} catch (e) {
|
|
3070
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
3071
|
+
return e.errno;
|
|
3072
|
+
}
|
|
3073
|
+
}
|
|
3074
|
+
function _random_get(buffer, size) {
|
|
3075
|
+
buffer >>>= 0;
|
|
3076
|
+
size >>>= 0;
|
|
3077
|
+
try {
|
|
3078
|
+
randomFill(HEAPU8.subarray(buffer >>> 0, buffer + size >>> 0));
|
|
3079
|
+
return 0;
|
|
3080
|
+
} catch (e) {
|
|
3081
|
+
if (typeof FS == "undefined" || !(e.name === "ErrnoError")) throw e;
|
|
3082
|
+
return e.errno;
|
|
3083
|
+
}
|
|
3084
|
+
}
|
|
3085
|
+
var getCFunc = (ident) => {
|
|
3086
|
+
return Module["_" + ident];
|
|
3087
|
+
};
|
|
3088
|
+
var writeArrayToMemory = (array, buffer) => {
|
|
3089
|
+
HEAP8.set(array, buffer >>> 0);
|
|
3090
|
+
};
|
|
3091
|
+
var stackAlloc = (sz) => __emscripten_stack_alloc(sz);
|
|
3092
|
+
var stringToUTF8OnStack = (str) => {
|
|
3093
|
+
var size = lengthBytesUTF8(str) + 1;
|
|
3094
|
+
var ret = stackAlloc(size);
|
|
3095
|
+
stringToUTF8(str, ret, size);
|
|
3096
|
+
return ret;
|
|
3097
|
+
};
|
|
3098
|
+
var stackSave = () => _emscripten_stack_get_current();
|
|
3099
|
+
var stackRestore = (val) => __emscripten_stack_restore(val);
|
|
3100
|
+
var ccall = (ident, returnType, argTypes, args, opts) => {
|
|
3101
|
+
var toC = {
|
|
3102
|
+
string: (str) => {
|
|
3103
|
+
var ret = 0;
|
|
3104
|
+
if (str !== null && str !== void 0 && str !== 0) ret = stringToUTF8OnStack(str);
|
|
3105
|
+
return ret;
|
|
3106
|
+
},
|
|
3107
|
+
array: (arr) => {
|
|
3108
|
+
var ret = stackAlloc(arr.length);
|
|
3109
|
+
writeArrayToMemory(arr, ret);
|
|
3110
|
+
return ret;
|
|
3111
|
+
}
|
|
3112
|
+
};
|
|
3113
|
+
function convertReturnValue(ret) {
|
|
3114
|
+
if (returnType === "string") return UTF8ToString(ret);
|
|
3115
|
+
if (returnType === "pointer") return ret >>> 0;
|
|
3116
|
+
if (returnType === "boolean") return Boolean(ret);
|
|
3117
|
+
return ret;
|
|
3118
|
+
}
|
|
3119
|
+
var func = getCFunc(ident);
|
|
3120
|
+
var cArgs = [];
|
|
3121
|
+
var stack = 0;
|
|
3122
|
+
if (args) for (var i = 0; i < args.length; i++) {
|
|
3123
|
+
var converter = toC[argTypes[i]];
|
|
3124
|
+
if (converter) {
|
|
3125
|
+
if (stack === 0) stack = stackSave();
|
|
3126
|
+
cArgs[i] = converter(args[i]);
|
|
3127
|
+
} else cArgs[i] = args[i];
|
|
3128
|
+
}
|
|
3129
|
+
var ret = func(...cArgs);
|
|
3130
|
+
function onDone(ret) {
|
|
3131
|
+
if (stack !== 0) stackRestore(stack);
|
|
3132
|
+
return convertReturnValue(ret);
|
|
3133
|
+
}
|
|
3134
|
+
ret = onDone(ret);
|
|
3135
|
+
return ret;
|
|
3136
|
+
};
|
|
3137
|
+
var FS_createPath = (...args) => FS.createPath(...args);
|
|
3138
|
+
var FS_unlink = (...args) => FS.unlink(...args);
|
|
3139
|
+
var FS_createLazyFile = (...args) => FS.createLazyFile(...args);
|
|
3140
|
+
var FS_createDevice = (...args) => FS.createDevice(...args);
|
|
3141
|
+
FS.createPreloadedFile = FS_createPreloadedFile;
|
|
3142
|
+
FS.preloadFile = FS_preloadFile;
|
|
3143
|
+
FS.staticInit();
|
|
3144
|
+
if (Module["noExitRuntime"]) noExitRuntime = Module["noExitRuntime"];
|
|
3145
|
+
if (Module["preloadPlugins"]) preloadPlugins = Module["preloadPlugins"];
|
|
3146
|
+
if (Module["print"]) out = Module["print"];
|
|
3147
|
+
if (Module["printErr"]) err = Module["printErr"];
|
|
3148
|
+
if (Module["wasmBinary"]) wasmBinary = Module["wasmBinary"];
|
|
3149
|
+
if (Module["arguments"]) Module["arguments"];
|
|
3150
|
+
if (Module["thisProgram"]) thisProgram = Module["thisProgram"];
|
|
3151
|
+
if (Module["preInit"]) {
|
|
3152
|
+
if (typeof Module["preInit"] == "function") Module["preInit"] = [Module["preInit"]];
|
|
3153
|
+
while (Module["preInit"].length > 0) Module["preInit"].shift()();
|
|
3154
|
+
}
|
|
3155
|
+
Module["addRunDependency"] = addRunDependency;
|
|
3156
|
+
Module["removeRunDependency"] = removeRunDependency;
|
|
3157
|
+
Module["ccall"] = ccall;
|
|
3158
|
+
Module["FS_preloadFile"] = FS_preloadFile;
|
|
3159
|
+
Module["FS_unlink"] = FS_unlink;
|
|
3160
|
+
Module["FS_createPath"] = FS_createPath;
|
|
3161
|
+
Module["FS_createDevice"] = FS_createDevice;
|
|
3162
|
+
Module["FS"] = FS;
|
|
3163
|
+
Module["FS_createDataFile"] = FS_createDataFile;
|
|
3164
|
+
Module["FS_createLazyFile"] = FS_createLazyFile;
|
|
3165
|
+
var _emscripten_builtin_memalign, __emscripten_timeout, ___trap, __emscripten_stack_restore, __emscripten_stack_alloc, _emscripten_stack_get_current, wasmMemory;
|
|
3166
|
+
function assignWasmExports(wasmExports) {
|
|
3167
|
+
Module["_free"] = wasmExports["R"];
|
|
3168
|
+
Module["_malloc"] = wasmExports["S"];
|
|
3169
|
+
Module["_rime_wasm_init"] = wasmExports["T"];
|
|
3170
|
+
Module["_rime_wasm_process_input"] = wasmExports["U"];
|
|
3171
|
+
Module["_rime_wasm_pick_candidate"] = wasmExports["V"];
|
|
3172
|
+
Module["_rime_wasm_flip_page"] = wasmExports["W"];
|
|
3173
|
+
Module["_rime_wasm_clear_input"] = wasmExports["X"];
|
|
3174
|
+
Module["_rime_wasm_set_option"] = wasmExports["Y"];
|
|
3175
|
+
Module["_rime_wasm_get_version"] = wasmExports["Z"];
|
|
3176
|
+
Module["_rime_wasm_destroy"] = wasmExports["_"];
|
|
3177
|
+
_emscripten_builtin_memalign = wasmExports["$"];
|
|
3178
|
+
__emscripten_timeout = wasmExports["aa"];
|
|
3179
|
+
___trap = wasmExports["ba"];
|
|
3180
|
+
__emscripten_stack_restore = wasmExports["ca"];
|
|
3181
|
+
__emscripten_stack_alloc = wasmExports["da"];
|
|
3182
|
+
_emscripten_stack_get_current = wasmExports["ea"];
|
|
3183
|
+
wasmMemory = wasmExports["P"];
|
|
3184
|
+
wasmExports["__indirect_function_table"];
|
|
3185
|
+
}
|
|
3186
|
+
var wasmImports = {
|
|
3187
|
+
r: ___syscall_chmod,
|
|
3188
|
+
k: ___syscall_faccessat,
|
|
3189
|
+
s: ___syscall_fchmod,
|
|
3190
|
+
c: ___syscall_fcntl64,
|
|
3191
|
+
j: ___syscall_fdatasync,
|
|
3192
|
+
N: ___syscall_fstat64,
|
|
3193
|
+
q: ___syscall_ftruncate64,
|
|
3194
|
+
u: ___syscall_getcwd,
|
|
3195
|
+
A: ___syscall_getdents64,
|
|
3196
|
+
O: ___syscall_ioctl,
|
|
3197
|
+
L: ___syscall_lstat64,
|
|
3198
|
+
G: ___syscall_mkdirat,
|
|
3199
|
+
K: ___syscall_newfstatat,
|
|
3200
|
+
d: ___syscall_openat,
|
|
3201
|
+
v: ___syscall_readlinkat,
|
|
3202
|
+
z: ___syscall_renameat,
|
|
3203
|
+
i: ___syscall_rmdir,
|
|
3204
|
+
M: ___syscall_stat64,
|
|
3205
|
+
t: ___syscall_truncate64,
|
|
3206
|
+
h: ___syscall_unlinkat,
|
|
3207
|
+
l: __abort_js,
|
|
3208
|
+
n: __emscripten_runtime_keepalive_clear,
|
|
3209
|
+
E: __localtime_js,
|
|
3210
|
+
C: __mmap_js,
|
|
3211
|
+
D: __munmap_js,
|
|
3212
|
+
o: __setitimer_js,
|
|
3213
|
+
F: __tzset_js,
|
|
3214
|
+
w: _clock_time_get,
|
|
3215
|
+
e: _emscripten_date_now,
|
|
3216
|
+
b: _emscripten_get_now,
|
|
3217
|
+
x: _emscripten_resize_heap,
|
|
3218
|
+
I: _environ_get,
|
|
3219
|
+
J: _environ_sizes_get,
|
|
3220
|
+
a: _fd_close,
|
|
3221
|
+
y: _fd_fdstat_get,
|
|
3222
|
+
B: _fd_pread,
|
|
3223
|
+
g: _fd_read,
|
|
3224
|
+
H: _fd_seek,
|
|
3225
|
+
f: _fd_write,
|
|
3226
|
+
m: _proc_exit,
|
|
3227
|
+
p: _random_get
|
|
3228
|
+
};
|
|
3229
|
+
function applySignatureConversions(wasmExports) {
|
|
3230
|
+
wasmExports = Object.assign({}, wasmExports);
|
|
3231
|
+
var makeWrapper_pp = (f) => (a0) => f(a0) >>> 0;
|
|
3232
|
+
var makeWrapper_ppp = (f) => (a0, a1) => f(a0, a1) >>> 0;
|
|
3233
|
+
var makeWrapper_p = (f) => () => f() >>> 0;
|
|
3234
|
+
wasmExports["S"] = makeWrapper_pp(wasmExports["S"]);
|
|
3235
|
+
wasmExports["$"] = makeWrapper_ppp(wasmExports["$"]);
|
|
3236
|
+
wasmExports["da"] = makeWrapper_pp(wasmExports["da"]);
|
|
3237
|
+
wasmExports["ea"] = makeWrapper_p(wasmExports["ea"]);
|
|
3238
|
+
return wasmExports;
|
|
3239
|
+
}
|
|
3240
|
+
function run() {
|
|
3241
|
+
if (runDependencies > 0) {
|
|
3242
|
+
dependenciesFulfilled = run;
|
|
3243
|
+
return;
|
|
3244
|
+
}
|
|
3245
|
+
preRun();
|
|
3246
|
+
if (runDependencies > 0) {
|
|
3247
|
+
dependenciesFulfilled = run;
|
|
3248
|
+
return;
|
|
3249
|
+
}
|
|
3250
|
+
function doRun() {
|
|
3251
|
+
Module["calledRun"] = true;
|
|
3252
|
+
if (ABORT) return;
|
|
3253
|
+
initRuntime();
|
|
3254
|
+
readyPromiseResolve?.(Module);
|
|
3255
|
+
Module["onRuntimeInitialized"]?.();
|
|
3256
|
+
postRun();
|
|
3257
|
+
}
|
|
3258
|
+
if (Module["setStatus"]) {
|
|
3259
|
+
Module["setStatus"]("Running...");
|
|
3260
|
+
setTimeout(() => {
|
|
3261
|
+
setTimeout(() => Module["setStatus"](""), 1);
|
|
3262
|
+
doRun();
|
|
3263
|
+
}, 1);
|
|
3264
|
+
} else doRun();
|
|
3265
|
+
}
|
|
3266
|
+
var wasmExports = await createWasm();
|
|
3267
|
+
run();
|
|
3268
|
+
if (runtimeInitialized) moduleRtn = Module;
|
|
3269
|
+
else moduleRtn = new Promise((resolve, reject) => {
|
|
3270
|
+
readyPromiseResolve = resolve;
|
|
3271
|
+
readyPromiseReject = reject;
|
|
3272
|
+
});
|
|
3273
|
+
return moduleRtn;
|
|
3274
|
+
}
|
|
3275
|
+
//#endregion
|
|
3276
|
+
//#region src/rime.ts
|
|
3277
|
+
function syncfs(module, populate) {
|
|
3278
|
+
return new Promise((resolve, reject) => {
|
|
3279
|
+
module.FS.syncfs(populate, (err) => {
|
|
3280
|
+
if (err) reject(err);
|
|
3281
|
+
else resolve();
|
|
3282
|
+
});
|
|
3283
|
+
});
|
|
3284
|
+
}
|
|
3285
|
+
async function createRimeEngine(options = {}) {
|
|
3286
|
+
const wasmDir = options.wasmDir ?? ".";
|
|
3287
|
+
const Module = await createRimeModule({ locateFile(file) {
|
|
3288
|
+
return `${wasmDir}/${file}`;
|
|
3289
|
+
} });
|
|
3290
|
+
try {
|
|
3291
|
+
Module.FS.mkdir("/rime_user");
|
|
3292
|
+
} catch {}
|
|
3293
|
+
Module.FS.mount(Module.FS.filesystems.IDBFS, {}, "/rime_user");
|
|
3294
|
+
await syncfs(Module, true);
|
|
3295
|
+
try {
|
|
3296
|
+
Module.FS.mkdir("/rime");
|
|
3297
|
+
} catch {}
|
|
3298
|
+
try {
|
|
3299
|
+
Module.FS.mkdir("/rime/build");
|
|
3300
|
+
} catch {}
|
|
3301
|
+
const allDataFiles = options.dataFiles ?? [
|
|
3302
|
+
"default.yaml",
|
|
3303
|
+
"luna_pinyin.schema.yaml",
|
|
3304
|
+
"luna_pinyin.table.bin",
|
|
3305
|
+
"luna_pinyin.prism.bin",
|
|
3306
|
+
"luna_pinyin.reverse.bin"
|
|
3307
|
+
];
|
|
3308
|
+
await Promise.all(allDataFiles.map(async (file) => {
|
|
3309
|
+
const url = `${wasmDir}/${file}`;
|
|
3310
|
+
const resp = await fetch(url);
|
|
3311
|
+
if (!resp.ok) throw new Error(`Failed to fetch ${url}: ${resp.status}`);
|
|
3312
|
+
const data = new Uint8Array(await resp.arrayBuffer());
|
|
3313
|
+
Module.FS.writeFile(`/rime/build/${file}`, data);
|
|
3314
|
+
}));
|
|
3315
|
+
const rc = Module.ccall("rime_wasm_init", "number", [], []);
|
|
3316
|
+
if (rc !== 0) throw new Error(`rime_wasm_init failed with code ${rc}`);
|
|
3317
|
+
await syncfs(Module, false);
|
|
3318
|
+
let destroyed = false;
|
|
3319
|
+
function callJson(fn, argTypes, args) {
|
|
3320
|
+
if (destroyed) throw new Error("Engine is destroyed");
|
|
3321
|
+
const json = Module.ccall(fn, "string", argTypes, args);
|
|
3322
|
+
return JSON.parse(json);
|
|
3323
|
+
}
|
|
3324
|
+
return {
|
|
3325
|
+
processInput(keys) {
|
|
3326
|
+
return callJson("rime_wasm_process_input", ["string"], [keys]);
|
|
3327
|
+
},
|
|
3328
|
+
pickCandidate(index) {
|
|
3329
|
+
const state = callJson("rime_wasm_pick_candidate", ["number"], [index]);
|
|
3330
|
+
syncfs(Module, false).catch(() => {});
|
|
3331
|
+
return state;
|
|
3332
|
+
},
|
|
3333
|
+
flipPage(forward) {
|
|
3334
|
+
return callJson("rime_wasm_flip_page", ["number"], [forward ? 0 : 1]);
|
|
3335
|
+
},
|
|
3336
|
+
clearInput() {
|
|
3337
|
+
if (destroyed) return;
|
|
3338
|
+
Module.ccall("rime_wasm_clear_input", null, [], []);
|
|
3339
|
+
},
|
|
3340
|
+
setOption(name, value) {
|
|
3341
|
+
if (destroyed) return;
|
|
3342
|
+
Module.ccall("rime_wasm_set_option", null, ["string", "number"], [name, value ? 1 : 0]);
|
|
3343
|
+
},
|
|
3344
|
+
getVersion() {
|
|
3345
|
+
if (destroyed) return "unknown";
|
|
3346
|
+
return Module.ccall("rime_wasm_get_version", "string", [], []);
|
|
3347
|
+
},
|
|
3348
|
+
destroy() {
|
|
3349
|
+
if (destroyed) return;
|
|
3350
|
+
destroyed = true;
|
|
3351
|
+
Module.ccall("rime_wasm_destroy", null, [], []);
|
|
3352
|
+
syncfs(Module, false).catch(() => {});
|
|
3353
|
+
}
|
|
3354
|
+
};
|
|
3355
|
+
}
|
|
3356
|
+
//#endregion
|
|
3357
|
+
//#region src/index.ts
|
|
3358
|
+
/**
|
|
3359
|
+
* 创建基于 RIME WASM 的拼音引擎,实现 PinyinEngine 通用接口。
|
|
3360
|
+
* 内部处理分页逻辑,processInput 返回所有页的候选词集合。
|
|
3361
|
+
*/
|
|
3362
|
+
var RimePinyinEngine = class {
|
|
3363
|
+
engine = null;
|
|
3364
|
+
prevRimeInput = "";
|
|
3365
|
+
initPromise = null;
|
|
3366
|
+
constructor(options = {}) {
|
|
3367
|
+
this.options = options;
|
|
3368
|
+
this.initialize();
|
|
3369
|
+
}
|
|
3370
|
+
async getEngine() {
|
|
3371
|
+
await this.initialize();
|
|
3372
|
+
return this.engine;
|
|
3373
|
+
}
|
|
3374
|
+
async initialize() {
|
|
3375
|
+
if (this.initPromise) return this.initPromise;
|
|
3376
|
+
this.initPromise = (async () => {
|
|
3377
|
+
this.engine = await createRimeEngine(this.options);
|
|
3378
|
+
this.engine.setOption("zh_simp", this.options.simplified ?? true);
|
|
3379
|
+
})();
|
|
3380
|
+
return this.initPromise;
|
|
3381
|
+
}
|
|
3382
|
+
async processInput(fullPinyin) {
|
|
3383
|
+
const engine = await this.getEngine();
|
|
3384
|
+
if (!fullPinyin) {
|
|
3385
|
+
engine.clearInput();
|
|
3386
|
+
this.prevRimeInput = "";
|
|
3387
|
+
return null;
|
|
3388
|
+
}
|
|
3389
|
+
let state;
|
|
3390
|
+
if (fullPinyin.startsWith(this.prevRimeInput)) {
|
|
3391
|
+
const delta = fullPinyin.slice(this.prevRimeInput.length);
|
|
3392
|
+
state = engine.processInput(delta);
|
|
3393
|
+
} else {
|
|
3394
|
+
engine.clearInput();
|
|
3395
|
+
state = engine.processInput(fullPinyin);
|
|
3396
|
+
}
|
|
3397
|
+
this.prevRimeInput = fullPinyin;
|
|
3398
|
+
return state;
|
|
3399
|
+
}
|
|
3400
|
+
async pickCandidate(index) {
|
|
3401
|
+
return (await this.getEngine()).pickCandidate(index);
|
|
3402
|
+
}
|
|
3403
|
+
async clearInput() {
|
|
3404
|
+
return (await this.getEngine()).clearInput();
|
|
3405
|
+
}
|
|
3406
|
+
async setSimplified(simplified) {
|
|
3407
|
+
return (await this.getEngine()).setOption("zh_simp", simplified);
|
|
3408
|
+
}
|
|
3409
|
+
async destroy() {
|
|
3410
|
+
return (await this.getEngine()).destroy();
|
|
3411
|
+
}
|
|
3412
|
+
};
|
|
3413
|
+
//#endregion
|
|
3414
|
+
exports.RimePinyinEngine = RimePinyinEngine;
|
|
3415
|
+
|
|
3416
|
+
//# sourceMappingURL=index.js.map
|