@agorapete/wllama 3.5.1-q2.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/.gitmodules +3 -0
- package/.prettierignore +38 -0
- package/AGENTS.md +1 -0
- package/CMakeLists.txt +131 -0
- package/LICENCE +21 -0
- package/README-dev.md +178 -0
- package/README.md +225 -0
- package/README_banner.png +0 -0
- package/assets/screenshot_0.png +0 -0
- package/cpp/generate_glue_prototype.js +115 -0
- package/cpp/glue.hpp +664 -0
- package/cpp/test_glue.cpp +80 -0
- package/cpp/wllama-context.h +1172 -0
- package/cpp/wllama-fs.h +148 -0
- package/cpp/wllama.cpp +187 -0
- package/cpp/wllama.h +6 -0
- package/esm/cache-manager.d.ts +130 -0
- package/esm/debug.d.ts +28 -0
- package/esm/glue/glue.d.ts +22 -0
- package/esm/glue/messages.d.ts +146 -0
- package/esm/huggingface.d.ts +31 -0
- package/esm/index.cjs +3406 -0
- package/esm/index.d.ts +8 -0
- package/esm/index.js +3387 -0
- package/esm/index.min.js +1 -0
- package/esm/index.min.js.map +1 -0
- package/esm/model-manager.d.ts +136 -0
- package/esm/storage/cos.d.ts +36 -0
- package/esm/storage/index.d.ts +33 -0
- package/esm/storage/opfs.d.ts +12 -0
- package/esm/types/oai-compat.d.ts +278 -0
- package/esm/types/types.d.ts +112 -0
- package/esm/utils.d.ts +119 -0
- package/esm/wasm/source-map.d.ts +1 -0
- package/esm/wasm/wllama.wasm +0 -0
- package/esm/wasm-from-cdn.d.ts +8 -0
- package/esm/wllama.d.ts +397 -0
- package/esm/worker.d.ts +92 -0
- package/esm/workers-code/generated.d.ts +4 -0
- package/guides/intro-v2.md +132 -0
- package/guides/intro-v3.1.md +40 -0
- package/guides/intro-v3.md +230 -0
- package/index.ts +1 -0
- package/package.json +71 -0
- package/scripts/bisect_test.sh +33 -0
- package/scripts/build_hf_space.sh +26 -0
- package/scripts/build_source_map.js +269 -0
- package/scripts/build_wasm.sh +19 -0
- package/scripts/build_worker.sh +38 -0
- package/scripts/check_debug_build.js +30 -0
- package/scripts/check_package_size.js +25 -0
- package/scripts/docker-compose.yml +76 -0
- package/scripts/generate_wasm_from_cdn.js +24 -0
- package/scripts/http_server.js +44 -0
- package/scripts/post_build.sh +32 -0
- package/src/cache-manager.ts +358 -0
- package/src/debug.ts +111 -0
- package/src/glue/glue.ts +291 -0
- package/src/glue/messages.ts +773 -0
- package/src/huggingface.ts +151 -0
- package/src/index.ts +8 -0
- package/src/mjs.test.ts +44 -0
- package/src/model-manager.test.ts +200 -0
- package/src/model-manager.ts +359 -0
- package/src/storage/cos.test.ts +83 -0
- package/src/storage/cos.ts +171 -0
- package/src/storage/index.ts +40 -0
- package/src/storage/opfs.ts +119 -0
- package/src/types/oai-compat.ts +342 -0
- package/src/types/types.ts +133 -0
- package/src/utils.test.ts +231 -0
- package/src/utils.ts +403 -0
- package/src/wasm/source-map.ts +7 -0
- package/src/wasm/wllama.js +1 -0
- package/src/wasm/wllama.wasm +0 -0
- package/src/wasm-from-cdn.ts +13 -0
- package/src/wllama.test.ts +392 -0
- package/src/wllama.ts +1138 -0
- package/src/wllama.wgpu.test.ts +62 -0
- package/src/worker.ts +443 -0
- package/src/workers-code/generated.ts +11 -0
- package/src/workers-code/llama-cpp.js +511 -0
- package/src/workers-code/opfs-utils.js +150 -0
- package/tsconfig.build.json +34 -0
- package/tsup.config.ts +23 -0
- package/vitest.config.ts +61 -0
package/esm/index.js
ADDED
|
@@ -0,0 +1,3387 @@
|
|
|
1
|
+
var __defProp = Object.defineProperty;
|
|
2
|
+
var __defProps = Object.defineProperties;
|
|
3
|
+
var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
|
|
4
|
+
var __getOwnPropSymbols = Object.getOwnPropertySymbols;
|
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
+
var __propIsEnum = Object.prototype.propertyIsEnumerable;
|
|
7
|
+
var __knownSymbol = (name, symbol) => (symbol = Symbol[name]) ? symbol : Symbol.for("Symbol." + name);
|
|
8
|
+
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
|
|
9
|
+
var __spreadValues = (a, b) => {
|
|
10
|
+
for (var prop in b || (b = {}))
|
|
11
|
+
if (__hasOwnProp.call(b, prop))
|
|
12
|
+
__defNormalProp(a, prop, b[prop]);
|
|
13
|
+
if (__getOwnPropSymbols)
|
|
14
|
+
for (var prop of __getOwnPropSymbols(b)) {
|
|
15
|
+
if (__propIsEnum.call(b, prop))
|
|
16
|
+
__defNormalProp(a, prop, b[prop]);
|
|
17
|
+
}
|
|
18
|
+
return a;
|
|
19
|
+
};
|
|
20
|
+
var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
|
|
21
|
+
var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
|
|
22
|
+
var __async = (__this, __arguments, generator) => {
|
|
23
|
+
return new Promise((resolve, reject) => {
|
|
24
|
+
var fulfilled = (value) => {
|
|
25
|
+
try {
|
|
26
|
+
step(generator.next(value));
|
|
27
|
+
} catch (e) {
|
|
28
|
+
reject(e);
|
|
29
|
+
}
|
|
30
|
+
};
|
|
31
|
+
var rejected = (value) => {
|
|
32
|
+
try {
|
|
33
|
+
step(generator.throw(value));
|
|
34
|
+
} catch (e) {
|
|
35
|
+
reject(e);
|
|
36
|
+
}
|
|
37
|
+
};
|
|
38
|
+
var step = (x) => x.done ? resolve(x.value) : Promise.resolve(x.value).then(fulfilled, rejected);
|
|
39
|
+
step((generator = generator.apply(__this, __arguments)).next());
|
|
40
|
+
});
|
|
41
|
+
};
|
|
42
|
+
var __await = function(promise, isYieldStar) {
|
|
43
|
+
this[0] = promise;
|
|
44
|
+
this[1] = isYieldStar;
|
|
45
|
+
};
|
|
46
|
+
var __asyncGenerator = (__this, __arguments, generator) => {
|
|
47
|
+
var resume = (k, v, yes, no) => {
|
|
48
|
+
try {
|
|
49
|
+
var x = generator[k](v), isAwait = (v = x.value) instanceof __await, done = x.done;
|
|
50
|
+
Promise.resolve(isAwait ? v[0] : v).then((y) => isAwait ? resume(k === "return" ? k : "next", v[1] ? { done: y.done, value: y.value } : y, yes, no) : yes({ value: y, done })).catch((e) => resume("throw", e, yes, no));
|
|
51
|
+
} catch (e) {
|
|
52
|
+
no(e);
|
|
53
|
+
}
|
|
54
|
+
}, method = (k) => it[k] = (x) => new Promise((yes, no) => resume(k, x, yes, no)), it = {};
|
|
55
|
+
return generator = generator.apply(__this, __arguments), it[__knownSymbol("asyncIterator")] = () => it, method("next"), method("throw"), method("return"), it;
|
|
56
|
+
};
|
|
57
|
+
var __forAwait = (obj, it, method) => (it = obj[__knownSymbol("asyncIterator")]) ? it.call(obj) : (obj = obj[__knownSymbol("iterator")](), it = {}, method = (key, fn) => (fn = obj[key]) && (it[key] = (arg) => new Promise((yes, no, done) => (arg = fn.call(obj, arg), done = arg.done, Promise.resolve(arg.value).then((value) => yes({ value, done }), no)))), method("next"), method("return"), it);
|
|
58
|
+
|
|
59
|
+
// src/glue/messages.ts
|
|
60
|
+
var GLUE_VERSION = 1;
|
|
61
|
+
var GLUE_MESSAGE_PROTOTYPES = {
|
|
62
|
+
"erro_evt": {
|
|
63
|
+
"name": "erro_evt",
|
|
64
|
+
"structName": "glue_msg_error",
|
|
65
|
+
"className": "GlueMsgError",
|
|
66
|
+
"fields": [
|
|
67
|
+
{
|
|
68
|
+
"type": "str",
|
|
69
|
+
"name": "message",
|
|
70
|
+
"isNullable": false
|
|
71
|
+
}
|
|
72
|
+
]
|
|
73
|
+
},
|
|
74
|
+
"load_req": {
|
|
75
|
+
"name": "load_req",
|
|
76
|
+
"structName": "glue_msg_load_req",
|
|
77
|
+
"className": "GlueMsgLoadReq",
|
|
78
|
+
"fields": [
|
|
79
|
+
{
|
|
80
|
+
"type": "arr_str",
|
|
81
|
+
"name": "model_paths",
|
|
82
|
+
"isNullable": false
|
|
83
|
+
},
|
|
84
|
+
{
|
|
85
|
+
"type": "str",
|
|
86
|
+
"name": "mmproj_path",
|
|
87
|
+
"isNullable": true
|
|
88
|
+
},
|
|
89
|
+
{
|
|
90
|
+
"type": "bool",
|
|
91
|
+
"name": "n_ctx_auto",
|
|
92
|
+
"isNullable": false
|
|
93
|
+
},
|
|
94
|
+
{
|
|
95
|
+
"type": "bool",
|
|
96
|
+
"name": "use_mmap",
|
|
97
|
+
"isNullable": false
|
|
98
|
+
},
|
|
99
|
+
{
|
|
100
|
+
"type": "bool",
|
|
101
|
+
"name": "use_mlock",
|
|
102
|
+
"isNullable": false
|
|
103
|
+
},
|
|
104
|
+
{
|
|
105
|
+
"type": "int",
|
|
106
|
+
"name": "n_gpu_layers",
|
|
107
|
+
"isNullable": false
|
|
108
|
+
},
|
|
109
|
+
{
|
|
110
|
+
"type": "int",
|
|
111
|
+
"name": "n_ctx",
|
|
112
|
+
"isNullable": false
|
|
113
|
+
},
|
|
114
|
+
{
|
|
115
|
+
"type": "int",
|
|
116
|
+
"name": "n_threads",
|
|
117
|
+
"isNullable": false
|
|
118
|
+
},
|
|
119
|
+
{
|
|
120
|
+
"type": "str",
|
|
121
|
+
"name": "model_alias",
|
|
122
|
+
"isNullable": true
|
|
123
|
+
},
|
|
124
|
+
{
|
|
125
|
+
"type": "int",
|
|
126
|
+
"name": "log_level",
|
|
127
|
+
"isNullable": true
|
|
128
|
+
},
|
|
129
|
+
{
|
|
130
|
+
"type": "bool",
|
|
131
|
+
"name": "embeddings",
|
|
132
|
+
"isNullable": true
|
|
133
|
+
},
|
|
134
|
+
{
|
|
135
|
+
"type": "bool",
|
|
136
|
+
"name": "offload_kqv",
|
|
137
|
+
"isNullable": true
|
|
138
|
+
},
|
|
139
|
+
{
|
|
140
|
+
"type": "int",
|
|
141
|
+
"name": "n_batch",
|
|
142
|
+
"isNullable": true
|
|
143
|
+
},
|
|
144
|
+
{
|
|
145
|
+
"type": "int",
|
|
146
|
+
"name": "n_ubatch",
|
|
147
|
+
"isNullable": true
|
|
148
|
+
},
|
|
149
|
+
{
|
|
150
|
+
"type": "int",
|
|
151
|
+
"name": "n_parallel",
|
|
152
|
+
"isNullable": true
|
|
153
|
+
},
|
|
154
|
+
{
|
|
155
|
+
"type": "str",
|
|
156
|
+
"name": "pooling_type",
|
|
157
|
+
"isNullable": true
|
|
158
|
+
},
|
|
159
|
+
{
|
|
160
|
+
"type": "str",
|
|
161
|
+
"name": "rope_scaling_type",
|
|
162
|
+
"isNullable": true
|
|
163
|
+
},
|
|
164
|
+
{
|
|
165
|
+
"type": "float",
|
|
166
|
+
"name": "rope_freq_base",
|
|
167
|
+
"isNullable": true
|
|
168
|
+
},
|
|
169
|
+
{
|
|
170
|
+
"type": "float",
|
|
171
|
+
"name": "rope_freq_scale",
|
|
172
|
+
"isNullable": true
|
|
173
|
+
},
|
|
174
|
+
{
|
|
175
|
+
"type": "float",
|
|
176
|
+
"name": "yarn_ext_factor",
|
|
177
|
+
"isNullable": true
|
|
178
|
+
},
|
|
179
|
+
{
|
|
180
|
+
"type": "float",
|
|
181
|
+
"name": "yarn_attn_factor",
|
|
182
|
+
"isNullable": true
|
|
183
|
+
},
|
|
184
|
+
{
|
|
185
|
+
"type": "float",
|
|
186
|
+
"name": "yarn_beta_fast",
|
|
187
|
+
"isNullable": true
|
|
188
|
+
},
|
|
189
|
+
{
|
|
190
|
+
"type": "float",
|
|
191
|
+
"name": "yarn_beta_slow",
|
|
192
|
+
"isNullable": true
|
|
193
|
+
},
|
|
194
|
+
{
|
|
195
|
+
"type": "int",
|
|
196
|
+
"name": "yarn_orig_ctx",
|
|
197
|
+
"isNullable": true
|
|
198
|
+
},
|
|
199
|
+
{
|
|
200
|
+
"type": "str",
|
|
201
|
+
"name": "cache_type_k",
|
|
202
|
+
"isNullable": true
|
|
203
|
+
},
|
|
204
|
+
{
|
|
205
|
+
"type": "str",
|
|
206
|
+
"name": "cache_type_v",
|
|
207
|
+
"isNullable": true
|
|
208
|
+
},
|
|
209
|
+
{
|
|
210
|
+
"type": "bool",
|
|
211
|
+
"name": "kv_unified",
|
|
212
|
+
"isNullable": true
|
|
213
|
+
},
|
|
214
|
+
{
|
|
215
|
+
"type": "bool",
|
|
216
|
+
"name": "flash_attn",
|
|
217
|
+
"isNullable": true
|
|
218
|
+
},
|
|
219
|
+
{
|
|
220
|
+
"type": "bool",
|
|
221
|
+
"name": "swa_full",
|
|
222
|
+
"isNullable": true
|
|
223
|
+
},
|
|
224
|
+
{
|
|
225
|
+
"type": "int",
|
|
226
|
+
"name": "n_ctx_checkpoints",
|
|
227
|
+
"isNullable": true
|
|
228
|
+
},
|
|
229
|
+
{
|
|
230
|
+
"type": "int",
|
|
231
|
+
"name": "checkpoint_min_step",
|
|
232
|
+
"isNullable": true
|
|
233
|
+
},
|
|
234
|
+
{
|
|
235
|
+
"type": "str",
|
|
236
|
+
"name": "chat_template",
|
|
237
|
+
"isNullable": true
|
|
238
|
+
},
|
|
239
|
+
{
|
|
240
|
+
"type": "bool",
|
|
241
|
+
"name": "jinja",
|
|
242
|
+
"isNullable": true
|
|
243
|
+
},
|
|
244
|
+
{
|
|
245
|
+
"type": "arr_str",
|
|
246
|
+
"name": "default_template_kwargs_keys",
|
|
247
|
+
"isNullable": true
|
|
248
|
+
},
|
|
249
|
+
{
|
|
250
|
+
"type": "arr_str",
|
|
251
|
+
"name": "default_template_kwargs_vals",
|
|
252
|
+
"isNullable": true
|
|
253
|
+
},
|
|
254
|
+
{
|
|
255
|
+
"type": "bool",
|
|
256
|
+
"name": "reasoning",
|
|
257
|
+
"isNullable": true
|
|
258
|
+
},
|
|
259
|
+
{
|
|
260
|
+
"type": "int",
|
|
261
|
+
"name": "image_min_tokens",
|
|
262
|
+
"isNullable": true
|
|
263
|
+
},
|
|
264
|
+
{
|
|
265
|
+
"type": "int",
|
|
266
|
+
"name": "image_max_tokens",
|
|
267
|
+
"isNullable": true
|
|
268
|
+
},
|
|
269
|
+
{
|
|
270
|
+
"type": "bool",
|
|
271
|
+
"name": "warmup",
|
|
272
|
+
"isNullable": true
|
|
273
|
+
},
|
|
274
|
+
{
|
|
275
|
+
"type": "bool",
|
|
276
|
+
"name": "no_kv_offload",
|
|
277
|
+
"isNullable": true
|
|
278
|
+
},
|
|
279
|
+
{
|
|
280
|
+
"type": "bool",
|
|
281
|
+
"name": "mmproj_offload",
|
|
282
|
+
"isNullable": true
|
|
283
|
+
},
|
|
284
|
+
{
|
|
285
|
+
"type": "bool",
|
|
286
|
+
"name": "cont_batching",
|
|
287
|
+
"isNullable": true
|
|
288
|
+
},
|
|
289
|
+
{
|
|
290
|
+
"type": "int",
|
|
291
|
+
"name": "n_keep",
|
|
292
|
+
"isNullable": true
|
|
293
|
+
},
|
|
294
|
+
{
|
|
295
|
+
"type": "bool",
|
|
296
|
+
"name": "ctx_shift",
|
|
297
|
+
"isNullable": true
|
|
298
|
+
},
|
|
299
|
+
{
|
|
300
|
+
"type": "bool",
|
|
301
|
+
"name": "cache_idle_slots",
|
|
302
|
+
"isNullable": true
|
|
303
|
+
},
|
|
304
|
+
{
|
|
305
|
+
"type": "int",
|
|
306
|
+
"name": "n_cache_reuse",
|
|
307
|
+
"isNullable": true
|
|
308
|
+
},
|
|
309
|
+
{
|
|
310
|
+
"type": "arr_str",
|
|
311
|
+
"name": "lora_paths",
|
|
312
|
+
"isNullable": true
|
|
313
|
+
},
|
|
314
|
+
{
|
|
315
|
+
"type": "arr_float",
|
|
316
|
+
"name": "lora_scales",
|
|
317
|
+
"isNullable": true
|
|
318
|
+
},
|
|
319
|
+
{
|
|
320
|
+
"type": "bool",
|
|
321
|
+
"name": "lora_init_without_apply",
|
|
322
|
+
"isNullable": true
|
|
323
|
+
},
|
|
324
|
+
{
|
|
325
|
+
"type": "str",
|
|
326
|
+
"name": "spec_draft_model",
|
|
327
|
+
"isNullable": true
|
|
328
|
+
},
|
|
329
|
+
{
|
|
330
|
+
"type": "int",
|
|
331
|
+
"name": "spec_draft_ngl",
|
|
332
|
+
"isNullable": true
|
|
333
|
+
},
|
|
334
|
+
{
|
|
335
|
+
"type": "int",
|
|
336
|
+
"name": "spec_draft_n_max",
|
|
337
|
+
"isNullable": true
|
|
338
|
+
},
|
|
339
|
+
{
|
|
340
|
+
"type": "int",
|
|
341
|
+
"name": "spec_draft_n_min",
|
|
342
|
+
"isNullable": true
|
|
343
|
+
},
|
|
344
|
+
{
|
|
345
|
+
"type": "float",
|
|
346
|
+
"name": "spec_draft_p_min",
|
|
347
|
+
"isNullable": true
|
|
348
|
+
},
|
|
349
|
+
{
|
|
350
|
+
"type": "int",
|
|
351
|
+
"name": "spec_draft_threads",
|
|
352
|
+
"isNullable": true
|
|
353
|
+
},
|
|
354
|
+
{
|
|
355
|
+
"type": "int",
|
|
356
|
+
"name": "spec_draft_threads_batch",
|
|
357
|
+
"isNullable": true
|
|
358
|
+
},
|
|
359
|
+
{
|
|
360
|
+
"type": "arr_str",
|
|
361
|
+
"name": "kv_overrides_keys",
|
|
362
|
+
"isNullable": true
|
|
363
|
+
},
|
|
364
|
+
{
|
|
365
|
+
"type": "arr_str",
|
|
366
|
+
"name": "kv_overrides_vals",
|
|
367
|
+
"isNullable": true
|
|
368
|
+
},
|
|
369
|
+
{
|
|
370
|
+
"type": "int",
|
|
371
|
+
"name": "reasoning_budget_tokens",
|
|
372
|
+
"isNullable": true
|
|
373
|
+
},
|
|
374
|
+
{
|
|
375
|
+
"type": "str",
|
|
376
|
+
"name": "reasoning_budget_message",
|
|
377
|
+
"isNullable": true
|
|
378
|
+
},
|
|
379
|
+
{
|
|
380
|
+
"type": "str",
|
|
381
|
+
"name": "reasoning_format",
|
|
382
|
+
"isNullable": true
|
|
383
|
+
},
|
|
384
|
+
{
|
|
385
|
+
"type": "bool",
|
|
386
|
+
"name": "skip_chat_parsing",
|
|
387
|
+
"isNullable": true
|
|
388
|
+
},
|
|
389
|
+
{
|
|
390
|
+
"type": "bool",
|
|
391
|
+
"name": "prefill_assistant",
|
|
392
|
+
"isNullable": true
|
|
393
|
+
}
|
|
394
|
+
]
|
|
395
|
+
},
|
|
396
|
+
"load_res": {
|
|
397
|
+
"name": "load_res",
|
|
398
|
+
"structName": "glue_msg_load_res",
|
|
399
|
+
"className": "GlueMsgLoadRes",
|
|
400
|
+
"fields": [
|
|
401
|
+
{
|
|
402
|
+
"type": "bool",
|
|
403
|
+
"name": "success",
|
|
404
|
+
"isNullable": false
|
|
405
|
+
},
|
|
406
|
+
{
|
|
407
|
+
"type": "int",
|
|
408
|
+
"name": "n_ctx",
|
|
409
|
+
"isNullable": false
|
|
410
|
+
},
|
|
411
|
+
{
|
|
412
|
+
"type": "int",
|
|
413
|
+
"name": "n_batch",
|
|
414
|
+
"isNullable": false
|
|
415
|
+
},
|
|
416
|
+
{
|
|
417
|
+
"type": "int",
|
|
418
|
+
"name": "n_ubatch",
|
|
419
|
+
"isNullable": false
|
|
420
|
+
},
|
|
421
|
+
{
|
|
422
|
+
"type": "int",
|
|
423
|
+
"name": "n_vocab",
|
|
424
|
+
"isNullable": false
|
|
425
|
+
},
|
|
426
|
+
{
|
|
427
|
+
"type": "int",
|
|
428
|
+
"name": "n_ctx_train",
|
|
429
|
+
"isNullable": false
|
|
430
|
+
},
|
|
431
|
+
{
|
|
432
|
+
"type": "int",
|
|
433
|
+
"name": "n_embd",
|
|
434
|
+
"isNullable": false
|
|
435
|
+
},
|
|
436
|
+
{
|
|
437
|
+
"type": "int",
|
|
438
|
+
"name": "n_layer",
|
|
439
|
+
"isNullable": false
|
|
440
|
+
},
|
|
441
|
+
{
|
|
442
|
+
"type": "arr_str",
|
|
443
|
+
"name": "metadata_key",
|
|
444
|
+
"isNullable": false
|
|
445
|
+
},
|
|
446
|
+
{
|
|
447
|
+
"type": "arr_str",
|
|
448
|
+
"name": "metadata_val",
|
|
449
|
+
"isNullable": false
|
|
450
|
+
},
|
|
451
|
+
{
|
|
452
|
+
"type": "int",
|
|
453
|
+
"name": "token_bos",
|
|
454
|
+
"isNullable": false
|
|
455
|
+
},
|
|
456
|
+
{
|
|
457
|
+
"type": "int",
|
|
458
|
+
"name": "token_eos",
|
|
459
|
+
"isNullable": false
|
|
460
|
+
},
|
|
461
|
+
{
|
|
462
|
+
"type": "int",
|
|
463
|
+
"name": "token_eot",
|
|
464
|
+
"isNullable": false
|
|
465
|
+
},
|
|
466
|
+
{
|
|
467
|
+
"type": "arr_int",
|
|
468
|
+
"name": "list_tokens_eog",
|
|
469
|
+
"isNullable": false
|
|
470
|
+
},
|
|
471
|
+
{
|
|
472
|
+
"type": "bool",
|
|
473
|
+
"name": "add_bos_token",
|
|
474
|
+
"isNullable": false
|
|
475
|
+
},
|
|
476
|
+
{
|
|
477
|
+
"type": "bool",
|
|
478
|
+
"name": "add_eos_token",
|
|
479
|
+
"isNullable": false
|
|
480
|
+
},
|
|
481
|
+
{
|
|
482
|
+
"type": "bool",
|
|
483
|
+
"name": "has_encoder",
|
|
484
|
+
"isNullable": false
|
|
485
|
+
},
|
|
486
|
+
{
|
|
487
|
+
"type": "int",
|
|
488
|
+
"name": "token_decoder_start",
|
|
489
|
+
"isNullable": false
|
|
490
|
+
},
|
|
491
|
+
{
|
|
492
|
+
"type": "str",
|
|
493
|
+
"name": "media_marker",
|
|
494
|
+
"isNullable": false
|
|
495
|
+
},
|
|
496
|
+
{
|
|
497
|
+
"type": "bool",
|
|
498
|
+
"name": "has_image_input",
|
|
499
|
+
"isNullable": false
|
|
500
|
+
},
|
|
501
|
+
{
|
|
502
|
+
"type": "bool",
|
|
503
|
+
"name": "has_audio_input",
|
|
504
|
+
"isNullable": false
|
|
505
|
+
}
|
|
506
|
+
]
|
|
507
|
+
},
|
|
508
|
+
"cmpl_req": {
|
|
509
|
+
"name": "cmpl_req",
|
|
510
|
+
"structName": "glue_msg_completion_req",
|
|
511
|
+
"className": "GlueMsgCompletionReq",
|
|
512
|
+
"fields": [
|
|
513
|
+
{
|
|
514
|
+
"type": "bool",
|
|
515
|
+
"name": "is_chat",
|
|
516
|
+
"isNullable": false
|
|
517
|
+
},
|
|
518
|
+
{
|
|
519
|
+
"type": "str",
|
|
520
|
+
"name": "data_json",
|
|
521
|
+
"isNullable": false
|
|
522
|
+
},
|
|
523
|
+
{
|
|
524
|
+
"type": "arr_raw",
|
|
525
|
+
"name": "files",
|
|
526
|
+
"isNullable": false
|
|
527
|
+
}
|
|
528
|
+
]
|
|
529
|
+
},
|
|
530
|
+
"cmpl_res": {
|
|
531
|
+
"name": "cmpl_res",
|
|
532
|
+
"structName": "glue_msg_completion_res",
|
|
533
|
+
"className": "GlueMsgCompletionRes",
|
|
534
|
+
"fields": [
|
|
535
|
+
{
|
|
536
|
+
"type": "bool",
|
|
537
|
+
"name": "success",
|
|
538
|
+
"isNullable": false
|
|
539
|
+
}
|
|
540
|
+
]
|
|
541
|
+
},
|
|
542
|
+
"embd_req": {
|
|
543
|
+
"name": "embd_req",
|
|
544
|
+
"structName": "glue_msg_embedding_req",
|
|
545
|
+
"className": "GlueMsgEmbeddingReq",
|
|
546
|
+
"fields": [
|
|
547
|
+
{
|
|
548
|
+
"type": "str",
|
|
549
|
+
"name": "data_json",
|
|
550
|
+
"isNullable": false
|
|
551
|
+
},
|
|
552
|
+
{
|
|
553
|
+
"type": "arr_raw",
|
|
554
|
+
"name": "files",
|
|
555
|
+
"isNullable": false
|
|
556
|
+
}
|
|
557
|
+
]
|
|
558
|
+
},
|
|
559
|
+
"embd_res": {
|
|
560
|
+
"name": "embd_res",
|
|
561
|
+
"structName": "glue_msg_embedding_res",
|
|
562
|
+
"className": "GlueMsgEmbeddingRes",
|
|
563
|
+
"fields": [
|
|
564
|
+
{
|
|
565
|
+
"type": "bool",
|
|
566
|
+
"name": "success",
|
|
567
|
+
"isNullable": false
|
|
568
|
+
}
|
|
569
|
+
]
|
|
570
|
+
},
|
|
571
|
+
"rrnk_req": {
|
|
572
|
+
"name": "rrnk_req",
|
|
573
|
+
"structName": "glue_msg_rerank_req",
|
|
574
|
+
"className": "GlueMsgRerankReq",
|
|
575
|
+
"fields": [
|
|
576
|
+
{
|
|
577
|
+
"type": "str",
|
|
578
|
+
"name": "data_json",
|
|
579
|
+
"isNullable": false
|
|
580
|
+
}
|
|
581
|
+
]
|
|
582
|
+
},
|
|
583
|
+
"rrnk_res": {
|
|
584
|
+
"name": "rrnk_res",
|
|
585
|
+
"structName": "glue_msg_rerank_res",
|
|
586
|
+
"className": "GlueMsgRerankRes",
|
|
587
|
+
"fields": [
|
|
588
|
+
{
|
|
589
|
+
"type": "bool",
|
|
590
|
+
"name": "success",
|
|
591
|
+
"isNullable": false
|
|
592
|
+
}
|
|
593
|
+
]
|
|
594
|
+
},
|
|
595
|
+
"gres_req": {
|
|
596
|
+
"name": "gres_req",
|
|
597
|
+
"structName": "glue_msg_get_result_req",
|
|
598
|
+
"className": "GlueMsgGetResultReq",
|
|
599
|
+
"fields": []
|
|
600
|
+
},
|
|
601
|
+
"gres_res": {
|
|
602
|
+
"name": "gres_res",
|
|
603
|
+
"structName": "glue_msg_get_result_res",
|
|
604
|
+
"className": "GlueMsgGetResultRes",
|
|
605
|
+
"fields": [
|
|
606
|
+
{
|
|
607
|
+
"type": "bool",
|
|
608
|
+
"name": "success",
|
|
609
|
+
"isNullable": false
|
|
610
|
+
},
|
|
611
|
+
{
|
|
612
|
+
"type": "bool",
|
|
613
|
+
"name": "has_more",
|
|
614
|
+
"isNullable": false
|
|
615
|
+
},
|
|
616
|
+
{
|
|
617
|
+
"type": "bool",
|
|
618
|
+
"name": "is_error",
|
|
619
|
+
"isNullable": false
|
|
620
|
+
},
|
|
621
|
+
{
|
|
622
|
+
"type": "str",
|
|
623
|
+
"name": "data_json",
|
|
624
|
+
"isNullable": false
|
|
625
|
+
}
|
|
626
|
+
]
|
|
627
|
+
},
|
|
628
|
+
"tbop_req": {
|
|
629
|
+
"name": "tbop_req",
|
|
630
|
+
"structName": "glue_msg_test_backend_ops_req",
|
|
631
|
+
"className": "GlueMsgTestBackendOpsReq",
|
|
632
|
+
"fields": [
|
|
633
|
+
{
|
|
634
|
+
"type": "arr_str",
|
|
635
|
+
"name": "args",
|
|
636
|
+
"isNullable": false
|
|
637
|
+
}
|
|
638
|
+
]
|
|
639
|
+
},
|
|
640
|
+
"tbop_res": {
|
|
641
|
+
"name": "tbop_res",
|
|
642
|
+
"structName": "glue_msg_test_backend_ops_res",
|
|
643
|
+
"className": "GlueMsgTestBackendOpsRes",
|
|
644
|
+
"fields": [
|
|
645
|
+
{
|
|
646
|
+
"type": "int",
|
|
647
|
+
"name": "retcode",
|
|
648
|
+
"isNullable": false
|
|
649
|
+
},
|
|
650
|
+
{
|
|
651
|
+
"type": "bool",
|
|
652
|
+
"name": "success",
|
|
653
|
+
"isNullable": false
|
|
654
|
+
}
|
|
655
|
+
]
|
|
656
|
+
}
|
|
657
|
+
};
|
|
658
|
+
|
|
659
|
+
// src/glue/glue.ts
|
|
660
|
+
var GLUE_MAGIC = new Uint8Array([71, 76, 85, 69]);
|
|
661
|
+
var GLUE_DTYPE_NULL = 0;
|
|
662
|
+
var GLUE_DTYPE_BOOL = 1;
|
|
663
|
+
var GLUE_DTYPE_INT = 2;
|
|
664
|
+
var GLUE_DTYPE_FLOAT = 3;
|
|
665
|
+
var GLUE_DTYPE_STRING = 4;
|
|
666
|
+
var GLUE_DTYPE_RAW = 5;
|
|
667
|
+
var GLUE_DTYPE_ARRAY_BOOL = 6;
|
|
668
|
+
var GLUE_DTYPE_ARRAY_INT = 7;
|
|
669
|
+
var GLUE_DTYPE_ARRAY_FLOAT = 8;
|
|
670
|
+
var GLUE_DTYPE_ARRAY_STRING = 9;
|
|
671
|
+
var GLUE_DTYPE_ARRAY_RAW = 10;
|
|
672
|
+
var TYPE_MAP = {
|
|
673
|
+
str: GLUE_DTYPE_STRING,
|
|
674
|
+
int: GLUE_DTYPE_INT,
|
|
675
|
+
float: GLUE_DTYPE_FLOAT,
|
|
676
|
+
bool: GLUE_DTYPE_BOOL,
|
|
677
|
+
raw: GLUE_DTYPE_RAW,
|
|
678
|
+
arr_str: GLUE_DTYPE_ARRAY_STRING,
|
|
679
|
+
arr_int: GLUE_DTYPE_ARRAY_INT,
|
|
680
|
+
arr_float: GLUE_DTYPE_ARRAY_FLOAT,
|
|
681
|
+
arr_bool: GLUE_DTYPE_ARRAY_BOOL,
|
|
682
|
+
arr_raw: GLUE_DTYPE_ARRAY_RAW,
|
|
683
|
+
null: GLUE_DTYPE_NULL
|
|
684
|
+
};
|
|
685
|
+
function glueDeserialize(buf) {
|
|
686
|
+
let offset = 0;
|
|
687
|
+
const view = new DataView(buf.buffer);
|
|
688
|
+
const readUint32 = () => {
|
|
689
|
+
const value = view.getUint32(offset, true);
|
|
690
|
+
offset += 4;
|
|
691
|
+
return value;
|
|
692
|
+
};
|
|
693
|
+
const readInt32 = () => {
|
|
694
|
+
const value = view.getInt32(offset, true);
|
|
695
|
+
offset += 4;
|
|
696
|
+
return value;
|
|
697
|
+
};
|
|
698
|
+
const readFloat = () => {
|
|
699
|
+
const value = view.getFloat32(offset, true);
|
|
700
|
+
offset += 4;
|
|
701
|
+
return value;
|
|
702
|
+
};
|
|
703
|
+
const readBool = () => {
|
|
704
|
+
return readUint32() !== 0;
|
|
705
|
+
};
|
|
706
|
+
const readString = (customLen) => {
|
|
707
|
+
const length = customLen != null ? customLen : readUint32();
|
|
708
|
+
const value = new TextDecoder().decode(buf.slice(offset, offset + length));
|
|
709
|
+
offset += length;
|
|
710
|
+
return value;
|
|
711
|
+
};
|
|
712
|
+
const readRaw = () => {
|
|
713
|
+
const length = readUint32();
|
|
714
|
+
const value = buf.slice(offset, offset + length);
|
|
715
|
+
offset += length;
|
|
716
|
+
return value;
|
|
717
|
+
};
|
|
718
|
+
const readArray = (readItem) => {
|
|
719
|
+
const length = readUint32();
|
|
720
|
+
const value = new Array(length);
|
|
721
|
+
for (let i = 0; i < length; i++) {
|
|
722
|
+
value[i] = readItem();
|
|
723
|
+
}
|
|
724
|
+
return value;
|
|
725
|
+
};
|
|
726
|
+
const readNull = () => null;
|
|
727
|
+
const readField = (field) => {
|
|
728
|
+
switch (field.type) {
|
|
729
|
+
case "str":
|
|
730
|
+
return readString();
|
|
731
|
+
case "int":
|
|
732
|
+
return readInt32();
|
|
733
|
+
case "float":
|
|
734
|
+
return readFloat();
|
|
735
|
+
case "bool":
|
|
736
|
+
return readBool();
|
|
737
|
+
case "raw":
|
|
738
|
+
return readRaw();
|
|
739
|
+
case "arr_str":
|
|
740
|
+
return readArray(readString);
|
|
741
|
+
case "arr_int":
|
|
742
|
+
return readArray(readInt32);
|
|
743
|
+
case "arr_float":
|
|
744
|
+
return readArray(readFloat);
|
|
745
|
+
case "arr_bool":
|
|
746
|
+
return readArray(readBool);
|
|
747
|
+
case "arr_raw":
|
|
748
|
+
return readArray(readRaw);
|
|
749
|
+
case "null":
|
|
750
|
+
return readNull();
|
|
751
|
+
}
|
|
752
|
+
};
|
|
753
|
+
const magicValid = buf[0] === GLUE_MAGIC[0] && buf[1] === GLUE_MAGIC[1] && buf[2] === GLUE_MAGIC[2] && buf[3] === GLUE_MAGIC[3];
|
|
754
|
+
offset += 4;
|
|
755
|
+
if (!magicValid) {
|
|
756
|
+
throw new Error("Invalid magic number");
|
|
757
|
+
}
|
|
758
|
+
const version = readUint32();
|
|
759
|
+
if (version !== GLUE_VERSION) {
|
|
760
|
+
throw new Error("Invalid version number");
|
|
761
|
+
}
|
|
762
|
+
const name = readString(8);
|
|
763
|
+
const msgProto = GLUE_MESSAGE_PROTOTYPES[name];
|
|
764
|
+
if (!msgProto) {
|
|
765
|
+
throw new Error(`Unknown message name: ${name}`);
|
|
766
|
+
}
|
|
767
|
+
const output = { _name: name };
|
|
768
|
+
for (const field of msgProto.fields) {
|
|
769
|
+
const readType = readUint32();
|
|
770
|
+
if (readType === GLUE_DTYPE_NULL) {
|
|
771
|
+
if (!field.isNullable) {
|
|
772
|
+
throw new Error(
|
|
773
|
+
`${name}: Expect field ${field.name} to be non-nullable`
|
|
774
|
+
);
|
|
775
|
+
}
|
|
776
|
+
output[field.name] = null;
|
|
777
|
+
continue;
|
|
778
|
+
}
|
|
779
|
+
if (readType !== TYPE_MAP[field.type]) {
|
|
780
|
+
throw new Error(
|
|
781
|
+
`${name}: Expect field ${field.name} to have type ${field.type}`
|
|
782
|
+
);
|
|
783
|
+
}
|
|
784
|
+
output[field.name] = readField(field);
|
|
785
|
+
}
|
|
786
|
+
return output;
|
|
787
|
+
}
|
|
788
|
+
function glueSerialize(msg) {
|
|
789
|
+
const msgProto = GLUE_MESSAGE_PROTOTYPES[msg._name];
|
|
790
|
+
if (!msgProto) {
|
|
791
|
+
throw new Error(`Unknown message name: ${msg._name}`);
|
|
792
|
+
}
|
|
793
|
+
const bufs = [];
|
|
794
|
+
const writeUint32 = (value) => {
|
|
795
|
+
const buf = new ArrayBuffer(4);
|
|
796
|
+
new DataView(buf).setUint32(0, value, true);
|
|
797
|
+
bufs.push(new Uint8Array(buf));
|
|
798
|
+
};
|
|
799
|
+
const writeInt32 = (value) => {
|
|
800
|
+
const buf = new ArrayBuffer(4);
|
|
801
|
+
new DataView(buf).setInt32(0, value, true);
|
|
802
|
+
bufs.push(new Uint8Array(buf));
|
|
803
|
+
};
|
|
804
|
+
const writeFloat = (value) => {
|
|
805
|
+
const buf = new ArrayBuffer(4);
|
|
806
|
+
new DataView(buf).setFloat32(0, value, true);
|
|
807
|
+
bufs.push(new Uint8Array(buf));
|
|
808
|
+
};
|
|
809
|
+
const writeBool = (value) => {
|
|
810
|
+
writeUint32(value ? 1 : 0);
|
|
811
|
+
};
|
|
812
|
+
const writeString = (value) => {
|
|
813
|
+
const utf8 = new TextEncoder().encode(value);
|
|
814
|
+
writeUint32(utf8.byteLength);
|
|
815
|
+
bufs.push(utf8);
|
|
816
|
+
};
|
|
817
|
+
const writeRaw = (value) => {
|
|
818
|
+
writeUint32(value.byteLength);
|
|
819
|
+
bufs.push(value);
|
|
820
|
+
};
|
|
821
|
+
const writeArray = (value, writeItem) => {
|
|
822
|
+
writeUint32(value.length);
|
|
823
|
+
for (const item of value) {
|
|
824
|
+
writeItem(item);
|
|
825
|
+
}
|
|
826
|
+
};
|
|
827
|
+
const writeNull = () => {
|
|
828
|
+
};
|
|
829
|
+
bufs.push(GLUE_MAGIC);
|
|
830
|
+
writeUint32(GLUE_VERSION);
|
|
831
|
+
{
|
|
832
|
+
const utf8 = new TextEncoder().encode(msg._name);
|
|
833
|
+
bufs.push(utf8);
|
|
834
|
+
}
|
|
835
|
+
for (const field of msgProto.fields) {
|
|
836
|
+
const val = msg[field.name];
|
|
837
|
+
if (!field.isNullable && (val === null || val === void 0)) {
|
|
838
|
+
throw new Error(
|
|
839
|
+
`${msg._name}: Expect field ${field.name} to be non-nullable`
|
|
840
|
+
);
|
|
841
|
+
}
|
|
842
|
+
if (val === null || val === void 0) {
|
|
843
|
+
writeUint32(GLUE_DTYPE_NULL);
|
|
844
|
+
continue;
|
|
845
|
+
}
|
|
846
|
+
writeUint32(TYPE_MAP[field.type]);
|
|
847
|
+
switch (field.type) {
|
|
848
|
+
case "str":
|
|
849
|
+
writeString(val);
|
|
850
|
+
break;
|
|
851
|
+
case "int":
|
|
852
|
+
writeInt32(val);
|
|
853
|
+
break;
|
|
854
|
+
case "float":
|
|
855
|
+
writeFloat(val);
|
|
856
|
+
break;
|
|
857
|
+
case "bool":
|
|
858
|
+
writeBool(val);
|
|
859
|
+
break;
|
|
860
|
+
case "raw":
|
|
861
|
+
writeRaw(val);
|
|
862
|
+
break;
|
|
863
|
+
case "arr_str":
|
|
864
|
+
writeArray(val, writeString);
|
|
865
|
+
break;
|
|
866
|
+
case "arr_int":
|
|
867
|
+
writeArray(val, writeInt32);
|
|
868
|
+
break;
|
|
869
|
+
case "arr_float":
|
|
870
|
+
writeArray(val, writeFloat);
|
|
871
|
+
break;
|
|
872
|
+
case "arr_bool":
|
|
873
|
+
writeArray(val, writeBool);
|
|
874
|
+
break;
|
|
875
|
+
case "arr_raw":
|
|
876
|
+
writeArray(val, writeRaw);
|
|
877
|
+
break;
|
|
878
|
+
case "null":
|
|
879
|
+
writeNull();
|
|
880
|
+
break;
|
|
881
|
+
}
|
|
882
|
+
}
|
|
883
|
+
const totalLength = bufs.reduce((acc, buf) => acc + buf.byteLength, 0);
|
|
884
|
+
const output = new Uint8Array(totalLength);
|
|
885
|
+
let offset = 0;
|
|
886
|
+
for (const buf of bufs) {
|
|
887
|
+
output.set(buf, offset);
|
|
888
|
+
offset += buf.byteLength;
|
|
889
|
+
}
|
|
890
|
+
return output;
|
|
891
|
+
}
|
|
892
|
+
|
|
893
|
+
// src/wasm/source-map.ts
|
|
894
|
+
var WASM_SOURCE_MAP = {
|
|
895
|
+
"default": "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",
|
|
896
|
+
"compat": "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"
|
|
897
|
+
};
|
|
898
|
+
|
|
899
|
+
// src/debug.ts
|
|
900
|
+
var cache = /* @__PURE__ */ new Map();
|
|
901
|
+
function loadMap(buildKey) {
|
|
902
|
+
return __async(this, null, function* () {
|
|
903
|
+
if (cache.has(buildKey)) return cache.get(buildKey);
|
|
904
|
+
const b64 = WASM_SOURCE_MAP[buildKey];
|
|
905
|
+
if (!b64) throw new Error(`No source map for build "${buildKey}"`);
|
|
906
|
+
const gzipped = Uint8Array.from(atob(b64), (c) => c.charCodeAt(0));
|
|
907
|
+
const ds = new DecompressionStream("gzip");
|
|
908
|
+
const writer = ds.writable.getWriter();
|
|
909
|
+
writer.write(gzipped);
|
|
910
|
+
writer.close();
|
|
911
|
+
const buf = yield new Response(ds.readable).arrayBuffer();
|
|
912
|
+
const dv = new DataView(buf);
|
|
913
|
+
const bytes = new Uint8Array(buf);
|
|
914
|
+
const firstId = dv.getUint32(0, true);
|
|
915
|
+
const funcCount = dv.getUint32(4, true);
|
|
916
|
+
const numNames = dv.getUint32(8, true);
|
|
917
|
+
const td = new TextDecoder();
|
|
918
|
+
const names = [];
|
|
919
|
+
let pos = 12;
|
|
920
|
+
for (let i = 0; i < numNames; i++) {
|
|
921
|
+
const len = bytes[pos++];
|
|
922
|
+
names.push(td.decode(bytes.subarray(pos, pos + len)));
|
|
923
|
+
pos += len;
|
|
924
|
+
}
|
|
925
|
+
const funcNames = [];
|
|
926
|
+
for (let i = 0; i < funcCount; i++) {
|
|
927
|
+
const idx = dv.getUint16(pos, true);
|
|
928
|
+
pos += 2;
|
|
929
|
+
funcNames.push(idx === 65535 ? null : names[idx]);
|
|
930
|
+
}
|
|
931
|
+
const entry = { firstId, funcNames };
|
|
932
|
+
cache.set(buildKey, entry);
|
|
933
|
+
return entry;
|
|
934
|
+
});
|
|
935
|
+
}
|
|
936
|
+
var Debug = {
|
|
937
|
+
/**
|
|
938
|
+
* Resolves a list of wasm function indices to their cleaned symbol names.
|
|
939
|
+
*/
|
|
940
|
+
decodeFuncIds: (funcIds, isCompatBuild) => __async(void 0, null, function* () {
|
|
941
|
+
const buildKey = isCompatBuild ? "compat" : "default";
|
|
942
|
+
const { firstId, funcNames } = yield loadMap(buildKey);
|
|
943
|
+
return funcIds.map((funcId) => {
|
|
944
|
+
const i = funcId - firstId;
|
|
945
|
+
const name = i >= 0 && i < funcNames.length && funcNames[i] ? funcNames[i] : "(unknown)";
|
|
946
|
+
return { funcId, name };
|
|
947
|
+
});
|
|
948
|
+
}),
|
|
949
|
+
/**
|
|
950
|
+
* Annotates a wasm stack trace string with resolved function names.
|
|
951
|
+
*
|
|
952
|
+
* Example input from Chrome:
|
|
953
|
+
* at http://localhost:8080/esm/wasm/wllama.wasm:wasm-function[775]:0x74251
|
|
954
|
+
* at async blob:http://localhost:8080/53a863cc-7227-45cc-8594-ddbbf5257f20:317:28
|
|
955
|
+
*
|
|
956
|
+
* Example input from Firefox:
|
|
957
|
+
* @http://localhost:8080/esm/wasm/wllama.wasm:wasm-function[796]:0x7dfe2
|
|
958
|
+
* at wModuleInit/WebAssembly.promising/< (9b6a2acd-d909-44e2-b021-d42fb9087cfb:15:32) index.js:1433:45
|
|
959
|
+
*
|
|
960
|
+
* Example input from Safari:
|
|
961
|
+
* 2441@wasm-function[2441]
|
|
962
|
+
* at wrapper (d746f19e-4523-4f36-ba06-d0969acc0b05:22:126009)
|
|
963
|
+
*
|
|
964
|
+
* Example output:
|
|
965
|
+
* wasm-func[775] (server_response::send)
|
|
966
|
+
*/
|
|
967
|
+
decodeStackTrace: (stack, isCompatBuild) => __async(void 0, null, function* () {
|
|
968
|
+
const re = /wasm-function\[(\d+)\]/g;
|
|
969
|
+
const funcIds = [
|
|
970
|
+
...new Set([...stack.matchAll(re)].map((m) => parseInt(m[1])))
|
|
971
|
+
];
|
|
972
|
+
if (funcIds.length === 0) return stack;
|
|
973
|
+
const resolved = yield Debug.decodeFuncIds(funcIds, isCompatBuild);
|
|
974
|
+
return resolved.map((r) => {
|
|
975
|
+
if (r.name === "(unknown)") {
|
|
976
|
+
return ` wasm-func[${r.funcId}] (unknown)`;
|
|
977
|
+
}
|
|
978
|
+
return ` wasm-func[${r.funcId}] (${r.name})`;
|
|
979
|
+
}).join("\n");
|
|
980
|
+
})
|
|
981
|
+
};
|
|
982
|
+
|
|
983
|
+
// src/utils.ts
|
|
984
|
+
var textDecoder = new TextDecoder();
|
|
985
|
+
var URL_PARTS_REGEX = /-(\d{5})-of-(\d{5})\.gguf(?:\?.*)?$/;
|
|
986
|
+
var parseShardNumber = (fnameOrUrl) => {
|
|
987
|
+
const matches = fnameOrUrl.match(URL_PARTS_REGEX);
|
|
988
|
+
if (!matches) {
|
|
989
|
+
return {
|
|
990
|
+
baseURL: fnameOrUrl,
|
|
991
|
+
current: 1,
|
|
992
|
+
total: 1
|
|
993
|
+
};
|
|
994
|
+
} else {
|
|
995
|
+
return {
|
|
996
|
+
baseURL: fnameOrUrl.replace(URL_PARTS_REGEX, ""),
|
|
997
|
+
current: parseInt(matches[1]),
|
|
998
|
+
total: parseInt(matches[2])
|
|
999
|
+
};
|
|
1000
|
+
}
|
|
1001
|
+
};
|
|
1002
|
+
var sortFileByShard = (blobs) => {
|
|
1003
|
+
const isFiles = blobs.every((b) => !!b.name);
|
|
1004
|
+
if (isFiles && blobs.length > 1) {
|
|
1005
|
+
const files = blobs;
|
|
1006
|
+
files.sort((a, b) => {
|
|
1007
|
+
const infoA = parseShardNumber(a.name);
|
|
1008
|
+
const infoB = parseShardNumber(b.name);
|
|
1009
|
+
return infoA.current - infoB.current;
|
|
1010
|
+
});
|
|
1011
|
+
}
|
|
1012
|
+
};
|
|
1013
|
+
var isMmproj = (blob) => __async(void 0, null, function* () {
|
|
1014
|
+
const META_NAME = "general.architecture";
|
|
1015
|
+
const META_VAL = "clip";
|
|
1016
|
+
const tmp = blob.slice(0, 128 * 1024);
|
|
1017
|
+
const header = yield tmp.arrayBuffer();
|
|
1018
|
+
const buf = new Uint8Array(header);
|
|
1019
|
+
const nameBytes = new TextEncoder().encode(META_NAME);
|
|
1020
|
+
const valBytes = new TextEncoder().encode(META_VAL);
|
|
1021
|
+
let offset = -1;
|
|
1022
|
+
outer: for (let i = 0; i <= buf.length - nameBytes.length; i++) {
|
|
1023
|
+
for (let j = 0; j < nameBytes.length; j++) {
|
|
1024
|
+
if (buf[i + j] !== nameBytes[j]) continue outer;
|
|
1025
|
+
}
|
|
1026
|
+
offset = i;
|
|
1027
|
+
break;
|
|
1028
|
+
}
|
|
1029
|
+
if (offset === -1) return false;
|
|
1030
|
+
if (offset + 8 * 4 + 4 > buf.length) return false;
|
|
1031
|
+
const view = new DataView(header);
|
|
1032
|
+
const valLen = view.getBigUint64(offset + 8 * 3, true);
|
|
1033
|
+
if (valLen !== /* @__PURE__ */ BigInt("4")) return false;
|
|
1034
|
+
for (let i = 0; i < valBytes.length; i++) {
|
|
1035
|
+
if (buf[offset + 8 * 4 + i] !== valBytes[i]) return false;
|
|
1036
|
+
}
|
|
1037
|
+
return true;
|
|
1038
|
+
});
|
|
1039
|
+
var absoluteUrl = (relativePath) => new URL(relativePath, document.baseURI).href;
|
|
1040
|
+
var padDigits = (number, digits) => {
|
|
1041
|
+
return Array(Math.max(digits - String(number).length + 1, 0)).join("0") + number;
|
|
1042
|
+
};
|
|
1043
|
+
var sumArr = (arr) => arr.reduce((prev, curr) => prev + curr, 0);
|
|
1044
|
+
var isString = (value) => !!(value == null ? void 0 : value.startsWith);
|
|
1045
|
+
var MMPROJ_FILE_NAME = "mmproj.gguf";
|
|
1046
|
+
var prepareBlobs = (blobsInp) => __async(void 0, null, function* () {
|
|
1047
|
+
const blobs = [];
|
|
1048
|
+
let blobMmproj = null;
|
|
1049
|
+
for (const blob of blobsInp) {
|
|
1050
|
+
if (yield isMmproj(blob)) {
|
|
1051
|
+
blobMmproj = blob;
|
|
1052
|
+
} else {
|
|
1053
|
+
blobs.push(blob);
|
|
1054
|
+
}
|
|
1055
|
+
}
|
|
1056
|
+
sortFileByShard(blobs);
|
|
1057
|
+
const result = blobs.map((blob, i) => ({
|
|
1058
|
+
blob,
|
|
1059
|
+
name: `model-${padDigits(i + 1, 5)}-of-${padDigits(blobs.length, 5)}.gguf`
|
|
1060
|
+
}));
|
|
1061
|
+
if (blobMmproj) {
|
|
1062
|
+
result.push({
|
|
1063
|
+
blob: blobMmproj,
|
|
1064
|
+
name: MMPROJ_FILE_NAME
|
|
1065
|
+
});
|
|
1066
|
+
}
|
|
1067
|
+
return {
|
|
1068
|
+
llm: result.filter((f) => f.name !== MMPROJ_FILE_NAME),
|
|
1069
|
+
mmproj: blobMmproj ? { blob: blobMmproj, name: MMPROJ_FILE_NAME } : null,
|
|
1070
|
+
all: result
|
|
1071
|
+
};
|
|
1072
|
+
});
|
|
1073
|
+
var isSupportMultiThread = () => ((e) => __async(void 0, null, function* () {
|
|
1074
|
+
try {
|
|
1075
|
+
return "undefined" != typeof MessageChannel && new MessageChannel().port1.postMessage(new SharedArrayBuffer(1)), WebAssembly.validate(e);
|
|
1076
|
+
} catch (e2) {
|
|
1077
|
+
return false;
|
|
1078
|
+
}
|
|
1079
|
+
}))(
|
|
1080
|
+
new Uint8Array([
|
|
1081
|
+
0,
|
|
1082
|
+
97,
|
|
1083
|
+
115,
|
|
1084
|
+
109,
|
|
1085
|
+
1,
|
|
1086
|
+
0,
|
|
1087
|
+
0,
|
|
1088
|
+
0,
|
|
1089
|
+
1,
|
|
1090
|
+
4,
|
|
1091
|
+
1,
|
|
1092
|
+
96,
|
|
1093
|
+
0,
|
|
1094
|
+
0,
|
|
1095
|
+
3,
|
|
1096
|
+
2,
|
|
1097
|
+
1,
|
|
1098
|
+
0,
|
|
1099
|
+
5,
|
|
1100
|
+
4,
|
|
1101
|
+
1,
|
|
1102
|
+
3,
|
|
1103
|
+
1,
|
|
1104
|
+
1,
|
|
1105
|
+
10,
|
|
1106
|
+
11,
|
|
1107
|
+
1,
|
|
1108
|
+
9,
|
|
1109
|
+
0,
|
|
1110
|
+
65,
|
|
1111
|
+
0,
|
|
1112
|
+
254,
|
|
1113
|
+
16,
|
|
1114
|
+
2,
|
|
1115
|
+
0,
|
|
1116
|
+
26,
|
|
1117
|
+
11
|
|
1118
|
+
])
|
|
1119
|
+
);
|
|
1120
|
+
var isSupportExceptions = () => __async(void 0, null, function* () {
|
|
1121
|
+
return WebAssembly.validate(
|
|
1122
|
+
new Uint8Array([
|
|
1123
|
+
0,
|
|
1124
|
+
97,
|
|
1125
|
+
115,
|
|
1126
|
+
109,
|
|
1127
|
+
1,
|
|
1128
|
+
0,
|
|
1129
|
+
0,
|
|
1130
|
+
0,
|
|
1131
|
+
1,
|
|
1132
|
+
4,
|
|
1133
|
+
1,
|
|
1134
|
+
96,
|
|
1135
|
+
0,
|
|
1136
|
+
0,
|
|
1137
|
+
3,
|
|
1138
|
+
2,
|
|
1139
|
+
1,
|
|
1140
|
+
0,
|
|
1141
|
+
10,
|
|
1142
|
+
8,
|
|
1143
|
+
1,
|
|
1144
|
+
6,
|
|
1145
|
+
0,
|
|
1146
|
+
6,
|
|
1147
|
+
64,
|
|
1148
|
+
25,
|
|
1149
|
+
11,
|
|
1150
|
+
11
|
|
1151
|
+
])
|
|
1152
|
+
);
|
|
1153
|
+
});
|
|
1154
|
+
var isSupportSIMD = () => __async(void 0, null, function* () {
|
|
1155
|
+
return WebAssembly.validate(
|
|
1156
|
+
new Uint8Array([
|
|
1157
|
+
0,
|
|
1158
|
+
97,
|
|
1159
|
+
115,
|
|
1160
|
+
109,
|
|
1161
|
+
1,
|
|
1162
|
+
0,
|
|
1163
|
+
0,
|
|
1164
|
+
0,
|
|
1165
|
+
1,
|
|
1166
|
+
5,
|
|
1167
|
+
1,
|
|
1168
|
+
96,
|
|
1169
|
+
0,
|
|
1170
|
+
1,
|
|
1171
|
+
123,
|
|
1172
|
+
3,
|
|
1173
|
+
2,
|
|
1174
|
+
1,
|
|
1175
|
+
0,
|
|
1176
|
+
10,
|
|
1177
|
+
10,
|
|
1178
|
+
1,
|
|
1179
|
+
8,
|
|
1180
|
+
0,
|
|
1181
|
+
65,
|
|
1182
|
+
0,
|
|
1183
|
+
253,
|
|
1184
|
+
15,
|
|
1185
|
+
253,
|
|
1186
|
+
98,
|
|
1187
|
+
11
|
|
1188
|
+
])
|
|
1189
|
+
);
|
|
1190
|
+
});
|
|
1191
|
+
var isSupportJSPI = () => {
|
|
1192
|
+
return !!WebAssembly.Suspending;
|
|
1193
|
+
};
|
|
1194
|
+
var isSupportWebGPU = () => {
|
|
1195
|
+
return !!navigator.gpu;
|
|
1196
|
+
};
|
|
1197
|
+
var isSupportMem64 = () => {
|
|
1198
|
+
try {
|
|
1199
|
+
new WebAssembly.Memory({
|
|
1200
|
+
address: "i64",
|
|
1201
|
+
initial: /* @__PURE__ */ BigInt("1")
|
|
1202
|
+
// 1 page (64 KiB)
|
|
1203
|
+
});
|
|
1204
|
+
return true;
|
|
1205
|
+
} catch (e) {
|
|
1206
|
+
return false;
|
|
1207
|
+
}
|
|
1208
|
+
};
|
|
1209
|
+
var checkEnvironmentCompatible = () => __async(void 0, null, function* () {
|
|
1210
|
+
if (!(yield isSupportExceptions())) {
|
|
1211
|
+
throw new Error("WebAssembly runtime does not support exception handling");
|
|
1212
|
+
}
|
|
1213
|
+
if (!(yield isSupportSIMD())) {
|
|
1214
|
+
throw new Error("WebAssembly runtime does not support SIMD");
|
|
1215
|
+
}
|
|
1216
|
+
});
|
|
1217
|
+
var isFirefox = () => {
|
|
1218
|
+
return !!navigator.userAgent.match(/Firefox\/([0-9\.]+)(?:\s|$)/);
|
|
1219
|
+
};
|
|
1220
|
+
var GGUF_FILE_REGEX = /^.*\.gguf(?:\?.*)?$/;
|
|
1221
|
+
var isValidGgufFile = (path) => {
|
|
1222
|
+
return GGUF_FILE_REGEX.test(path);
|
|
1223
|
+
};
|
|
1224
|
+
var isSafariMobile = () => {
|
|
1225
|
+
return !!navigator.userAgent.match(/Version\/([0-9\._]+).*Mobile.*Safari.*/);
|
|
1226
|
+
};
|
|
1227
|
+
var createWorker = (workerCode) => {
|
|
1228
|
+
const workerURL = URL.createObjectURL(
|
|
1229
|
+
isString(workerCode) ? new Blob([workerCode], { type: "text/javascript" }) : workerCode
|
|
1230
|
+
);
|
|
1231
|
+
return new Worker(workerURL, { type: "module" });
|
|
1232
|
+
};
|
|
1233
|
+
var cbToAsyncIter = (fn) => (...args) => {
|
|
1234
|
+
let values = [];
|
|
1235
|
+
let resolve;
|
|
1236
|
+
let reject;
|
|
1237
|
+
values.push(
|
|
1238
|
+
new Promise((res, rej) => {
|
|
1239
|
+
resolve = res;
|
|
1240
|
+
reject = rej;
|
|
1241
|
+
})
|
|
1242
|
+
);
|
|
1243
|
+
fn(...args, (val, done, err) => {
|
|
1244
|
+
if (err) {
|
|
1245
|
+
reject(err);
|
|
1246
|
+
return;
|
|
1247
|
+
}
|
|
1248
|
+
resolve([val, done]);
|
|
1249
|
+
values.push(
|
|
1250
|
+
new Promise((res, rej) => {
|
|
1251
|
+
resolve = res;
|
|
1252
|
+
reject = rej;
|
|
1253
|
+
})
|
|
1254
|
+
);
|
|
1255
|
+
});
|
|
1256
|
+
return function() {
|
|
1257
|
+
return __asyncGenerator(this, null, function* () {
|
|
1258
|
+
let val;
|
|
1259
|
+
for (let i = 0, done = false; !done; i++) {
|
|
1260
|
+
[val, done] = yield new __await(values[i]);
|
|
1261
|
+
delete values[i];
|
|
1262
|
+
if (val !== void 0) yield val;
|
|
1263
|
+
}
|
|
1264
|
+
});
|
|
1265
|
+
}();
|
|
1266
|
+
};
|
|
1267
|
+
var canUseAsyncFileRead = (compat) => isSupportJSPI() || compat;
|
|
1268
|
+
var needCompat = () => !isSupportJSPI() || !isSupportMem64();
|
|
1269
|
+
|
|
1270
|
+
// src/workers-code/generated.ts
|
|
1271
|
+
var LIBLLAMA_VERSION = "b9643-dd37ee8";
|
|
1272
|
+
var LLAMA_CPP_WORKER_CODE = "// Start the main llama.cpp\nlet wllamaMalloc;\nlet wllamaStart;\nlet wllamaAction;\nlet wllamaExit;\nlet wllamaDebug;\n\nlet Module = null;\nlet isCompat = false;\nlet lastStack = '';\nlet isAborted = false;\nlet hasMultithread = false;\n\n//////////////////////////////////////////////////////////////\n// UTILS\n//////////////////////////////////////////////////////////////\n\n// send message back to main thread\nconst msg = (data, transfer) => postMessage(data, transfer);\n\n// Convert CPP log into JS log\nconst cppLogToJSLog = (line) => {\n const matched = line.match(/@@(DEBUG|INFO|WARN|ERROR)@@(.*)/);\n return !!matched\n ? {\n level: (matched[1] === 'INFO' ? 'debug' : matched[1]).toLowerCase(),\n text: matched[2],\n }\n : { level: 'log', text: line };\n};\n\nconst getHeapU8 = () => {\n const buffer = Module.wasmMemory.buffer;\n return new Uint8Array(buffer);\n};\n\nconst toSizeT = (num) => {\n return isCompat ? Number(num) : BigInt(num);\n};\n\n// Get module config that forwards stdout/err to main thread\nconst getWModuleConfig = (_argMainScriptBlob) => {\n var pathConfig = RUN_OPTIONS.pathConfig;\n var pthreadPoolSize = RUN_OPTIONS.nbThread;\n var argMainScriptBlob = _argMainScriptBlob;\n\n isCompat = RUN_OPTIONS.compat;\n hasMultithread = pthreadPoolSize > 1;\n\n msg({\n verb: 'console.debug',\n args: [\n `Multithread enabled: ${hasMultithread}, pthreadPoolSize: ${pthreadPoolSize}`,\n ],\n });\n\n if (!pathConfig['wllama.wasm']) {\n throw new Error('\"wllama.wasm\" is missing in pathConfig');\n }\n return {\n noInitialRun: true,\n print: function (text) {\n if (arguments.length > 1)\n text = Array.prototype.slice.call(arguments).join(' ');\n msg({ verb: 'console.log', args: [text] });\n },\n printErr: function (text) {\n if (arguments.length > 1)\n text = Array.prototype.slice.call(arguments).join(' ');\n if (text.startsWith('@@STACK@@')) {\n lastStack = text.slice('@@STACK@@'.length);\n return;\n }\n const logLine = cppLogToJSLog(text);\n msg({ verb: 'console.' + logLine.level, args: [logLine.text] });\n },\n locateFile: function (filename, basePath) {\n const p = pathConfig[filename];\n const truncate = (str) =>\n str.length > 128 ? `${str.substr(0, 128)}...` : str;\n if (filename.match(/wllama\\.worker\\.js/)) {\n msg({\n verb: 'console.error',\n args: [\n '\"wllama.worker.js\" is removed from v2.2.1. Hint: make sure to clear browser\\'s cache.',\n ],\n });\n } else {\n msg({\n verb: 'console.debug',\n args: [`Loading \"${filename}\" from \"${truncate(p)}\"`],\n });\n return p;\n }\n },\n mainScriptUrlOrBlob: hasMultithread\n ? argMainScriptBlob\n : 'throw new Error(\"Multithreading is not enabled\")',\n pthreadPoolSize: hasMultithread ? pthreadPoolSize : 0,\n wasmMemory: hasMultithread ? getWasmMemory() : null,\n onAbort: function (message) {\n isAborted = true;\n msg({ verb: 'signal.abort', args: ['abort', message, lastStack, null] });\n },\n onExit: function (code) {\n isAborted = true;\n const callstack = new Error().stack.toString();\n msg({\n verb: 'signal.abort',\n args: ['abort', 'exit(' + code + ')', callstack, null],\n });\n },\n };\n};\n\n// Get the memory to be used by wasm. (Only used in multi-thread mode)\n// Because we have a weird OOM issue on iOS, we need to try some values\n// See: https://github.com/emscripten-core/emscripten/issues/19144\n// https://github.com/godotengine/godot/issues/70621\nconst getWasmMemory = () => {\n let minBytes = 128 * 1024 * 1024;\n let maxBytes = 4096 * 1024 * 1024;\n let stepBytes = 128 * 1024 * 1024;\n while (maxBytes > minBytes) {\n try {\n const wasmMemory = new WebAssembly.Memory({\n initial: toSizeT(minBytes / 65536),\n maximum: toSizeT(maxBytes / 65536),\n shared: true,\n address: isCompat ? undefined : 'i64',\n });\n return wasmMemory;\n } catch (e) {\n maxBytes -= stepBytes;\n continue; // retry\n }\n }\n throw new Error('Cannot allocate WebAssembly.Memory');\n};\n\n//////////////////////////////////////////////////////////////\n// HEAPFS PATCH\n//////////////////////////////////////////////////////////////\n\n/**\n * By default, emscripten uses memfs. The way it works is by\n * allocating new Uint8Array in javascript heap. This is not good\n * because it requires files to be copied to wasm heap each time\n * a file is read.\n *\n * HeapFS is an alternative, which resolves this problem by\n * allocating space for file directly inside wasm heap. This\n * allows us to mmap without doing any copy.\n *\n * For llama.cpp, this is great because we use MAP_SHARED\n *\n * Ref: https://github.com/ngxson/wllama/pull/39\n * Ref: https://github.com/emscripten-core/emscripten/blob/main/src/library_memfs.js\n *\n * Note 29/05/2024 @ngxson\n * Due to ftell() being limited to MAX_LONG, we cannot load files bigger than 2^31 bytes (or 2GB)\n * Ref: https://github.com/emscripten-core/emscripten/blob/main/system/lib/libc/musl/src/stdio/ftell.c\n */\n\nconst fsNameToFile = {}; // map Name => File\nconst fsIdToFile = {}; // map ID => File\nlet currFileId = 0;\n\n// Patch and redirect memfs calls to wllama\nconst patchHeapFS = () => {\n const m = Module;\n // save functions\n m.MEMFS.stream_ops._read = m.MEMFS.stream_ops.read;\n m.MEMFS.stream_ops._write = m.MEMFS.stream_ops.write;\n m.MEMFS.stream_ops._llseek = m.MEMFS.stream_ops.llseek;\n m.MEMFS.stream_ops._allocate = m.MEMFS.stream_ops.allocate;\n m.MEMFS.stream_ops._mmap = m.MEMFS.stream_ops.mmap;\n m.MEMFS.stream_ops._msync = m.MEMFS.stream_ops.msync;\n\n const patchStream = (stream) => {\n const name = stream.node.name;\n if (fsNameToFile[name]) {\n const f = fsNameToFile[name];\n const ptr = Number(f.ptr);\n stream.node.contents = getHeapU8().subarray(ptr, ptr + f.size);\n stream.node.usedBytes = f.size;\n }\n };\n\n // replace \"read\" functions\n m.MEMFS.stream_ops.read = function (\n stream,\n buffer,\n offset,\n length,\n position\n ) {\n patchStream(stream);\n return m.MEMFS.stream_ops._read(stream, buffer, offset, length, position);\n };\n m.MEMFS.ops_table.file.stream.read = m.MEMFS.stream_ops.read;\n\n // replace \"llseek\" functions\n m.MEMFS.stream_ops.llseek = function (stream, offset, whence) {\n patchStream(stream);\n return m.MEMFS.stream_ops._llseek(stream, offset, whence);\n };\n m.MEMFS.ops_table.file.stream.llseek = m.MEMFS.stream_ops.llseek;\n\n // replace \"mmap\" functions\n m.MEMFS.stream_ops.mmap = function (stream, length, position, prot, flags) {\n patchStream(stream);\n const name = stream.node.name;\n if (fsNameToFile[name]) {\n const f = fsNameToFile[name];\n const mmapPtr = f.ptr + toSizeT(position);\n return {\n ptr: mmapPtr,\n allocated: false,\n };\n } else {\n return m.MEMFS.stream_ops._mmap(stream, length, position, prot, flags);\n }\n };\n m.MEMFS.ops_table.file.stream.mmap = m.MEMFS.stream_ops.mmap;\n\n // mount FS\n m.FS.mkdir('/models');\n m.FS.mount(m.MEMFS, { root: '.' }, '/models');\n};\n\n// Allocate a new file in wllama heapfs, returns file ID\nconst heapfsAlloc = (name, size, allocBuffer) => {\n if (size < 1) {\n throw new Error('File size must be bigger than 0');\n }\n const m = Module;\n const ptr = toSizeT(allocBuffer ? m.mmapAlloc(size) : 0);\n const file = {\n ptr: ptr,\n size: size,\n id: currFileId++,\n };\n fsIdToFile[file.id] = file;\n fsNameToFile[name] = file;\n return file.id;\n};\n\n// Add new file to wllama heapfs, return number of written bytes\nconst heapfsWrite = (id, buffer, offset) => {\n if (fsIdToFile[id]) {\n const { ptr, size } = fsIdToFile[id];\n const afterWriteByte = offset + buffer.byteLength;\n if (afterWriteByte > size) {\n throw new Error(\n `File ID ${id} write out of bound, afterWriteByte = ${afterWriteByte} while size = ${size}`\n );\n }\n getHeapU8().set(buffer, Number(ptr) + offset);\n return buffer.byteLength;\n } else {\n throw new Error(`File ID ${id} not found in heapfs`);\n }\n};\n\n//////////////////////////////////////////////////////////////\n// ASYNC FILE READ\n//////////////////////////////////////////////////////////////\n\nlet isAwaitReading = false;\nlet pendingReadPromise = null;\nlet pendingReadResolve = null;\nlet pendingReadReject = null;\n\nconst _stripModelsPrefix = (path) => path.replace(/^\\/?models\\//, '');\n\n// Called from EM_ASYNC_JS stub in wllama-fs.h (path is already a JS string)\nconst _wllama_js_file_read = async (path, offset, req_size, out_ptr) => {\n const name = _stripModelsPrefix(path);\n\n pendingReadPromise = new Promise((res, rej) => {\n pendingReadResolve = res;\n pendingReadReject = rej;\n });\n isAwaitReading = true;\n\n postMessage({ verb: 'fs.read_req', args: [name, offset, req_size] });\n\n let data;\n try {\n data = await pendingReadPromise;\n } finally {\n isAwaitReading = false;\n pendingReadResolve = null;\n pendingReadReject = null;\n }\n\n const bytes = new Uint8Array(data);\n getHeapU8().set(bytes, out_ptr);\n return toSizeT(bytes.length);\n};\n\n//////////////////////////////////////////////////////////////\n// MAIN CODE\n//////////////////////////////////////////////////////////////\n\nconst callWrapper = (name, ret, args, isAsync) => {\n const fn = Module.cwrap(\n name,\n ret,\n args,\n isAsync ? { async: true } : undefined\n );\n return async (action, req) => {\n // console.log(`Calling ${name} with action:`, action, 'and req:', req);\n let result;\n try {\n if (args.length === 2) {\n result = isAsync ? await fn(action, req) : fn(action, req);\n } else {\n result = fn();\n }\n } catch (ex) {\n console.error(ex);\n throw ex;\n }\n return result;\n };\n};\n\nfunction handleError(err) {\n // If WASM already aborted, onAbort already sent signal.abort; skip to avoid\n // re-reporting the resulting WebAssembly.RuntimeError as a JS exception.\n if (isAborted) return;\n\n const message = err ? err.message || String(err) : 'Unknown error';\n const stack = err ? err.stack || String(err) : '';\n msg({\n verb: 'signal.abort',\n args: ['exception', message, stack, err],\n });\n}\n\nonmessage = async (e) => {\n if (!e.data) return;\n const { verb, args, callbackId } = e.data;\n\n // fs.read_res arrives while wasm is JSPI-suspended; resolve the pending promise.\n if (verb === 'fs.read_res') {\n if (pendingReadResolve) {\n pendingReadResolve(args[0]);\n }\n return;\n }\n\n // Guard: while awaiting a file read, reject any other incoming task.\n if (isAwaitReading) {\n if (callbackId) {\n msg({\n callbackId,\n err: 'Worker is suspended waiting for file data (JSPI)',\n });\n }\n return;\n }\n\n if (!callbackId) {\n msg({ verb: 'console.error', args: ['callbackId is required', e.data] });\n return;\n }\n\n if (verb === 'module.init') {\n const argMainScriptBlob = args[0];\n const argUseAsyncFile = args[1];\n try {\n Module = getWModuleConfig(argMainScriptBlob);\n Module.preRun = () => {\n if (argUseAsyncFile) {\n Module.ENV['USE_ASYNC_FILE'] = '1';\n }\n };\n Module.onRuntimeInitialized = () => {\n // async call once module is ready\n // init FS\n patchHeapFS();\n // init cwrap\n const pointer = isCompat ? 'number' : 'bigint';\n // TODO: note sure why emscripten cannot bind if there is only 1 argument\n wllamaMalloc = callWrapper('wllama_malloc', pointer, [\n 'number',\n pointer,\n ]);\n wllamaStart = callWrapper('wllama_start', 'string', [], true);\n wllamaAction = callWrapper(\n 'wllama_action',\n pointer,\n ['string', pointer],\n true\n );\n wllamaExit = callWrapper('wllama_exit', 'string', []);\n wllamaDebug = callWrapper('wllama_debug', 'string', []);\n msg({ callbackId, result: null });\n };\n wModuleInit();\n } catch (err) {\n handleError(err);\n }\n return;\n }\n\n if (verb === 'fs.alloc') {\n const argFilename = args[0];\n const argSize = args[1];\n const argAllocBuffer = args[2];\n try {\n // create blank file\n const emptyBuffer = new ArrayBuffer(0);\n Module['FS_createDataFile'](\n '/models',\n argFilename,\n emptyBuffer,\n true,\n true,\n true\n );\n // alloc data on heap\n const fileId = heapfsAlloc(argFilename, argSize, argAllocBuffer);\n msg({ callbackId, result: { fileId } });\n } catch (err) {\n handleError(err);\n }\n return;\n }\n\n if (verb === 'fs.write') {\n const argFileId = args[0];\n const argBuffer = args[1];\n const argOffset = args[2];\n try {\n const writtenBytes = heapfsWrite(argFileId, argBuffer, argOffset);\n msg({ callbackId, result: { writtenBytes } });\n } catch (err) {\n handleError(err);\n }\n return;\n }\n\n if (verb === 'wllama.start') {\n try {\n const result = await wllamaStart();\n msg({ callbackId, result });\n } catch (err) {\n handleError(err);\n }\n return;\n }\n\n if (verb === 'wllama.action') {\n const argAction = args[0];\n const argEncodedMsg = args[1];\n try {\n const inputPtr = await wllamaMalloc(toSizeT(argEncodedMsg.byteLength), 0);\n // copy data to wasm heap\n const inputBuffer = new Uint8Array(\n getHeapU8().buffer,\n Number(inputPtr),\n argEncodedMsg.byteLength\n );\n inputBuffer.set(argEncodedMsg, 0);\n const outputPtr = await wllamaAction(argAction, inputPtr);\n // length of output buffer is written at the first 4 bytes of input buffer\n const outputLen = new Uint32Array(\n getHeapU8().buffer,\n Number(inputPtr),\n 1\n )[0];\n // copy the output buffer to JS heap\n const outputBuffer = new Uint8Array(outputLen);\n const outputSrcView = new Uint8Array(\n getHeapU8().buffer,\n Number(outputPtr),\n outputLen\n );\n outputBuffer.set(outputSrcView, 0); // copy it\n msg({ callbackId, result: outputBuffer }, [outputBuffer.buffer]);\n } catch (err) {\n handleError(err);\n }\n return;\n }\n\n if (verb === 'wllama.exit') {\n try {\n const result = await wllamaExit();\n msg({ callbackId, result });\n } catch (err) {\n handleError(err);\n }\n return;\n }\n\n if (verb === 'wllama.debug') {\n try {\n const result = await wllamaDebug();\n msg({ callbackId, result });\n } catch (err) {\n handleError(err);\n }\n return;\n }\n};\n";
|
|
1273
|
+
var OPFS_UTILS_WORKER_CODE = "let accessHandle;\nlet abortController = new AbortController();\n\nasync function openFile(filename) {\n const opfsRoot = await navigator.storage.getDirectory();\n const cacheDir = await opfsRoot.getDirectoryHandle('cache', { create: true });\n const fileHandler = await cacheDir.getFileHandle(filename, { create: true });\n accessHandle = await fileHandler.createSyncAccessHandle();\n accessHandle.truncate(0); // clear file content\n}\n\nasync function writeFile(buf) {\n accessHandle.write(buf);\n}\n\nasync function closeFile() {\n accessHandle.flush();\n accessHandle.close();\n}\n\nasync function writeTextFile(filename, str) {\n await openFile(filename);\n await writeFile(new TextEncoder().encode(str));\n await closeFile();\n}\n\nconst throttled = (func, delay) => {\n let lastRun = 0;\n return (...args) => {\n const now = Date.now();\n if (now - lastRun > delay) {\n lastRun = now;\n func.apply(null, args);\n }\n };\n};\n\nconst assertNonNull = (val) => {\n if (val === null || val === undefined) {\n throw new Error('OPFS Worker: Assertion failed');\n }\n};\n\n// respond to main thread\nconst resOK = () => postMessage({ ok: true });\nconst resProgress = (loaded, total) =>\n postMessage({ progress: { loaded, total } });\nconst resErr = (err) => postMessage({ err });\n\nonmessage = async (e) => {\n try {\n if (!e.data) return;\n\n /**\n * @param {Object} e.data\n *\n * Fine-control FS actions:\n * - { action: 'open', filename: 'string' }\n * - { action: 'write', buf: ArrayBuffer }\n * - { action: 'close' }\n *\n * Simple write API:\n * - { action: 'write-simple', filename: 'string', buf: ArrayBuffer }\n *\n * Download API:\n * - { action: 'download', url: 'string', filename: 'string', options: Object, metadataFileName: 'string' }\n * - { action: 'download-abort' }\n */\n const {\n action,\n filename,\n buf,\n url,\n options,\n metadataFileName,\n metadataAdditional,\n } = e.data;\n\n if (action === 'open') {\n assertNonNull(filename);\n await openFile(filename);\n return resOK();\n } else if (action === 'write') {\n assertNonNull(buf);\n await writeFile(buf);\n return resOK();\n } else if (action === 'close') {\n await closeFile();\n return resOK();\n } else if (action === 'write-simple') {\n assertNonNull(filename);\n assertNonNull(buf);\n await openFile(filename);\n await writeFile(buf);\n await closeFile();\n return resOK();\n } else if (action === 'download') {\n assertNonNull(url);\n assertNonNull(filename);\n assertNonNull(metadataFileName);\n assertNonNull(options);\n assertNonNull(options.aborted);\n abortController = new AbortController();\n if (options.aborted) abortController.abort();\n const response = await fetch(url, {\n ...options,\n signal: abortController.signal,\n });\n const contentLength = response.headers.get('content-length');\n const etag = (response.headers.get('etag') || '').replace(\n /[^A-Za-z0-9]/g,\n ''\n );\n const total = parseInt(contentLength, 10);\n const reader = response.body.getReader();\n await openFile(filename);\n let loaded = 0;\n const throttledProgress = throttled(resProgress, 100);\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n loaded += value.byteLength;\n await writeFile(value);\n throttledProgress(loaded, total);\n }\n resProgress(total, total); // 100% done\n await closeFile();\n // make sure this is in-sync with CacheEntryMetadata\n await writeTextFile(\n metadataFileName,\n JSON.stringify({\n originalURL: url,\n originalSize: total,\n etag,\n ...metadataAdditional,\n })\n );\n return resOK();\n } else if (action === 'download-abort') {\n if (abortController) {\n abortController.abort();\n }\n return;\n }\n\n throw new Error('OPFS Worker: Invalid action', e.data);\n } catch (err) {\n return resErr(err);\n }\n};\n";
|
|
1274
|
+
var WLLAMA_EMSCRIPTEN_CODE = 'var Module=typeof Module!="undefined"?Module:{};var ENVIRONMENT_IS_WEB=!!globalThis.window;var ENVIRONMENT_IS_WORKER=!!globalThis.WorkerGlobalScope;var ENVIRONMENT_IS_NODE=globalThis.process?.versions?.node&&globalThis.process?.type!="renderer";var ENVIRONMENT_IS_PTHREAD=ENVIRONMENT_IS_WORKER&&self.name?.startsWith("em-pthread");if(ENVIRONMENT_IS_NODE){var worker_threads=require("worker_threads");global.Worker=worker_threads.Worker;ENVIRONMENT_IS_WORKER=!worker_threads.isMainThread;ENVIRONMENT_IS_PTHREAD=ENVIRONMENT_IS_WORKER&&worker_threads["workerData"]=="em-pthread"}var arguments_=[];var thisProgram="./this.program";var quit_=(status,toThrow)=>{throw toThrow};var _scriptName=globalThis.document?.currentScript?.src;if(typeof __filename!="undefined"){_scriptName=__filename}else if(ENVIRONMENT_IS_WORKER){_scriptName=self.location.href}var scriptDirectory="";function locateFile(path){if(Module["locateFile"]){return Module["locateFile"](path,scriptDirectory)}return scriptDirectory+path}var readAsync,readBinary;if(ENVIRONMENT_IS_NODE){var fs=require("fs");scriptDirectory=__dirname+"/";readBinary=filename=>{filename=isFileURI(filename)?new URL(filename):filename;var ret=fs.readFileSync(filename);return ret};readAsync=async(filename,binary=true)=>{filename=isFileURI(filename)?new URL(filename):filename;var ret=fs.readFileSync(filename,binary?undefined:"utf8");return ret};if(process.argv.length>1){thisProgram=process.argv[1].replace(/\\\\/g,"/")}arguments_=process.argv.slice(2);if(typeof module!="undefined"){module["exports"]=Module}quit_=(status,toThrow)=>{process.exitCode=status;throw toThrow}}else if(ENVIRONMENT_IS_WEB||ENVIRONMENT_IS_WORKER){try{scriptDirectory=new URL(".",_scriptName).href}catch{}if(!ENVIRONMENT_IS_NODE){if(ENVIRONMENT_IS_WORKER){readBinary=url=>{var xhr=new XMLHttpRequest;xhr.open("GET",url,false);xhr.responseType="arraybuffer";xhr.send(null);return new Uint8Array(xhr.response)}}readAsync=async url=>{if(isFileURI(url)){return new Promise((resolve,reject)=>{var xhr=new XMLHttpRequest;xhr.open("GET",url,true);xhr.responseType="arraybuffer";xhr.onload=()=>{if(xhr.status==200||xhr.status==0&&xhr.response){resolve(xhr.response);return}reject(xhr.status)};xhr.onerror=reject;xhr.send(null)})}var response=await fetch(url,{credentials:"same-origin"});if(response.ok){return response.arrayBuffer()}throw new Error(response.status+" : "+response.url)}}}else{}var defaultPrint=console.log.bind(console);var defaultPrintErr=console.error.bind(console);if(ENVIRONMENT_IS_NODE){var utils=require("util");var stringify=a=>typeof a=="object"?utils.inspect(a):a;defaultPrint=(...args)=>fs.writeSync(1,args.map(stringify).join(" ")+"\\n");defaultPrintErr=(...args)=>fs.writeSync(2,args.map(stringify).join(" ")+"\\n")}var out=defaultPrint;var err=defaultPrintErr;var wasmBinary;var wasmModule;var ABORT=false;var EXITSTATUS;function assert(condition,text){if(!condition){abort(text)}}var isFileURI=filename=>filename.startsWith("file://");function growMemViews(){if(wasmMemory.buffer!=HEAP8.buffer){updateMemoryViews()}}if(ENVIRONMENT_IS_NODE&&ENVIRONMENT_IS_PTHREAD){var parentPort=worker_threads["parentPort"];parentPort.on("message",msg=>global.onmessage?.({data:msg}));Object.assign(globalThis,{self:global,postMessage:msg=>parentPort["postMessage"](msg)});process.on("uncaughtException",err=>{postMessage({cmd:"uncaughtException",error:err});process.exit(1)})}var startWorker;if(ENVIRONMENT_IS_PTHREAD){var initializedJS=false;self.onunhandledrejection=e=>{throw e.reason||e};async function handleMessage(e){try{var msgData=e["data"];var cmd=msgData.cmd;if(cmd==="load"){let messageQueue=[];self.onmessage=e=>messageQueue.push(e);startWorker=()=>{postMessage({cmd:"loaded"});for(let msg of messageQueue){handleMessage(msg)}self.onmessage=handleMessage};for(const handler of msgData.handlers){if(!Module[handler]||Module[handler].proxy){Module[handler]=(...args)=>{postMessage({cmd:"callHandler",handler,args})};if(handler=="print")out=Module[handler];if(handler=="printErr")err=Module[handler]}}wasmMemory=msgData.wasmMemory;updateMemoryViews();wasmModule=msgData.wasmModule;createWasm();run()}else if(cmd==="run"){establishStackSpace(msgData.pthread_ptr);__emscripten_thread_init(msgData.pthread_ptr,0,0,1,0,0);PThread.threadInitTLS();__emscripten_thread_mailbox_await(msgData.pthread_ptr);if(!initializedJS){initializedJS=true}try{await invokeEntryPoint(msgData.start_routine,msgData.arg)}catch(ex){if(ex!="unwind"){throw ex}}}else if(msgData.target==="setimmediate"){}else if(cmd==="checkMailbox"){if(initializedJS){checkMailbox()}}else if(cmd){err(`worker: received unknown command ${cmd}`);err(msgData)}}catch(ex){__emscripten_thread_crashed();throw ex}}self.onmessage=handleMessage}var HEAP8,HEAPU8,HEAP16,HEAPU16,HEAP32,HEAPU32,HEAPF32,HEAPF64;var HEAP64,HEAPU64;var runtimeInitialized=false;function updateMemoryViews(){var b=wasmMemory.buffer;HEAP8=new Int8Array(b);HEAP16=new Int16Array(b);Module["HEAPU8"]=HEAPU8=new Uint8Array(b);HEAPU16=new Uint16Array(b);HEAP32=new Int32Array(b);HEAPU32=new Uint32Array(b);HEAPF32=new Float32Array(b);HEAPF64=new Float64Array(b);HEAP64=new BigInt64Array(b);HEAPU64=new BigUint64Array(b)}function initMemory(){if(ENVIRONMENT_IS_PTHREAD){return}if(Module["wasmMemory"]){wasmMemory=Module["wasmMemory"]}else{var INITIAL_MEMORY=Module["INITIAL_MEMORY"]||134217728;wasmMemory=new WebAssembly.Memory({initial:BigInt(INITIAL_MEMORY/65536),maximum:65536n,shared:true,address:"i64"})}updateMemoryViews()}function preRun(){if(Module["preRun"]){if(typeof Module["preRun"]=="function")Module["preRun"]=[Module["preRun"]];while(Module["preRun"].length){addOnPreRun(Module["preRun"].shift())}}callRuntimeCallbacks(onPreRuns)}function initRuntime(){runtimeInitialized=true;if(ENVIRONMENT_IS_PTHREAD)return startWorker();if(!Module["noFSInit"]&&!FS.initialized)FS.init();TTY.init();wasmExports["__wasm_call_ctors"]();FS.ignorePermissions=false}function preMain(){}function postRun(){if(ENVIRONMENT_IS_PTHREAD){return}if(Module["postRun"]){if(typeof Module["postRun"]=="function")Module["postRun"]=[Module["postRun"]];while(Module["postRun"].length){addOnPostRun(Module["postRun"].shift())}}callRuntimeCallbacks(onPostRuns)}function abort(what){Module["onAbort"]?.(what);what="Aborted("+what+")";err(what);ABORT=true;what+=". Build with -sASSERTIONS for more info.";if(runtimeInitialized){___trap()}var e=new WebAssembly.RuntimeError(what);throw e}var wasmBinaryFile;function findWasmBinary(){return locateFile("wllama.wasm")}function getBinarySync(file){if(file==wasmBinaryFile&&wasmBinary){return new Uint8Array(wasmBinary)}if(readBinary){return readBinary(file)}throw"both async and sync fetching of the wasm failed"}async function getWasmBinary(binaryFile){if(!wasmBinary){try{var response=await readAsync(binaryFile);return new Uint8Array(response)}catch{}}return getBinarySync(binaryFile)}async function instantiateArrayBuffer(binaryFile,imports){try{var binary=await getWasmBinary(binaryFile);var instance=await WebAssembly.instantiate(binary,imports);return instance}catch(reason){err(`failed to asynchronously prepare wasm: ${reason}`);abort(reason)}}async function instantiateAsync(binary,binaryFile,imports){if(!binary&&!isFileURI(binaryFile)&&!ENVIRONMENT_IS_NODE){try{var response=fetch(binaryFile,{credentials:"same-origin"});var instantiationResult=await WebAssembly.instantiateStreaming(response,imports);return instantiationResult}catch(reason){err(`wasm streaming compile failed: ${reason}`);err("falling back to ArrayBuffer instantiation")}}return instantiateArrayBuffer(binaryFile,imports)}function getWasmImports(){assignWasmImports();if(!wasmImports.__instrumented){wasmImports.__instrumented=true;Asyncify.instrumentWasmImports(wasmImports)}var imports={env:wasmImports,wasi_snapshot_preview1:wasmImports};return imports}async function createWasm(){function receiveInstance(instance,module){wasmExports=instance.exports;wasmExports=Asyncify.instrumentWasmExports(wasmExports);wasmExports=applySignatureConversions(wasmExports);registerTLSInit(wasmExports["_emscripten_tls_init"]);assignWasmExports(wasmExports);wasmModule=module;removeRunDependency("wasm-instantiate");return wasmExports}addRunDependency("wasm-instantiate");function receiveInstantiationResult(result){return receiveInstance(result["instance"],result["module"])}var info=getWasmImports();if(Module["instantiateWasm"]){return new Promise((resolve,reject)=>{Module["instantiateWasm"](info,(inst,mod)=>{resolve(receiveInstance(inst,mod))})})}if(ENVIRONMENT_IS_PTHREAD){var instance=new WebAssembly.Instance(wasmModule,getWasmImports());return receiveInstance(instance,wasmModule)}wasmBinaryFile??=findWasmBinary();var result=await instantiateAsync(wasmBinary,wasmBinaryFile,info);var exports=receiveInstantiationResult(result);return exports}class ExitStatus{name="ExitStatus";constructor(status){this.message=`Program terminated with exit(${status})`;this.status=status}}var terminateWorker=worker=>{worker.terminate();worker.onmessage=e=>{}};var cleanupThread=pthread_ptr=>{var worker=PThread.pthreads[pthread_ptr];PThread.returnWorkerToPool(worker)};var callRuntimeCallbacks=callbacks=>{while(callbacks.length>0){callbacks.shift()(Module)}};var onPreRuns=[];var addOnPreRun=cb=>onPreRuns.push(cb);var runDependencies=0;var dependenciesFulfilled=null;var removeRunDependency=id=>{runDependencies--;Module["monitorRunDependencies"]?.(runDependencies);if(runDependencies==0){if(dependenciesFulfilled){var callback=dependenciesFulfilled;dependenciesFulfilled=null;callback()}}};var addRunDependency=id=>{runDependencies++;Module["monitorRunDependencies"]?.(runDependencies)};var spawnThread=threadParams=>{var worker=PThread.getNewWorker();if(!worker){return 6}PThread.runningWorkers.push(worker);PThread.pthreads[threadParams.pthread_ptr]=worker;worker.pthread_ptr=threadParams.pthread_ptr;var msg={cmd:"run",start_routine:threadParams.startRoutine,arg:threadParams.arg,pthread_ptr:threadParams.pthread_ptr};if(ENVIRONMENT_IS_NODE){worker.unref()}worker.postMessage(msg,threadParams.transferList);return 0};var runtimeKeepaliveCounter=0;var keepRuntimeAlive=()=>noExitRuntime||runtimeKeepaliveCounter>0;var stackSave=()=>_emscripten_stack_get_current();var stackRestore=val=>__emscripten_stack_restore(val);var stackAlloc=sz=>__emscripten_stack_alloc(sz);var proxyToMainThread=(funcIndex,emAsmAddr,sync,...callArgs)=>{var serializedNumCallArgs=callArgs.length*2;var sp=stackSave();var args=stackAlloc(serializedNumCallArgs*8);var b=args/8;for(var i=0;i<callArgs.length;i++){var arg=callArgs[i];if(typeof arg=="bigint"){(growMemViews(),HEAP64)[b+2*i]=1n;(growMemViews(),HEAP64)[b+2*i+1]=arg}else{(growMemViews(),HEAP64)[b+2*i]=0n;(growMemViews(),HEAPF64)[b+2*i+1]=arg}}var rtn=__emscripten_run_js_on_main_thread(funcIndex,emAsmAddr,serializedNumCallArgs,args,sync);stackRestore(sp);return rtn};function _proc_exit(code){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(0,0,1,code);EXITSTATUS=code;if(!keepRuntimeAlive()){PThread.terminateAllThreads();Module["onExit"]?.(code);ABORT=true}quit_(code,new ExitStatus(code))}function exitOnMainThread(returnCode){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(1,0,0,returnCode);_exit(returnCode)}var exitJS=(status,implicit)=>{EXITSTATUS=status;if(ENVIRONMENT_IS_PTHREAD){exitOnMainThread(status);throw"unwind"}_proc_exit(status)};var _exit=exitJS;var PThread={unusedWorkers:[],runningWorkers:[],tlsInitFunctions:[],pthreads:{},init(){if(!ENVIRONMENT_IS_PTHREAD){PThread.initMainThread()}},initMainThread(){var pthreadPoolSize=Module["pthreadPoolSize"];while(pthreadPoolSize--){PThread.allocateUnusedWorker()}addOnPreRun(async()=>{var pthreadPoolReady=PThread.loadWasmModuleToAllWorkers();addRunDependency("loading-workers");await pthreadPoolReady;removeRunDependency("loading-workers")})},terminateAllThreads:()=>{for(var worker of PThread.runningWorkers){terminateWorker(worker)}for(var worker of PThread.unusedWorkers){terminateWorker(worker)}PThread.unusedWorkers=[];PThread.runningWorkers=[];PThread.pthreads={}},returnWorkerToPool:worker=>{var pthread_ptr=worker.pthread_ptr;delete PThread.pthreads[pthread_ptr];PThread.unusedWorkers.push(worker);PThread.runningWorkers.splice(PThread.runningWorkers.indexOf(worker),1);worker.pthread_ptr=0;__emscripten_thread_free_data(pthread_ptr)},threadInitTLS(){PThread.tlsInitFunctions.forEach(f=>f())},loadWasmModuleToWorker:worker=>new Promise(onFinishedLoading=>{worker.onmessage=e=>{var d=e["data"];var cmd=d.cmd;if(d.targetThread&&d.targetThread!=_pthread_self()){var targetWorker=PThread.pthreads[d.targetThread];if(targetWorker){targetWorker.postMessage(d,d.transferList)}else{err(`Internal error! Worker sent a message "${cmd}" to target pthread ${d.targetThread}, but that thread no longer exists!`)}return}if(cmd==="checkMailbox"){checkMailbox()}else if(cmd==="spawnThread"){spawnThread(d)}else if(cmd==="cleanupThread"){callUserCallback(()=>cleanupThread(d.thread))}else if(cmd==="loaded"){worker.loaded=true;if(ENVIRONMENT_IS_NODE&&!worker.pthread_ptr){worker.unref()}onFinishedLoading(worker)}else if(d.target==="setimmediate"){worker.postMessage(d)}else if(cmd==="uncaughtException"){worker.onerror(d.error)}else if(cmd==="callHandler"){Module[d.handler](...d.args)}else if(cmd){err(`worker sent an unknown command ${cmd}`)}};worker.onerror=e=>{var message="worker sent an error!";err(`${message} ${e.filename}:${e.lineno}: ${e.message}`);throw e};if(ENVIRONMENT_IS_NODE){worker.on("message",data=>worker.onmessage({data}));worker.on("error",e=>worker.onerror(e))}var handlers=[];var knownHandlers=["onExit","onAbort","print","printErr"];for(var handler of knownHandlers){if(Module.propertyIsEnumerable(handler)){handlers.push(handler)}}worker.postMessage({cmd:"load",handlers,wasmMemory,wasmModule})}),async loadWasmModuleToAllWorkers(){if(ENVIRONMENT_IS_PTHREAD){return}let pthreadPoolReady=Promise.all(PThread.unusedWorkers.map(PThread.loadWasmModuleToWorker));return pthreadPoolReady},allocateUnusedWorker(){var worker;var pthreadMainJs=_scriptName;if(Module["mainScriptUrlOrBlob"]){pthreadMainJs=Module["mainScriptUrlOrBlob"];if(typeof pthreadMainJs!="string"){pthreadMainJs=URL.createObjectURL(pthreadMainJs)}}worker=new Worker(pthreadMainJs,{workerData:"em-pthread",name:"em-pthread"});PThread.unusedWorkers.push(worker)},getNewWorker(){if(PThread.unusedWorkers.length==0){PThread.allocateUnusedWorker();PThread.loadWasmModuleToWorker(PThread.unusedWorkers[0])}return PThread.unusedWorkers.pop()}};var onPostRuns=[];var addOnPostRun=cb=>onPostRuns.push(cb);function establishStackSpace(pthread_ptr){var stackHigh=Number((growMemViews(),HEAPU64)[(pthread_ptr+88)/8]);var stackSize=Number((growMemViews(),HEAPU64)[(pthread_ptr+96)/8]);var stackLow=stackHigh-stackSize;_emscripten_stack_set_limits(stackHigh,stackLow);stackRestore(stackHigh)}var wasmTableMirror=[];var getWasmTableEntry=funcPtr=>{funcPtr=Number(funcPtr);var func=wasmTableMirror[funcPtr];if(!func){wasmTableMirror[funcPtr]=func=wasmTable.get(BigInt(funcPtr));if(Asyncify.isAsyncExport(func)){wasmTableMirror[funcPtr]=func=Asyncify.makeAsyncFunction(func)}}return func};var invokeEntryPoint=async(ptr,arg)=>{runtimeKeepaliveCounter=0;noExitRuntime=0;var result=(a1=>WebAssembly.promising(getWasmTableEntry(ptr)).call(null,BigInt(a1)))(arg);function finish(result){if(keepRuntimeAlive()){EXITSTATUS=result;return}__emscripten_thread_exit(result)}result=await result;finish(result)};invokeEntryPoint.isAsync=true;var noExitRuntime=true;var registerTLSInit=tlsInitFunc=>PThread.tlsInitFunctions.push(tlsInitFunc);var wasmMemory;function pthreadCreateProxied(pthread_ptr,attr,startRoutine,arg){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(2,0,1,pthread_ptr,attr,startRoutine,arg);return ___pthread_create_js(pthread_ptr,attr,startRoutine,arg)}var _emscripten_has_threading_support=()=>!!globalThis.SharedArrayBuffer;var INT53_MAX=9007199254740992;var INT53_MIN=-9007199254740992;var bigintToI53Checked=num=>num<INT53_MIN||num>INT53_MAX?NaN:Number(num);function ___pthread_create_js(pthread_ptr,attr,startRoutine,arg){pthread_ptr=bigintToI53Checked(pthread_ptr);attr=bigintToI53Checked(attr);startRoutine=bigintToI53Checked(startRoutine);arg=bigintToI53Checked(arg);if(!_emscripten_has_threading_support()){return 6}var transferList=[];var error=0;if(ENVIRONMENT_IS_PTHREAD&&(transferList.length===0||error)){return pthreadCreateProxied(pthread_ptr,attr,startRoutine,arg)}if(error)return error;var threadParams={startRoutine,pthread_ptr,arg,transferList};if(ENVIRONMENT_IS_PTHREAD){threadParams.cmd="spawnThread";postMessage(threadParams,transferList);return 0}return spawnThread(threadParams)}var syscallGetVarargP=()=>{var ret=Number((growMemViews(),HEAPU64)[SYSCALLS.varargs/8]);SYSCALLS.varargs+=8;return ret};var syscallGetVarargI=()=>{var ret=(growMemViews(),HEAP32)[+SYSCALLS.varargs/4];SYSCALLS.varargs+=4;return ret};var PATH={isAbs:path=>path.charAt(0)==="/",splitPath:filename=>{var splitPathRe=/^(\\/?|)([\\s\\S]*?)((?:\\.{1,2}|[^\\/]+?|)(\\.[^.\\/]*|))(?:[\\/]*)$/;return splitPathRe.exec(filename).slice(1)},normalizeArray:(parts,allowAboveRoot)=>{var up=0;for(var i=parts.length-1;i>=0;i--){var last=parts[i];if(last==="."){parts.splice(i,1)}else if(last===".."){parts.splice(i,1);up++}else if(up){parts.splice(i,1);up--}}if(allowAboveRoot){for(;up;up--){parts.unshift("..")}}return parts},normalize:path=>{var isAbsolute=PATH.isAbs(path),trailingSlash=path.slice(-1)==="/";path=PATH.normalizeArray(path.split("/").filter(p=>!!p),!isAbsolute).join("/");if(!path&&!isAbsolute){path="."}if(path&&trailingSlash){path+="/"}return(isAbsolute?"/":"")+path},dirname:path=>{var result=PATH.splitPath(path),root=result[0],dir=result[1];if(!root&&!dir){return"."}if(dir){dir=dir.slice(0,-1)}return root+dir},basename:path=>path&&path.match(/([^\\/]+|\\/)\\/*$/)[1],join:(...paths)=>PATH.normalize(paths.join("/")),join2:(l,r)=>PATH.normalize(l+"/"+r)};var initRandomFill=()=>view=>view.set(crypto.getRandomValues(new Uint8Array(view.byteLength)));var randomFill=view=>{(randomFill=initRandomFill())(view)};var PATH_FS={resolve:(...args)=>{var resolvedPath="",resolvedAbsolute=false;for(var i=args.length-1;i>=-1&&!resolvedAbsolute;i--){var path=i>=0?args[i]:FS.cwd();if(typeof path!="string"){throw new TypeError("Arguments to path.resolve must be strings")}else if(!path){return""}resolvedPath=path+"/"+resolvedPath;resolvedAbsolute=PATH.isAbs(path)}resolvedPath=PATH.normalizeArray(resolvedPath.split("/").filter(p=>!!p),!resolvedAbsolute).join("/");return(resolvedAbsolute?"/":"")+resolvedPath||"."},relative:(from,to)=>{from=PATH_FS.resolve(from).slice(1);to=PATH_FS.resolve(to).slice(1);function trim(arr){var start=0;for(;start<arr.length;start++){if(arr[start]!=="")break}var end=arr.length-1;for(;end>=0;end--){if(arr[end]!=="")break}if(start>end)return[];return arr.slice(start,end-start+1)}var fromParts=trim(from.split("/"));var toParts=trim(to.split("/"));var length=Math.min(fromParts.length,toParts.length);var samePartsLength=length;for(var i=0;i<length;i++){if(fromParts[i]!==toParts[i]){samePartsLength=i;break}}var outputParts=[];for(var i=samePartsLength;i<fromParts.length;i++){outputParts.push("..")}outputParts=outputParts.concat(toParts.slice(samePartsLength));return outputParts.join("/")}};var UTF8Decoder=globalThis.TextDecoder&&new TextDecoder;var findStringEnd=(heapOrArray,idx,maxBytesToRead,ignoreNul)=>{var maxIdx=idx+maxBytesToRead;if(ignoreNul)return maxIdx;while(heapOrArray[idx]&&!(idx>=maxIdx))++idx;return idx};var UTF8ArrayToString=(heapOrArray,idx=0,maxBytesToRead,ignoreNul)=>{var endPtr=findStringEnd(heapOrArray,idx,maxBytesToRead,ignoreNul);if(endPtr-idx>16&&heapOrArray.buffer&&UTF8Decoder){return UTF8Decoder.decode(heapOrArray.buffer instanceof ArrayBuffer?heapOrArray.subarray(idx,endPtr):heapOrArray.slice(idx,endPtr))}var str="";while(idx<endPtr){var u0=heapOrArray[idx++];if(!(u0&128)){str+=String.fromCharCode(u0);continue}var u1=heapOrArray[idx++]&63;if((u0&224)==192){str+=String.fromCharCode((u0&31)<<6|u1);continue}var u2=heapOrArray[idx++]&63;if((u0&240)==224){u0=(u0&15)<<12|u1<<6|u2}else{u0=(u0&7)<<18|u1<<12|u2<<6|heapOrArray[idx++]&63}if(u0<65536){str+=String.fromCharCode(u0)}else{var ch=u0-65536;str+=String.fromCharCode(55296|ch>>10,56320|ch&1023)}}return str};var FS_stdin_getChar_buffer=[];var lengthBytesUTF8=str=>{var len=0;for(var i=0;i<str.length;++i){var c=str.charCodeAt(i);if(c<=127){len++}else if(c<=2047){len+=2}else if(c>=55296&&c<=57343){len+=4;++i}else{len+=3}}return len};var stringToUTF8Array=(str,heap,outIdx,maxBytesToWrite)=>{if(!(maxBytesToWrite>0))return 0;var startIdx=outIdx;var endIdx=outIdx+maxBytesToWrite-1;for(var i=0;i<str.length;++i){var u=str.codePointAt(i);if(u<=127){if(outIdx>=endIdx)break;heap[outIdx++]=u}else if(u<=2047){if(outIdx+1>=endIdx)break;heap[outIdx++]=192|u>>6;heap[outIdx++]=128|u&63}else if(u<=65535){if(outIdx+2>=endIdx)break;heap[outIdx++]=224|u>>12;heap[outIdx++]=128|u>>6&63;heap[outIdx++]=128|u&63}else{if(outIdx+3>=endIdx)break;heap[outIdx++]=240|u>>18;heap[outIdx++]=128|u>>12&63;heap[outIdx++]=128|u>>6&63;heap[outIdx++]=128|u&63;i++}}heap[outIdx]=0;return outIdx-startIdx};var intArrayFromString=(stringy,dontAddNull,length)=>{var len=length>0?length:lengthBytesUTF8(stringy)+1;var u8array=new Array(len);var numBytesWritten=stringToUTF8Array(stringy,u8array,0,u8array.length);if(dontAddNull)u8array.length=numBytesWritten;return u8array};var FS_stdin_getChar=()=>{if(!FS_stdin_getChar_buffer.length){var result=null;if(ENVIRONMENT_IS_NODE){var BUFSIZE=256;var buf=Buffer.alloc(BUFSIZE);var bytesRead=0;var fd=process.stdin.fd;try{bytesRead=fs.readSync(fd,buf,0,BUFSIZE)}catch(e){if(e.toString().includes("EOF"))bytesRead=0;else throw e}if(bytesRead>0){result=buf.slice(0,bytesRead).toString("utf-8")}}else if(globalThis.window?.prompt){result=window.prompt("Input: ");if(result!==null){result+="\\n"}}else{}if(!result){return null}FS_stdin_getChar_buffer=intArrayFromString(result,true)}return FS_stdin_getChar_buffer.shift()};var TTY={ttys:[],init(){},shutdown(){},register(dev,ops){TTY.ttys[dev]={input:[],output:[],ops};FS.registerDevice(dev,TTY.stream_ops)},stream_ops:{open(stream){var tty=TTY.ttys[stream.node.rdev];if(!tty){throw new FS.ErrnoError(43)}stream.tty=tty;stream.seekable=false},close(stream){stream.tty.ops.fsync(stream.tty)},fsync(stream){stream.tty.ops.fsync(stream.tty)},read(stream,buffer,offset,length,pos){if(!stream.tty||!stream.tty.ops.get_char){throw new FS.ErrnoError(60)}var bytesRead=0;for(var i=0;i<length;i++){var result;try{result=stream.tty.ops.get_char(stream.tty)}catch(e){throw new FS.ErrnoError(29)}if(result===undefined&&bytesRead===0){throw new FS.ErrnoError(6)}if(result===null||result===undefined)break;bytesRead++;buffer[offset+i]=result}if(bytesRead){stream.node.atime=Date.now()}return bytesRead},write(stream,buffer,offset,length,pos){if(!stream.tty||!stream.tty.ops.put_char){throw new FS.ErrnoError(60)}try{for(var i=0;i<length;i++){stream.tty.ops.put_char(stream.tty,buffer[offset+i])}}catch(e){throw new FS.ErrnoError(29)}if(length){stream.node.mtime=stream.node.ctime=Date.now()}return i}},default_tty_ops:{get_char(tty){return FS_stdin_getChar()},put_char(tty,val){if(val===null||val===10){out(UTF8ArrayToString(tty.output));tty.output=[]}else{if(val!=0)tty.output.push(val)}},fsync(tty){if(tty.output?.length>0){out(UTF8ArrayToString(tty.output));tty.output=[]}},ioctl_tcgets(tty){return{c_iflag:25856,c_oflag:5,c_cflag:191,c_lflag:35387,c_cc:[3,28,127,21,4,0,1,0,17,19,26,0,18,15,23,22,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0]}},ioctl_tcsets(tty,optional_actions,data){return 0},ioctl_tiocgwinsz(tty){return[24,80]}},default_tty1_ops:{put_char(tty,val){if(val===null||val===10){err(UTF8ArrayToString(tty.output));tty.output=[]}else{if(val!=0)tty.output.push(val)}},fsync(tty){if(tty.output?.length>0){err(UTF8ArrayToString(tty.output));tty.output=[]}}}};var zeroMemory=(ptr,size)=>(growMemViews(),HEAPU8).fill(0,ptr,ptr+size);var alignMemory=(size,alignment)=>Math.ceil(size/alignment)*alignment;var mmapAlloc=size=>{size=alignMemory(size,65536);var ptr=_emscripten_builtin_memalign(65536,size);if(ptr)zeroMemory(ptr,size);return ptr};var MEMFS={ops_table:null,mount(mount){return MEMFS.createNode(null,"/",16895,0)},createNode(parent,name,mode,dev){if(FS.isBlkdev(mode)||FS.isFIFO(mode)){throw new FS.ErrnoError(63)}MEMFS.ops_table||={dir:{node:{getattr:MEMFS.node_ops.getattr,setattr:MEMFS.node_ops.setattr,lookup:MEMFS.node_ops.lookup,mknod:MEMFS.node_ops.mknod,rename:MEMFS.node_ops.rename,unlink:MEMFS.node_ops.unlink,rmdir:MEMFS.node_ops.rmdir,readdir:MEMFS.node_ops.readdir,symlink:MEMFS.node_ops.symlink},stream:{llseek:MEMFS.stream_ops.llseek}},file:{node:{getattr:MEMFS.node_ops.getattr,setattr:MEMFS.node_ops.setattr},stream:{llseek:MEMFS.stream_ops.llseek,read:MEMFS.stream_ops.read,write:MEMFS.stream_ops.write,mmap:MEMFS.stream_ops.mmap,msync:MEMFS.stream_ops.msync}},link:{node:{getattr:MEMFS.node_ops.getattr,setattr:MEMFS.node_ops.setattr,readlink:MEMFS.node_ops.readlink},stream:{}},chrdev:{node:{getattr:MEMFS.node_ops.getattr,setattr:MEMFS.node_ops.setattr},stream:FS.chrdev_stream_ops}};var node=FS.createNode(parent,name,mode,dev);if(FS.isDir(node.mode)){node.node_ops=MEMFS.ops_table.dir.node;node.stream_ops=MEMFS.ops_table.dir.stream;node.contents={}}else if(FS.isFile(node.mode)){node.node_ops=MEMFS.ops_table.file.node;node.stream_ops=MEMFS.ops_table.file.stream;node.usedBytes=0;node.contents=null}else if(FS.isLink(node.mode)){node.node_ops=MEMFS.ops_table.link.node;node.stream_ops=MEMFS.ops_table.link.stream}else if(FS.isChrdev(node.mode)){node.node_ops=MEMFS.ops_table.chrdev.node;node.stream_ops=MEMFS.ops_table.chrdev.stream}node.atime=node.mtime=node.ctime=Date.now();if(parent){parent.contents[name]=node;parent.atime=parent.mtime=parent.ctime=node.atime}return node},getFileDataAsTypedArray(node){if(!node.contents)return new Uint8Array(0);if(node.contents.subarray)return node.contents.subarray(0,node.usedBytes);return new Uint8Array(node.contents)},expandFileStorage(node,newCapacity){var prevCapacity=node.contents?node.contents.length:0;if(prevCapacity>=newCapacity)return;var CAPACITY_DOUBLING_MAX=1024*1024;newCapacity=Math.max(newCapacity,prevCapacity*(prevCapacity<CAPACITY_DOUBLING_MAX?2:1.125)>>>0);if(prevCapacity!=0)newCapacity=Math.max(newCapacity,256);var oldContents=node.contents;node.contents=new Uint8Array(newCapacity);if(node.usedBytes>0)node.contents.set(oldContents.subarray(0,node.usedBytes),0)},resizeFileStorage(node,newSize){if(node.usedBytes==newSize)return;if(newSize==0){node.contents=null;node.usedBytes=0}else{var oldContents=node.contents;node.contents=new Uint8Array(newSize);if(oldContents){node.contents.set(oldContents.subarray(0,Math.min(newSize,node.usedBytes)))}node.usedBytes=newSize}},node_ops:{getattr(node){var attr={};attr.dev=FS.isChrdev(node.mode)?node.id:1;attr.ino=node.id;attr.mode=node.mode;attr.nlink=1;attr.uid=0;attr.gid=0;attr.rdev=node.rdev;if(FS.isDir(node.mode)){attr.size=4096}else if(FS.isFile(node.mode)){attr.size=node.usedBytes}else if(FS.isLink(node.mode)){attr.size=node.link.length}else{attr.size=0}attr.atime=new Date(node.atime);attr.mtime=new Date(node.mtime);attr.ctime=new Date(node.ctime);attr.blksize=4096;attr.blocks=Math.ceil(attr.size/attr.blksize);return attr},setattr(node,attr){for(const key of["mode","atime","mtime","ctime"]){if(attr[key]!=null){node[key]=attr[key]}}if(attr.size!==undefined){MEMFS.resizeFileStorage(node,attr.size)}},lookup(parent,name){if(!MEMFS.doesNotExistError){MEMFS.doesNotExistError=new FS.ErrnoError(44);MEMFS.doesNotExistError.stack="<generic error, no stack>"}throw MEMFS.doesNotExistError},mknod(parent,name,mode,dev){return MEMFS.createNode(parent,name,mode,dev)},rename(old_node,new_dir,new_name){var new_node;try{new_node=FS.lookupNode(new_dir,new_name)}catch(e){}if(new_node){if(FS.isDir(old_node.mode)){for(var i in new_node.contents){throw new FS.ErrnoError(55)}}FS.hashRemoveNode(new_node)}delete old_node.parent.contents[old_node.name];new_dir.contents[new_name]=old_node;old_node.name=new_name;new_dir.ctime=new_dir.mtime=old_node.parent.ctime=old_node.parent.mtime=Date.now()},unlink(parent,name){delete parent.contents[name];parent.ctime=parent.mtime=Date.now()},rmdir(parent,name){var node=FS.lookupNode(parent,name);for(var i in node.contents){throw new FS.ErrnoError(55)}delete parent.contents[name];parent.ctime=parent.mtime=Date.now()},readdir(node){return[".","..",...Object.keys(node.contents)]},symlink(parent,newname,oldpath){var node=MEMFS.createNode(parent,newname,511|40960,0);node.link=oldpath;return node},readlink(node){if(!FS.isLink(node.mode)){throw new FS.ErrnoError(28)}return node.link}},stream_ops:{read(stream,buffer,offset,length,position){var contents=stream.node.contents;if(position>=stream.node.usedBytes)return 0;var size=Math.min(stream.node.usedBytes-position,length);if(size>8&&contents.subarray){buffer.set(contents.subarray(position,position+size),offset)}else{for(var i=0;i<size;i++)buffer[offset+i]=contents[position+i]}return size},write(stream,buffer,offset,length,position,canOwn){if(buffer.buffer===(growMemViews(),HEAP8).buffer){canOwn=false}if(!length)return 0;var node=stream.node;node.mtime=node.ctime=Date.now();if(buffer.subarray&&(!node.contents||node.contents.subarray)){if(canOwn){node.contents=buffer.subarray(offset,offset+length);node.usedBytes=length;return length}else if(node.usedBytes===0&&position===0){node.contents=buffer.slice(offset,offset+length);node.usedBytes=length;return length}else if(position+length<=node.usedBytes){node.contents.set(buffer.subarray(offset,offset+length),position);return length}}MEMFS.expandFileStorage(node,position+length);if(node.contents.subarray&&buffer.subarray){node.contents.set(buffer.subarray(offset,offset+length),position)}else{for(var i=0;i<length;i++){node.contents[position+i]=buffer[offset+i]}}node.usedBytes=Math.max(node.usedBytes,position+length);return length},llseek(stream,offset,whence){var position=offset;if(whence===1){position+=stream.position}else if(whence===2){if(FS.isFile(stream.node.mode)){position+=stream.node.usedBytes}}if(position<0){throw new FS.ErrnoError(28)}return position},mmap(stream,length,position,prot,flags){if(!FS.isFile(stream.node.mode)){throw new FS.ErrnoError(43)}var ptr;var allocated;var contents=stream.node.contents;if(!(flags&2)&&contents&&contents.buffer===(growMemViews(),HEAP8).buffer){allocated=false;ptr=contents.byteOffset}else{allocated=true;ptr=mmapAlloc(length);if(!ptr){throw new FS.ErrnoError(48)}if(contents){if(position>0||position+length<contents.length){if(contents.subarray){contents=contents.subarray(position,position+length)}else{contents=Array.prototype.slice.call(contents,position,position+length)}}(growMemViews(),HEAP8).set(contents,ptr)}}return{ptr,allocated}},msync(stream,buffer,offset,length,mmapFlags){MEMFS.stream_ops.write(stream,buffer,0,length,offset,false);return 0}}};var FS_modeStringToFlags=str=>{var flagModes={r:0,"r+":2,w:512|64|1,"w+":512|64|2,a:1024|64|1,"a+":1024|64|2};var flags=flagModes[str];if(typeof flags=="undefined"){throw new Error(`Unknown file open mode: ${str}`)}return flags};var FS_getMode=(canRead,canWrite)=>{var mode=0;if(canRead)mode|=292|73;if(canWrite)mode|=146;return mode};var asyncLoad=async url=>{var arrayBuffer=await readAsync(url);return new Uint8Array(arrayBuffer)};var FS_createDataFile=(...args)=>FS.createDataFile(...args);var getUniqueRunDependency=id=>id;var preloadPlugins=[];var FS_handledByPreloadPlugin=async(byteArray,fullname)=>{if(typeof Browser!="undefined")Browser.init();for(var plugin of preloadPlugins){if(plugin["canHandle"](fullname)){return plugin["handle"](byteArray,fullname)}}return byteArray};var FS_preloadFile=async(parent,name,url,canRead,canWrite,dontCreateFile,canOwn,preFinish)=>{var fullname=name?PATH_FS.resolve(PATH.join2(parent,name)):parent;var dep=getUniqueRunDependency(`cp ${fullname}`);addRunDependency(dep);try{var byteArray=url;if(typeof url=="string"){byteArray=await asyncLoad(url)}byteArray=await FS_handledByPreloadPlugin(byteArray,fullname);preFinish?.();if(!dontCreateFile){FS_createDataFile(parent,name,byteArray,canRead,canWrite,canOwn)}}finally{removeRunDependency(dep)}};var FS_createPreloadedFile=(parent,name,url,canRead,canWrite,onload,onerror,dontCreateFile,canOwn,preFinish)=>{FS_preloadFile(parent,name,url,canRead,canWrite,dontCreateFile,canOwn,preFinish).then(onload).catch(onerror)};var FS={root:null,mounts:[],devices:{},streams:[],nextInode:1,nameTable:null,currentPath:"/",initialized:false,ignorePermissions:true,filesystems:null,syncFSRequests:0,readFiles:{},ErrnoError:class{name="ErrnoError";constructor(errno){this.errno=errno}},FSStream:class{shared={};get object(){return this.node}set object(val){this.node=val}get isRead(){return(this.flags&2097155)!==1}get isWrite(){return(this.flags&2097155)!==0}get isAppend(){return this.flags&1024}get flags(){return this.shared.flags}set flags(val){this.shared.flags=val}get position(){return this.shared.position}set position(val){this.shared.position=val}},FSNode:class{node_ops={};stream_ops={};readMode=292|73;writeMode=146;mounted=null;constructor(parent,name,mode,rdev){if(!parent){parent=this}this.parent=parent;this.mount=parent.mount;this.id=FS.nextInode++;this.name=name;this.mode=mode;this.rdev=rdev;this.atime=this.mtime=this.ctime=Date.now()}get read(){return(this.mode&this.readMode)===this.readMode}set read(val){val?this.mode|=this.readMode:this.mode&=~this.readMode}get write(){return(this.mode&this.writeMode)===this.writeMode}set write(val){val?this.mode|=this.writeMode:this.mode&=~this.writeMode}get isFolder(){return FS.isDir(this.mode)}get isDevice(){return FS.isChrdev(this.mode)}},lookupPath(path,opts={}){if(!path){throw new FS.ErrnoError(44)}opts.follow_mount??=true;if(!PATH.isAbs(path)){path=FS.cwd()+"/"+path}linkloop:for(var nlinks=0;nlinks<40;nlinks++){var parts=path.split("/").filter(p=>!!p);var current=FS.root;var current_path="/";for(var i=0;i<parts.length;i++){var islast=i===parts.length-1;if(islast&&opts.parent){break}if(parts[i]==="."){continue}if(parts[i]===".."){current_path=PATH.dirname(current_path);if(FS.isRoot(current)){path=current_path+"/"+parts.slice(i+1).join("/");nlinks--;continue linkloop}else{current=current.parent}continue}current_path=PATH.join2(current_path,parts[i]);try{current=FS.lookupNode(current,parts[i])}catch(e){if(e?.errno===44&&islast&&opts.noent_okay){return{path:current_path}}throw e}if(FS.isMountpoint(current)&&(!islast||opts.follow_mount)){current=current.mounted.root}if(FS.isLink(current.mode)&&(!islast||opts.follow)){if(!current.node_ops.readlink){throw new FS.ErrnoError(52)}var link=current.node_ops.readlink(current);if(!PATH.isAbs(link)){link=PATH.dirname(current_path)+"/"+link}path=link+"/"+parts.slice(i+1).join("/");continue linkloop}}return{path:current_path,node:current}}throw new FS.ErrnoError(32)},getPath(node){var path;while(true){if(FS.isRoot(node)){var mount=node.mount.mountpoint;if(!path)return mount;return mount[mount.length-1]!=="/"?`${mount}/${path}`:mount+path}path=path?`${node.name}/${path}`:node.name;node=node.parent}},hashName(parentid,name){var hash=0;for(var i=0;i<name.length;i++){hash=(hash<<5)-hash+name.charCodeAt(i)|0}return(parentid+hash>>>0)%FS.nameTable.length},hashAddNode(node){var hash=FS.hashName(node.parent.id,node.name);node.name_next=FS.nameTable[hash];FS.nameTable[hash]=node},hashRemoveNode(node){var hash=FS.hashName(node.parent.id,node.name);if(FS.nameTable[hash]===node){FS.nameTable[hash]=node.name_next}else{var current=FS.nameTable[hash];while(current){if(current.name_next===node){current.name_next=node.name_next;break}current=current.name_next}}},lookupNode(parent,name){var errCode=FS.mayLookup(parent);if(errCode){throw new FS.ErrnoError(errCode)}var hash=FS.hashName(parent.id,name);for(var node=FS.nameTable[hash];node;node=node.name_next){var nodeName=node.name;if(node.parent.id===parent.id&&nodeName===name){return node}}return FS.lookup(parent,name)},createNode(parent,name,mode,rdev){var node=new FS.FSNode(parent,name,mode,rdev);FS.hashAddNode(node);return node},destroyNode(node){FS.hashRemoveNode(node)},isRoot(node){return node===node.parent},isMountpoint(node){return!!node.mounted},isFile(mode){return(mode&61440)===32768},isDir(mode){return(mode&61440)===16384},isLink(mode){return(mode&61440)===40960},isChrdev(mode){return(mode&61440)===8192},isBlkdev(mode){return(mode&61440)===24576},isFIFO(mode){return(mode&61440)===4096},isSocket(mode){return(mode&49152)===49152},flagsToPermissionString(flag){var perms=["r","w","rw"][flag&3];if(flag&512){perms+="w"}return perms},nodePermissions(node,perms){if(FS.ignorePermissions){return 0}if(perms.includes("r")&&!(node.mode&292)){return 2}else if(perms.includes("w")&&!(node.mode&146)){return 2}else if(perms.includes("x")&&!(node.mode&73)){return 2}return 0},mayLookup(dir){if(!FS.isDir(dir.mode))return 54;var errCode=FS.nodePermissions(dir,"x");if(errCode)return errCode;if(!dir.node_ops.lookup)return 2;return 0},mayCreate(dir,name){if(!FS.isDir(dir.mode)){return 54}try{var node=FS.lookupNode(dir,name);return 20}catch(e){}return FS.nodePermissions(dir,"wx")},mayDelete(dir,name,isdir){var node;try{node=FS.lookupNode(dir,name)}catch(e){return e.errno}var errCode=FS.nodePermissions(dir,"wx");if(errCode){return errCode}if(isdir){if(!FS.isDir(node.mode)){return 54}if(FS.isRoot(node)||FS.getPath(node)===FS.cwd()){return 10}}else{if(FS.isDir(node.mode)){return 31}}return 0},mayOpen(node,flags){if(!node){return 44}if(FS.isLink(node.mode)){return 32}else if(FS.isDir(node.mode)){if(FS.flagsToPermissionString(flags)!=="r"||flags&(512|64)){return 31}}return FS.nodePermissions(node,FS.flagsToPermissionString(flags))},checkOpExists(op,err){if(!op){throw new FS.ErrnoError(err)}return op},MAX_OPEN_FDS:4096,nextfd(){for(var fd=0;fd<=FS.MAX_OPEN_FDS;fd++){if(!FS.streams[fd]){return fd}}throw new FS.ErrnoError(33)},getStreamChecked(fd){var stream=FS.getStream(fd);if(!stream){throw new FS.ErrnoError(8)}return stream},getStream:fd=>FS.streams[fd],createStream(stream,fd=-1){stream=Object.assign(new FS.FSStream,stream);if(fd==-1){fd=FS.nextfd()}stream.fd=fd;FS.streams[fd]=stream;return stream},closeStream(fd){FS.streams[fd]=null},dupStream(origStream,fd=-1){var stream=FS.createStream(origStream,fd);stream.stream_ops?.dup?.(stream);return stream},doSetAttr(stream,node,attr){var setattr=stream?.stream_ops.setattr;var arg=setattr?stream:node;setattr??=node.node_ops.setattr;FS.checkOpExists(setattr,63);setattr(arg,attr)},chrdev_stream_ops:{open(stream){var device=FS.getDevice(stream.node.rdev);stream.stream_ops=device.stream_ops;stream.stream_ops.open?.(stream)},llseek(){throw new FS.ErrnoError(70)}},major:dev=>dev>>8,minor:dev=>dev&255,makedev:(ma,mi)=>ma<<8|mi,registerDevice(dev,ops){FS.devices[dev]={stream_ops:ops}},getDevice:dev=>FS.devices[dev],getMounts(mount){var mounts=[];var check=[mount];while(check.length){var m=check.pop();mounts.push(m);check.push(...m.mounts)}return mounts},syncfs(populate,callback){if(typeof populate=="function"){callback=populate;populate=false}FS.syncFSRequests++;if(FS.syncFSRequests>1){err(`warning: ${FS.syncFSRequests} FS.syncfs operations in flight at once, probably just doing extra work`)}var mounts=FS.getMounts(FS.root.mount);var completed=0;function doCallback(errCode){FS.syncFSRequests--;return callback(errCode)}function done(errCode){if(errCode){if(!done.errored){done.errored=true;return doCallback(errCode)}return}if(++completed>=mounts.length){doCallback(null)}}for(var mount of mounts){if(mount.type.syncfs){mount.type.syncfs(mount,populate,done)}else{done(null)}}},mount(type,opts,mountpoint){var root=mountpoint==="/";var pseudo=!mountpoint;var node;if(root&&FS.root){throw new FS.ErrnoError(10)}else if(!root&&!pseudo){var lookup=FS.lookupPath(mountpoint,{follow_mount:false});mountpoint=lookup.path;node=lookup.node;if(FS.isMountpoint(node)){throw new FS.ErrnoError(10)}if(!FS.isDir(node.mode)){throw new FS.ErrnoError(54)}}var mount={type,opts,mountpoint,mounts:[]};var mountRoot=type.mount(mount);mountRoot.mount=mount;mount.root=mountRoot;if(root){FS.root=mountRoot}else if(node){node.mounted=mount;if(node.mount){node.mount.mounts.push(mount)}}return mountRoot},unmount(mountpoint){var lookup=FS.lookupPath(mountpoint,{follow_mount:false});if(!FS.isMountpoint(lookup.node)){throw new FS.ErrnoError(28)}var node=lookup.node;var mount=node.mounted;var mounts=FS.getMounts(mount);for(var[hash,current]of Object.entries(FS.nameTable)){while(current){var next=current.name_next;if(mounts.includes(current.mount)){FS.destroyNode(current)}current=next}}node.mounted=null;var idx=node.mount.mounts.indexOf(mount);node.mount.mounts.splice(idx,1)},lookup(parent,name){return parent.node_ops.lookup(parent,name)},mknod(path,mode,dev){var lookup=FS.lookupPath(path,{parent:true});var parent=lookup.node;var name=PATH.basename(path);if(!name){throw new FS.ErrnoError(28)}if(name==="."||name===".."){throw new FS.ErrnoError(20)}var errCode=FS.mayCreate(parent,name);if(errCode){throw new FS.ErrnoError(errCode)}if(!parent.node_ops.mknod){throw new FS.ErrnoError(63)}return parent.node_ops.mknod(parent,name,mode,dev)},statfs(path){return FS.statfsNode(FS.lookupPath(path,{follow:true}).node)},statfsStream(stream){return FS.statfsNode(stream.node)},statfsNode(node){var rtn={bsize:4096,frsize:4096,blocks:1e6,bfree:5e5,bavail:5e5,files:FS.nextInode,ffree:FS.nextInode-1,fsid:42,flags:2,namelen:255};if(node.node_ops.statfs){Object.assign(rtn,node.node_ops.statfs(node.mount.opts.root))}return rtn},create(path,mode=438){mode&=4095;mode|=32768;return FS.mknod(path,mode,0)},mkdir(path,mode=511){mode&=511|512;mode|=16384;return FS.mknod(path,mode,0)},mkdirTree(path,mode){var dirs=path.split("/");var d="";for(var dir of dirs){if(!dir)continue;if(d||PATH.isAbs(path))d+="/";d+=dir;try{FS.mkdir(d,mode)}catch(e){if(e.errno!=20)throw e}}},mkdev(path,mode,dev){if(typeof dev=="undefined"){dev=mode;mode=438}mode|=8192;return FS.mknod(path,mode,dev)},symlink(oldpath,newpath){if(!PATH_FS.resolve(oldpath)){throw new FS.ErrnoError(44)}var lookup=FS.lookupPath(newpath,{parent:true});var parent=lookup.node;if(!parent){throw new FS.ErrnoError(44)}var newname=PATH.basename(newpath);var errCode=FS.mayCreate(parent,newname);if(errCode){throw new FS.ErrnoError(errCode)}if(!parent.node_ops.symlink){throw new FS.ErrnoError(63)}return parent.node_ops.symlink(parent,newname,oldpath)},rename(old_path,new_path){var old_dirname=PATH.dirname(old_path);var new_dirname=PATH.dirname(new_path);var old_name=PATH.basename(old_path);var new_name=PATH.basename(new_path);var lookup,old_dir,new_dir;lookup=FS.lookupPath(old_path,{parent:true});old_dir=lookup.node;lookup=FS.lookupPath(new_path,{parent:true});new_dir=lookup.node;if(!old_dir||!new_dir)throw new FS.ErrnoError(44);if(old_dir.mount!==new_dir.mount){throw new FS.ErrnoError(75)}var old_node=FS.lookupNode(old_dir,old_name);var relative=PATH_FS.relative(old_path,new_dirname);if(relative.charAt(0)!=="."){throw new FS.ErrnoError(28)}relative=PATH_FS.relative(new_path,old_dirname);if(relative.charAt(0)!=="."){throw new FS.ErrnoError(55)}var new_node;try{new_node=FS.lookupNode(new_dir,new_name)}catch(e){}if(old_node===new_node){return}var isdir=FS.isDir(old_node.mode);var errCode=FS.mayDelete(old_dir,old_name,isdir);if(errCode){throw new FS.ErrnoError(errCode)}errCode=new_node?FS.mayDelete(new_dir,new_name,isdir):FS.mayCreate(new_dir,new_name);if(errCode){throw new FS.ErrnoError(errCode)}if(!old_dir.node_ops.rename){throw new FS.ErrnoError(63)}if(FS.isMountpoint(old_node)||new_node&&FS.isMountpoint(new_node)){throw new FS.ErrnoError(10)}if(new_dir!==old_dir){errCode=FS.nodePermissions(old_dir,"w");if(errCode){throw new FS.ErrnoError(errCode)}}FS.hashRemoveNode(old_node);try{old_dir.node_ops.rename(old_node,new_dir,new_name);old_node.parent=new_dir}catch(e){throw e}finally{FS.hashAddNode(old_node)}},rmdir(path){var lookup=FS.lookupPath(path,{parent:true});var parent=lookup.node;var name=PATH.basename(path);var node=FS.lookupNode(parent,name);var errCode=FS.mayDelete(parent,name,true);if(errCode){throw new FS.ErrnoError(errCode)}if(!parent.node_ops.rmdir){throw new FS.ErrnoError(63)}if(FS.isMountpoint(node)){throw new FS.ErrnoError(10)}parent.node_ops.rmdir(parent,name);FS.destroyNode(node)},readdir(path){var lookup=FS.lookupPath(path,{follow:true});var node=lookup.node;var readdir=FS.checkOpExists(node.node_ops.readdir,54);return readdir(node)},unlink(path){var lookup=FS.lookupPath(path,{parent:true});var parent=lookup.node;if(!parent){throw new FS.ErrnoError(44)}var name=PATH.basename(path);var node=FS.lookupNode(parent,name);var errCode=FS.mayDelete(parent,name,false);if(errCode){throw new FS.ErrnoError(errCode)}if(!parent.node_ops.unlink){throw new FS.ErrnoError(63)}if(FS.isMountpoint(node)){throw new FS.ErrnoError(10)}parent.node_ops.unlink(parent,name);FS.destroyNode(node)},readlink(path){var lookup=FS.lookupPath(path);var link=lookup.node;if(!link){throw new FS.ErrnoError(44)}if(!link.node_ops.readlink){throw new FS.ErrnoError(28)}return link.node_ops.readlink(link)},stat(path,dontFollow){var lookup=FS.lookupPath(path,{follow:!dontFollow});var node=lookup.node;var getattr=FS.checkOpExists(node.node_ops.getattr,63);return getattr(node)},fstat(fd){var stream=FS.getStreamChecked(fd);var node=stream.node;var getattr=stream.stream_ops.getattr;var arg=getattr?stream:node;getattr??=node.node_ops.getattr;FS.checkOpExists(getattr,63);return getattr(arg)},lstat(path){return FS.stat(path,true)},doChmod(stream,node,mode,dontFollow){FS.doSetAttr(stream,node,{mode:mode&4095|node.mode&~4095,ctime:Date.now(),dontFollow})},chmod(path,mode,dontFollow){var node;if(typeof path=="string"){var lookup=FS.lookupPath(path,{follow:!dontFollow});node=lookup.node}else{node=path}FS.doChmod(null,node,mode,dontFollow)},lchmod(path,mode){FS.chmod(path,mode,true)},fchmod(fd,mode){var stream=FS.getStreamChecked(fd);FS.doChmod(stream,stream.node,mode,false)},doChown(stream,node,dontFollow){FS.doSetAttr(stream,node,{timestamp:Date.now(),dontFollow})},chown(path,uid,gid,dontFollow){var node;if(typeof path=="string"){var lookup=FS.lookupPath(path,{follow:!dontFollow});node=lookup.node}else{node=path}FS.doChown(null,node,dontFollow)},lchown(path,uid,gid){FS.chown(path,uid,gid,true)},fchown(fd,uid,gid){var stream=FS.getStreamChecked(fd);FS.doChown(stream,stream.node,false)},doTruncate(stream,node,len){if(FS.isDir(node.mode)){throw new FS.ErrnoError(31)}if(!FS.isFile(node.mode)){throw new FS.ErrnoError(28)}var errCode=FS.nodePermissions(node,"w");if(errCode){throw new FS.ErrnoError(errCode)}FS.doSetAttr(stream,node,{size:len,timestamp:Date.now()})},truncate(path,len){if(len<0){throw new FS.ErrnoError(28)}var node;if(typeof path=="string"){var lookup=FS.lookupPath(path,{follow:true});node=lookup.node}else{node=path}FS.doTruncate(null,node,len)},ftruncate(fd,len){var stream=FS.getStreamChecked(fd);if(len<0||(stream.flags&2097155)===0){throw new FS.ErrnoError(28)}FS.doTruncate(stream,stream.node,len)},utime(path,atime,mtime){var lookup=FS.lookupPath(path,{follow:true});var node=lookup.node;var setattr=FS.checkOpExists(node.node_ops.setattr,63);setattr(node,{atime,mtime})},open(path,flags,mode=438){if(path===""){throw new FS.ErrnoError(44)}flags=typeof flags=="string"?FS_modeStringToFlags(flags):flags;if(flags&64){mode=mode&4095|32768}else{mode=0}var node;var isDirPath;if(typeof path=="object"){node=path}else{isDirPath=path.endsWith("/");var lookup=FS.lookupPath(path,{follow:!(flags&131072),noent_okay:true});node=lookup.node;path=lookup.path}var created=false;if(flags&64){if(node){if(flags&128){throw new FS.ErrnoError(20)}}else if(isDirPath){throw new FS.ErrnoError(31)}else{node=FS.mknod(path,mode|511,0);created=true}}if(!node){throw new FS.ErrnoError(44)}if(FS.isChrdev(node.mode)){flags&=~512}if(flags&65536&&!FS.isDir(node.mode)){throw new FS.ErrnoError(54)}if(!created){var errCode=FS.mayOpen(node,flags);if(errCode){throw new FS.ErrnoError(errCode)}}if(flags&512&&!created){FS.truncate(node,0)}flags&=~(128|512|131072);var stream=FS.createStream({node,path:FS.getPath(node),flags,seekable:true,position:0,stream_ops:node.stream_ops,ungotten:[],error:false});if(stream.stream_ops.open){stream.stream_ops.open(stream)}if(created){FS.chmod(node,mode&511)}if(Module["logReadFiles"]&&!(flags&1)){if(!(path in FS.readFiles)){FS.readFiles[path]=1}}return stream},close(stream){if(FS.isClosed(stream)){throw new FS.ErrnoError(8)}if(stream.getdents)stream.getdents=null;try{if(stream.stream_ops.close){stream.stream_ops.close(stream)}}catch(e){throw e}finally{FS.closeStream(stream.fd)}stream.fd=null},isClosed(stream){return stream.fd===null},llseek(stream,offset,whence){if(FS.isClosed(stream)){throw new FS.ErrnoError(8)}if(!stream.seekable||!stream.stream_ops.llseek){throw new FS.ErrnoError(70)}if(whence!=0&&whence!=1&&whence!=2){throw new FS.ErrnoError(28)}stream.position=stream.stream_ops.llseek(stream,offset,whence);stream.ungotten=[];return stream.position},read(stream,buffer,offset,length,position){if(length<0||position<0){throw new FS.ErrnoError(28)}if(FS.isClosed(stream)){throw new FS.ErrnoError(8)}if((stream.flags&2097155)===1){throw new FS.ErrnoError(8)}if(FS.isDir(stream.node.mode)){throw new FS.ErrnoError(31)}if(!stream.stream_ops.read){throw new FS.ErrnoError(28)}var seeking=typeof position!="undefined";if(!seeking){position=stream.position}else if(!stream.seekable){throw new FS.ErrnoError(70)}var bytesRead=stream.stream_ops.read(stream,buffer,offset,length,position);if(!seeking)stream.position+=bytesRead;return bytesRead},write(stream,buffer,offset,length,position,canOwn){if(length<0||position<0){throw new FS.ErrnoError(28)}if(FS.isClosed(stream)){throw new FS.ErrnoError(8)}if((stream.flags&2097155)===0){throw new FS.ErrnoError(8)}if(FS.isDir(stream.node.mode)){throw new FS.ErrnoError(31)}if(!stream.stream_ops.write){throw new FS.ErrnoError(28)}if(stream.seekable&&stream.flags&1024){FS.llseek(stream,0,2)}var seeking=typeof position!="undefined";if(!seeking){position=stream.position}else if(!stream.seekable){throw new FS.ErrnoError(70)}var bytesWritten=stream.stream_ops.write(stream,buffer,offset,length,position,canOwn);if(!seeking)stream.position+=bytesWritten;return bytesWritten},mmap(stream,length,position,prot,flags){if((prot&2)!==0&&(flags&2)===0&&(stream.flags&2097155)!==2){throw new FS.ErrnoError(2)}if((stream.flags&2097155)===1){throw new FS.ErrnoError(2)}if(!stream.stream_ops.mmap){throw new FS.ErrnoError(43)}if(!length){throw new FS.ErrnoError(28)}return stream.stream_ops.mmap(stream,length,position,prot,flags)},msync(stream,buffer,offset,length,mmapFlags){if(!stream.stream_ops.msync){return 0}return stream.stream_ops.msync(stream,buffer,offset,length,mmapFlags)},ioctl(stream,cmd,arg){if(!stream.stream_ops.ioctl){throw new FS.ErrnoError(59)}return stream.stream_ops.ioctl(stream,cmd,arg)},readFile(path,opts={}){opts.flags=opts.flags||0;opts.encoding=opts.encoding||"binary";if(opts.encoding!=="utf8"&&opts.encoding!=="binary"){abort(`Invalid encoding type "${opts.encoding}"`)}var stream=FS.open(path,opts.flags);var stat=FS.stat(path);var length=stat.size;var buf=new Uint8Array(length);FS.read(stream,buf,0,length,0);if(opts.encoding==="utf8"){buf=UTF8ArrayToString(buf)}FS.close(stream);return buf},writeFile(path,data,opts={}){opts.flags=opts.flags||577;var stream=FS.open(path,opts.flags,opts.mode);if(typeof data=="string"){data=new Uint8Array(intArrayFromString(data,true))}if(ArrayBuffer.isView(data)){FS.write(stream,data,0,data.byteLength,undefined,opts.canOwn)}else{abort("Unsupported data type")}FS.close(stream)},cwd:()=>FS.currentPath,chdir(path){var lookup=FS.lookupPath(path,{follow:true});if(lookup.node===null){throw new FS.ErrnoError(44)}if(!FS.isDir(lookup.node.mode)){throw new FS.ErrnoError(54)}var errCode=FS.nodePermissions(lookup.node,"x");if(errCode){throw new FS.ErrnoError(errCode)}FS.currentPath=lookup.path},createDefaultDirectories(){FS.mkdir("/tmp");FS.mkdir("/home");FS.mkdir("/home/web_user")},createDefaultDevices(){FS.mkdir("/dev");FS.registerDevice(FS.makedev(1,3),{read:()=>0,write:(stream,buffer,offset,length,pos)=>length,llseek:()=>0});FS.mkdev("/dev/null",FS.makedev(1,3));TTY.register(FS.makedev(5,0),TTY.default_tty_ops);TTY.register(FS.makedev(6,0),TTY.default_tty1_ops);FS.mkdev("/dev/tty",FS.makedev(5,0));FS.mkdev("/dev/tty1",FS.makedev(6,0));var randomBuffer=new Uint8Array(1024),randomLeft=0;var randomByte=()=>{if(randomLeft===0){randomFill(randomBuffer);randomLeft=randomBuffer.byteLength}return randomBuffer[--randomLeft]};FS.createDevice("/dev","random",randomByte);FS.createDevice("/dev","urandom",randomByte);FS.mkdir("/dev/shm");FS.mkdir("/dev/shm/tmp")},createSpecialDirectories(){FS.mkdir("/proc");var proc_self=FS.mkdir("/proc/self");FS.mkdir("/proc/self/fd");FS.mount({mount(){var node=FS.createNode(proc_self,"fd",16895,73);node.stream_ops={llseek:MEMFS.stream_ops.llseek};node.node_ops={lookup(parent,name){var fd=+name;var stream=FS.getStreamChecked(fd);var ret={parent:null,mount:{mountpoint:"fake"},node_ops:{readlink:()=>stream.path},id:fd+1};ret.parent=ret;return ret},readdir(){return Array.from(FS.streams.entries()).filter(([k,v])=>v).map(([k,v])=>k.toString())}};return node}},{},"/proc/self/fd")},createStandardStreams(input,output,error){if(input){FS.createDevice("/dev","stdin",input)}else{FS.symlink("/dev/tty","/dev/stdin")}if(output){FS.createDevice("/dev","stdout",null,output)}else{FS.symlink("/dev/tty","/dev/stdout")}if(error){FS.createDevice("/dev","stderr",null,error)}else{FS.symlink("/dev/tty1","/dev/stderr")}var stdin=FS.open("/dev/stdin",0);var stdout=FS.open("/dev/stdout",1);var stderr=FS.open("/dev/stderr",1)},staticInit(){FS.nameTable=new Array(4096);FS.mount(MEMFS,{},"/");FS.createDefaultDirectories();FS.createDefaultDevices();FS.createSpecialDirectories();FS.filesystems={MEMFS}},init(input,output,error){FS.initialized=true;input??=Module["stdin"];output??=Module["stdout"];error??=Module["stderr"];FS.createStandardStreams(input,output,error)},quit(){FS.initialized=false;for(var stream of FS.streams){if(stream){FS.close(stream)}}},findObject(path,dontResolveLastLink){var ret=FS.analyzePath(path,dontResolveLastLink);if(!ret.exists){return null}return ret.object},analyzePath(path,dontResolveLastLink){try{var lookup=FS.lookupPath(path,{follow:!dontResolveLastLink});path=lookup.path}catch(e){}var ret={isRoot:false,exists:false,error:0,name:null,path:null,object:null,parentExists:false,parentPath:null,parentObject:null};try{var lookup=FS.lookupPath(path,{parent:true});ret.parentExists=true;ret.parentPath=lookup.path;ret.parentObject=lookup.node;ret.name=PATH.basename(path);lookup=FS.lookupPath(path,{follow:!dontResolveLastLink});ret.exists=true;ret.path=lookup.path;ret.object=lookup.node;ret.name=lookup.node.name;ret.isRoot=lookup.path==="/"}catch(e){ret.error=e.errno}return ret},createPath(parent,path,canRead,canWrite){parent=typeof parent=="string"?parent:FS.getPath(parent);var parts=path.split("/").reverse();while(parts.length){var part=parts.pop();if(!part)continue;var current=PATH.join2(parent,part);try{FS.mkdir(current)}catch(e){if(e.errno!=20)throw e}parent=current}return current},createFile(parent,name,properties,canRead,canWrite){var path=PATH.join2(typeof parent=="string"?parent:FS.getPath(parent),name);var mode=FS_getMode(canRead,canWrite);return FS.create(path,mode)},createDataFile(parent,name,data,canRead,canWrite,canOwn){var path=name;if(parent){parent=typeof parent=="string"?parent:FS.getPath(parent);path=name?PATH.join2(parent,name):parent}var mode=FS_getMode(canRead,canWrite);var node=FS.create(path,mode);if(data){if(typeof data=="string"){var arr=new Array(data.length);for(var i=0,len=data.length;i<len;++i)arr[i]=data.charCodeAt(i);data=arr}FS.chmod(node,mode|146);var stream=FS.open(node,577);FS.write(stream,data,0,data.length,0,canOwn);FS.close(stream);FS.chmod(node,mode)}},createDevice(parent,name,input,output){var path=PATH.join2(typeof parent=="string"?parent:FS.getPath(parent),name);var mode=FS_getMode(!!input,!!output);FS.createDevice.major??=64;var dev=FS.makedev(FS.createDevice.major++,0);FS.registerDevice(dev,{open(stream){stream.seekable=false},close(stream){if(output?.buffer?.length){output(10)}},read(stream,buffer,offset,length,pos){var bytesRead=0;for(var i=0;i<length;i++){var result;try{result=input()}catch(e){throw new FS.ErrnoError(29)}if(result===undefined&&bytesRead===0){throw new FS.ErrnoError(6)}if(result===null||result===undefined)break;bytesRead++;buffer[offset+i]=result}if(bytesRead){stream.node.atime=Date.now()}return bytesRead},write(stream,buffer,offset,length,pos){for(var i=0;i<length;i++){try{output(buffer[offset+i])}catch(e){throw new FS.ErrnoError(29)}}if(length){stream.node.mtime=stream.node.ctime=Date.now()}return i}});return FS.mkdev(path,mode,dev)},forceLoadFile(obj){if(obj.isDevice||obj.isFolder||obj.link||obj.contents)return true;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.")}else{try{obj.contents=readBinary(obj.url)}catch(e){throw new FS.ErrnoError(29)}}},createLazyFile(parent,name,url,canRead,canWrite){class LazyUint8Array{lengthKnown=false;chunks=[];get(idx){if(idx>this.length-1||idx<0){return undefined}var chunkOffset=idx%this.chunkSize;var chunkNum=idx/this.chunkSize|0;return this.getter(chunkNum)[chunkOffset]}setDataGetter(getter){this.getter=getter}cacheLength(){var xhr=new XMLHttpRequest;xhr.open("HEAD",url,false);xhr.send(null);if(!(xhr.status>=200&&xhr.status<300||xhr.status===304))abort("Couldn\'t load "+url+". Status: "+xhr.status);var datalength=Number(xhr.getResponseHeader("Content-length"));var header;var hasByteServing=(header=xhr.getResponseHeader("Accept-Ranges"))&&header==="bytes";var usesGzip=(header=xhr.getResponseHeader("Content-Encoding"))&&header==="gzip";var chunkSize=1024*1024;if(!hasByteServing)chunkSize=datalength;var doXHR=(from,to)=>{if(from>to)abort("invalid range ("+from+", "+to+") or no bytes requested!");if(to>datalength-1)abort("only "+datalength+" bytes available! programmer error!");var xhr=new XMLHttpRequest;xhr.open("GET",url,false);if(datalength!==chunkSize)xhr.setRequestHeader("Range","bytes="+from+"-"+to);xhr.responseType="arraybuffer";if(xhr.overrideMimeType){xhr.overrideMimeType("text/plain; charset=x-user-defined")}xhr.send(null);if(!(xhr.status>=200&&xhr.status<300||xhr.status===304))abort("Couldn\'t load "+url+". Status: "+xhr.status);if(xhr.response!==undefined){return new Uint8Array(xhr.response||[])}return intArrayFromString(xhr.responseText||"",true)};var lazyArray=this;lazyArray.setDataGetter(chunkNum=>{var start=chunkNum*chunkSize;var end=(chunkNum+1)*chunkSize-1;end=Math.min(end,datalength-1);if(typeof lazyArray.chunks[chunkNum]=="undefined"){lazyArray.chunks[chunkNum]=doXHR(start,end)}if(typeof lazyArray.chunks[chunkNum]=="undefined")abort("doXHR failed!");return lazyArray.chunks[chunkNum]});if(usesGzip||!datalength){chunkSize=datalength=1;datalength=this.getter(0).length;chunkSize=datalength;out("LazyFiles on gzip forces download of the whole file when length is accessed")}this._length=datalength;this._chunkSize=chunkSize;this.lengthKnown=true}get length(){if(!this.lengthKnown){this.cacheLength()}return this._length}get chunkSize(){if(!this.lengthKnown){this.cacheLength()}return this._chunkSize}}if(globalThis.XMLHttpRequest){if(!ENVIRONMENT_IS_WORKER)abort("Cannot do synchronous binary XHRs outside webworkers in modern browsers. Use --embed-file or --preload-file in emcc");var lazyArray=new LazyUint8Array;var properties={isDevice:false,contents:lazyArray}}else{var properties={isDevice:false,url}}var node=FS.createFile(parent,name,properties,canRead,canWrite);if(properties.contents){node.contents=properties.contents}else if(properties.url){node.contents=null;node.url=properties.url}Object.defineProperties(node,{usedBytes:{get:function(){return this.contents.length}}});var stream_ops={};for(const[key,fn]of Object.entries(node.stream_ops)){stream_ops[key]=(...args)=>{FS.forceLoadFile(node);return fn(...args)}}function writeChunks(stream,buffer,offset,length,position){var contents=stream.node.contents;if(position>=contents.length)return 0;var size=Math.min(contents.length-position,length);if(contents.slice){for(var i=0;i<size;i++){buffer[offset+i]=contents[position+i]}}else{for(var i=0;i<size;i++){buffer[offset+i]=contents.get(position+i)}}return size}stream_ops.read=(stream,buffer,offset,length,position)=>{FS.forceLoadFile(node);return writeChunks(stream,buffer,offset,length,position)};stream_ops.mmap=(stream,length,position,prot,flags)=>{FS.forceLoadFile(node);var ptr=mmapAlloc(length);if(!ptr){throw new FS.ErrnoError(48)}writeChunks(stream,(growMemViews(),HEAP8),ptr,length,position);return{ptr,allocated:true}};node.stream_ops=stream_ops;return node}};var UTF8ToString=(ptr,maxBytesToRead,ignoreNul)=>ptr?UTF8ArrayToString((growMemViews(),HEAPU8),ptr,maxBytesToRead,ignoreNul):"";var SYSCALLS={DEFAULT_POLLMASK:5,calculateAt(dirfd,path,allowEmpty){if(PATH.isAbs(path)){return path}var dir;if(dirfd===-100){dir=FS.cwd()}else{var dirstream=SYSCALLS.getStreamFromFD(dirfd);dir=dirstream.path}if(path.length==0){if(!allowEmpty){throw new FS.ErrnoError(44)}return dir}return dir+"/"+path},writeStat(buf,stat){(growMemViews(),HEAPU32)[buf/4]=stat.dev;(growMemViews(),HEAPU32)[(buf+4)/4]=stat.mode;(growMemViews(),HEAPU64)[(buf+8)/8]=BigInt(stat.nlink);(growMemViews(),HEAPU32)[(buf+16)/4]=stat.uid;(growMemViews(),HEAPU32)[(buf+20)/4]=stat.gid;(growMemViews(),HEAPU32)[(buf+24)/4]=stat.rdev;(growMemViews(),HEAP64)[(buf+32)/8]=BigInt(stat.size);(growMemViews(),HEAP32)[(buf+40)/4]=4096;(growMemViews(),HEAP32)[(buf+44)/4]=stat.blocks;var atime=stat.atime.getTime();var mtime=stat.mtime.getTime();var ctime=stat.ctime.getTime();(growMemViews(),HEAP64)[(buf+48)/8]=BigInt(Math.floor(atime/1e3));(growMemViews(),HEAPU64)[(buf+56)/8]=BigInt(atime%1e3*1e3*1e3);(growMemViews(),HEAP64)[(buf+64)/8]=BigInt(Math.floor(mtime/1e3));(growMemViews(),HEAPU64)[(buf+72)/8]=BigInt(mtime%1e3*1e3*1e3);(growMemViews(),HEAP64)[(buf+80)/8]=BigInt(Math.floor(ctime/1e3));(growMemViews(),HEAPU64)[(buf+88)/8]=BigInt(ctime%1e3*1e3*1e3);(growMemViews(),HEAP64)[(buf+96)/8]=BigInt(stat.ino);return 0},writeStatFs(buf,stats){(growMemViews(),HEAPU32)[(buf+8)/4]=stats.bsize;(growMemViews(),HEAPU32)[(buf+72)/4]=stats.bsize;(growMemViews(),HEAP64)[(buf+16)/8]=BigInt(stats.blocks);(growMemViews(),HEAP64)[(buf+24)/8]=BigInt(stats.bfree);(growMemViews(),HEAP64)[(buf+32)/8]=BigInt(stats.bavail);(growMemViews(),HEAP64)[(buf+40)/8]=BigInt(stats.files);(growMemViews(),HEAP64)[(buf+48)/8]=BigInt(stats.ffree);(growMemViews(),HEAPU32)[(buf+56)/4]=stats.fsid;(growMemViews(),HEAPU32)[(buf+80)/4]=stats.flags;(growMemViews(),HEAPU32)[(buf+64)/4]=stats.namelen},doMsync(addr,stream,len,flags,offset){if(!FS.isFile(stream.node.mode)){throw new FS.ErrnoError(43)}if(flags&2){return 0}var buffer=(growMemViews(),HEAPU8).slice(addr,addr+len);FS.msync(stream,buffer,offset,len,flags)},getStreamFromFD(fd){var stream=FS.getStreamChecked(fd);return stream},varargs:undefined,getStr(ptr){var ret=UTF8ToString(ptr);return ret}};function ___syscall_fcntl64(fd,cmd,varargs){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(3,0,1,fd,cmd,varargs);varargs=bigintToI53Checked(varargs);SYSCALLS.varargs=varargs;try{var stream=SYSCALLS.getStreamFromFD(fd);switch(cmd){case 0:{var arg=syscallGetVarargI();if(arg<0){return-28}while(FS.streams[arg]){arg++}var newStream;newStream=FS.dupStream(stream,arg);return newStream.fd}case 1:case 2:return 0;case 3:return stream.flags;case 4:{var arg=syscallGetVarargI();stream.flags|=arg;return 0}case 5:{var arg=syscallGetVarargP();var offset=0;(growMemViews(),HEAP16)[(arg+offset)/2]=2;return 0}case 6:case 7:return 0}return-28}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return-e.errno}}function ___syscall_fstat64(fd,buf){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(4,0,1,fd,buf);buf=bigintToI53Checked(buf);try{return SYSCALLS.writeStat(buf,FS.fstat(fd))}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return-e.errno}}var stringToUTF8=(str,outPtr,maxBytesToWrite)=>stringToUTF8Array(str,(growMemViews(),HEAPU8),outPtr,maxBytesToWrite);function ___syscall_getcwd(buf,size){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(5,0,1,buf,size);buf=bigintToI53Checked(buf);size=bigintToI53Checked(size);try{if(size===0)return-28;var cwd=FS.cwd();var cwdLengthInBytes=lengthBytesUTF8(cwd)+1;if(size<cwdLengthInBytes)return-68;stringToUTF8(cwd,buf,size);return cwdLengthInBytes}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return-e.errno}}function ___syscall_getdents64(fd,dirp,count){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(6,0,1,fd,dirp,count);dirp=bigintToI53Checked(dirp);count=bigintToI53Checked(count);try{var stream=SYSCALLS.getStreamFromFD(fd);stream.getdents||=FS.readdir(stream.path);var struct_size=280;var pos=0;var off=FS.llseek(stream,0,1);var startIdx=Math.floor(off/struct_size);var endIdx=Math.min(stream.getdents.length,startIdx+Math.floor(count/struct_size));for(var idx=startIdx;idx<endIdx;idx++){var id;var type;var name=stream.getdents[idx];if(name==="."){id=stream.node.id;type=4}else if(name===".."){var lookup=FS.lookupPath(stream.path,{parent:true});id=lookup.node.id;type=4}else{var child;try{child=FS.lookupNode(stream.node,name)}catch(e){if(e?.errno===28){continue}throw e}id=child.id;type=FS.isChrdev(child.mode)?2:FS.isDir(child.mode)?4:FS.isLink(child.mode)?10:8}(growMemViews(),HEAP64)[(dirp+pos)/8]=BigInt(id);(growMemViews(),HEAP64)[(dirp+pos+8)/8]=BigInt((idx+1)*struct_size);(growMemViews(),HEAP16)[(dirp+pos+16)/2]=280;(growMemViews(),HEAP8)[dirp+pos+18]=type;stringToUTF8(name,dirp+pos+19,256);pos+=struct_size}FS.llseek(stream,idx*struct_size,0);return pos}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return-e.errno}}function ___syscall_ioctl(fd,op,varargs){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(7,0,1,fd,op,varargs);varargs=bigintToI53Checked(varargs);SYSCALLS.varargs=varargs;try{var stream=SYSCALLS.getStreamFromFD(fd);switch(op){case 21509:{if(!stream.tty)return-59;return 0}case 21505:{if(!stream.tty)return-59;if(stream.tty.ops.ioctl_tcgets){var termios=stream.tty.ops.ioctl_tcgets(stream);var argp=syscallGetVarargP();(growMemViews(),HEAP32)[argp/4]=termios.c_iflag||0;(growMemViews(),HEAP32)[(argp+4)/4]=termios.c_oflag||0;(growMemViews(),HEAP32)[(argp+8)/4]=termios.c_cflag||0;(growMemViews(),HEAP32)[(argp+12)/4]=termios.c_lflag||0;for(var i=0;i<32;i++){(growMemViews(),HEAP8)[argp+i+17]=termios.c_cc[i]||0}return 0}return 0}case 21510:case 21511:case 21512:{if(!stream.tty)return-59;return 0}case 21506:case 21507:case 21508:{if(!stream.tty)return-59;if(stream.tty.ops.ioctl_tcsets){var argp=syscallGetVarargP();var c_iflag=(growMemViews(),HEAP32)[argp/4];var c_oflag=(growMemViews(),HEAP32)[(argp+4)/4];var c_cflag=(growMemViews(),HEAP32)[(argp+8)/4];var c_lflag=(growMemViews(),HEAP32)[(argp+12)/4];var c_cc=[];for(var i=0;i<32;i++){c_cc.push((growMemViews(),HEAP8)[argp+i+17])}return stream.tty.ops.ioctl_tcsets(stream.tty,op,{c_iflag,c_oflag,c_cflag,c_lflag,c_cc})}return 0}case 21519:{if(!stream.tty)return-59;var argp=syscallGetVarargP();(growMemViews(),HEAP32)[argp/4]=0;return 0}case 21520:{if(!stream.tty)return-59;return-28}case 21537:case 21531:{var argp=syscallGetVarargP();return FS.ioctl(stream,op,argp)}case 21523:{if(!stream.tty)return-59;if(stream.tty.ops.ioctl_tiocgwinsz){var winsize=stream.tty.ops.ioctl_tiocgwinsz(stream.tty);var argp=syscallGetVarargP();(growMemViews(),HEAP16)[argp/2]=winsize[0];(growMemViews(),HEAP16)[(argp+2)/2]=winsize[1]}return 0}case 21524:{if(!stream.tty)return-59;return 0}case 21515:{if(!stream.tty)return-59;return 0}default:return-28}}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return-e.errno}}function ___syscall_lstat64(path,buf){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(8,0,1,path,buf);path=bigintToI53Checked(path);buf=bigintToI53Checked(buf);try{path=SYSCALLS.getStr(path);return SYSCALLS.writeStat(buf,FS.lstat(path))}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return-e.errno}}function ___syscall_newfstatat(dirfd,path,buf,flags){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(9,0,1,dirfd,path,buf,flags);path=bigintToI53Checked(path);buf=bigintToI53Checked(buf);try{path=SYSCALLS.getStr(path);var nofollow=flags&256;var allowEmpty=flags&4096;flags=flags&~6400;path=SYSCALLS.calculateAt(dirfd,path,allowEmpty);return SYSCALLS.writeStat(buf,nofollow?FS.lstat(path):FS.stat(path))}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return-e.errno}}function ___syscall_openat(dirfd,path,flags,varargs){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(10,0,1,dirfd,path,flags,varargs);path=bigintToI53Checked(path);varargs=bigintToI53Checked(varargs);SYSCALLS.varargs=varargs;try{path=SYSCALLS.getStr(path);path=SYSCALLS.calculateAt(dirfd,path);var mode=varargs?syscallGetVarargI():0;return FS.open(path,flags,mode).fd}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return-e.errno}}function ___syscall_stat64(path,buf){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(11,0,1,path,buf);path=bigintToI53Checked(path);buf=bigintToI53Checked(buf);try{path=SYSCALLS.getStr(path);return SYSCALLS.writeStat(buf,FS.stat(path))}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return-e.errno}}var __abort_js=()=>abort("");function __emscripten_init_main_thread_js(tb){tb=bigintToI53Checked(tb);__emscripten_thread_init(tb,!ENVIRONMENT_IS_WORKER,1,!ENVIRONMENT_IS_WEB,5242880,false);PThread.threadInitTLS()}var handleException=e=>{if(e instanceof ExitStatus||e=="unwind"){return EXITSTATUS}quit_(1,e)};var maybeExit=()=>{if(!keepRuntimeAlive()){try{if(ENVIRONMENT_IS_PTHREAD){if(_pthread_self())__emscripten_thread_exit(EXITSTATUS);return}_exit(EXITSTATUS)}catch(e){handleException(e)}}};var callUserCallback=func=>{if(ABORT){return}try{func();maybeExit()}catch(e){handleException(e)}};function __emscripten_thread_mailbox_await(pthread_ptr){pthread_ptr=bigintToI53Checked(pthread_ptr);if(Atomics.waitAsync){var wait=Atomics.waitAsync((growMemViews(),HEAP32),pthread_ptr/4,pthread_ptr);wait.value.then(checkMailbox);var waitingAsync=pthread_ptr+228;Atomics.store((growMemViews(),HEAP32),waitingAsync/4,1)}}var checkMailbox=()=>callUserCallback(()=>{var pthread_ptr=_pthread_self();if(pthread_ptr){__emscripten_thread_mailbox_await(pthread_ptr);__emscripten_check_mailbox()}});function __emscripten_notify_mailbox_postmessage(targetThread,currThreadId){targetThread=bigintToI53Checked(targetThread);currThreadId=bigintToI53Checked(currThreadId);if(targetThread==currThreadId){setTimeout(checkMailbox)}else if(ENVIRONMENT_IS_PTHREAD){postMessage({targetThread,cmd:"checkMailbox"})}else{var worker=PThread.pthreads[targetThread];if(!worker){return}worker.postMessage({cmd:"checkMailbox"})}}var proxiedJSCallArgs=[];function __emscripten_receive_on_main_thread_js(funcIndex,emAsmAddr,callingThread,numCallArgs,args){emAsmAddr=bigintToI53Checked(emAsmAddr);callingThread=bigintToI53Checked(callingThread);args=bigintToI53Checked(args);numCallArgs/=2;proxiedJSCallArgs.length=numCallArgs;var b=args/8;for(var i=0;i<numCallArgs;i++){if((growMemViews(),HEAP64)[b+2*i]){proxiedJSCallArgs[i]=(growMemViews(),HEAP64)[b+2*i+1]}else{proxiedJSCallArgs[i]=(growMemViews(),HEAPF64)[b+2*i+1]}}var func=proxiedFunctionTable[funcIndex];PThread.currentProxiedOperationCallerThread=callingThread;var rtn=func(...proxiedJSCallArgs);PThread.currentProxiedOperationCallerThread=0;if(typeof rtn=="bigint"){rtn=bigintToI53Checked(rtn)}return rtn}function __emscripten_thread_cleanup(thread){thread=bigintToI53Checked(thread);if(!ENVIRONMENT_IS_PTHREAD)cleanupThread(thread);else postMessage({cmd:"cleanupThread",thread})}function __emscripten_thread_set_strongref(thread){thread=bigintToI53Checked(thread);if(ENVIRONMENT_IS_NODE){PThread.pthreads[thread].ref()}}var isLeapYear=year=>year%4===0&&(year%100!==0||year%400===0);var MONTH_DAYS_LEAP_CUMULATIVE=[0,31,60,91,121,152,182,213,244,274,305,335];var MONTH_DAYS_REGULAR_CUMULATIVE=[0,31,59,90,120,151,181,212,243,273,304,334];var ydayFromDate=date=>{var leap=isLeapYear(date.getFullYear());var monthDaysCumulative=leap?MONTH_DAYS_LEAP_CUMULATIVE:MONTH_DAYS_REGULAR_CUMULATIVE;var yday=monthDaysCumulative[date.getMonth()]+date.getDate()-1;return yday};function __localtime_js(time,tmPtr){time=bigintToI53Checked(time);tmPtr=bigintToI53Checked(tmPtr);var date=new Date(time*1e3);(growMemViews(),HEAP32)[tmPtr/4]=date.getSeconds();(growMemViews(),HEAP32)[(tmPtr+4)/4]=date.getMinutes();(growMemViews(),HEAP32)[(tmPtr+8)/4]=date.getHours();(growMemViews(),HEAP32)[(tmPtr+12)/4]=date.getDate();(growMemViews(),HEAP32)[(tmPtr+16)/4]=date.getMonth();(growMemViews(),HEAP32)[(tmPtr+20)/4]=date.getFullYear()-1900;(growMemViews(),HEAP32)[(tmPtr+24)/4]=date.getDay();var yday=ydayFromDate(date)|0;(growMemViews(),HEAP32)[(tmPtr+28)/4]=yday;(growMemViews(),HEAP64)[(tmPtr+40)/8]=BigInt(-(date.getTimezoneOffset()*60));var start=new Date(date.getFullYear(),0,1);var summerOffset=new Date(date.getFullYear(),6,1).getTimezoneOffset();var winterOffset=start.getTimezoneOffset();var dst=(summerOffset!=winterOffset&&date.getTimezoneOffset()==Math.min(winterOffset,summerOffset))|0;(growMemViews(),HEAP32)[(tmPtr+32)/4]=dst}function __mmap_js(len,prot,flags,fd,offset,allocated,addr){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(12,0,1,len,prot,flags,fd,offset,allocated,addr);len=bigintToI53Checked(len);offset=bigintToI53Checked(offset);allocated=bigintToI53Checked(allocated);addr=bigintToI53Checked(addr);try{var stream=SYSCALLS.getStreamFromFD(fd);var res=FS.mmap(stream,len,offset,prot,flags);var ptr=res.ptr;(growMemViews(),HEAP32)[allocated/4]=res.allocated;(growMemViews(),HEAPU64)[addr/8]=BigInt(ptr);return 0}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return-e.errno}}function __munmap_js(addr,len,prot,flags,fd,offset){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(13,0,1,addr,len,prot,flags,fd,offset);addr=bigintToI53Checked(addr);len=bigintToI53Checked(len);offset=bigintToI53Checked(offset);try{var stream=SYSCALLS.getStreamFromFD(fd);if(prot&2){SYSCALLS.doMsync(addr,stream,len,flags,offset)}}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return-e.errno}}var __tzset_js=function(timezone,daylight,std_name,dst_name){timezone=bigintToI53Checked(timezone);daylight=bigintToI53Checked(daylight);std_name=bigintToI53Checked(std_name);dst_name=bigintToI53Checked(dst_name);var currentYear=(new Date).getFullYear();var winter=new Date(currentYear,0,1);var summer=new Date(currentYear,6,1);var winterOffset=winter.getTimezoneOffset();var summerOffset=summer.getTimezoneOffset();var stdTimezoneOffset=Math.max(winterOffset,summerOffset);(growMemViews(),HEAPU64)[timezone/8]=BigInt(stdTimezoneOffset*60);(growMemViews(),HEAP32)[daylight/4]=Number(winterOffset!=summerOffset);var extractZone=timezoneOffset=>{var sign=timezoneOffset>=0?"-":"+";var absOffset=Math.abs(timezoneOffset);var hours=String(Math.floor(absOffset/60)).padStart(2,"0");var minutes=String(absOffset%60).padStart(2,"0");return`UTC${sign}${hours}${minutes}`};var winterName=extractZone(winterOffset);var summerName=extractZone(summerOffset);if(summerOffset<winterOffset){stringToUTF8(winterName,std_name,17);stringToUTF8(summerName,dst_name,17)}else{stringToUTF8(winterName,dst_name,17);stringToUTF8(summerName,std_name,17)}};var _emscripten_get_now=()=>performance.timeOrigin+performance.now();var _emscripten_date_now=()=>Date.now();var nowIsMonotonic=1;var checkWasiClock=clock_id=>clock_id>=0&&clock_id<=3;function _clock_time_get(clk_id,ignored_precision,ptime){ignored_precision=bigintToI53Checked(ignored_precision);ptime=bigintToI53Checked(ptime);if(!checkWasiClock(clk_id)){return 28}var now;if(clk_id===0){now=_emscripten_date_now()}else if(nowIsMonotonic){now=_emscripten_get_now()}else{return 52}var nsec=Math.round(now*1e3*1e3);(growMemViews(),HEAP64)[ptime/8]=BigInt(nsec);return 0}var _emscripten_check_blocking_allowed=()=>{};var runtimeKeepalivePush=()=>{runtimeKeepaliveCounter+=1};var _emscripten_exit_with_live_runtime=()=>{runtimeKeepalivePush();throw"unwind"};var jsStackTrace=()=>(new Error).stack.toString();var getCallstack=flags=>{var callstack=jsStackTrace();var lines=callstack.split("\\n");callstack="";var firefoxRe=new RegExp("\\\\s*(.*?)@(.*?):([0-9]+):([0-9]+)");var chromeRe=new RegExp("\\\\s*at (.*?) \\\\((.*):(.*):(.*)\\\\)");for(var line of lines){var symbolName="";var file="";var lineno=0;var column=0;var parts=chromeRe.exec(line);if(parts?.length==5){symbolName=parts[1];file=parts[2];lineno=parts[3];column=parts[4]}else{parts=firefoxRe.exec(line);if(parts?.length>=4){symbolName=parts[1];file=parts[2];lineno=parts[3];column=parts[4]|0}else{callstack+=line+"\\n";continue}}if(symbolName=="_emscripten_log"||symbolName=="_emscripten_get_callstack"){callstack="";continue}if(flags&24){if(flags&64){file=file.substring(file.replace(/\\\\/g,"/").lastIndexOf("/")+1)}callstack+=` at ${symbolName} (${file}:${lineno}:${column})\\n`}}callstack=callstack.replace(/\\s+$/,"");return callstack};function _emscripten_get_callstack(flags,str,maxbytes){str=bigintToI53Checked(str);var callstack=getCallstack(flags);if(!str||maxbytes<=0){return lengthBytesUTF8(callstack)+1}var bytesWrittenExcludingNull=stringToUTF8(callstack,str,maxbytes);return bytesWrittenExcludingNull+1}var getHeapMax=()=>4294967296;var _emscripten_get_heap_max=()=>BigInt(getHeapMax());var _emscripten_has_asyncify=()=>2;var _emscripten_num_logical_cores=()=>ENVIRONMENT_IS_NODE?require("os").cpus().length:navigator["hardwareConcurrency"];var growMemory=size=>{var oldHeapSize=wasmMemory.buffer.byteLength;var pages=(size-oldHeapSize+65535)/65536|0;try{wasmMemory.grow(BigInt(pages));updateMemoryViews();return 1}catch(e){}};function _emscripten_resize_heap(requestedSize){requestedSize=bigintToI53Checked(requestedSize);var oldSize=(growMemViews(),HEAPU8).length;if(requestedSize<=oldSize){return false}var maxHeapSize=getHeapMax();if(requestedSize>maxHeapSize){return false}for(var cutDown=1;cutDown<=4;cutDown*=2){var overGrownHeapSize=oldSize*(1+.2/cutDown);overGrownHeapSize=Math.min(overGrownHeapSize,requestedSize+100663296);var newSize=Math.min(maxHeapSize,alignMemory(Math.max(requestedSize,overGrownHeapSize),65536));var replacement=growMemory(newSize);if(replacement){return true}}return false}var stringToUTF8OnStack=str=>{var size=lengthBytesUTF8(str)+1;var ret=stackAlloc(size);stringToUTF8(str,ret,size);return ret};var writeI53ToI64=(ptr,num)=>{(growMemViews(),HEAPU32)[ptr/4]=num;var lower=(growMemViews(),HEAPU32)[ptr/4];(growMemViews(),HEAPU32)[(ptr+4)/4]=(num-lower)/4294967296};var stringToNewUTF8=str=>{var size=lengthBytesUTF8(str)+1;var ret=_malloc(size);if(ret)stringToUTF8(str,ret,size);return ret};var readI53FromI64=ptr=>(growMemViews(),HEAPU32)[ptr/4]+(growMemViews(),HEAP32)[(ptr+4)/4]*4294967296;var WebGPU={Internals:{jsObjects:[],jsObjectInsert:(ptr,jsObject)=>{WebGPU.Internals.jsObjects[ptr]=jsObject},bufferOnUnmaps:[],futures:[],futureInsert:(futureId,promise)=>{WebGPU.Internals.futures[futureId]=new Promise(resolve=>promise.finally(()=>resolve(futureId)))}},getJsObject:ptr=>{if(!ptr)return undefined;return WebGPU.Internals.jsObjects[ptr]},importJsAdapter:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateAdapter(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsBindGroup:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateBindGroup(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsBindGroupLayout:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateBindGroupLayout(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsBuffer:(buffer,parentPtr=0)=>{assert(buffer.mapState==="unmapped");var bufferPtr=_emwgpuCreateBuffer(parentPtr);WebGPU.Internals.jsObjectInsert(bufferPtr,buffer);return bufferPtr},importJsCommandBuffer:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateCommandBuffer(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsCommandEncoder:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateCommandEncoder(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsComputePassEncoder:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateComputePassEncoder(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsComputePipeline:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateComputePipeline(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsDevice:(device,parentPtr=0)=>{var queuePtr=_emwgpuCreateQueue(parentPtr);var devicePtr=_emwgpuCreateDevice(parentPtr,queuePtr);WebGPU.Internals.jsObjectInsert(queuePtr,device.queue);WebGPU.Internals.jsObjectInsert(devicePtr,device);return devicePtr},importJsExternalTexture:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateExternalTexture(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsPipelineLayout:(obj,parentPtr=0)=>{var ptr=_emwgpuCreatePipelineLayout(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsQuerySet:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateQuerySet(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsQueue:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateQueue(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsRenderBundle:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateRenderBundle(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsRenderBundleEncoder:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateRenderBundleEncoder(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsRenderPassEncoder:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateRenderPassEncoder(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsRenderPipeline:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateRenderPipeline(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsSampler:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateSampler(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsShaderModule:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateShaderModule(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsSurface:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateSurface(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsTexture:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateTexture(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},importJsTextureView:(obj,parentPtr=0)=>{var ptr=_emwgpuCreateTextureView(parentPtr);WebGPU.Internals.jsObjects[ptr]=obj;return ptr},errorCallback:(callback,type,message,userdata)=>{var sp=stackSave();var messagePtr=stringToUTF8OnStack(message);((a1,a2,a3)=>getWasmTableEntry(callback).call(null,a1,BigInt(a2),BigInt(a3)))(type,BigInt(messagePtr),userdata);stackRestore(sp)},iterateExtensions:(root,handlers)=>{for(var ptr=Number((growMemViews(),HEAPU64)[root/8]);ptr;ptr=Number((growMemViews(),HEAPU64)[ptr/8])){var sType=(growMemViews(),HEAP32)[(ptr+8)/4];var handler=handlers[sType](ptr)}},setStringView:(ptr,data,length)=>{(growMemViews(),HEAPU64)[ptr/8]=BigInt(data);(growMemViews(),HEAPU64)[(ptr+8)/8]=BigInt(length)},makeStringFromStringView:stringViewPtr=>{var ptr=Number((growMemViews(),HEAPU64)[stringViewPtr/8]);var length=Number((growMemViews(),HEAPU64)[(stringViewPtr+8)/8]);return UTF8ToString(ptr,length)},makeStringFromOptionalStringView:stringViewPtr=>{var ptr=Number((growMemViews(),HEAPU64)[stringViewPtr/8]);var length=Number((growMemViews(),HEAPU64)[(stringViewPtr+8)/8]);if(!ptr){if(length===0){return""}return undefined}return UTF8ToString(ptr,length)},makeColor:ptr=>({r:(growMemViews(),HEAPF64)[ptr/8],g:(growMemViews(),HEAPF64)[(ptr+8)/8],b:(growMemViews(),HEAPF64)[(ptr+16)/8],a:(growMemViews(),HEAPF64)[(ptr+24)/8]}),makeExtent3D:ptr=>({width:(growMemViews(),HEAPU32)[ptr/4],height:(growMemViews(),HEAPU32)[(ptr+4)/4],depthOrArrayLayers:(growMemViews(),HEAPU32)[(ptr+8)/4]}),makeOrigin3D:ptr=>({x:(growMemViews(),HEAPU32)[ptr/4],y:(growMemViews(),HEAPU32)[(ptr+4)/4],z:(growMemViews(),HEAPU32)[(ptr+8)/4]}),makeTexelCopyTextureInfo:ptr=>({texture:WebGPU.getJsObject(Number((growMemViews(),HEAPU64)[ptr/8])),mipLevel:(growMemViews(),HEAPU32)[(ptr+8)/4],origin:WebGPU.makeOrigin3D(ptr+12),aspect:WebGPU.TextureAspect[(growMemViews(),HEAP32)[(ptr+24)/4]]}),makeTexelCopyBufferLayout:ptr=>{var bytesPerRow=(growMemViews(),HEAPU32)[(ptr+8)/4];var rowsPerImage=(growMemViews(),HEAPU32)[(ptr+12)/4];return{offset:readI53FromI64(ptr),bytesPerRow:bytesPerRow===4294967295?undefined:bytesPerRow,rowsPerImage:rowsPerImage===4294967295?undefined:rowsPerImage}},makeTexelCopyBufferInfo:ptr=>{var layoutPtr=ptr+0;var bufferCopyView=WebGPU.makeTexelCopyBufferLayout(layoutPtr);bufferCopyView["buffer"]=WebGPU.getJsObject(Number((growMemViews(),HEAPU64)[(ptr+16)/8]));return bufferCopyView},makePassTimestampWrites:ptr=>{if(ptr===0)return undefined;return{querySet:WebGPU.getJsObject(Number((growMemViews(),HEAPU64)[(ptr+8)/8])),beginningOfPassWriteIndex:(growMemViews(),HEAPU32)[(ptr+16)/4],endOfPassWriteIndex:(growMemViews(),HEAPU32)[(ptr+20)/4]}},makePipelineConstants:(constantCount,constantsPtr)=>{if(!constantCount)return;var constants={};for(var i=0;i<constantCount;++i){var entryPtr=constantsPtr+32*i;var key=WebGPU.makeStringFromStringView(entryPtr+8);constants[key]=(growMemViews(),HEAPF64)[(entryPtr+24)/8]}return constants},makePipelineLayout:layoutPtr=>{if(!layoutPtr)return"auto";return WebGPU.getJsObject(layoutPtr)},makeComputeState:ptr=>{if(!ptr)return undefined;var desc={module:WebGPU.getJsObject(Number((growMemViews(),HEAPU64)[(ptr+8)/8])),constants:WebGPU.makePipelineConstants(Number((growMemViews(),HEAPU64)[(ptr+32)/8]),Number((growMemViews(),HEAPU64)[(ptr+40)/8])),entryPoint:WebGPU.makeStringFromOptionalStringView(ptr+16)};return desc},makeComputePipelineDesc:descriptor=>{var desc={label:WebGPU.makeStringFromOptionalStringView(descriptor+8),layout:WebGPU.makePipelineLayout(Number((growMemViews(),HEAPU64)[(descriptor+24)/8])),compute:WebGPU.makeComputeState(descriptor+32)};return desc},makeRenderPipelineDesc:descriptor=>{function makePrimitiveState(psPtr){if(!psPtr)return undefined;return{topology:WebGPU.PrimitiveTopology[(growMemViews(),HEAP32)[(psPtr+8)/4]],stripIndexFormat:WebGPU.IndexFormat[(growMemViews(),HEAP32)[(psPtr+12)/4]],frontFace:WebGPU.FrontFace[(growMemViews(),HEAP32)[(psPtr+16)/4]],cullMode:WebGPU.CullMode[(growMemViews(),HEAP32)[(psPtr+20)/4]],unclippedDepth:!!(growMemViews(),HEAPU32)[(psPtr+24)/4]}}function makeBlendComponent(bdPtr){if(!bdPtr)return undefined;return{operation:WebGPU.BlendOperation[(growMemViews(),HEAP32)[bdPtr/4]],srcFactor:WebGPU.BlendFactor[(growMemViews(),HEAP32)[(bdPtr+4)/4]],dstFactor:WebGPU.BlendFactor[(growMemViews(),HEAP32)[(bdPtr+8)/4]]}}function makeBlendState(bsPtr){if(!bsPtr)return undefined;return{alpha:makeBlendComponent(bsPtr+12),color:makeBlendComponent(bsPtr+0)}}function makeColorState(csPtr){var format=WebGPU.TextureFormat[(growMemViews(),HEAP32)[(csPtr+8)/4]];return format?{format,blend:makeBlendState(Number((growMemViews(),HEAPU64)[(csPtr+16)/8])),writeMask:(growMemViews(),HEAPU32)[(csPtr+24)/4]}:undefined}function makeColorStates(count,csArrayPtr){var states=[];for(var i=0;i<count;++i){states.push(makeColorState(csArrayPtr+32*i))}return states}function makeStencilStateFace(ssfPtr){return{compare:WebGPU.CompareFunction[(growMemViews(),HEAP32)[ssfPtr/4]],failOp:WebGPU.StencilOperation[(growMemViews(),HEAP32)[(ssfPtr+4)/4]],depthFailOp:WebGPU.StencilOperation[(growMemViews(),HEAP32)[(ssfPtr+8)/4]],passOp:WebGPU.StencilOperation[(growMemViews(),HEAP32)[(ssfPtr+12)/4]]}}function makeDepthStencilState(dssPtr){if(!dssPtr)return undefined;return{format:WebGPU.TextureFormat[(growMemViews(),HEAP32)[(dssPtr+8)/4]],depthWriteEnabled:!!(growMemViews(),HEAPU32)[(dssPtr+12)/4],depthCompare:WebGPU.CompareFunction[(growMemViews(),HEAP32)[(dssPtr+16)/4]],stencilFront:makeStencilStateFace(dssPtr+20),stencilBack:makeStencilStateFace(dssPtr+36),stencilReadMask:(growMemViews(),HEAPU32)[(dssPtr+52)/4],stencilWriteMask:(growMemViews(),HEAPU32)[(dssPtr+56)/4],depthBias:(growMemViews(),HEAP32)[(dssPtr+60)/4],depthBiasSlopeScale:(growMemViews(),HEAPF32)[(dssPtr+64)/4],depthBiasClamp:(growMemViews(),HEAPF32)[(dssPtr+68)/4]}}function makeVertexAttribute(vaPtr){return{format:WebGPU.VertexFormat[(growMemViews(),HEAP32)[(vaPtr+8)/4]],offset:readI53FromI64(vaPtr+16),shaderLocation:(growMemViews(),HEAPU32)[(vaPtr+24)/4]}}function makeVertexAttributes(count,vaArrayPtr){var vas=[];for(var i=0;i<count;++i){vas.push(makeVertexAttribute(vaArrayPtr+i*32))}return vas}function makeVertexBuffer(vbPtr){if(!vbPtr)return undefined;var stepMode=WebGPU.VertexStepMode[(growMemViews(),HEAP32)[(vbPtr+8)/4]];var attributeCount=Number((growMemViews(),HEAPU64)[(vbPtr+24)/8]);if(!stepMode&&!attributeCount){return null}return{arrayStride:readI53FromI64(vbPtr+16),stepMode,attributes:makeVertexAttributes(attributeCount,Number((growMemViews(),HEAPU64)[(vbPtr+32)/8]))}}function makeVertexBuffers(count,vbArrayPtr){if(!count)return undefined;var vbs=[];for(var i=0;i<count;++i){vbs.push(makeVertexBuffer(vbArrayPtr+i*40))}return vbs}function makeVertexState(viPtr){if(!viPtr)return undefined;var desc={module:WebGPU.getJsObject(Number((growMemViews(),HEAPU64)[(viPtr+8)/8])),constants:WebGPU.makePipelineConstants(Number((growMemViews(),HEAPU64)[(viPtr+32)/8]),Number((growMemViews(),HEAPU64)[(viPtr+40)/8])),buffers:makeVertexBuffers(Number((growMemViews(),HEAPU64)[(viPtr+48)/8]),Number((growMemViews(),HEAPU64)[(viPtr+56)/8])),entryPoint:WebGPU.makeStringFromOptionalStringView(viPtr+16)};return desc}function makeMultisampleState(msPtr){if(!msPtr)return undefined;return{count:(growMemViews(),HEAPU32)[(msPtr+8)/4],mask:(growMemViews(),HEAPU32)[(msPtr+12)/4],alphaToCoverageEnabled:!!(growMemViews(),HEAPU32)[(msPtr+16)/4]}}function makeFragmentState(fsPtr){if(!fsPtr)return undefined;var desc={module:WebGPU.getJsObject(Number((growMemViews(),HEAPU64)[(fsPtr+8)/8])),constants:WebGPU.makePipelineConstants(Number((growMemViews(),HEAPU64)[(fsPtr+32)/8]),Number((growMemViews(),HEAPU64)[(fsPtr+40)/8])),targets:makeColorStates(Number((growMemViews(),HEAPU64)[(fsPtr+48)/8]),Number((growMemViews(),HEAPU64)[(fsPtr+56)/8])),entryPoint:WebGPU.makeStringFromOptionalStringView(fsPtr+16)};return desc}var desc={label:WebGPU.makeStringFromOptionalStringView(descriptor+8),layout:WebGPU.makePipelineLayout(Number((growMemViews(),HEAPU64)[(descriptor+24)/8])),vertex:makeVertexState(descriptor+32),primitive:makePrimitiveState(descriptor+96),depthStencil:makeDepthStencilState(Number((growMemViews(),HEAPU64)[(descriptor+128)/8])),multisample:makeMultisampleState(descriptor+136),fragment:makeFragmentState(Number((growMemViews(),HEAPU64)[(descriptor+160)/8]))};return desc},fillLimitStruct:(limits,limitsOutPtr)=>{var nextInChainPtr=Number((growMemViews(),HEAPU64)[limitsOutPtr/8]);function setLimitValueU32(name,basePtr,limitOffset,fallbackValue=0){var limitValue=limits[name]??fallbackValue;(growMemViews(),HEAPU32)[(basePtr+limitOffset)/4]=limitValue}function setLimitValueU64(name,basePtr,limitOffset,fallbackValue=0){var limitValue=limits[name]??fallbackValue;writeI53ToI64(basePtr+limitOffset,limitValue)}setLimitValueU32("maxTextureDimension1D",limitsOutPtr,8);setLimitValueU32("maxTextureDimension2D",limitsOutPtr,12);setLimitValueU32("maxTextureDimension3D",limitsOutPtr,16);setLimitValueU32("maxTextureArrayLayers",limitsOutPtr,20);setLimitValueU32("maxBindGroups",limitsOutPtr,24);setLimitValueU32("maxBindGroupsPlusVertexBuffers",limitsOutPtr,28);setLimitValueU32("maxBindingsPerBindGroup",limitsOutPtr,32);setLimitValueU32("maxDynamicUniformBuffersPerPipelineLayout",limitsOutPtr,36);setLimitValueU32("maxDynamicStorageBuffersPerPipelineLayout",limitsOutPtr,40);setLimitValueU32("maxSampledTexturesPerShaderStage",limitsOutPtr,44);setLimitValueU32("maxSamplersPerShaderStage",limitsOutPtr,48);setLimitValueU32("maxStorageBuffersPerShaderStage",limitsOutPtr,52);setLimitValueU32("maxStorageTexturesPerShaderStage",limitsOutPtr,56);setLimitValueU32("maxUniformBuffersPerShaderStage",limitsOutPtr,60);setLimitValueU32("minUniformBufferOffsetAlignment",limitsOutPtr,80);setLimitValueU32("minStorageBufferOffsetAlignment",limitsOutPtr,84);setLimitValueU64("maxUniformBufferBindingSize",limitsOutPtr,64);setLimitValueU64("maxStorageBufferBindingSize",limitsOutPtr,72);setLimitValueU32("maxVertexBuffers",limitsOutPtr,88);setLimitValueU64("maxBufferSize",limitsOutPtr,96);setLimitValueU32("maxVertexAttributes",limitsOutPtr,104);setLimitValueU32("maxVertexBufferArrayStride",limitsOutPtr,108);setLimitValueU32("maxInterStageShaderVariables",limitsOutPtr,112);setLimitValueU32("maxColorAttachments",limitsOutPtr,116);setLimitValueU32("maxColorAttachmentBytesPerSample",limitsOutPtr,120);setLimitValueU32("maxComputeWorkgroupStorageSize",limitsOutPtr,124);setLimitValueU32("maxComputeInvocationsPerWorkgroup",limitsOutPtr,128);setLimitValueU32("maxComputeWorkgroupSizeX",limitsOutPtr,132);setLimitValueU32("maxComputeWorkgroupSizeY",limitsOutPtr,136);setLimitValueU32("maxComputeWorkgroupSizeZ",limitsOutPtr,140);setLimitValueU32("maxComputeWorkgroupsPerDimension",limitsOutPtr,144);setLimitValueU32("maxImmediateSize",limitsOutPtr,148);if(nextInChainPtr!==0){var sType=(growMemViews(),HEAP32)[(nextInChainPtr+8)/4];var compatibilityModeLimitsPtr=nextInChainPtr;setLimitValueU32("maxStorageBuffersInVertexStage",compatibilityModeLimitsPtr,16,limits.maxStorageBuffersPerShaderStage);setLimitValueU32("maxStorageBuffersInFragmentStage",compatibilityModeLimitsPtr,24,limits.maxStorageBuffersPerShaderStage);setLimitValueU32("maxStorageTexturesInVertexStage",compatibilityModeLimitsPtr,20,limits.maxStorageTexturesPerShaderStage);setLimitValueU32("maxStorageTexturesInFragmentStage",compatibilityModeLimitsPtr,28,limits.maxStorageTexturesPerShaderStage)}},fillAdapterInfoStruct:(info,infoStruct)=>{(growMemViews(),HEAPU32)[(infoStruct+88)/4]=info.subgroupMinSize;(growMemViews(),HEAPU32)[(infoStruct+92)/4]=info.subgroupMaxSize;var strs=info.vendor+info.architecture+info.device+info.description;var strPtr=stringToNewUTF8(strs);var vendorLen=lengthBytesUTF8(info.vendor);WebGPU.setStringView(infoStruct+8,strPtr,vendorLen);strPtr+=vendorLen;var architectureLen=lengthBytesUTF8(info.architecture);WebGPU.setStringView(infoStruct+24,strPtr,architectureLen);strPtr+=architectureLen;var deviceLen=lengthBytesUTF8(info.device);WebGPU.setStringView(infoStruct+40,strPtr,deviceLen);strPtr+=deviceLen;var descriptionLen=lengthBytesUTF8(info.description);WebGPU.setStringView(infoStruct+56,strPtr,descriptionLen);strPtr+=descriptionLen;(growMemViews(),HEAP32)[(infoStruct+72)/4]=2;var adapterType=info.isFallbackAdapter?3:4;(growMemViews(),HEAP32)[(infoStruct+76)/4]=adapterType;(growMemViews(),HEAPU32)[(infoStruct+80)/4]=0;(growMemViews(),HEAPU32)[(infoStruct+84)/4]=0},AddressMode:[,"clamp-to-edge","repeat","mirror-repeat"],BlendFactor:[,"zero","one","src","one-minus-src","src-alpha","one-minus-src-alpha","dst","one-minus-dst","dst-alpha","one-minus-dst-alpha","src-alpha-saturated","constant","one-minus-constant","src1","one-minus-src1","src1-alpha","one-minus-src1-alpha"],BlendOperation:[,"add","subtract","reverse-subtract","min","max"],BufferBindingType:[,,"uniform","storage","read-only-storage"],BufferMapState:[,"unmapped","pending","mapped"],CompareFunction:[,"never","less","equal","less-equal","greater","not-equal","greater-equal","always"],CompilationInfoRequestStatus:[,"success","callback-cancelled"],ComponentSwizzle:[,"0","1","r","g","b","a"],CompositeAlphaMode:[,"opaque","premultiplied","unpremultiplied","inherit"],CullMode:[,"none","front","back"],ErrorFilter:[,"validation","out-of-memory","internal"],FeatureLevel:[,"compatibility","core"],FeatureName:{1:"core-features-and-limits",2:"depth-clip-control",3:"depth32float-stencil8",4:"texture-compression-bc",5:"texture-compression-bc-sliced-3d",6:"texture-compression-etc2",7:"texture-compression-astc",8:"texture-compression-astc-sliced-3d",9:"timestamp-query",10:"indirect-first-instance",11:"shader-f16",12:"rg11b10ufloat-renderable",13:"bgra8unorm-storage",14:"float32-filterable",15:"float32-blendable",16:"clip-distances",17:"dual-source-blending",18:"subgroups",19:"texture-formats-tier1",20:"texture-formats-tier2",21:"primitive-index",22:"texture-component-swizzle",327692:"chromium-experimental-unorm16-texture-formats",327729:"chromium-experimental-multi-draw-indirect"},FilterMode:[,"nearest","linear"],FrontFace:[,"ccw","cw"],IndexFormat:[,"uint16","uint32"],InstanceFeatureName:[,"timed-wait-any","shader-source-spirv","multiple-devices-per-adapter"],LoadOp:[,"load","clear"],MipmapFilterMode:[,"nearest","linear"],OptionalBool:["false","true"],PowerPreference:[,"low-power","high-performance"],PredefinedColorSpace:[,"srgb","display-p3"],PrimitiveTopology:[,"point-list","line-list","line-strip","triangle-list","triangle-strip"],QueryType:[,"occlusion","timestamp"],SamplerBindingType:[,,"filtering","non-filtering","comparison"],Status:[,"success","error"],StencilOperation:[,"keep","zero","replace","invert","increment-clamp","decrement-clamp","increment-wrap","decrement-wrap"],StorageTextureAccess:[,,"write-only","read-only","read-write"],StoreOp:[,"store","discard"],SurfaceGetCurrentTextureStatus:[,"success-optimal","success-suboptimal","timeout","outdated","lost","error"],TextureAspect:[,"all","stencil-only","depth-only"],TextureDimension:[,"1d","2d","3d"],TextureFormat:[,"r8unorm","r8snorm","r8uint","r8sint","r16unorm","r16snorm","r16uint","r16sint","r16float","rg8unorm","rg8snorm","rg8uint","rg8sint","r32float","r32uint","r32sint","rg16unorm","rg16snorm","rg16uint","rg16sint","rg16float","rgba8unorm","rgba8unorm-srgb","rgba8snorm","rgba8uint","rgba8sint","bgra8unorm","bgra8unorm-srgb","rgb10a2uint","rgb10a2unorm","rg11b10ufloat","rgb9e5ufloat","rg32float","rg32uint","rg32sint","rgba16unorm","rgba16snorm","rgba16uint","rgba16sint","rgba16float","rgba32float","rgba32uint","rgba32sint","stencil8","depth16unorm","depth24plus","depth24plus-stencil8","depth32float","depth32float-stencil8","bc1-rgba-unorm","bc1-rgba-unorm-srgb","bc2-rgba-unorm","bc2-rgba-unorm-srgb","bc3-rgba-unorm","bc3-rgba-unorm-srgb","bc4-r-unorm","bc4-r-snorm","bc5-rg-unorm","bc5-rg-snorm","bc6h-rgb-ufloat","bc6h-rgb-float","bc7-rgba-unorm","bc7-rgba-unorm-srgb","etc2-rgb8unorm","etc2-rgb8unorm-srgb","etc2-rgb8a1unorm","etc2-rgb8a1unorm-srgb","etc2-rgba8unorm","etc2-rgba8unorm-srgb","eac-r11unorm","eac-r11snorm","eac-rg11unorm","eac-rg11snorm","astc-4x4-unorm","astc-4x4-unorm-srgb","astc-5x4-unorm","astc-5x4-unorm-srgb","astc-5x5-unorm","astc-5x5-unorm-srgb","astc-6x5-unorm","astc-6x5-unorm-srgb","astc-6x6-unorm","astc-6x6-unorm-srgb","astc-8x5-unorm","astc-8x5-unorm-srgb","astc-8x6-unorm","astc-8x6-unorm-srgb","astc-8x8-unorm","astc-8x8-unorm-srgb","astc-10x5-unorm","astc-10x5-unorm-srgb","astc-10x6-unorm","astc-10x6-unorm-srgb","astc-10x8-unorm","astc-10x8-unorm-srgb","astc-10x10-unorm","astc-10x10-unorm-srgb","astc-12x10-unorm","astc-12x10-unorm-srgb","astc-12x12-unorm","astc-12x12-unorm-srgb"],TextureSampleType:[,,"float","unfilterable-float","depth","sint","uint"],TextureViewDimension:[,"1d","2d","2d-array","cube","cube-array","3d"],ToneMappingMode:[,"standard","extended"],VertexFormat:[,"uint8","uint8x2","uint8x4","sint8","sint8x2","sint8x4","unorm8","unorm8x2","unorm8x4","snorm8","snorm8x2","snorm8x4","uint16","uint16x2","uint16x4","sint16","sint16x2","sint16x4","unorm16","unorm16x2","unorm16x4","snorm16","snorm16x2","snorm16x4","float16","float16x2","float16x4","float32","float32x2","float32x3","float32x4","uint32","uint32x2","uint32x3","uint32x4","sint32","sint32x2","sint32x3","sint32x4","unorm10-10-10-2","unorm8x4-bgra"],VertexStepMode:[,"vertex","instance"],WGSLLanguageFeatureName:[,"readonly_and_readwrite_storage_textures","packed_4x8_integer_dot_product","unrestricted_pointer_parameters","pointer_composite_access","uniform_buffer_standard_layout","subgroup_id","texture_and_sampler_let","subgroup_uniformity","texture_formats_tier1"]};var emwgpuStringToInt_DeviceLostReason={undefined:1,unknown:1,destroyed:2};var runtimeKeepalivePop=()=>{runtimeKeepaliveCounter-=1};function _emwgpuAdapterRequestDevice(adapterPtr,futureId,deviceLostFutureId,devicePtr,queuePtr,descriptor){adapterPtr=bigintToI53Checked(adapterPtr);futureId=bigintToI53Checked(futureId);deviceLostFutureId=bigintToI53Checked(deviceLostFutureId);devicePtr=bigintToI53Checked(devicePtr);queuePtr=bigintToI53Checked(queuePtr);descriptor=bigintToI53Checked(descriptor);var adapter=WebGPU.getJsObject(adapterPtr);var desc={};if(descriptor){var requiredFeatureCount=Number((growMemViews(),HEAPU64)[(descriptor+24)/8]);if(requiredFeatureCount){var requiredFeaturesPtr=Number((growMemViews(),HEAPU64)[(descriptor+32)/8]);desc["requiredFeatures"]=Array.from((growMemViews(),HEAPU32).subarray(requiredFeaturesPtr/4,(requiredFeaturesPtr+requiredFeatureCount*4)/4),feature=>WebGPU.FeatureName[feature])}var limitsPtr=Number((growMemViews(),HEAPU64)[(descriptor+40)/8]);if(limitsPtr){var nextInChainPtr=Number((growMemViews(),HEAPU64)[limitsPtr/8]);var requiredLimits={};function setLimitU32IfDefined(name,basePtr,limitOffset,ignoreIfZero=false){var ptr=basePtr+limitOffset;var value=(growMemViews(),HEAPU32)[ptr/4];if(value!=4294967295&&(!ignoreIfZero||value!=0)){requiredLimits[name]=value}}function setLimitU64IfDefined(name,basePtr,limitOffset){var ptr=basePtr+limitOffset;var limitPart1=(growMemViews(),HEAPU32)[ptr/4];var limitPart2=(growMemViews(),HEAPU32)[(ptr+4)/4];if(limitPart1!=4294967295||limitPart2!=4294967295){requiredLimits[name]=readI53FromI64(ptr)}}setLimitU32IfDefined("maxTextureDimension1D",limitsPtr,8);setLimitU32IfDefined("maxTextureDimension2D",limitsPtr,12);setLimitU32IfDefined("maxTextureDimension3D",limitsPtr,16);setLimitU32IfDefined("maxTextureArrayLayers",limitsPtr,20);setLimitU32IfDefined("maxBindGroups",limitsPtr,24);setLimitU32IfDefined("maxBindGroupsPlusVertexBuffers",limitsPtr,28);setLimitU32IfDefined("maxBindingsPerBindGroup",limitsPtr,32);setLimitU32IfDefined("maxDynamicUniformBuffersPerPipelineLayout",limitsPtr,36);setLimitU32IfDefined("maxDynamicStorageBuffersPerPipelineLayout",limitsPtr,40);setLimitU32IfDefined("maxSampledTexturesPerShaderStage",limitsPtr,44);setLimitU32IfDefined("maxSamplersPerShaderStage",limitsPtr,48);setLimitU32IfDefined("maxStorageBuffersPerShaderStage",limitsPtr,52);setLimitU32IfDefined("maxStorageTexturesPerShaderStage",limitsPtr,56);setLimitU32IfDefined("maxUniformBuffersPerShaderStage",limitsPtr,60);setLimitU32IfDefined("minUniformBufferOffsetAlignment",limitsPtr,80);setLimitU32IfDefined("minStorageBufferOffsetAlignment",limitsPtr,84);setLimitU64IfDefined("maxUniformBufferBindingSize",limitsPtr,64);setLimitU64IfDefined("maxStorageBufferBindingSize",limitsPtr,72);setLimitU32IfDefined("maxVertexBuffers",limitsPtr,88);setLimitU64IfDefined("maxBufferSize",limitsPtr,96);setLimitU32IfDefined("maxVertexAttributes",limitsPtr,104);setLimitU32IfDefined("maxVertexBufferArrayStride",limitsPtr,108);setLimitU32IfDefined("maxInterStageShaderVariables",limitsPtr,112);setLimitU32IfDefined("maxColorAttachments",limitsPtr,116);setLimitU32IfDefined("maxColorAttachmentBytesPerSample",limitsPtr,120);setLimitU32IfDefined("maxComputeWorkgroupStorageSize",limitsPtr,124);setLimitU32IfDefined("maxComputeInvocationsPerWorkgroup",limitsPtr,128);setLimitU32IfDefined("maxComputeWorkgroupSizeX",limitsPtr,132);setLimitU32IfDefined("maxComputeWorkgroupSizeY",limitsPtr,136);setLimitU32IfDefined("maxComputeWorkgroupSizeZ",limitsPtr,140);setLimitU32IfDefined("maxComputeWorkgroupsPerDimension",limitsPtr,144);setLimitU32IfDefined("maxImmediateSize",limitsPtr,148,true);if(nextInChainPtr!==0){var sType=(growMemViews(),HEAP32)[(nextInChainPtr+8)/4];var compatibilityModeLimitsPtr=nextInChainPtr;if("maxStorageBuffersInVertexStage"in GPUSupportedLimits.prototype){setLimitU32IfDefined("maxStorageBuffersInVertexStage",compatibilityModeLimitsPtr,16);setLimitU32IfDefined("maxStorageTexturesInVertexStage",compatibilityModeLimitsPtr,20);setLimitU32IfDefined("maxStorageBuffersInFragmentStage",compatibilityModeLimitsPtr,24);setLimitU32IfDefined("maxStorageTexturesInFragmentStage",compatibilityModeLimitsPtr,28)}}desc["requiredLimits"]=requiredLimits}var defaultQueuePtr=Number((growMemViews(),HEAPU64)[(descriptor+48)/8]);if(defaultQueuePtr){var defaultQueueDesc={label:WebGPU.makeStringFromOptionalStringView(defaultQueuePtr+8)};desc["defaultQueue"]=defaultQueueDesc}desc["label"]=WebGPU.makeStringFromOptionalStringView(descriptor+8)}runtimeKeepalivePush();WebGPU.Internals.futureInsert(futureId,adapter.requestDevice(desc).then(device=>{runtimeKeepalivePop();callUserCallback(()=>{WebGPU.Internals.jsObjectInsert(queuePtr,device.queue);WebGPU.Internals.jsObjectInsert(devicePtr,device);devicePtr=BigInt(devicePtr);WebGPU.Internals.futureInsert(deviceLostFutureId,device.lost.then(info=>{callUserCallback(()=>{device.onuncapturederror=ev=>{};var sp=stackSave();var messagePtr=stringToUTF8OnStack(info.message);_emwgpuOnDeviceLostCompleted(deviceLostFutureId,emwgpuStringToInt_DeviceLostReason[info.reason],BigInt(messagePtr));stackRestore(sp)})}));device.onuncapturederror=ev=>{var type=5;if(ev.error instanceof GPUValidationError)type=2;else if(ev.error instanceof GPUOutOfMemoryError)type=3;else if(ev.error instanceof GPUInternalError)type=4;var sp=stackSave();var messagePtr=stringToUTF8OnStack(ev.error.message);_emwgpuOnUncapturedError(BigInt(devicePtr),type,BigInt(messagePtr));stackRestore(sp)};_emwgpuOnRequestDeviceCompleted(futureId,1,BigInt(devicePtr),0n)})},ex=>{runtimeKeepalivePop();callUserCallback(()=>{var sp=stackSave();var messagePtr=stringToUTF8OnStack(ex.message);_emwgpuOnRequestDeviceCompleted(futureId,3,BigInt(devicePtr),BigInt(messagePtr));if(deviceLostFutureId){_emwgpuOnDeviceLostCompleted(deviceLostFutureId,4,BigInt(messagePtr))}stackRestore(sp)})}))}function _emwgpuBufferDestroy(bufferPtr){bufferPtr=bigintToI53Checked(bufferPtr);var buffer=WebGPU.getJsObject(bufferPtr);var onUnmap=WebGPU.Internals.bufferOnUnmaps[bufferPtr];if(onUnmap){for(var i=0;i<onUnmap.length;++i){onUnmap[i]()}delete WebGPU.Internals.bufferOnUnmaps[bufferPtr]}buffer.destroy()}var warnOnce=text=>{warnOnce.shown||={};if(!warnOnce.shown[text]){warnOnce.shown[text]=1;if(ENVIRONMENT_IS_NODE)text="warning: "+text;err(text)}};var _emwgpuBufferGetConstMappedRange=function(bufferPtr,offset,size){bufferPtr=bigintToI53Checked(bufferPtr);offset=bigintToI53Checked(offset);size=bigintToI53Checked(size);var ret=(()=>{var buffer=WebGPU.getJsObject(bufferPtr);if(size==-1)size=undefined;var mapped;try{mapped=buffer.getMappedRange(offset,size)}catch(ex){return 0n}var data=_memalign(16,mapped.byteLength);(growMemViews(),HEAPU8).set(new Uint8Array(mapped),data);WebGPU.Internals.bufferOnUnmaps[bufferPtr].push(()=>_free(data));return data})();return BigInt(ret)};var _emwgpuBufferMapAsync=function(bufferPtr,futureId,mode,offset,size){bufferPtr=bigintToI53Checked(bufferPtr);futureId=bigintToI53Checked(futureId);mode=bigintToI53Checked(mode);offset=bigintToI53Checked(offset);size=bigintToI53Checked(size);var buffer=WebGPU.getJsObject(bufferPtr);WebGPU.Internals.bufferOnUnmaps[bufferPtr]=[];if(size==-1)size=undefined;runtimeKeepalivePush();WebGPU.Internals.futureInsert(futureId,buffer.mapAsync(mode,offset,size).then(()=>{runtimeKeepalivePop();callUserCallback(()=>{_emwgpuOnMapAsyncCompleted(futureId,1,0n)})},ex=>{runtimeKeepalivePop();callUserCallback(()=>{var sp=stackSave();var messagePtr=stringToUTF8OnStack(ex.message);var status=ex.name==="AbortError"?4:ex.name==="OperationError"?3:0;_emwgpuOnMapAsyncCompleted(futureId,status,BigInt(messagePtr));delete WebGPU.Internals.bufferOnUnmaps[bufferPtr]})}))};function _emwgpuBufferUnmap(bufferPtr){bufferPtr=bigintToI53Checked(bufferPtr);var buffer=WebGPU.getJsObject(bufferPtr);var onUnmap=WebGPU.Internals.bufferOnUnmaps[bufferPtr];if(!onUnmap){return}for(var i=0;i<onUnmap.length;++i){onUnmap[i]()}delete WebGPU.Internals.bufferOnUnmaps[bufferPtr];buffer.unmap()}function _emwgpuDelete(ptr){ptr=bigintToI53Checked(ptr);delete WebGPU.Internals.jsObjects[ptr]}function _emwgpuDeviceCreateBuffer(devicePtr,descriptor,bufferPtr){devicePtr=bigintToI53Checked(devicePtr);descriptor=bigintToI53Checked(descriptor);bufferPtr=bigintToI53Checked(bufferPtr);var mappedAtCreation=!!(growMemViews(),HEAPU32)[(descriptor+40)/4];var desc={label:WebGPU.makeStringFromOptionalStringView(descriptor+8),usage:(growMemViews(),HEAPU32)[(descriptor+24)/4],size:readI53FromI64(descriptor+32),mappedAtCreation};var device=WebGPU.getJsObject(devicePtr);var buffer;try{buffer=device.createBuffer(desc)}catch(ex){return false}WebGPU.Internals.jsObjectInsert(bufferPtr,buffer);if(mappedAtCreation){WebGPU.Internals.bufferOnUnmaps[bufferPtr]=[]}return true}function _emwgpuDeviceCreateShaderModule(devicePtr,descriptor,shaderModulePtr){devicePtr=bigintToI53Checked(devicePtr);descriptor=bigintToI53Checked(descriptor);shaderModulePtr=bigintToI53Checked(shaderModulePtr);var nextInChainPtr=Number((growMemViews(),HEAPU64)[descriptor/8]);var sType=(growMemViews(),HEAP32)[(nextInChainPtr+8)/4];var desc={label:WebGPU.makeStringFromOptionalStringView(descriptor+8),code:""};switch(sType){case 2:{desc["code"]=WebGPU.makeStringFromStringView(nextInChainPtr+16);break}}var device=WebGPU.getJsObject(devicePtr);WebGPU.Internals.jsObjectInsert(shaderModulePtr,device.createShaderModule(desc))}var _emwgpuDeviceDestroy=devicePtr=>{const device=WebGPU.getJsObject(devicePtr);device.onuncapturederror=null;device.destroy()};function _emwgpuInstanceRequestAdapter(instancePtr,futureId,options,adapterPtr){instancePtr=bigintToI53Checked(instancePtr);futureId=bigintToI53Checked(futureId);options=bigintToI53Checked(options);adapterPtr=bigintToI53Checked(adapterPtr);var opts;if(options){opts={featureLevel:WebGPU.FeatureLevel[(growMemViews(),HEAP32)[(options+8)/4]],powerPreference:WebGPU.PowerPreference[(growMemViews(),HEAP32)[(options+12)/4]],forceFallbackAdapter:!!(growMemViews(),HEAPU32)[(options+16)/4]};var nextInChainPtr=Number((growMemViews(),HEAPU64)[options/8]);if(nextInChainPtr!==0){var sType=(growMemViews(),HEAP32)[(nextInChainPtr+8)/4];var webxrOptions=nextInChainPtr;opts.xrCompatible=!!(growMemViews(),HEAPU32)[(webxrOptions+16)/4]}}if(!("gpu"in navigator)){var sp=stackSave();var messagePtr=stringToUTF8OnStack("WebGPU not available on this browser (navigator.gpu is not available)");_emwgpuOnRequestAdapterCompleted(futureId,3,BigInt(adapterPtr),BigInt(messagePtr));stackRestore(sp);return}runtimeKeepalivePush();WebGPU.Internals.futureInsert(futureId,navigator.gpu.requestAdapter(opts).then(adapter=>{runtimeKeepalivePop();callUserCallback(()=>{if(adapter){WebGPU.Internals.jsObjectInsert(adapterPtr,adapter);_emwgpuOnRequestAdapterCompleted(futureId,1,BigInt(adapterPtr),0n)}else{var sp=stackSave();var messagePtr=stringToUTF8OnStack("WebGPU not available on this browser (requestAdapter returned null)");_emwgpuOnRequestAdapterCompleted(futureId,3,BigInt(adapterPtr),BigInt(messagePtr));stackRestore(sp)}})},ex=>{runtimeKeepalivePop();callUserCallback(()=>{var sp=stackSave();var messagePtr=stringToUTF8OnStack(ex.message);_emwgpuOnRequestAdapterCompleted(futureId,4,BigInt(adapterPtr),BigInt(messagePtr));stackRestore(sp)})}))}var _emwgpuQueueOnSubmittedWorkDone=function(queuePtr,futureId){queuePtr=bigintToI53Checked(queuePtr);futureId=bigintToI53Checked(futureId);var queue=WebGPU.getJsObject(queuePtr);runtimeKeepalivePush();WebGPU.Internals.futureInsert(futureId,queue.onSubmittedWorkDone().then(()=>{runtimeKeepalivePop();callUserCallback(()=>{_emwgpuOnWorkDoneCompleted(futureId,1)})}))};var _emwgpuWaitAny=function(futurePtr,futureCount,timeoutMSPtr){futurePtr=bigintToI53Checked(futurePtr);futureCount=bigintToI53Checked(futureCount);timeoutMSPtr=bigintToI53Checked(timeoutMSPtr);return Asyncify.handleAsync(async()=>{var promises=[];if(timeoutMSPtr){var timeoutMS=(growMemViews(),HEAP32)[timeoutMSPtr/4];promises.length=futureCount+1;promises[futureCount]=new Promise(resolve=>setTimeout(resolve,timeoutMS,0))}else{promises.length=futureCount}for(var i=0;i<futureCount;++i){var futureId=readI53FromI64(futurePtr+i*8);if(!(futureId in WebGPU.Internals.futures)){return futureId}promises[i]=WebGPU.Internals.futures[futureId]}const firstResolvedFuture=await Promise.race(promises);delete WebGPU.Internals.futures[firstResolvedFuture];return firstResolvedFuture})};_emwgpuWaitAny.isAsync=true;var ENV={};var getExecutableName=()=>thisProgram||"./this.program";var getEnvStrings=()=>{if(!getEnvStrings.strings){var lang=(typeof navigator=="object"&&navigator.language||"C").replace("-","_")+".UTF-8";var env={USER:"web_user",LOGNAME:"web_user",PATH:"/",PWD:"/",HOME:"/home/web_user",LANG:lang,_:getExecutableName()};for(var x in ENV){if(ENV[x]===undefined)delete env[x];else env[x]=ENV[x]}var strings=[];for(var x in env){strings.push(`${x}=${env[x]}`)}getEnvStrings.strings=strings}return getEnvStrings.strings};function _environ_get(__environ,environ_buf){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(14,0,1,__environ,environ_buf);__environ=bigintToI53Checked(__environ);environ_buf=bigintToI53Checked(environ_buf);var bufSize=0;var envp=0;for(var string of getEnvStrings()){var ptr=environ_buf+bufSize;(growMemViews(),HEAPU64)[(__environ+envp)/8]=BigInt(ptr);bufSize+=stringToUTF8(string,ptr,Infinity)+1;envp+=8}return 0}function _environ_sizes_get(penviron_count,penviron_buf_size){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(15,0,1,penviron_count,penviron_buf_size);penviron_count=bigintToI53Checked(penviron_count);penviron_buf_size=bigintToI53Checked(penviron_buf_size);var strings=getEnvStrings();(growMemViews(),HEAPU64)[penviron_count/8]=BigInt(strings.length);var bufSize=0;for(var string of strings){bufSize+=lengthBytesUTF8(string)+1}(growMemViews(),HEAPU64)[penviron_buf_size/8]=BigInt(bufSize);return 0}function _fd_close(fd){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(16,0,1,fd);try{var stream=SYSCALLS.getStreamFromFD(fd);FS.close(stream);return 0}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return e.errno}}var doReadv=(stream,iov,iovcnt,offset)=>{var ret=0;for(var i=0;i<iovcnt;i++){var ptr=Number((growMemViews(),HEAPU64)[iov/8]);var len=Number((growMemViews(),HEAPU64)[(iov+8)/8]);iov+=16;var curr=FS.read(stream,(growMemViews(),HEAP8),ptr,len,offset);if(curr<0)return-1;ret+=curr;if(curr<len)break;if(typeof offset!="undefined"){offset+=curr}}return ret};function _fd_read(fd,iov,iovcnt,pnum){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(17,0,1,fd,iov,iovcnt,pnum);iov=bigintToI53Checked(iov);iovcnt=bigintToI53Checked(iovcnt);pnum=bigintToI53Checked(pnum);try{var stream=SYSCALLS.getStreamFromFD(fd);var num=doReadv(stream,iov,iovcnt);(growMemViews(),HEAPU64)[pnum/8]=BigInt(num);return 0}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return e.errno}}function _fd_seek(fd,offset,whence,newOffset){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(18,0,1,fd,offset,whence,newOffset);offset=bigintToI53Checked(offset);newOffset=bigintToI53Checked(newOffset);try{if(isNaN(offset))return 61;var stream=SYSCALLS.getStreamFromFD(fd);FS.llseek(stream,offset,whence);(growMemViews(),HEAP64)[newOffset/8]=BigInt(stream.position);if(stream.getdents&&offset===0&&whence===0)stream.getdents=null;return 0}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return e.errno}}var doWritev=(stream,iov,iovcnt,offset)=>{var ret=0;for(var i=0;i<iovcnt;i++){var ptr=Number((growMemViews(),HEAPU64)[iov/8]);var len=Number((growMemViews(),HEAPU64)[(iov+8)/8]);iov+=16;var curr=FS.write(stream,(growMemViews(),HEAP8),ptr,len,offset);if(curr<0)return-1;ret+=curr;if(curr<len){break}if(typeof offset!="undefined"){offset+=curr}}return ret};function _fd_write(fd,iov,iovcnt,pnum){if(ENVIRONMENT_IS_PTHREAD)return proxyToMainThread(19,0,1,fd,iov,iovcnt,pnum);iov=bigintToI53Checked(iov);iovcnt=bigintToI53Checked(iovcnt);pnum=bigintToI53Checked(pnum);try{var stream=SYSCALLS.getStreamFromFD(fd);var num=doWritev(stream,iov,iovcnt);(growMemViews(),HEAPU64)[pnum/8]=BigInt(num);return 0}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return e.errno}}function _random_get(buffer,size){buffer=bigintToI53Checked(buffer);size=bigintToI53Checked(size);try{randomFill((growMemViews(),HEAPU8).subarray(buffer,buffer+size));return 0}catch(e){if(typeof FS=="undefined"||!(e.name==="ErrnoError"))throw e;return e.errno}}function _wgpuAdapterGetInfo(adapterPtr,info){adapterPtr=bigintToI53Checked(adapterPtr);info=bigintToI53Checked(info);var adapter=WebGPU.getJsObject(adapterPtr);WebGPU.fillAdapterInfoStruct(adapter.info,info);return 1}function _wgpuAdapterGetLimits(adapterPtr,limitsOutPtr){adapterPtr=bigintToI53Checked(adapterPtr);limitsOutPtr=bigintToI53Checked(limitsOutPtr);var adapter=WebGPU.getJsObject(adapterPtr);WebGPU.fillLimitStruct(adapter.limits,limitsOutPtr);return 1}function _wgpuAdapterHasFeature(adapterPtr,featureEnumValue){adapterPtr=bigintToI53Checked(adapterPtr);var adapter=WebGPU.getJsObject(adapterPtr);return adapter.features.has(WebGPU.FeatureName[featureEnumValue])}var _wgpuBufferGetSize=function(bufferPtr){bufferPtr=bigintToI53Checked(bufferPtr);var ret=(()=>{var buffer=WebGPU.getJsObject(bufferPtr);return buffer.size})();return BigInt(ret)};var _wgpuCommandEncoderBeginComputePass=function(encoderPtr,descriptor){encoderPtr=bigintToI53Checked(encoderPtr);descriptor=bigintToI53Checked(descriptor);var ret=(()=>{var desc;if(descriptor){desc={label:WebGPU.makeStringFromOptionalStringView(descriptor+8),timestampWrites:WebGPU.makePassTimestampWrites(Number((growMemViews(),HEAPU64)[(descriptor+24)/8]))}}var commandEncoder=WebGPU.getJsObject(encoderPtr);var ptr=_emwgpuCreateComputePassEncoder(0n);WebGPU.Internals.jsObjectInsert(ptr,commandEncoder.beginComputePass(desc));return ptr})();return BigInt(ret)};function _wgpuCommandEncoderCopyBufferToBuffer(encoderPtr,srcPtr,srcOffset,dstPtr,dstOffset,size){encoderPtr=bigintToI53Checked(encoderPtr);srcPtr=bigintToI53Checked(srcPtr);srcOffset=bigintToI53Checked(srcOffset);dstPtr=bigintToI53Checked(dstPtr);dstOffset=bigintToI53Checked(dstOffset);size=bigintToI53Checked(size);var commandEncoder=WebGPU.getJsObject(encoderPtr);var src=WebGPU.getJsObject(srcPtr);var dst=WebGPU.getJsObject(dstPtr);commandEncoder.copyBufferToBuffer(src,srcOffset,dst,dstOffset,size)}var _wgpuCommandEncoderFinish=function(encoderPtr,descriptor){encoderPtr=bigintToI53Checked(encoderPtr);descriptor=bigintToI53Checked(descriptor);var ret=(()=>{var commandEncoder=WebGPU.getJsObject(encoderPtr);var ptr=_emwgpuCreateCommandBuffer(0n);WebGPU.Internals.jsObjectInsert(ptr,commandEncoder.finish());return ptr})();return BigInt(ret)};function _wgpuComputePassEncoderDispatchWorkgroups(passPtr,x,y,z){passPtr=bigintToI53Checked(passPtr);var pass=WebGPU.getJsObject(passPtr);pass.dispatchWorkgroups(x,y,z)}function _wgpuComputePassEncoderEnd(passPtr){passPtr=bigintToI53Checked(passPtr);var pass=WebGPU.getJsObject(passPtr);pass.end()}function _wgpuComputePassEncoderSetBindGroup(passPtr,groupIndex,groupPtr,dynamicOffsetCount,dynamicOffsetsPtr){passPtr=bigintToI53Checked(passPtr);groupPtr=bigintToI53Checked(groupPtr);dynamicOffsetCount=bigintToI53Checked(dynamicOffsetCount);dynamicOffsetsPtr=bigintToI53Checked(dynamicOffsetsPtr);var pass=WebGPU.getJsObject(passPtr);var group=WebGPU.getJsObject(groupPtr);if(dynamicOffsetCount==0){pass.setBindGroup(groupIndex,group)}else{pass.setBindGroup(groupIndex,group,(growMemViews(),HEAPU32),dynamicOffsetsPtr/4,dynamicOffsetCount)}}function _wgpuComputePassEncoderSetPipeline(passPtr,pipelinePtr){passPtr=bigintToI53Checked(passPtr);pipelinePtr=bigintToI53Checked(pipelinePtr);var pass=WebGPU.getJsObject(passPtr);var pipeline=WebGPU.getJsObject(pipelinePtr);pass.setPipeline(pipeline)}var _wgpuComputePipelineGetBindGroupLayout=function(pipelinePtr,groupIndex){pipelinePtr=bigintToI53Checked(pipelinePtr);var ret=(()=>{var pipeline=WebGPU.getJsObject(pipelinePtr);var ptr=_emwgpuCreateBindGroupLayout(0n);WebGPU.Internals.jsObjectInsert(ptr,pipeline.getBindGroupLayout(groupIndex));return ptr})();return BigInt(ret)};var _wgpuDeviceCreateBindGroup=function(devicePtr,descriptor){devicePtr=bigintToI53Checked(devicePtr);descriptor=bigintToI53Checked(descriptor);var ret=(()=>{function makeEntry(entryPtr){var bufferPtr=Number((growMemViews(),HEAPU64)[(entryPtr+16)/8]);var samplerPtr=Number((growMemViews(),HEAPU64)[(entryPtr+40)/8]);var textureViewPtr=Number((growMemViews(),HEAPU64)[(entryPtr+48)/8]);var externalTexturePtr=0;WebGPU.iterateExtensions(entryPtr,{327681:ptr=>{externalTexturePtr=Number((growMemViews(),HEAPU64)[(ptr+16)/8])}});var resource;if(bufferPtr){var size=readI53FromI64(entryPtr+32);if(size==-1)size=undefined;resource={buffer:WebGPU.getJsObject(bufferPtr),offset:readI53FromI64(entryPtr+24),size}}else{resource=WebGPU.getJsObject(samplerPtr||textureViewPtr||externalTexturePtr)}return{binding:(growMemViews(),HEAPU32)[(entryPtr+8)/4],resource}}function makeEntries(count,entriesPtrs){var entries=[];for(var i=0;i<count;++i){entries.push(makeEntry(entriesPtrs+56*i))}return entries}var desc={label:WebGPU.makeStringFromOptionalStringView(descriptor+8),layout:WebGPU.getJsObject(Number((growMemViews(),HEAPU64)[(descriptor+24)/8])),entries:makeEntries(Number((growMemViews(),HEAPU64)[(descriptor+32)/8]),Number((growMemViews(),HEAPU64)[(descriptor+40)/8]))};var device=WebGPU.getJsObject(devicePtr);var ptr=_emwgpuCreateBindGroup(0n);WebGPU.Internals.jsObjectInsert(ptr,device.createBindGroup(desc));return ptr})();return BigInt(ret)};var _wgpuDeviceCreateCommandEncoder=function(devicePtr,descriptor){devicePtr=bigintToI53Checked(devicePtr);descriptor=bigintToI53Checked(descriptor);var ret=(()=>{var desc;if(descriptor){desc={label:WebGPU.makeStringFromOptionalStringView(descriptor+8)}}var device=WebGPU.getJsObject(devicePtr);var ptr=_emwgpuCreateCommandEncoder(0n);WebGPU.Internals.jsObjectInsert(ptr,device.createCommandEncoder(desc));return ptr})();return BigInt(ret)};var _wgpuDeviceCreateComputePipeline=function(devicePtr,descriptor){devicePtr=bigintToI53Checked(devicePtr);descriptor=bigintToI53Checked(descriptor);var ret=(()=>{var desc=WebGPU.makeComputePipelineDesc(descriptor);var device=WebGPU.getJsObject(devicePtr);var ptr=_emwgpuCreateComputePipeline(0n);WebGPU.Internals.jsObjectInsert(ptr,device.createComputePipeline(desc));return ptr})();return BigInt(ret)};function _wgpuInstanceHasWGSLLanguageFeature(instance,featureEnumValue){instance=bigintToI53Checked(instance);if(!("wgslLanguageFeatures"in navigator.gpu)){return false}return navigator.gpu.wgslLanguageFeatures.has(WebGPU.WGSLLanguageFeatureName[featureEnumValue])}var _wgpuQueueSubmit=function(queuePtr,commandCount,commands){queuePtr=bigintToI53Checked(queuePtr);commandCount=bigintToI53Checked(commandCount);commands=bigintToI53Checked(commands);var queue=WebGPU.getJsObject(queuePtr);var cmds=Array.from((growMemViews(),HEAP64).subarray(commands/8,(commands+commandCount*8)/8),id=>WebGPU.getJsObject(id));queue.submit(cmds)};function _wgpuQueueWriteBuffer(queuePtr,bufferPtr,bufferOffset,data,size){queuePtr=bigintToI53Checked(queuePtr);bufferPtr=bigintToI53Checked(bufferPtr);bufferOffset=bigintToI53Checked(bufferOffset);data=bigintToI53Checked(data);size=bigintToI53Checked(size);var queue=WebGPU.getJsObject(queuePtr);var buffer=WebGPU.getJsObject(bufferPtr);var subarray=(growMemViews(),HEAPU8).subarray(data,data+size);queue.writeBuffer(buffer,bufferOffset,subarray,0,size)}var Asyncify={instrumentWasmImports(imports){var importPattern=/^(invoke_.*|__asyncjs__.*)$/;for(let[x,original]of Object.entries(imports)){if(typeof original=="function"){let isAsyncifyImport=original.isAsync||importPattern.test(x);if(isAsyncifyImport){imports[x]=original=new WebAssembly.Suspending(original)}}}},instrumentFunction(original){var wrapper=(...args)=>original(...args);return wrapper},instrumentWasmExports(exports){var exportPattern=/^(wllama_start|wllama_action|main|__main_argc_argv)$/;Asyncify.asyncExports=new Set;var ret={};for(let[x,original]of Object.entries(exports)){if(typeof original=="function"){let isAsyncifyExport=exportPattern.test(x);if(isAsyncifyExport){Asyncify.asyncExports.add(original);original=Asyncify.makeAsyncFunction(original)}var wrapper=Asyncify.instrumentFunction(original);ret[x]=wrapper}else{ret[x]=original}}return ret},asyncExports:null,isAsyncExport(func){return Asyncify.asyncExports?.has(func)},handleAsync:async startAsync=>{runtimeKeepalivePush();try{return await startAsync()}finally{runtimeKeepalivePop()}},handleSleep:startAsync=>Asyncify.handleAsync(()=>new Promise(startAsync)),makeAsyncFunction(original){return WebAssembly.promising(original)}};var getCFunc=ident=>{var func=Module["_"+ident];return func};var writeArrayToMemory=(array,buffer)=>{(growMemViews(),HEAP8).set(array,buffer)};var ccall=(ident,returnType,argTypes,args,opts)=>{var toC={pointer:p=>BigInt(p),string:str=>{var ret=0;if(str!==null&&str!==undefined&&str!==0){ret=stringToUTF8OnStack(str)}return BigInt(ret)},array:arr=>{var ret=stackAlloc(arr.length);writeArrayToMemory(arr,ret);return BigInt(ret)}};function convertReturnValue(ret){if(returnType==="string"){return UTF8ToString(Number(ret))}if(returnType==="pointer")return Number(ret);if(returnType==="boolean")return Boolean(ret);return ret}var func=getCFunc(ident);var cArgs=[];var stack=0;if(args){for(var i=0;i<args.length;i++){var converter=toC[argTypes[i]];if(converter){if(stack===0)stack=stackSave();cArgs[i]=converter(args[i])}else{cArgs[i]=args[i]}}}var ret=func(...cArgs);function onDone(ret){if(stack!==0)stackRestore(stack);return convertReturnValue(ret)}var asyncMode=opts?.async;if(asyncMode)return ret.then(onDone);ret=onDone(ret);return ret};var cwrap=(ident,returnType,argTypes,opts)=>{var numericArgs=!argTypes||argTypes.every(type=>type==="number"||type==="boolean");var numericRet=returnType!=="string";if(numericRet&&numericArgs&&!opts){return getCFunc(ident)}return(...args)=>ccall(ident,returnType,argTypes,args,opts)};var FS_createPath=(...args)=>FS.createPath(...args);var FS_unlink=(...args)=>FS.unlink(...args);var FS_createLazyFile=(...args)=>FS.createLazyFile(...args);var FS_createDevice=(...args)=>FS.createDevice(...args);PThread.init();FS.createPreloadedFile=FS_createPreloadedFile;FS.preloadFile=FS_preloadFile;FS.staticInit();{initMemory();if(Module["noExitRuntime"])noExitRuntime=Module["noExitRuntime"];if(Module["preloadPlugins"])preloadPlugins=Module["preloadPlugins"];if(Module["print"])out=Module["print"];if(Module["printErr"])err=Module["printErr"];if(Module["wasmBinary"])wasmBinary=Module["wasmBinary"];if(Module["arguments"])arguments_=Module["arguments"];if(Module["thisProgram"])thisProgram=Module["thisProgram"];if(Module["preInit"]){if(typeof Module["preInit"]=="function")Module["preInit"]=[Module["preInit"]];while(Module["preInit"].length>0){Module["preInit"].shift()()}}}Module["ENV"]=ENV;Module["mmapAlloc"]=mmapAlloc;Module["wasmMemory"]=wasmMemory;Module["addRunDependency"]=addRunDependency;Module["removeRunDependency"]=removeRunDependency;Module["ccall"]=ccall;Module["cwrap"]=cwrap;Module["FS_preloadFile"]=FS_preloadFile;Module["FS_unlink"]=FS_unlink;Module["FS_createPath"]=FS_createPath;Module["FS_createDevice"]=FS_createDevice;Module["FS"]=FS;Module["FS_createDataFile"]=FS_createDataFile;Module["FS_createLazyFile"]=FS_createLazyFile;Module["MEMFS"]=MEMFS;var proxiedFunctionTable=[_proc_exit,exitOnMainThread,pthreadCreateProxied,___syscall_fcntl64,___syscall_fstat64,___syscall_getcwd,___syscall_getdents64,___syscall_ioctl,___syscall_lstat64,___syscall_newfstatat,___syscall_openat,___syscall_stat64,__mmap_js,__munmap_js,_environ_get,_environ_sizes_get,_fd_close,_fd_read,_fd_seek,_fd_write];function __asyncjs__js_file_read(path_ptr,offset,req_size,out_ptr){return Asyncify.handleAsync(async()=>await _wllama_js_file_read(UTF8ToString(Number(path_ptr)),Number(offset),Number(req_size),Number(out_ptr)))}__asyncjs__js_file_read.sig="jjjjj";var _malloc,_free,_wllama_malloc,_wllama_start,_wllama_action,_wllama_exit,_wllama_debug,_main,_emwgpuCreateBindGroup,_emwgpuCreateBindGroupLayout,_emwgpuCreateCommandBuffer,_emwgpuCreateCommandEncoder,_emwgpuCreateComputePassEncoder,_emwgpuCreateComputePipeline,_emwgpuCreateExternalTexture,_emwgpuCreatePipelineLayout,_emwgpuCreateQuerySet,_emwgpuCreateRenderBundle,_emwgpuCreateRenderBundleEncoder,_emwgpuCreateRenderPassEncoder,_emwgpuCreateRenderPipeline,_emwgpuCreateSampler,_emwgpuCreateSurface,_emwgpuCreateTexture,_emwgpuCreateTextureView,_emwgpuCreateAdapter,_emwgpuCreateBuffer,_emwgpuCreateDevice,_emwgpuCreateQueue,_emwgpuCreateShaderModule,_emwgpuOnDeviceLostCompleted,_emwgpuOnMapAsyncCompleted,_emwgpuOnRequestAdapterCompleted,_emwgpuOnRequestDeviceCompleted,_emwgpuOnWorkDoneCompleted,_emwgpuOnUncapturedError,__emscripten_tls_init,_pthread_self,_emscripten_builtin_memalign,__emscripten_thread_init,__emscripten_thread_crashed,__emscripten_run_js_on_main_thread,__emscripten_thread_free_data,__emscripten_thread_exit,__emscripten_check_mailbox,_memalign,___trap,_emscripten_stack_set_limits,__emscripten_stack_restore,__emscripten_stack_alloc,_emscripten_stack_get_current,__indirect_function_table,wasmTable;function assignWasmExports(wasmExports){_malloc=wasmExports["malloc"];_free=wasmExports["free"];_wllama_malloc=Module["_wllama_malloc"]=wasmExports["wllama_malloc"];_wllama_start=Module["_wllama_start"]=wasmExports["wllama_start"];_wllama_action=Module["_wllama_action"]=wasmExports["wllama_action"];_wllama_exit=Module["_wllama_exit"]=wasmExports["wllama_exit"];_wllama_debug=Module["_wllama_debug"]=wasmExports["wllama_debug"];_main=Module["_main"]=wasmExports["main"];_emwgpuCreateBindGroup=wasmExports["emwgpuCreateBindGroup"];_emwgpuCreateBindGroupLayout=wasmExports["emwgpuCreateBindGroupLayout"];_emwgpuCreateCommandBuffer=wasmExports["emwgpuCreateCommandBuffer"];_emwgpuCreateCommandEncoder=wasmExports["emwgpuCreateCommandEncoder"];_emwgpuCreateComputePassEncoder=wasmExports["emwgpuCreateComputePassEncoder"];_emwgpuCreateComputePipeline=wasmExports["emwgpuCreateComputePipeline"];_emwgpuCreateExternalTexture=wasmExports["emwgpuCreateExternalTexture"];_emwgpuCreatePipelineLayout=wasmExports["emwgpuCreatePipelineLayout"];_emwgpuCreateQuerySet=wasmExports["emwgpuCreateQuerySet"];_emwgpuCreateRenderBundle=wasmExports["emwgpuCreateRenderBundle"];_emwgpuCreateRenderBundleEncoder=wasmExports["emwgpuCreateRenderBundleEncoder"];_emwgpuCreateRenderPassEncoder=wasmExports["emwgpuCreateRenderPassEncoder"];_emwgpuCreateRenderPipeline=wasmExports["emwgpuCreateRenderPipeline"];_emwgpuCreateSampler=wasmExports["emwgpuCreateSampler"];_emwgpuCreateSurface=wasmExports["emwgpuCreateSurface"];_emwgpuCreateTexture=wasmExports["emwgpuCreateTexture"];_emwgpuCreateTextureView=wasmExports["emwgpuCreateTextureView"];_emwgpuCreateAdapter=wasmExports["emwgpuCreateAdapter"];_emwgpuCreateBuffer=wasmExports["emwgpuCreateBuffer"];_emwgpuCreateDevice=wasmExports["emwgpuCreateDevice"];_emwgpuCreateQueue=wasmExports["emwgpuCreateQueue"];_emwgpuCreateShaderModule=wasmExports["emwgpuCreateShaderModule"];_emwgpuOnDeviceLostCompleted=wasmExports["emwgpuOnDeviceLostCompleted"];_emwgpuOnMapAsyncCompleted=wasmExports["emwgpuOnMapAsyncCompleted"];_emwgpuOnRequestAdapterCompleted=wasmExports["emwgpuOnRequestAdapterCompleted"];_emwgpuOnRequestDeviceCompleted=wasmExports["emwgpuOnRequestDeviceCompleted"];_emwgpuOnWorkDoneCompleted=wasmExports["emwgpuOnWorkDoneCompleted"];_emwgpuOnUncapturedError=wasmExports["emwgpuOnUncapturedError"];__emscripten_tls_init=wasmExports["_emscripten_tls_init"];_pthread_self=wasmExports["pthread_self"];_emscripten_builtin_memalign=wasmExports["emscripten_builtin_memalign"];__emscripten_thread_init=wasmExports["_emscripten_thread_init"];__emscripten_thread_crashed=wasmExports["_emscripten_thread_crashed"];__emscripten_run_js_on_main_thread=wasmExports["_emscripten_run_js_on_main_thread"];__emscripten_thread_free_data=wasmExports["_emscripten_thread_free_data"];__emscripten_thread_exit=wasmExports["_emscripten_thread_exit"];__emscripten_check_mailbox=wasmExports["_emscripten_check_mailbox"];_memalign=wasmExports["memalign"];___trap=wasmExports["__trap"];_emscripten_stack_set_limits=wasmExports["emscripten_stack_set_limits"];__emscripten_stack_restore=wasmExports["_emscripten_stack_restore"];__emscripten_stack_alloc=wasmExports["_emscripten_stack_alloc"];_emscripten_stack_get_current=wasmExports["emscripten_stack_get_current"];__indirect_function_table=wasmTable=wasmExports["__indirect_function_table"]}var wasmImports;function assignWasmImports(){wasmImports={__asyncjs__js_file_read,__pthread_create_js:___pthread_create_js,__syscall_fcntl64:___syscall_fcntl64,__syscall_getcwd:___syscall_getcwd,__syscall_getdents64:___syscall_getdents64,__syscall_ioctl:___syscall_ioctl,__syscall_openat:___syscall_openat,__syscall_stat64:___syscall_stat64,_abort_js:__abort_js,_emscripten_init_main_thread_js:__emscripten_init_main_thread_js,_emscripten_notify_mailbox_postmessage:__emscripten_notify_mailbox_postmessage,_emscripten_receive_on_main_thread_js:__emscripten_receive_on_main_thread_js,_emscripten_thread_cleanup:__emscripten_thread_cleanup,_emscripten_thread_mailbox_await:__emscripten_thread_mailbox_await,_emscripten_thread_set_strongref:__emscripten_thread_set_strongref,_localtime_js:__localtime_js,_mmap_js:__mmap_js,_munmap_js:__munmap_js,_tzset_js:__tzset_js,clock_time_get:_clock_time_get,emscripten_check_blocking_allowed:_emscripten_check_blocking_allowed,emscripten_date_now:_emscripten_date_now,emscripten_exit_with_live_runtime:_emscripten_exit_with_live_runtime,emscripten_get_callstack:_emscripten_get_callstack,emscripten_get_heap_max:_emscripten_get_heap_max,emscripten_get_now:_emscripten_get_now,emscripten_has_asyncify:_emscripten_has_asyncify,emscripten_num_logical_cores:_emscripten_num_logical_cores,emscripten_resize_heap:_emscripten_resize_heap,emwgpuAdapterRequestDevice:_emwgpuAdapterRequestDevice,emwgpuBufferDestroy:_emwgpuBufferDestroy,emwgpuBufferGetConstMappedRange:_emwgpuBufferGetConstMappedRange,emwgpuBufferMapAsync:_emwgpuBufferMapAsync,emwgpuBufferUnmap:_emwgpuBufferUnmap,emwgpuDelete:_emwgpuDelete,emwgpuDeviceCreateBuffer:_emwgpuDeviceCreateBuffer,emwgpuDeviceCreateShaderModule:_emwgpuDeviceCreateShaderModule,emwgpuDeviceDestroy:_emwgpuDeviceDestroy,emwgpuInstanceRequestAdapter:_emwgpuInstanceRequestAdapter,emwgpuQueueOnSubmittedWorkDone:_emwgpuQueueOnSubmittedWorkDone,emwgpuWaitAny:_emwgpuWaitAny,environ_get:_environ_get,environ_sizes_get:_environ_sizes_get,exit:_exit,fd_close:_fd_close,fd_read:_fd_read,fd_seek:_fd_seek,fd_write:_fd_write,memory:wasmMemory,random_get:_random_get,wgpuAdapterGetInfo:_wgpuAdapterGetInfo,wgpuAdapterGetLimits:_wgpuAdapterGetLimits,wgpuAdapterHasFeature:_wgpuAdapterHasFeature,wgpuBufferGetSize:_wgpuBufferGetSize,wgpuCommandEncoderBeginComputePass:_wgpuCommandEncoderBeginComputePass,wgpuCommandEncoderCopyBufferToBuffer:_wgpuCommandEncoderCopyBufferToBuffer,wgpuCommandEncoderFinish:_wgpuCommandEncoderFinish,wgpuComputePassEncoderDispatchWorkgroups:_wgpuComputePassEncoderDispatchWorkgroups,wgpuComputePassEncoderEnd:_wgpuComputePassEncoderEnd,wgpuComputePassEncoderSetBindGroup:_wgpuComputePassEncoderSetBindGroup,wgpuComputePassEncoderSetPipeline:_wgpuComputePassEncoderSetPipeline,wgpuComputePipelineGetBindGroupLayout:_wgpuComputePipelineGetBindGroupLayout,wgpuDeviceCreateBindGroup:_wgpuDeviceCreateBindGroup,wgpuDeviceCreateCommandEncoder:_wgpuDeviceCreateCommandEncoder,wgpuDeviceCreateComputePipeline:_wgpuDeviceCreateComputePipeline,wgpuInstanceHasWGSLLanguageFeature:_wgpuInstanceHasWGSLLanguageFeature,wgpuQueueSubmit:_wgpuQueueSubmit,wgpuQueueWriteBuffer:_wgpuQueueWriteBuffer}}function applySignatureConversions(wasmExports){wasmExports=Object.assign({},wasmExports);var makeWrapper_pp=f=>a0=>Number(f(BigInt(a0)));var makeWrapper__p=f=>a0=>f(BigInt(a0));var makeWrapper___PP=f=>(a0,a1,a2)=>f(a0,BigInt(a1?a1:0),BigInt(a2?a2:0));var makeWrapper_p=f=>()=>Number(f());var makeWrapper_ppp=f=>(a0,a1)=>Number(f(BigInt(a0),BigInt(a1)));var makeWrapper__p_____=f=>(a0,a1,a2,a3,a4,a5)=>f(BigInt(a0),a1,a2,a3,a4,a5);var makeWrapper___p_p_=f=>(a0,a1,a2,a3,a4)=>f(a0,BigInt(a1),a2,BigInt(a3),a4);var makeWrapper__pp=f=>(a0,a1)=>f(BigInt(a0),BigInt(a1));wasmExports["malloc"]=makeWrapper_pp(wasmExports["malloc"]);wasmExports["free"]=makeWrapper__p(wasmExports["free"]);wasmExports["main"]=makeWrapper___PP(wasmExports["main"]);wasmExports["pthread_self"]=makeWrapper_p(wasmExports["pthread_self"]);wasmExports["emscripten_builtin_memalign"]=makeWrapper_ppp(wasmExports["emscripten_builtin_memalign"]);wasmExports["_emscripten_thread_init"]=makeWrapper__p_____(wasmExports["_emscripten_thread_init"]);wasmExports["_emscripten_run_js_on_main_thread"]=makeWrapper___p_p_(wasmExports["_emscripten_run_js_on_main_thread"]);wasmExports["_emscripten_thread_free_data"]=makeWrapper__p(wasmExports["_emscripten_thread_free_data"]);wasmExports["_emscripten_thread_exit"]=makeWrapper__p(wasmExports["_emscripten_thread_exit"]);wasmExports["memalign"]=makeWrapper_ppp(wasmExports["memalign"]);wasmExports["emscripten_stack_set_limits"]=makeWrapper__pp(wasmExports["emscripten_stack_set_limits"]);wasmExports["_emscripten_stack_restore"]=makeWrapper__p(wasmExports["_emscripten_stack_restore"]);wasmExports["_emscripten_stack_alloc"]=makeWrapper_pp(wasmExports["_emscripten_stack_alloc"]);wasmExports["emscripten_stack_get_current"]=makeWrapper_p(wasmExports["emscripten_stack_get_current"]);return wasmExports}async function callMain(){var entryFunction=_main;var argc=0;var argv=0;try{var ret=entryFunction(argc,BigInt(argv));ret=await ret;exitJS(ret,true);return ret}catch(e){return handleException(e)}}function run(){if(runDependencies>0){dependenciesFulfilled=run;return}if(ENVIRONMENT_IS_PTHREAD){initRuntime();return}preRun();if(runDependencies>0){dependenciesFulfilled=run;return}async function doRun(){Module["calledRun"]=true;if(ABORT)return;initRuntime();preMain();Module["onRuntimeInitialized"]?.();var noInitialRun=Module["noInitialRun"]||false;if(!noInitialRun)await callMain();postRun()}if(Module["setStatus"]){Module["setStatus"]("Running...");setTimeout(()=>{setTimeout(()=>Module["setStatus"](""),1);doRun()},1)}else{doRun()}}var wasmExports;if(!ENVIRONMENT_IS_PTHREAD){createWasm();run()}\n';
|
|
1275
|
+
|
|
1276
|
+
// src/worker.ts
|
|
1277
|
+
var FILE_READ_REQ_EVENT = "fs.read_req";
|
|
1278
|
+
var JSPI_STUB = `
|
|
1279
|
+
if (!WebAssembly.Suspending) {
|
|
1280
|
+
// JSPI not available - stubs that keep the import/export tables valid.
|
|
1281
|
+
// Suspending wraps imports: identity is fine since async imports won't be called.
|
|
1282
|
+
WebAssembly.Suspending = function (fn) {
|
|
1283
|
+
// console.log(fn.toString());
|
|
1284
|
+
return fn;
|
|
1285
|
+
};
|
|
1286
|
+
// promising wraps exports: must return a Promise so ccall's ret.then() works.
|
|
1287
|
+
WebAssembly.promising = function (fn) {
|
|
1288
|
+
return function (...args) {
|
|
1289
|
+
try {
|
|
1290
|
+
return Promise.resolve(fn(...args));
|
|
1291
|
+
} catch (e) {
|
|
1292
|
+
return Promise.reject(e);
|
|
1293
|
+
}
|
|
1294
|
+
};
|
|
1295
|
+
};
|
|
1296
|
+
}
|
|
1297
|
+
`;
|
|
1298
|
+
var ProxyToWorker = class {
|
|
1299
|
+
// filename -> Blob for async reads
|
|
1300
|
+
constructor(resources, nbThread, suppressNativeLog, logger) {
|
|
1301
|
+
__publicField(this, "resources");
|
|
1302
|
+
__publicField(this, "logger");
|
|
1303
|
+
__publicField(this, "suppressNativeLog");
|
|
1304
|
+
__publicField(this, "taskQueue", []);
|
|
1305
|
+
__publicField(this, "taskId", 1);
|
|
1306
|
+
__publicField(this, "resultQueue", []);
|
|
1307
|
+
__publicField(this, "busy", false);
|
|
1308
|
+
// is the work loop is running?
|
|
1309
|
+
__publicField(this, "worker");
|
|
1310
|
+
__publicField(this, "multiThread");
|
|
1311
|
+
__publicField(this, "nbThread");
|
|
1312
|
+
__publicField(this, "useAsyncFile");
|
|
1313
|
+
__publicField(this, "fileBlobs", /* @__PURE__ */ new Map());
|
|
1314
|
+
this.resources = resources;
|
|
1315
|
+
this.nbThread = nbThread;
|
|
1316
|
+
this.multiThread = nbThread > 0;
|
|
1317
|
+
this.logger = logger;
|
|
1318
|
+
this.suppressNativeLog = suppressNativeLog;
|
|
1319
|
+
this.useAsyncFile = canUseAsyncFileRead(resources.compat);
|
|
1320
|
+
}
|
|
1321
|
+
getModuleCode() {
|
|
1322
|
+
return __async(this, null, function* () {
|
|
1323
|
+
if (!this.resources.jsPath) {
|
|
1324
|
+
if (this.resources.compat) {
|
|
1325
|
+
throw new Error(
|
|
1326
|
+
"compat mode is enabled but no jsPath was provided. Pass a worker JS via setCompat() or install @wllama/wllama-compat."
|
|
1327
|
+
);
|
|
1328
|
+
}
|
|
1329
|
+
return WLLAMA_EMSCRIPTEN_CODE;
|
|
1330
|
+
} else if (this.resources.jsPath.code) {
|
|
1331
|
+
return this.resources.jsPath.code;
|
|
1332
|
+
} else if (isString(this.resources.jsPath)) {
|
|
1333
|
+
const response = yield fetch(this.resources.jsPath);
|
|
1334
|
+
if (!response.ok) {
|
|
1335
|
+
throw new Error(
|
|
1336
|
+
`Failed to fetch worker code from ${this.resources.jsPath}`
|
|
1337
|
+
);
|
|
1338
|
+
}
|
|
1339
|
+
return yield response.text();
|
|
1340
|
+
} else {
|
|
1341
|
+
throw new Error("No JS code provided for worker");
|
|
1342
|
+
}
|
|
1343
|
+
});
|
|
1344
|
+
}
|
|
1345
|
+
moduleInit(ggufFiles) {
|
|
1346
|
+
return __async(this, null, function* () {
|
|
1347
|
+
let moduleCode = JSPI_STUB + (yield this.getModuleCode());
|
|
1348
|
+
let mainModuleCode = moduleCode.replace("var Module", "var ___Module");
|
|
1349
|
+
const runOptions = {
|
|
1350
|
+
pathConfig: {
|
|
1351
|
+
"wllama.wasm": this.resources.wasmPath
|
|
1352
|
+
},
|
|
1353
|
+
nbThread: this.nbThread,
|
|
1354
|
+
compat: this.resources.compat
|
|
1355
|
+
};
|
|
1356
|
+
const completeCode = [
|
|
1357
|
+
`const RUN_OPTIONS = ${JSON.stringify(runOptions)};`,
|
|
1358
|
+
`function wModuleInit() { ${mainModuleCode}; return Module; }`,
|
|
1359
|
+
LLAMA_CPP_WORKER_CODE
|
|
1360
|
+
].join(";\n\n");
|
|
1361
|
+
this.worker = createWorker(completeCode);
|
|
1362
|
+
this.worker.onmessage = this.onRecvMsg.bind(this);
|
|
1363
|
+
this.worker.onerror = this.logger.error;
|
|
1364
|
+
const res = yield this.pushTask({
|
|
1365
|
+
verb: "module.init",
|
|
1366
|
+
args: [
|
|
1367
|
+
new Blob([moduleCode], { type: "text/javascript" }),
|
|
1368
|
+
this.useAsyncFile
|
|
1369
|
+
],
|
|
1370
|
+
callbackId: this.taskId++
|
|
1371
|
+
});
|
|
1372
|
+
const nativeFiles = [];
|
|
1373
|
+
for (const file of ggufFiles) {
|
|
1374
|
+
const needAllocBuffer = !this.useAsyncFile;
|
|
1375
|
+
const id = yield this.fileAlloc(
|
|
1376
|
+
file.name,
|
|
1377
|
+
file.blob.size,
|
|
1378
|
+
needAllocBuffer
|
|
1379
|
+
);
|
|
1380
|
+
nativeFiles.push(__spreadValues({ id }, file));
|
|
1381
|
+
if (this.useAsyncFile) {
|
|
1382
|
+
this.fileBlobs.set(file.name, file.blob);
|
|
1383
|
+
}
|
|
1384
|
+
}
|
|
1385
|
+
if (!this.useAsyncFile) {
|
|
1386
|
+
yield Promise.all(
|
|
1387
|
+
nativeFiles.map((file) => {
|
|
1388
|
+
return this.fileWrite(file.id, file.blob);
|
|
1389
|
+
})
|
|
1390
|
+
);
|
|
1391
|
+
}
|
|
1392
|
+
return res;
|
|
1393
|
+
});
|
|
1394
|
+
}
|
|
1395
|
+
wllamaStart() {
|
|
1396
|
+
return __async(this, null, function* () {
|
|
1397
|
+
const result = yield this.pushTask({
|
|
1398
|
+
verb: "wllama.start",
|
|
1399
|
+
args: [],
|
|
1400
|
+
callbackId: this.taskId++
|
|
1401
|
+
});
|
|
1402
|
+
const parsedResult = this.parseResult(result);
|
|
1403
|
+
return parsedResult;
|
|
1404
|
+
});
|
|
1405
|
+
}
|
|
1406
|
+
wllamaAction(name, body) {
|
|
1407
|
+
return __async(this, null, function* () {
|
|
1408
|
+
const encodedMsg = glueSerialize(body);
|
|
1409
|
+
const result = yield this.pushTask({
|
|
1410
|
+
verb: "wllama.action",
|
|
1411
|
+
args: [name, encodedMsg],
|
|
1412
|
+
callbackId: this.taskId++
|
|
1413
|
+
});
|
|
1414
|
+
const parsedResult = glueDeserialize(result);
|
|
1415
|
+
return parsedResult;
|
|
1416
|
+
});
|
|
1417
|
+
}
|
|
1418
|
+
wllamaExit() {
|
|
1419
|
+
return __async(this, null, function* () {
|
|
1420
|
+
if (this.worker) {
|
|
1421
|
+
this.worker.terminate();
|
|
1422
|
+
}
|
|
1423
|
+
});
|
|
1424
|
+
}
|
|
1425
|
+
wllamaDebug() {
|
|
1426
|
+
return __async(this, null, function* () {
|
|
1427
|
+
const result = yield this.pushTask({
|
|
1428
|
+
verb: "wllama.debug",
|
|
1429
|
+
args: [],
|
|
1430
|
+
callbackId: this.taskId++
|
|
1431
|
+
});
|
|
1432
|
+
return JSON.parse(result);
|
|
1433
|
+
});
|
|
1434
|
+
}
|
|
1435
|
+
///////////////////////////////////////
|
|
1436
|
+
/**
|
|
1437
|
+
* Allocate a new file in heapfs
|
|
1438
|
+
* @returns fileId, to be used by fileWrite()
|
|
1439
|
+
*/
|
|
1440
|
+
fileAlloc(fileName, size, allocBuffer) {
|
|
1441
|
+
return __async(this, null, function* () {
|
|
1442
|
+
const result = yield this.pushTask({
|
|
1443
|
+
verb: "fs.alloc",
|
|
1444
|
+
args: [fileName, size, allocBuffer],
|
|
1445
|
+
callbackId: this.taskId++
|
|
1446
|
+
});
|
|
1447
|
+
return result.fileId;
|
|
1448
|
+
});
|
|
1449
|
+
}
|
|
1450
|
+
/**
|
|
1451
|
+
* Write a Blob to heapfs
|
|
1452
|
+
*/
|
|
1453
|
+
fileWrite(fileId, blob) {
|
|
1454
|
+
return __async(this, null, function* () {
|
|
1455
|
+
const reader = blob.stream().getReader();
|
|
1456
|
+
let offset = 0;
|
|
1457
|
+
while (true) {
|
|
1458
|
+
const { done, value } = yield reader.read();
|
|
1459
|
+
if (done) break;
|
|
1460
|
+
const size = value.byteLength;
|
|
1461
|
+
yield this.pushTask(
|
|
1462
|
+
{
|
|
1463
|
+
verb: "fs.write",
|
|
1464
|
+
args: [fileId, value, offset],
|
|
1465
|
+
callbackId: this.taskId++
|
|
1466
|
+
},
|
|
1467
|
+
// @ts-ignore Type 'ArrayBufferLike' is not assignable to type 'ArrayBuffer'
|
|
1468
|
+
[value.buffer]
|
|
1469
|
+
);
|
|
1470
|
+
offset += size;
|
|
1471
|
+
}
|
|
1472
|
+
});
|
|
1473
|
+
}
|
|
1474
|
+
fileReadResponse(name, offset, size) {
|
|
1475
|
+
return __async(this, null, function* () {
|
|
1476
|
+
var _a;
|
|
1477
|
+
try {
|
|
1478
|
+
const blob = this.fileBlobs.get(name);
|
|
1479
|
+
if (!blob) {
|
|
1480
|
+
throw new Error(`blob not found for name="${name}"`);
|
|
1481
|
+
}
|
|
1482
|
+
const chunk = blob.slice(offset, offset + size);
|
|
1483
|
+
const buffer = yield chunk.arrayBuffer();
|
|
1484
|
+
this.worker.postMessage(
|
|
1485
|
+
{ verb: "fs.read_res", args: [buffer] },
|
|
1486
|
+
{ transfer: [buffer] }
|
|
1487
|
+
);
|
|
1488
|
+
} catch (err) {
|
|
1489
|
+
this.logger.error("fileReadResponse failed, terminating worker:", err);
|
|
1490
|
+
(_a = this.worker) == null ? void 0 : _a.terminate();
|
|
1491
|
+
this.worker = void 0;
|
|
1492
|
+
this.abort(`File read failed: ${err}`, err.stack || "");
|
|
1493
|
+
}
|
|
1494
|
+
});
|
|
1495
|
+
}
|
|
1496
|
+
/**
|
|
1497
|
+
* Parse JSON result returned by cpp code.
|
|
1498
|
+
* Throw new Error if "__exception" is present in the response
|
|
1499
|
+
*
|
|
1500
|
+
* TODO: get rid of this function once everything is migrated to Glue
|
|
1501
|
+
*/
|
|
1502
|
+
parseResult(result) {
|
|
1503
|
+
const parsedResult = JSON.parse(result);
|
|
1504
|
+
if (parsedResult && parsedResult["error"]) {
|
|
1505
|
+
throw new WllamaRuntimeError("Unknown error, please see console.log", "");
|
|
1506
|
+
}
|
|
1507
|
+
return parsedResult;
|
|
1508
|
+
}
|
|
1509
|
+
/**
|
|
1510
|
+
* Push a new task to taskQueue
|
|
1511
|
+
*/
|
|
1512
|
+
pushTask(param, buffers) {
|
|
1513
|
+
return new Promise((resolve, reject) => {
|
|
1514
|
+
this.taskQueue.push({ resolve, reject, param, buffers });
|
|
1515
|
+
this.runTaskLoop();
|
|
1516
|
+
});
|
|
1517
|
+
}
|
|
1518
|
+
/**
|
|
1519
|
+
* Main loop for processing tasks
|
|
1520
|
+
*/
|
|
1521
|
+
runTaskLoop() {
|
|
1522
|
+
return __async(this, null, function* () {
|
|
1523
|
+
var _a;
|
|
1524
|
+
if (this.busy) {
|
|
1525
|
+
return;
|
|
1526
|
+
}
|
|
1527
|
+
this.busy = true;
|
|
1528
|
+
while (true) {
|
|
1529
|
+
const task = this.taskQueue.shift();
|
|
1530
|
+
if (!task) break;
|
|
1531
|
+
this.resultQueue.push(task);
|
|
1532
|
+
this.worker.postMessage(
|
|
1533
|
+
task.param,
|
|
1534
|
+
isSafariMobile() ? void 0 : {
|
|
1535
|
+
transfer: (_a = task.buffers) != null ? _a : []
|
|
1536
|
+
}
|
|
1537
|
+
);
|
|
1538
|
+
}
|
|
1539
|
+
this.busy = false;
|
|
1540
|
+
});
|
|
1541
|
+
}
|
|
1542
|
+
/**
|
|
1543
|
+
* Handle messages from worker
|
|
1544
|
+
*/
|
|
1545
|
+
onRecvMsg(e) {
|
|
1546
|
+
if (!e.data) return;
|
|
1547
|
+
const { verb, args } = e.data;
|
|
1548
|
+
const isCompatBuild = this.resources.compat;
|
|
1549
|
+
if (verb && verb.startsWith("console.")) {
|
|
1550
|
+
if (this.suppressNativeLog) {
|
|
1551
|
+
return;
|
|
1552
|
+
}
|
|
1553
|
+
if (verb.endsWith("debug")) this.logger.debug(...args);
|
|
1554
|
+
if (verb.endsWith("log")) this.logger.log(...args);
|
|
1555
|
+
if (verb.endsWith("warn")) this.logger.warn(...args);
|
|
1556
|
+
if (verb.endsWith("error")) this.logger.error(...args);
|
|
1557
|
+
return;
|
|
1558
|
+
} else if (verb === "signal.abort") {
|
|
1559
|
+
const [signalType, message, rawStack, originalErr] = args;
|
|
1560
|
+
if (originalErr) {
|
|
1561
|
+
this.logger.error(originalErr);
|
|
1562
|
+
}
|
|
1563
|
+
(() => __async(this, null, function* () {
|
|
1564
|
+
let stack = "";
|
|
1565
|
+
let newMsg = message.replace(
|
|
1566
|
+
"Build with -sASSERTIONS for more info.",
|
|
1567
|
+
""
|
|
1568
|
+
);
|
|
1569
|
+
if (signalType === "abort") {
|
|
1570
|
+
newMsg = `(ABORT) ${newMsg}`;
|
|
1571
|
+
stack = rawStack.replace(/\|/g, "\n");
|
|
1572
|
+
} else if (signalType === "exception") {
|
|
1573
|
+
stack = rawStack;
|
|
1574
|
+
}
|
|
1575
|
+
const decoded = yield Debug.decodeStackTrace(stack, isCompatBuild);
|
|
1576
|
+
this.logger.error(`Stack trace (${signalType}):
|
|
1577
|
+
` + decoded);
|
|
1578
|
+
this.abort(newMsg, decoded);
|
|
1579
|
+
}))();
|
|
1580
|
+
return;
|
|
1581
|
+
}
|
|
1582
|
+
if (verb === FILE_READ_REQ_EVENT) {
|
|
1583
|
+
const [name, offset, size] = args;
|
|
1584
|
+
this.fileReadResponse(name, offset, size).catch(() => {
|
|
1585
|
+
});
|
|
1586
|
+
return;
|
|
1587
|
+
}
|
|
1588
|
+
const { callbackId, result, err } = e.data;
|
|
1589
|
+
if (callbackId) {
|
|
1590
|
+
const idx = this.resultQueue.findIndex(
|
|
1591
|
+
(t) => t.param.callbackId === callbackId
|
|
1592
|
+
);
|
|
1593
|
+
if (idx !== -1) {
|
|
1594
|
+
const waitingTask = this.resultQueue.splice(idx, 1)[0];
|
|
1595
|
+
if (err) waitingTask.reject(err);
|
|
1596
|
+
else waitingTask.resolve(result);
|
|
1597
|
+
} else {
|
|
1598
|
+
this.logger.error(
|
|
1599
|
+
`Cannot find waiting task with callbackId = ${callbackId}`
|
|
1600
|
+
);
|
|
1601
|
+
}
|
|
1602
|
+
}
|
|
1603
|
+
}
|
|
1604
|
+
abort(text, stack) {
|
|
1605
|
+
const error = new WllamaRuntimeError(
|
|
1606
|
+
text.length == 0 ? "(unknown error)" : text,
|
|
1607
|
+
stack
|
|
1608
|
+
);
|
|
1609
|
+
while (this.resultQueue.length > 0) {
|
|
1610
|
+
const waitingTask = this.resultQueue.pop();
|
|
1611
|
+
if (!waitingTask) break;
|
|
1612
|
+
waitingTask.reject(error);
|
|
1613
|
+
}
|
|
1614
|
+
while (this.taskQueue.length > 0) {
|
|
1615
|
+
const pendingTask = this.taskQueue.pop();
|
|
1616
|
+
if (!pendingTask) break;
|
|
1617
|
+
pendingTask.reject(error);
|
|
1618
|
+
}
|
|
1619
|
+
}
|
|
1620
|
+
};
|
|
1621
|
+
|
|
1622
|
+
// src/huggingface.ts
|
|
1623
|
+
var HF_BASE = "https://huggingface.co";
|
|
1624
|
+
var DEFAULT_QUANTS = ["Q4_K_M", "Q8_0"];
|
|
1625
|
+
function fetchRepoFiles(repo, token) {
|
|
1626
|
+
return __async(this, null, function* () {
|
|
1627
|
+
var _a;
|
|
1628
|
+
const url = `${HF_BASE}/api/models/${repo}/tree/main?recursive=true`;
|
|
1629
|
+
const headers = { Accept: "application/json" };
|
|
1630
|
+
if (token) {
|
|
1631
|
+
headers["Authorization"] = `Bearer ${token}`;
|
|
1632
|
+
}
|
|
1633
|
+
const res = yield fetch(url, { headers });
|
|
1634
|
+
if (!res.ok) {
|
|
1635
|
+
let msg = res.statusText;
|
|
1636
|
+
try {
|
|
1637
|
+
msg = (_a = (yield res.json()).error) != null ? _a : msg;
|
|
1638
|
+
} catch (e) {
|
|
1639
|
+
}
|
|
1640
|
+
throw new Error(`HF API error (${res.status}): ${msg}`);
|
|
1641
|
+
}
|
|
1642
|
+
return res.json();
|
|
1643
|
+
});
|
|
1644
|
+
}
|
|
1645
|
+
function firstShardPath(files, path) {
|
|
1646
|
+
const m = path.match(/^(.+)-(\d{5})-of-(\d{5})\.gguf$/i);
|
|
1647
|
+
if (!m) return path;
|
|
1648
|
+
const first = `${m[1]}-00001-of-${m[3]}.gguf`;
|
|
1649
|
+
return files.some((f) => f.path === first) ? first : path;
|
|
1650
|
+
}
|
|
1651
|
+
function selectFile(files, quant, mmprojOnly) {
|
|
1652
|
+
const candidates = files.filter((f) => {
|
|
1653
|
+
if (f.type !== "file" || !f.path.toLowerCase().endsWith(".gguf"))
|
|
1654
|
+
return false;
|
|
1655
|
+
const ismmproj = f.path.toLowerCase().includes("mmproj");
|
|
1656
|
+
return mmprojOnly ? ismmproj : !ismmproj;
|
|
1657
|
+
});
|
|
1658
|
+
if (candidates.length === 0) return null;
|
|
1659
|
+
if (quant) {
|
|
1660
|
+
const upper = quant.toUpperCase();
|
|
1661
|
+
const match = candidates.find((f) => f.path.toUpperCase().includes(upper));
|
|
1662
|
+
if (match) return firstShardPath(candidates, match.path);
|
|
1663
|
+
return null;
|
|
1664
|
+
}
|
|
1665
|
+
for (const q of DEFAULT_QUANTS) {
|
|
1666
|
+
const match = candidates.find((f) => f.path.toUpperCase().includes(q));
|
|
1667
|
+
if (match) return firstShardPath(candidates, match.path);
|
|
1668
|
+
}
|
|
1669
|
+
return firstShardPath(candidates, candidates[0].path);
|
|
1670
|
+
}
|
|
1671
|
+
function getHFModelSource(config) {
|
|
1672
|
+
return __async(this, null, function* () {
|
|
1673
|
+
const { repo, file, quant, mmprojFile, mmprojQuant, hfToken } = config;
|
|
1674
|
+
const files = yield fetchRepoFiles(repo, hfToken);
|
|
1675
|
+
const modelPath = file != null ? file : selectFile(files, quant, false);
|
|
1676
|
+
if (!modelPath) {
|
|
1677
|
+
throw new Error(`No GGUF file found in repo "${repo}"`);
|
|
1678
|
+
}
|
|
1679
|
+
const source = {
|
|
1680
|
+
url: `${HF_BASE}/${repo}/resolve/main/${modelPath}`
|
|
1681
|
+
};
|
|
1682
|
+
if (mmprojFile || mmprojQuant !== void 0) {
|
|
1683
|
+
const mmpath = mmprojFile != null ? mmprojFile : selectFile(files, mmprojQuant, true);
|
|
1684
|
+
if (mmpath) {
|
|
1685
|
+
source.mmprojUrl = `${HF_BASE}/${repo}/resolve/main/${mmpath}`;
|
|
1686
|
+
}
|
|
1687
|
+
}
|
|
1688
|
+
if (hfToken) {
|
|
1689
|
+
const params = new URLSearchParams({ token: hfToken });
|
|
1690
|
+
source.url += `?${params}`;
|
|
1691
|
+
if (source.mmprojUrl) {
|
|
1692
|
+
source.mmprojUrl += `?${params}`;
|
|
1693
|
+
}
|
|
1694
|
+
}
|
|
1695
|
+
return source;
|
|
1696
|
+
});
|
|
1697
|
+
}
|
|
1698
|
+
function getHFFileSHA256(url, headers) {
|
|
1699
|
+
return __async(this, null, function* () {
|
|
1700
|
+
if (!url.includes("/resolve/")) return void 0;
|
|
1701
|
+
const rawUrl = url.replace("/resolve/", "/raw/");
|
|
1702
|
+
try {
|
|
1703
|
+
const text = yield fetch(rawUrl, { headers }).then((r) => r.text());
|
|
1704
|
+
const match = text.match(/^oid sha256:([0-9a-f]{64})$/m);
|
|
1705
|
+
return match ? match[1] : void 0;
|
|
1706
|
+
} catch (e) {
|
|
1707
|
+
return void 0;
|
|
1708
|
+
}
|
|
1709
|
+
});
|
|
1710
|
+
}
|
|
1711
|
+
|
|
1712
|
+
// src/storage/opfs.ts
|
|
1713
|
+
var OPFSBackend = class {
|
|
1714
|
+
isSupported() {
|
|
1715
|
+
var _a;
|
|
1716
|
+
return typeof navigator !== "undefined" && "storage" in navigator && !!((_a = navigator.storage) == null ? void 0 : _a.getDirectory);
|
|
1717
|
+
}
|
|
1718
|
+
read(key) {
|
|
1719
|
+
return __async(this, null, function* () {
|
|
1720
|
+
try {
|
|
1721
|
+
const cacheDir = yield getCacheDir();
|
|
1722
|
+
const fileHandle = yield cacheDir.getFileHandle(key);
|
|
1723
|
+
return yield fileHandle.getFile();
|
|
1724
|
+
} catch (e) {
|
|
1725
|
+
return null;
|
|
1726
|
+
}
|
|
1727
|
+
});
|
|
1728
|
+
}
|
|
1729
|
+
write(key, stream) {
|
|
1730
|
+
return __async(this, null, function* () {
|
|
1731
|
+
const writable = yield openWritable(key);
|
|
1732
|
+
yield writable.truncate(0);
|
|
1733
|
+
const reader = stream.getReader();
|
|
1734
|
+
try {
|
|
1735
|
+
while (true) {
|
|
1736
|
+
const { done, value } = yield reader.read();
|
|
1737
|
+
if (done) break;
|
|
1738
|
+
yield writable.write(value);
|
|
1739
|
+
}
|
|
1740
|
+
} finally {
|
|
1741
|
+
yield writable.close();
|
|
1742
|
+
}
|
|
1743
|
+
});
|
|
1744
|
+
}
|
|
1745
|
+
getSize(key) {
|
|
1746
|
+
return __async(this, null, function* () {
|
|
1747
|
+
try {
|
|
1748
|
+
const cacheDir = yield getCacheDir();
|
|
1749
|
+
const fileHandle = yield cacheDir.getFileHandle(key);
|
|
1750
|
+
const file = yield fileHandle.getFile();
|
|
1751
|
+
return file.size;
|
|
1752
|
+
} catch (e) {
|
|
1753
|
+
return -1;
|
|
1754
|
+
}
|
|
1755
|
+
});
|
|
1756
|
+
}
|
|
1757
|
+
list() {
|
|
1758
|
+
return __async(this, null, function* () {
|
|
1759
|
+
const cacheDir = yield getCacheDir();
|
|
1760
|
+
const result = [];
|
|
1761
|
+
try {
|
|
1762
|
+
for (var iter = __forAwait(cacheDir.entries()), more, temp, error; more = !(temp = yield iter.next()).done; more = false) {
|
|
1763
|
+
const [name, handle] = temp.value;
|
|
1764
|
+
if (handle.kind === "file") {
|
|
1765
|
+
const file = yield handle.getFile();
|
|
1766
|
+
result.push({ key: name, size: file.size });
|
|
1767
|
+
}
|
|
1768
|
+
}
|
|
1769
|
+
} catch (temp) {
|
|
1770
|
+
error = [temp];
|
|
1771
|
+
} finally {
|
|
1772
|
+
try {
|
|
1773
|
+
more && (temp = iter.return) && (yield temp.call(iter));
|
|
1774
|
+
} finally {
|
|
1775
|
+
if (error)
|
|
1776
|
+
throw error[0];
|
|
1777
|
+
}
|
|
1778
|
+
}
|
|
1779
|
+
return result;
|
|
1780
|
+
});
|
|
1781
|
+
}
|
|
1782
|
+
delete(key) {
|
|
1783
|
+
return __async(this, null, function* () {
|
|
1784
|
+
try {
|
|
1785
|
+
const cacheDir = yield getCacheDir();
|
|
1786
|
+
yield cacheDir.removeEntry(key);
|
|
1787
|
+
} catch (e) {
|
|
1788
|
+
if ((e == null ? void 0 : e.name) !== "NotFoundError") throw e;
|
|
1789
|
+
}
|
|
1790
|
+
});
|
|
1791
|
+
}
|
|
1792
|
+
};
|
|
1793
|
+
function getCacheDir() {
|
|
1794
|
+
return __async(this, null, function* () {
|
|
1795
|
+
const opfsRoot = yield navigator.storage.getDirectory();
|
|
1796
|
+
return opfsRoot.getDirectoryHandle("cache", { create: true });
|
|
1797
|
+
});
|
|
1798
|
+
}
|
|
1799
|
+
function openWritable(fileName) {
|
|
1800
|
+
return __async(this, null, function* () {
|
|
1801
|
+
const worker = createWorker(OPFS_UTILS_WORKER_CODE);
|
|
1802
|
+
let pResolve;
|
|
1803
|
+
let pReject;
|
|
1804
|
+
worker.onmessage = (e) => {
|
|
1805
|
+
if (e.data.ok) pResolve(null);
|
|
1806
|
+
else if (e.data.err) pReject(e.data.err);
|
|
1807
|
+
};
|
|
1808
|
+
worker.onerror = (e) => {
|
|
1809
|
+
var _a;
|
|
1810
|
+
return pReject == null ? void 0 : pReject((_a = e.message) != null ? _a : e);
|
|
1811
|
+
};
|
|
1812
|
+
const workerExec = (data) => new Promise((resolve, reject) => {
|
|
1813
|
+
pResolve = resolve;
|
|
1814
|
+
pReject = reject;
|
|
1815
|
+
worker.postMessage(
|
|
1816
|
+
data,
|
|
1817
|
+
isSafariMobile() ? void 0 : { transfer: "buf" in data && data.buf ? [data.buf.buffer] : [] }
|
|
1818
|
+
);
|
|
1819
|
+
});
|
|
1820
|
+
yield workerExec({ action: "open", filename: fileName });
|
|
1821
|
+
return {
|
|
1822
|
+
truncate: () => __async(this, null, function* () {
|
|
1823
|
+
}),
|
|
1824
|
+
write: (value) => workerExec({ action: "write", buf: value }),
|
|
1825
|
+
close: () => __async(this, null, function* () {
|
|
1826
|
+
yield workerExec({ action: "close" });
|
|
1827
|
+
worker.terminate();
|
|
1828
|
+
})
|
|
1829
|
+
};
|
|
1830
|
+
});
|
|
1831
|
+
}
|
|
1832
|
+
|
|
1833
|
+
// src/storage/cos.ts
|
|
1834
|
+
function makeHash(key) {
|
|
1835
|
+
return { algorithm: "SHA-256", value: key };
|
|
1836
|
+
}
|
|
1837
|
+
var COSInternalBackend = class {
|
|
1838
|
+
isSupported() {
|
|
1839
|
+
return typeof navigator !== "undefined" && "crossOriginStorage" in navigator;
|
|
1840
|
+
}
|
|
1841
|
+
// IMPORTANT: key must be SHA-256 hash of the data
|
|
1842
|
+
read(key) {
|
|
1843
|
+
return __async(this, null, function* () {
|
|
1844
|
+
try {
|
|
1845
|
+
const handle = yield navigator.crossOriginStorage.requestFileHandle(
|
|
1846
|
+
makeHash(key)
|
|
1847
|
+
);
|
|
1848
|
+
return handle.getFile();
|
|
1849
|
+
} catch (e) {
|
|
1850
|
+
return null;
|
|
1851
|
+
}
|
|
1852
|
+
});
|
|
1853
|
+
}
|
|
1854
|
+
// IMPORTANT: key must be SHA-256 hash of the data
|
|
1855
|
+
write(key, stream) {
|
|
1856
|
+
return __async(this, null, function* () {
|
|
1857
|
+
const handle = yield navigator.crossOriginStorage.requestFileHandle(
|
|
1858
|
+
makeHash(key),
|
|
1859
|
+
{ create: true }
|
|
1860
|
+
);
|
|
1861
|
+
const writable = yield handle.createWritable();
|
|
1862
|
+
const reader = stream.getReader();
|
|
1863
|
+
try {
|
|
1864
|
+
while (true) {
|
|
1865
|
+
const { done, value } = yield reader.read();
|
|
1866
|
+
if (done) break;
|
|
1867
|
+
yield writable.write(value);
|
|
1868
|
+
}
|
|
1869
|
+
} finally {
|
|
1870
|
+
yield writable.close();
|
|
1871
|
+
}
|
|
1872
|
+
});
|
|
1873
|
+
}
|
|
1874
|
+
// IMPORTANT: key must be SHA-256 hash of the data
|
|
1875
|
+
getSize(key) {
|
|
1876
|
+
return __async(this, null, function* () {
|
|
1877
|
+
try {
|
|
1878
|
+
const handle = yield navigator.crossOriginStorage.requestFileHandle(
|
|
1879
|
+
makeHash(key)
|
|
1880
|
+
);
|
|
1881
|
+
const file = yield handle.getFile();
|
|
1882
|
+
return file.size;
|
|
1883
|
+
} catch (e) {
|
|
1884
|
+
return -1;
|
|
1885
|
+
}
|
|
1886
|
+
});
|
|
1887
|
+
}
|
|
1888
|
+
list() {
|
|
1889
|
+
return __async(this, null, function* () {
|
|
1890
|
+
throw new Error("not implemented");
|
|
1891
|
+
});
|
|
1892
|
+
}
|
|
1893
|
+
delete(_key) {
|
|
1894
|
+
return __async(this, null, function* () {
|
|
1895
|
+
throw new Error("not implemented");
|
|
1896
|
+
});
|
|
1897
|
+
}
|
|
1898
|
+
};
|
|
1899
|
+
var COSBackend = class {
|
|
1900
|
+
constructor() {
|
|
1901
|
+
__publicField(this, "cos", new COSInternalBackend());
|
|
1902
|
+
__publicField(this, "priv", new OPFSBackend());
|
|
1903
|
+
}
|
|
1904
|
+
isSupported() {
|
|
1905
|
+
return this.priv.isSupported();
|
|
1906
|
+
}
|
|
1907
|
+
read(key, hint) {
|
|
1908
|
+
return __async(this, null, function* () {
|
|
1909
|
+
if ((hint == null ? void 0 : hint.sha256) && this.cos.isSupported()) {
|
|
1910
|
+
const blob = yield this.cos.read(hint.sha256);
|
|
1911
|
+
if (blob) return blob;
|
|
1912
|
+
}
|
|
1913
|
+
return this.priv.read(key);
|
|
1914
|
+
});
|
|
1915
|
+
}
|
|
1916
|
+
write(key, stream, hint) {
|
|
1917
|
+
return __async(this, null, function* () {
|
|
1918
|
+
if ((hint == null ? void 0 : hint.sha256) && this.cos.isSupported()) {
|
|
1919
|
+
yield this.cos.write(hint.sha256, stream);
|
|
1920
|
+
} else {
|
|
1921
|
+
yield this.priv.write(key, stream);
|
|
1922
|
+
}
|
|
1923
|
+
});
|
|
1924
|
+
}
|
|
1925
|
+
getSize(key, hint) {
|
|
1926
|
+
return __async(this, null, function* () {
|
|
1927
|
+
if ((hint == null ? void 0 : hint.sha256) && this.cos.isSupported()) {
|
|
1928
|
+
const size = yield this.cos.getSize(hint.sha256);
|
|
1929
|
+
if (size !== -1) return size;
|
|
1930
|
+
}
|
|
1931
|
+
return this.priv.getSize(key);
|
|
1932
|
+
});
|
|
1933
|
+
}
|
|
1934
|
+
list() {
|
|
1935
|
+
return __async(this, null, function* () {
|
|
1936
|
+
return this.priv.list();
|
|
1937
|
+
});
|
|
1938
|
+
}
|
|
1939
|
+
delete(key) {
|
|
1940
|
+
return __async(this, null, function* () {
|
|
1941
|
+
return this.priv.delete(key);
|
|
1942
|
+
});
|
|
1943
|
+
}
|
|
1944
|
+
};
|
|
1945
|
+
|
|
1946
|
+
// src/cache-manager.ts
|
|
1947
|
+
var PREFIX_METADATA = "__metadata__";
|
|
1948
|
+
var POLYFILL_ETAG = "polyfill_for_older_version";
|
|
1949
|
+
function hintFromMetadata(metadata) {
|
|
1950
|
+
if (!metadata) return void 0;
|
|
1951
|
+
if (metadata.sha256) return { sha256: metadata.sha256 };
|
|
1952
|
+
return void 0;
|
|
1953
|
+
}
|
|
1954
|
+
var CacheManager = class {
|
|
1955
|
+
/**
|
|
1956
|
+
* @param backends Array of storage backends to use, in order of preference ; if first is available, use it, otherwise try the next one.
|
|
1957
|
+
*/
|
|
1958
|
+
constructor(backends = [new COSBackend()]) {
|
|
1959
|
+
__publicField(this, "sb");
|
|
1960
|
+
for (const backend of backends) {
|
|
1961
|
+
if (backend.isSupported()) {
|
|
1962
|
+
this.sb = backend;
|
|
1963
|
+
return;
|
|
1964
|
+
}
|
|
1965
|
+
}
|
|
1966
|
+
throw new Error("No supported storage backend found");
|
|
1967
|
+
}
|
|
1968
|
+
/**
|
|
1969
|
+
* Convert a given URL into a storage key.
|
|
1970
|
+
*
|
|
1971
|
+
* Format: `${hashSHA1(fullURL)}_${fileName}`
|
|
1972
|
+
*/
|
|
1973
|
+
getNameFromURL(url) {
|
|
1974
|
+
return __async(this, null, function* () {
|
|
1975
|
+
return urlToFileName(url, "");
|
|
1976
|
+
});
|
|
1977
|
+
}
|
|
1978
|
+
/**
|
|
1979
|
+
* @deprecated Use `download()` instead
|
|
1980
|
+
*
|
|
1981
|
+
* Write a new file to cache. This will overwrite existing file.
|
|
1982
|
+
*
|
|
1983
|
+
* @param name The file name returned by `getNameFromURL()` or `list()`
|
|
1984
|
+
*/
|
|
1985
|
+
write(name, stream, metadata) {
|
|
1986
|
+
return __async(this, null, function* () {
|
|
1987
|
+
yield this.sb.write(name, stream);
|
|
1988
|
+
yield this.writeMetadata(name, metadata);
|
|
1989
|
+
});
|
|
1990
|
+
}
|
|
1991
|
+
download(_0) {
|
|
1992
|
+
return __async(this, arguments, function* (url, options = {}) {
|
|
1993
|
+
var _a, _b, _c;
|
|
1994
|
+
const fileKey = yield urlToFileName(url, "");
|
|
1995
|
+
const sha256 = yield getHFFileSHA256(url, (_a = options.headers) != null ? _a : {});
|
|
1996
|
+
const hint = sha256 ? { sha256 } : void 0;
|
|
1997
|
+
if (hint && (yield this.sb.getSize(fileKey, hint)) !== -1) {
|
|
1998
|
+
if (!(yield this.getMetadata(fileKey))) {
|
|
1999
|
+
const head = yield fetch(url, __spreadValues({
|
|
2000
|
+
method: "HEAD"
|
|
2001
|
+
}, options.headers ? { headers: options.headers } : {}));
|
|
2002
|
+
const contentLength2 = head.headers.get("content-length");
|
|
2003
|
+
const etag2 = (head.headers.get("etag") || "").replace(
|
|
2004
|
+
/[^A-Za-z0-9]/g,
|
|
2005
|
+
""
|
|
2006
|
+
);
|
|
2007
|
+
yield this.writeMetadata(fileKey, __spreadValues({
|
|
2008
|
+
originalURL: url,
|
|
2009
|
+
originalSize: parseInt(contentLength2 != null ? contentLength2 : "0", 10),
|
|
2010
|
+
etag: etag2,
|
|
2011
|
+
sha256
|
|
2012
|
+
}, (_b = options.metadataAdditional) != null ? _b : {}));
|
|
2013
|
+
}
|
|
2014
|
+
return;
|
|
2015
|
+
}
|
|
2016
|
+
const response = yield fetch(url, __spreadValues(__spreadValues({}, options.headers ? { headers: options.headers } : {}), options.signal ? { signal: options.signal } : {}));
|
|
2017
|
+
if (!response.ok || !response.body) {
|
|
2018
|
+
throw new Error(`Failed to fetch ${url}: HTTP ${response.status}`);
|
|
2019
|
+
}
|
|
2020
|
+
const contentLength = response.headers.get("content-length");
|
|
2021
|
+
const etag = (response.headers.get("etag") || "").replace(
|
|
2022
|
+
/[^A-Za-z0-9]/g,
|
|
2023
|
+
""
|
|
2024
|
+
);
|
|
2025
|
+
const total = parseInt(contentLength != null ? contentLength : "0", 10);
|
|
2026
|
+
const progressCallback = options.progressCallback;
|
|
2027
|
+
let loaded = 0;
|
|
2028
|
+
let lastProgressAt = 0;
|
|
2029
|
+
const progressStream = new TransformStream({
|
|
2030
|
+
transform(chunk, controller) {
|
|
2031
|
+
loaded += chunk.byteLength;
|
|
2032
|
+
if (progressCallback) {
|
|
2033
|
+
const now = Date.now();
|
|
2034
|
+
if (now - lastProgressAt > 100) {
|
|
2035
|
+
lastProgressAt = now;
|
|
2036
|
+
progressCallback({ loaded, total });
|
|
2037
|
+
}
|
|
2038
|
+
}
|
|
2039
|
+
controller.enqueue(chunk);
|
|
2040
|
+
},
|
|
2041
|
+
flush() {
|
|
2042
|
+
progressCallback == null ? void 0 : progressCallback({ loaded, total: total || loaded });
|
|
2043
|
+
}
|
|
2044
|
+
});
|
|
2045
|
+
const metadata = __spreadValues({
|
|
2046
|
+
originalURL: url,
|
|
2047
|
+
originalSize: total,
|
|
2048
|
+
etag
|
|
2049
|
+
}, (_c = options.metadataAdditional) != null ? _c : {});
|
|
2050
|
+
if (sha256) {
|
|
2051
|
+
metadata.sha256 = sha256;
|
|
2052
|
+
}
|
|
2053
|
+
yield this.sb.write(
|
|
2054
|
+
fileKey,
|
|
2055
|
+
response.body.pipeThrough(progressStream),
|
|
2056
|
+
hint
|
|
2057
|
+
);
|
|
2058
|
+
yield this.writeMetadata(fileKey, metadata);
|
|
2059
|
+
});
|
|
2060
|
+
}
|
|
2061
|
+
/**
|
|
2062
|
+
* Open a file in cache for reading
|
|
2063
|
+
*
|
|
2064
|
+
* @param nameOrURL The file name returned by `getNameFromURL()` or `list()`, or the original URL of the remote file
|
|
2065
|
+
* @returns Blob, or null if file does not exist
|
|
2066
|
+
*/
|
|
2067
|
+
open(nameOrURL) {
|
|
2068
|
+
return __async(this, null, function* () {
|
|
2069
|
+
const hint1 = hintFromMetadata(yield this.getMetadata(nameOrURL));
|
|
2070
|
+
const direct = yield this.sb.read(nameOrURL, hint1);
|
|
2071
|
+
if (direct) return direct;
|
|
2072
|
+
const key = yield urlToFileName(nameOrURL, "");
|
|
2073
|
+
const hint2 = hintFromMetadata(yield this.getMetadata(key));
|
|
2074
|
+
return this.sb.read(key, hint2);
|
|
2075
|
+
});
|
|
2076
|
+
}
|
|
2077
|
+
/**
|
|
2078
|
+
* Get the size of a file in stored cache
|
|
2079
|
+
*
|
|
2080
|
+
* NOTE: in case the download is stopped mid-way (i.e. user close browser tab), the file maybe corrupted, size maybe different from `metadata.originalSize`
|
|
2081
|
+
*
|
|
2082
|
+
* @param name The file name returned by `getNameFromURL()` or `list()`
|
|
2083
|
+
* @returns number of bytes, or -1 if file does not exist
|
|
2084
|
+
*/
|
|
2085
|
+
getSize(name) {
|
|
2086
|
+
return __async(this, null, function* () {
|
|
2087
|
+
const hint = hintFromMetadata(yield this.getMetadata(name));
|
|
2088
|
+
return this.sb.getSize(name, hint);
|
|
2089
|
+
});
|
|
2090
|
+
}
|
|
2091
|
+
/**
|
|
2092
|
+
* Get metadata of a cached file
|
|
2093
|
+
*/
|
|
2094
|
+
getMetadata(name) {
|
|
2095
|
+
return __async(this, null, function* () {
|
|
2096
|
+
const blob = yield this.sb.read(`${PREFIX_METADATA}${name}`);
|
|
2097
|
+
const cachedSize = yield this.sb.getSize(name);
|
|
2098
|
+
if (!blob) {
|
|
2099
|
+
return cachedSize > 0 ? (
|
|
2100
|
+
// files created by older version of wllama don't have metadata; polyfill it
|
|
2101
|
+
{
|
|
2102
|
+
etag: POLYFILL_ETAG,
|
|
2103
|
+
originalSize: cachedSize,
|
|
2104
|
+
originalURL: ""
|
|
2105
|
+
}
|
|
2106
|
+
) : (
|
|
2107
|
+
// cached file not found
|
|
2108
|
+
null
|
|
2109
|
+
);
|
|
2110
|
+
}
|
|
2111
|
+
try {
|
|
2112
|
+
return yield new Response(blob).json();
|
|
2113
|
+
} catch (e) {
|
|
2114
|
+
return null;
|
|
2115
|
+
}
|
|
2116
|
+
});
|
|
2117
|
+
}
|
|
2118
|
+
/**
|
|
2119
|
+
* List all files currently in cache
|
|
2120
|
+
*/
|
|
2121
|
+
list() {
|
|
2122
|
+
return __async(this, null, function* () {
|
|
2123
|
+
const all = yield this.sb.list();
|
|
2124
|
+
const metadataMap = {};
|
|
2125
|
+
for (const { key } of all) {
|
|
2126
|
+
if (key.startsWith(PREFIX_METADATA)) {
|
|
2127
|
+
const blob = yield this.sb.read(key);
|
|
2128
|
+
if (blob) {
|
|
2129
|
+
const meta = yield new Response(blob).json().catch(() => null);
|
|
2130
|
+
metadataMap[key.slice(PREFIX_METADATA.length)] = meta;
|
|
2131
|
+
}
|
|
2132
|
+
}
|
|
2133
|
+
}
|
|
2134
|
+
const result = [];
|
|
2135
|
+
for (const { key, size } of all) {
|
|
2136
|
+
if (!key.startsWith(PREFIX_METADATA)) {
|
|
2137
|
+
result.push({
|
|
2138
|
+
name: key,
|
|
2139
|
+
size,
|
|
2140
|
+
metadata: metadataMap[key] || {
|
|
2141
|
+
originalSize: size,
|
|
2142
|
+
originalURL: "",
|
|
2143
|
+
etag: ""
|
|
2144
|
+
}
|
|
2145
|
+
});
|
|
2146
|
+
}
|
|
2147
|
+
}
|
|
2148
|
+
return result;
|
|
2149
|
+
});
|
|
2150
|
+
}
|
|
2151
|
+
/**
|
|
2152
|
+
* Clear all files currently in cache
|
|
2153
|
+
*/
|
|
2154
|
+
clear() {
|
|
2155
|
+
return __async(this, null, function* () {
|
|
2156
|
+
yield this.deleteMany(() => true);
|
|
2157
|
+
});
|
|
2158
|
+
}
|
|
2159
|
+
/**
|
|
2160
|
+
* Delete a single file in cache
|
|
2161
|
+
*
|
|
2162
|
+
* @param nameOrURL Can be either an URL or a name returned by `getNameFromURL()` or `list()`
|
|
2163
|
+
*/
|
|
2164
|
+
delete(nameOrURL) {
|
|
2165
|
+
return __async(this, null, function* () {
|
|
2166
|
+
const name2 = yield this.getNameFromURL(nameOrURL);
|
|
2167
|
+
yield this.deleteMany(
|
|
2168
|
+
(entry) => entry.name === nameOrURL || entry.name === name2
|
|
2169
|
+
);
|
|
2170
|
+
});
|
|
2171
|
+
}
|
|
2172
|
+
/**
|
|
2173
|
+
* Delete multiple files in cache.
|
|
2174
|
+
*
|
|
2175
|
+
* @param predicate A predicate like `array.filter(item => boolean)`
|
|
2176
|
+
*/
|
|
2177
|
+
deleteMany(predicate) {
|
|
2178
|
+
return __async(this, null, function* () {
|
|
2179
|
+
const list = yield this.list();
|
|
2180
|
+
for (const item of list) {
|
|
2181
|
+
if (predicate(item)) {
|
|
2182
|
+
yield this.sb.delete(item.name);
|
|
2183
|
+
yield this.sb.delete(`${PREFIX_METADATA}${item.name}`);
|
|
2184
|
+
}
|
|
2185
|
+
}
|
|
2186
|
+
});
|
|
2187
|
+
}
|
|
2188
|
+
/**
|
|
2189
|
+
* Write the metadata of the file to disk.
|
|
2190
|
+
*/
|
|
2191
|
+
writeMetadata(name, metadata) {
|
|
2192
|
+
return __async(this, null, function* () {
|
|
2193
|
+
const blob = new Blob([JSON.stringify(metadata)], { type: "text/plain" });
|
|
2194
|
+
yield this.sb.write(`${PREFIX_METADATA}${name}`, blob.stream());
|
|
2195
|
+
});
|
|
2196
|
+
}
|
|
2197
|
+
};
|
|
2198
|
+
var cache_manager_default = CacheManager;
|
|
2199
|
+
function urlToFileName(url, prefix) {
|
|
2200
|
+
return __async(this, null, function* () {
|
|
2201
|
+
const hashBuffer = yield crypto.subtle.digest(
|
|
2202
|
+
"SHA-1",
|
|
2203
|
+
new TextEncoder().encode(url)
|
|
2204
|
+
);
|
|
2205
|
+
const hashArray = Array.from(new Uint8Array(hashBuffer));
|
|
2206
|
+
const hashHex = hashArray.map((b) => b.toString(16).padStart(2, "0")).join("");
|
|
2207
|
+
return `${prefix}${hashHex}_${url.split("/").pop()}`;
|
|
2208
|
+
});
|
|
2209
|
+
}
|
|
2210
|
+
|
|
2211
|
+
// src/model-manager.ts
|
|
2212
|
+
var DEFAULT_PARALLEL_DOWNLOADS = 3;
|
|
2213
|
+
var ModelValidationStatus = /* @__PURE__ */ ((ModelValidationStatus2) => {
|
|
2214
|
+
ModelValidationStatus2["VALID"] = "valid";
|
|
2215
|
+
ModelValidationStatus2["INVALID"] = "invalid";
|
|
2216
|
+
ModelValidationStatus2["DELETED"] = "deleted";
|
|
2217
|
+
return ModelValidationStatus2;
|
|
2218
|
+
})(ModelValidationStatus || {});
|
|
2219
|
+
var Model = class {
|
|
2220
|
+
constructor(modelManager, url, mmprojUrl, savedFiles) {
|
|
2221
|
+
__publicField(this, "modelManager");
|
|
2222
|
+
/**
|
|
2223
|
+
* URL to the GGUF file (in case it contains multiple shards, the URL should point to the first shard)
|
|
2224
|
+
*
|
|
2225
|
+
* This URL will be used to identify the model in the cache. There can't be 2 models with the same URL.
|
|
2226
|
+
*/
|
|
2227
|
+
__publicField(this, "url");
|
|
2228
|
+
/**
|
|
2229
|
+
* URL to mmproj file, if exists
|
|
2230
|
+
*/
|
|
2231
|
+
__publicField(this, "mmprojUrl");
|
|
2232
|
+
/**
|
|
2233
|
+
* Size in bytes (total size of all shards).
|
|
2234
|
+
*
|
|
2235
|
+
* A value of -1 means the model is deleted from the cache. You must call `ModelManager.downloadModel` to re-download the model.
|
|
2236
|
+
*/
|
|
2237
|
+
__publicField(this, "size");
|
|
2238
|
+
/**
|
|
2239
|
+
* List of all shards in the cache, sorted by original URL (ascending order)
|
|
2240
|
+
*/
|
|
2241
|
+
__publicField(this, "files");
|
|
2242
|
+
this.modelManager = modelManager;
|
|
2243
|
+
this.url = url;
|
|
2244
|
+
this.mmprojUrl = mmprojUrl;
|
|
2245
|
+
if (savedFiles) {
|
|
2246
|
+
this.files = this.getAllFiles(savedFiles);
|
|
2247
|
+
this.size = sumArr(this.files.map((f) => f.metadata.originalSize));
|
|
2248
|
+
} else {
|
|
2249
|
+
this.files = [];
|
|
2250
|
+
this.size = 0;
|
|
2251
|
+
}
|
|
2252
|
+
}
|
|
2253
|
+
/**
|
|
2254
|
+
* Open and get a list of all shards as Blobs
|
|
2255
|
+
*/
|
|
2256
|
+
open() {
|
|
2257
|
+
return __async(this, null, function* () {
|
|
2258
|
+
if (this.size === -1) {
|
|
2259
|
+
throw new WllamaError(
|
|
2260
|
+
`Model is deleted from the cache; Call ModelManager.downloadModel to re-download the model`,
|
|
2261
|
+
"load_error"
|
|
2262
|
+
);
|
|
2263
|
+
}
|
|
2264
|
+
const blobs = [];
|
|
2265
|
+
for (const file of this.files) {
|
|
2266
|
+
const blob = yield this.modelManager.cacheManager.open(file.name);
|
|
2267
|
+
if (!blob) {
|
|
2268
|
+
throw new Error(
|
|
2269
|
+
`Failed to open file ${file.name}; Hint: the model may be invalid, please refresh it`
|
|
2270
|
+
);
|
|
2271
|
+
}
|
|
2272
|
+
blobs.push(blob);
|
|
2273
|
+
}
|
|
2274
|
+
return blobs;
|
|
2275
|
+
});
|
|
2276
|
+
}
|
|
2277
|
+
/**
|
|
2278
|
+
* Validate the model files.
|
|
2279
|
+
*
|
|
2280
|
+
* If the model is invalid, the model manager will not be able to use it. You must call `refresh` to re-download the model.
|
|
2281
|
+
*
|
|
2282
|
+
* Cases that model is invalid:
|
|
2283
|
+
* - The model is deleted from the cache
|
|
2284
|
+
* - The model files are missing (or the download is interrupted)
|
|
2285
|
+
*/
|
|
2286
|
+
validate() {
|
|
2287
|
+
let nbShards = ModelManager.parseModelUrl(this.url).length;
|
|
2288
|
+
if (this.mmprojUrl) {
|
|
2289
|
+
nbShards += 1;
|
|
2290
|
+
}
|
|
2291
|
+
if (this.size === -1) {
|
|
2292
|
+
return "deleted" /* DELETED */;
|
|
2293
|
+
}
|
|
2294
|
+
if (this.size < 16 || this.files.length !== nbShards) {
|
|
2295
|
+
return "invalid" /* INVALID */;
|
|
2296
|
+
}
|
|
2297
|
+
for (const file of this.files) {
|
|
2298
|
+
if (!file.metadata || file.metadata.originalSize !== file.size) {
|
|
2299
|
+
return "invalid" /* INVALID */;
|
|
2300
|
+
}
|
|
2301
|
+
}
|
|
2302
|
+
return "valid" /* VALID */;
|
|
2303
|
+
}
|
|
2304
|
+
/**
|
|
2305
|
+
* In case the model is invalid, call this function to re-download the model
|
|
2306
|
+
*/
|
|
2307
|
+
refresh() {
|
|
2308
|
+
return __async(this, arguments, function* (options = {}) {
|
|
2309
|
+
var _a;
|
|
2310
|
+
const urls = ModelManager.parseModelUrl(this.url);
|
|
2311
|
+
if (this.mmprojUrl) {
|
|
2312
|
+
urls.push(this.mmprojUrl);
|
|
2313
|
+
}
|
|
2314
|
+
const works = urls.map((url, index) => ({
|
|
2315
|
+
url,
|
|
2316
|
+
index
|
|
2317
|
+
}));
|
|
2318
|
+
this.modelManager.logger.debug("Downloading model files:", urls);
|
|
2319
|
+
const nParallel = (_a = this.modelManager.params.parallelDownloads) != null ? _a : DEFAULT_PARALLEL_DOWNLOADS;
|
|
2320
|
+
const totalSize = yield this.getTotalDownloadSize(urls);
|
|
2321
|
+
const loadedSize = [];
|
|
2322
|
+
const worker = () => __async(this, null, function* () {
|
|
2323
|
+
while (works.length > 0) {
|
|
2324
|
+
const w = works.shift();
|
|
2325
|
+
if (!w) break;
|
|
2326
|
+
yield this.modelManager.cacheManager.download(w.url, __spreadProps(__spreadValues({}, options), {
|
|
2327
|
+
metadataAdditional: {
|
|
2328
|
+
originalURL: w.url,
|
|
2329
|
+
mmprojURL: this.mmprojUrl
|
|
2330
|
+
},
|
|
2331
|
+
progressCallback: ({ loaded }) => {
|
|
2332
|
+
var _a2;
|
|
2333
|
+
loadedSize[w.index] = loaded;
|
|
2334
|
+
(_a2 = options.progressCallback) == null ? void 0 : _a2.call(options, {
|
|
2335
|
+
loaded: sumArr(loadedSize),
|
|
2336
|
+
total: totalSize
|
|
2337
|
+
});
|
|
2338
|
+
}
|
|
2339
|
+
}));
|
|
2340
|
+
}
|
|
2341
|
+
});
|
|
2342
|
+
const promises = [];
|
|
2343
|
+
for (let i = 0; i < nParallel; i++) {
|
|
2344
|
+
promises.push(worker());
|
|
2345
|
+
loadedSize.push(0);
|
|
2346
|
+
}
|
|
2347
|
+
yield Promise.all(promises);
|
|
2348
|
+
this.files = this.getAllFiles(yield this.modelManager.cacheManager.list());
|
|
2349
|
+
this.size = this.files.reduce((acc, f) => acc + f.metadata.originalSize, 0);
|
|
2350
|
+
});
|
|
2351
|
+
}
|
|
2352
|
+
/**
|
|
2353
|
+
* Remove the model from the cache
|
|
2354
|
+
*/
|
|
2355
|
+
remove() {
|
|
2356
|
+
return __async(this, null, function* () {
|
|
2357
|
+
this.files = this.getAllFiles(yield this.modelManager.cacheManager.list());
|
|
2358
|
+
yield this.modelManager.cacheManager.deleteMany(
|
|
2359
|
+
(f) => !!this.files.find((file) => file.name === f.name)
|
|
2360
|
+
);
|
|
2361
|
+
this.size = -1;
|
|
2362
|
+
});
|
|
2363
|
+
}
|
|
2364
|
+
getAllFiles(savedFiles) {
|
|
2365
|
+
const allUrls = new Set(ModelManager.parseModelUrl(this.url));
|
|
2366
|
+
if (this.mmprojUrl) {
|
|
2367
|
+
allUrls.add(this.mmprojUrl);
|
|
2368
|
+
}
|
|
2369
|
+
const allFiles = [];
|
|
2370
|
+
for (const url of allUrls) {
|
|
2371
|
+
const file = savedFiles.find((f) => f.metadata.originalURL === url);
|
|
2372
|
+
if (!file) {
|
|
2373
|
+
throw new Error(`Model file not found: ${url}`);
|
|
2374
|
+
}
|
|
2375
|
+
allFiles.push(file);
|
|
2376
|
+
}
|
|
2377
|
+
allFiles.sort(
|
|
2378
|
+
(a, b) => a.metadata.originalURL.localeCompare(b.metadata.originalURL)
|
|
2379
|
+
);
|
|
2380
|
+
return allFiles;
|
|
2381
|
+
}
|
|
2382
|
+
getTotalDownloadSize(urls) {
|
|
2383
|
+
return __async(this, null, function* () {
|
|
2384
|
+
const responses = yield Promise.all(
|
|
2385
|
+
urls.map((url) => fetch(url, { method: "HEAD" }))
|
|
2386
|
+
);
|
|
2387
|
+
const sizes = responses.map(
|
|
2388
|
+
(res) => Number(res.headers.get("content-length") || "0")
|
|
2389
|
+
);
|
|
2390
|
+
return sumArr(sizes);
|
|
2391
|
+
});
|
|
2392
|
+
}
|
|
2393
|
+
};
|
|
2394
|
+
var ModelManager = class _ModelManager {
|
|
2395
|
+
constructor(params = {}) {
|
|
2396
|
+
// The CacheManager singleton, can be accessed by user
|
|
2397
|
+
__publicField(this, "cacheManager");
|
|
2398
|
+
__publicField(this, "params");
|
|
2399
|
+
__publicField(this, "logger");
|
|
2400
|
+
this.cacheManager = params.cacheManager || new cache_manager_default();
|
|
2401
|
+
this.params = params;
|
|
2402
|
+
this.logger = params.logger || console;
|
|
2403
|
+
}
|
|
2404
|
+
/**
|
|
2405
|
+
* Parses a model URL and returns an array of URLs based on the following patterns:
|
|
2406
|
+
* - If the input URL is an array, it returns the array itself.
|
|
2407
|
+
* - If the input URL is a string in the `gguf-split` format, it returns an array containing the URL of each shard in ascending order.
|
|
2408
|
+
* - Otherwise, it returns an array containing the input URL as a single element array.
|
|
2409
|
+
* @param modelUrl URL or list of URLs
|
|
2410
|
+
*/
|
|
2411
|
+
static parseModelUrl(modelUrl) {
|
|
2412
|
+
var _a;
|
|
2413
|
+
if (Array.isArray(modelUrl)) {
|
|
2414
|
+
return modelUrl;
|
|
2415
|
+
}
|
|
2416
|
+
const urlPartsRegex = /-(\d{5})-of-(\d{5})\.gguf(?:\?.*)?$/;
|
|
2417
|
+
const queryMatch = modelUrl.match(/\.gguf(\?.*)?$/);
|
|
2418
|
+
const queryParams = (_a = queryMatch == null ? void 0 : queryMatch[1]) != null ? _a : "";
|
|
2419
|
+
const matches = modelUrl.match(urlPartsRegex);
|
|
2420
|
+
if (!matches) {
|
|
2421
|
+
return [modelUrl];
|
|
2422
|
+
}
|
|
2423
|
+
const baseURL = modelUrl.replace(urlPartsRegex, "");
|
|
2424
|
+
const total = matches[2];
|
|
2425
|
+
const paddedShardIds = Array.from(
|
|
2426
|
+
{ length: Number(total) },
|
|
2427
|
+
(_, index) => (index + 1).toString().padStart(5, "0")
|
|
2428
|
+
);
|
|
2429
|
+
return paddedShardIds.map(
|
|
2430
|
+
(current) => `${baseURL}-${current}-of-${total}.gguf${queryParams}`
|
|
2431
|
+
);
|
|
2432
|
+
}
|
|
2433
|
+
/**
|
|
2434
|
+
* Get all models in the cache
|
|
2435
|
+
*/
|
|
2436
|
+
getModels() {
|
|
2437
|
+
return __async(this, arguments, function* (opts = {}) {
|
|
2438
|
+
const cachedFiles = yield this.cacheManager.list();
|
|
2439
|
+
let models = [];
|
|
2440
|
+
for (const file of cachedFiles) {
|
|
2441
|
+
const shards = _ModelManager.parseModelUrl(file.metadata.originalURL);
|
|
2442
|
+
const mmprojUrl = file.metadata.mmprojURL;
|
|
2443
|
+
const isFirstShard = shards.length === 1 || shards[0] === file.metadata.originalURL;
|
|
2444
|
+
if (isFirstShard) {
|
|
2445
|
+
models.push(
|
|
2446
|
+
new Model(this, file.metadata.originalURL, mmprojUrl, cachedFiles)
|
|
2447
|
+
);
|
|
2448
|
+
}
|
|
2449
|
+
}
|
|
2450
|
+
if (!opts.includeInvalid) {
|
|
2451
|
+
models = models.filter(
|
|
2452
|
+
(m) => m.validate() === "valid" /* VALID */
|
|
2453
|
+
);
|
|
2454
|
+
}
|
|
2455
|
+
return models;
|
|
2456
|
+
});
|
|
2457
|
+
}
|
|
2458
|
+
/**
|
|
2459
|
+
* Download a model from the given URL.
|
|
2460
|
+
*
|
|
2461
|
+
* The URL must end with `.gguf`
|
|
2462
|
+
*/
|
|
2463
|
+
downloadModel(_0) {
|
|
2464
|
+
return __async(this, arguments, function* (sourceOrURL, options = {}) {
|
|
2465
|
+
const source = isString(sourceOrURL) ? { url: sourceOrURL } : sourceOrURL;
|
|
2466
|
+
if (!isValidGgufFile(source.url)) {
|
|
2467
|
+
throw new WllamaError(
|
|
2468
|
+
`Invalid model URL: ${source.url}; URL must ends with ".gguf"`,
|
|
2469
|
+
"download_error"
|
|
2470
|
+
);
|
|
2471
|
+
}
|
|
2472
|
+
const model = new Model(this, source.url, source.mmprojUrl);
|
|
2473
|
+
const validity = model.validate();
|
|
2474
|
+
if (validity !== "valid" /* VALID */) {
|
|
2475
|
+
yield model.refresh(options);
|
|
2476
|
+
}
|
|
2477
|
+
return model;
|
|
2478
|
+
});
|
|
2479
|
+
}
|
|
2480
|
+
/**
|
|
2481
|
+
* Get a model from the cache or download it if it's not available.
|
|
2482
|
+
*/
|
|
2483
|
+
getModelOrDownload(_0) {
|
|
2484
|
+
return __async(this, arguments, function* (source, options = {}) {
|
|
2485
|
+
var _a;
|
|
2486
|
+
const models = yield this.getModels();
|
|
2487
|
+
const model = models.find((m) => m.url === source.url);
|
|
2488
|
+
if (model) {
|
|
2489
|
+
(_a = options.progressCallback) == null ? void 0 : _a.call(options, { loaded: model.size, total: model.size });
|
|
2490
|
+
return model;
|
|
2491
|
+
}
|
|
2492
|
+
return this.downloadModel(source, options);
|
|
2493
|
+
});
|
|
2494
|
+
}
|
|
2495
|
+
/**
|
|
2496
|
+
* Remove all models from the cache
|
|
2497
|
+
*/
|
|
2498
|
+
clear() {
|
|
2499
|
+
return __async(this, null, function* () {
|
|
2500
|
+
yield this.cacheManager.clear();
|
|
2501
|
+
});
|
|
2502
|
+
}
|
|
2503
|
+
};
|
|
2504
|
+
|
|
2505
|
+
// src/types/types.ts
|
|
2506
|
+
var LogLevel = /* @__PURE__ */ ((LogLevel2) => {
|
|
2507
|
+
LogLevel2[LogLevel2["DEBUG"] = 1] = "DEBUG";
|
|
2508
|
+
LogLevel2[LogLevel2["INFO"] = 2] = "INFO";
|
|
2509
|
+
LogLevel2[LogLevel2["WARN"] = 3] = "WARN";
|
|
2510
|
+
LogLevel2[LogLevel2["ERROR"] = 4] = "ERROR";
|
|
2511
|
+
return LogLevel2;
|
|
2512
|
+
})(LogLevel || {});
|
|
2513
|
+
|
|
2514
|
+
// src/wasm-from-cdn.ts
|
|
2515
|
+
var WasmCompatFromCDN = {
|
|
2516
|
+
worker: "https://cdn.jsdelivr.net/npm/@wllama/wllama-compat@3.5.1-q2.0/wasm/wllama.js",
|
|
2517
|
+
wasm: "https://cdn.jsdelivr.net/npm/@wllama/wllama-compat@3.5.1-q2.0/wasm/wllama.wasm"
|
|
2518
|
+
};
|
|
2519
|
+
|
|
2520
|
+
// src/wllama.ts
|
|
2521
|
+
var LoggerWithoutDebug = __spreadProps(__spreadValues({}, console), {
|
|
2522
|
+
debug: () => {
|
|
2523
|
+
}
|
|
2524
|
+
});
|
|
2525
|
+
var WllamaError = class extends Error {
|
|
2526
|
+
constructor(message, type = "unknown_error") {
|
|
2527
|
+
super(message);
|
|
2528
|
+
__publicField(this, "type");
|
|
2529
|
+
this.type = type;
|
|
2530
|
+
}
|
|
2531
|
+
};
|
|
2532
|
+
var WllamaAbortError = class extends Error {
|
|
2533
|
+
constructor() {
|
|
2534
|
+
super("Operation aborted");
|
|
2535
|
+
__publicField(this, "name", "AbortError");
|
|
2536
|
+
}
|
|
2537
|
+
};
|
|
2538
|
+
var WllamaRuntimeError = class extends Error {
|
|
2539
|
+
constructor(message, stack) {
|
|
2540
|
+
super(message);
|
|
2541
|
+
__publicField(this, "name", "RuntimeError");
|
|
2542
|
+
__publicField(this, "stack");
|
|
2543
|
+
this.stack = stack;
|
|
2544
|
+
}
|
|
2545
|
+
};
|
|
2546
|
+
var Wllama = class {
|
|
2547
|
+
constructor(pathConfig, wllamaConfig = {}) {
|
|
2548
|
+
// The CacheManager and ModelManager are singleton, can be accessed by user
|
|
2549
|
+
__publicField(this, "cacheManager");
|
|
2550
|
+
__publicField(this, "modelManager");
|
|
2551
|
+
__publicField(this, "compat", null);
|
|
2552
|
+
__publicField(this, "proxy", null);
|
|
2553
|
+
__publicField(this, "config");
|
|
2554
|
+
__publicField(this, "pathConfig");
|
|
2555
|
+
__publicField(this, "useMultiThread", false);
|
|
2556
|
+
__publicField(this, "nbThreads", 1);
|
|
2557
|
+
__publicField(this, "useEmbeddings", false);
|
|
2558
|
+
__publicField(this, "useRerank", false);
|
|
2559
|
+
// available when loaded
|
|
2560
|
+
__publicField(this, "loadedContextInfo", null);
|
|
2561
|
+
__publicField(this, "seed");
|
|
2562
|
+
__publicField(this, "bosToken", -1);
|
|
2563
|
+
__publicField(this, "eosToken", -1);
|
|
2564
|
+
__publicField(this, "eotToken", -1);
|
|
2565
|
+
__publicField(this, "eogTokens", /* @__PURE__ */ new Set());
|
|
2566
|
+
__publicField(this, "addBosToken", false);
|
|
2567
|
+
__publicField(this, "addEosToken", false);
|
|
2568
|
+
__publicField(this, "mediaMarker");
|
|
2569
|
+
__publicField(this, "chatTemplate");
|
|
2570
|
+
__publicField(this, "metadata");
|
|
2571
|
+
__publicField(this, "hasEncoder", false);
|
|
2572
|
+
__publicField(this, "decoderStartToken", -1);
|
|
2573
|
+
// note: we overlay instead of using llama-server default_template_kwargs, because we cannot transfer complex data structure via GLUE
|
|
2574
|
+
// overlay allow mixed data type or nested structure for kwargs
|
|
2575
|
+
__publicField(this, "chatTemplateKwargs", {});
|
|
2576
|
+
var _a, _b, _c;
|
|
2577
|
+
checkEnvironmentCompatible();
|
|
2578
|
+
if (!pathConfig) throw new WllamaError("AssetsPathConfig is required");
|
|
2579
|
+
this.pathConfig = pathConfig;
|
|
2580
|
+
this.config = wllamaConfig;
|
|
2581
|
+
this.cacheManager = (_a = wllamaConfig.cacheManager) != null ? _a : new cache_manager_default();
|
|
2582
|
+
this.modelManager = (_c = wllamaConfig.modelManager) != null ? _c : new ModelManager({
|
|
2583
|
+
cacheManager: this.cacheManager,
|
|
2584
|
+
logger: (_b = wllamaConfig.logger) != null ? _b : console,
|
|
2585
|
+
parallelDownloads: wllamaConfig.parallelDownloads,
|
|
2586
|
+
allowOffline: wllamaConfig.allowOffline
|
|
2587
|
+
});
|
|
2588
|
+
this.setCompat("default");
|
|
2589
|
+
}
|
|
2590
|
+
logger() {
|
|
2591
|
+
var _a;
|
|
2592
|
+
return (_a = this.config.logger) != null ? _a : console;
|
|
2593
|
+
}
|
|
2594
|
+
checkModelLoaded() {
|
|
2595
|
+
if (!this.isModelLoaded()) {
|
|
2596
|
+
throw new WllamaError(
|
|
2597
|
+
"loadModel() is not yet called",
|
|
2598
|
+
"model_not_loaded"
|
|
2599
|
+
);
|
|
2600
|
+
}
|
|
2601
|
+
}
|
|
2602
|
+
/**
|
|
2603
|
+
* Get the libllama version string, e.g. "b6327-4d74393".
|
|
2604
|
+
*
|
|
2605
|
+
* @returns version string embedded at build time.
|
|
2606
|
+
*/
|
|
2607
|
+
static getLibllamaVersion() {
|
|
2608
|
+
return LIBLLAMA_VERSION;
|
|
2609
|
+
}
|
|
2610
|
+
/**
|
|
2611
|
+
* Set compatibility options for Wllama.
|
|
2612
|
+
* @param compat Set to null to disable compatibility, or 'default' to use the default compat resources from CDN.
|
|
2613
|
+
* @param mode 'safari' by default; If set to 'firefox_safari', the compat mode will **also** be enabled on Firefox, which will significantly degrade the performance but allow using WebGPU on Firefox.
|
|
2614
|
+
*/
|
|
2615
|
+
setCompat(compat, mode = "safari") {
|
|
2616
|
+
if (mode === "safari") {
|
|
2617
|
+
if (isFirefox()) {
|
|
2618
|
+
this.compat = null;
|
|
2619
|
+
return;
|
|
2620
|
+
}
|
|
2621
|
+
}
|
|
2622
|
+
this.compat = compat === "default" ? WasmCompatFromCDN : compat;
|
|
2623
|
+
}
|
|
2624
|
+
/**
|
|
2625
|
+
* Check if the model is loaded via `loadModel()`
|
|
2626
|
+
*/
|
|
2627
|
+
isModelLoaded() {
|
|
2628
|
+
return !!this.proxy && !!this.metadata;
|
|
2629
|
+
}
|
|
2630
|
+
/**
|
|
2631
|
+
* Get token ID associated to BOS (begin of sentence) token.
|
|
2632
|
+
*
|
|
2633
|
+
* NOTE: This can only being used after `loadModel` is called.
|
|
2634
|
+
*
|
|
2635
|
+
* @returns -1 if the model is not loaded.
|
|
2636
|
+
*/
|
|
2637
|
+
getBOS() {
|
|
2638
|
+
return this.bosToken;
|
|
2639
|
+
}
|
|
2640
|
+
/**
|
|
2641
|
+
* Get token ID associated to EOS (end of sentence) token.
|
|
2642
|
+
*
|
|
2643
|
+
* NOTE: This can only being used after `loadModel` is called.
|
|
2644
|
+
*
|
|
2645
|
+
* @returns -1 if the model is not loaded.
|
|
2646
|
+
*/
|
|
2647
|
+
getEOS() {
|
|
2648
|
+
return this.eosToken;
|
|
2649
|
+
}
|
|
2650
|
+
/**
|
|
2651
|
+
* Get token ID associated to EOT (end of turn) token.
|
|
2652
|
+
*
|
|
2653
|
+
* NOTE: This can only being used after `loadModel` is called.
|
|
2654
|
+
*
|
|
2655
|
+
* @returns -1 if the model is not loaded.
|
|
2656
|
+
*/
|
|
2657
|
+
getEOT() {
|
|
2658
|
+
return this.eotToken;
|
|
2659
|
+
}
|
|
2660
|
+
/**
|
|
2661
|
+
* Check if a given token is end-of-generation token (e.g. EOS, EOT, etc.)
|
|
2662
|
+
*
|
|
2663
|
+
* @param token the token ID to be checked
|
|
2664
|
+
* @returns true if the token is EOS, EOT, or any other end-of-generation tokens
|
|
2665
|
+
*/
|
|
2666
|
+
isTokenEOG(token) {
|
|
2667
|
+
return token === this.eosToken || token === this.eotToken || this.eogTokens.has(token);
|
|
2668
|
+
}
|
|
2669
|
+
/**
|
|
2670
|
+
* Get token ID associated to token used by decoder, to start generating output sequence(only usable for encoder-decoder architecture). In other words, encoder uses normal BOS and decoder uses this token.
|
|
2671
|
+
*
|
|
2672
|
+
* NOTE: This can only being used after `loadModel` is called.
|
|
2673
|
+
*
|
|
2674
|
+
* @returns -1 if the model is not loaded.
|
|
2675
|
+
*/
|
|
2676
|
+
getDecoderStartToken() {
|
|
2677
|
+
return this.decoderStartToken;
|
|
2678
|
+
}
|
|
2679
|
+
/**
|
|
2680
|
+
* Get model hyper-parameters and metadata
|
|
2681
|
+
*
|
|
2682
|
+
* NOTE: This can only being used after `loadModel` is called.
|
|
2683
|
+
*
|
|
2684
|
+
* @returns ModelMetadata
|
|
2685
|
+
*/
|
|
2686
|
+
getModelMetadata() {
|
|
2687
|
+
this.checkModelLoaded();
|
|
2688
|
+
return this.metadata;
|
|
2689
|
+
}
|
|
2690
|
+
/**
|
|
2691
|
+
* Check if we're currently using multi-thread build.
|
|
2692
|
+
*
|
|
2693
|
+
* NOTE: This can only being used after `loadModel` is called.
|
|
2694
|
+
*
|
|
2695
|
+
* @returns true if multi-thread is used.
|
|
2696
|
+
*/
|
|
2697
|
+
isMultithread() {
|
|
2698
|
+
this.checkModelLoaded();
|
|
2699
|
+
return this.useMultiThread;
|
|
2700
|
+
}
|
|
2701
|
+
/**
|
|
2702
|
+
* Get number of threads used in the current context.
|
|
2703
|
+
*
|
|
2704
|
+
* NOTE: This can only being used after `loadModel` is called.
|
|
2705
|
+
*
|
|
2706
|
+
* @returns number of threads
|
|
2707
|
+
*/
|
|
2708
|
+
getNumThreads() {
|
|
2709
|
+
this.checkModelLoaded();
|
|
2710
|
+
return this.useMultiThread ? this.nbThreads : 1;
|
|
2711
|
+
}
|
|
2712
|
+
/**
|
|
2713
|
+
* Check if the current model uses encoder-decoder architecture
|
|
2714
|
+
*
|
|
2715
|
+
* NOTE: This can only being used after `loadModel` is called.
|
|
2716
|
+
*
|
|
2717
|
+
* @returns true if multi-thread is used.
|
|
2718
|
+
*/
|
|
2719
|
+
isEncoderDecoderArchitecture() {
|
|
2720
|
+
this.checkModelLoaded();
|
|
2721
|
+
return this.hasEncoder;
|
|
2722
|
+
}
|
|
2723
|
+
/**
|
|
2724
|
+
* Must we add BOS token to the tokenized sequence?
|
|
2725
|
+
*
|
|
2726
|
+
* NOTE: This can only being used after `loadModel` is called.
|
|
2727
|
+
*
|
|
2728
|
+
* @returns true if BOS token must be added to the sequence
|
|
2729
|
+
*/
|
|
2730
|
+
mustAddBosToken() {
|
|
2731
|
+
this.checkModelLoaded();
|
|
2732
|
+
return this.addBosToken;
|
|
2733
|
+
}
|
|
2734
|
+
/**
|
|
2735
|
+
* Must we add EOS token to the tokenized sequence?
|
|
2736
|
+
*
|
|
2737
|
+
* NOTE: This can only being used after `loadModel` is called.
|
|
2738
|
+
*
|
|
2739
|
+
* @returns true if EOS token must be added to the sequence
|
|
2740
|
+
*/
|
|
2741
|
+
mustAddEosToken() {
|
|
2742
|
+
this.checkModelLoaded();
|
|
2743
|
+
return this.addEosToken;
|
|
2744
|
+
}
|
|
2745
|
+
/**
|
|
2746
|
+
* Get the jinja chat template comes with the model. It only available if the original model (before converting to gguf) has the template in `tokenizer_config.json`
|
|
2747
|
+
*
|
|
2748
|
+
* NOTE: This can only being used after `loadModel` is called.
|
|
2749
|
+
*
|
|
2750
|
+
* @returns the jinja template. null if there is no template in gguf
|
|
2751
|
+
*/
|
|
2752
|
+
getChatTemplate() {
|
|
2753
|
+
var _a;
|
|
2754
|
+
this.checkModelLoaded();
|
|
2755
|
+
return (_a = this.chatTemplate) != null ? _a : null;
|
|
2756
|
+
}
|
|
2757
|
+
/**
|
|
2758
|
+
* Check if WebGPU is supported by the current environment.
|
|
2759
|
+
* @returns true if WebGPU is supported
|
|
2760
|
+
*/
|
|
2761
|
+
isSupportWebGPU() {
|
|
2762
|
+
return isSupportWebGPU();
|
|
2763
|
+
}
|
|
2764
|
+
/**
|
|
2765
|
+
* Load model from a given URL (or a list of URLs, in case the model is splitted into smaller files)
|
|
2766
|
+
* - If the model already been downloaded (via `downloadModel()`), then we will use the cached model
|
|
2767
|
+
* - Else, we download the model from internet
|
|
2768
|
+
* @param modelSourceOrURL
|
|
2769
|
+
* @param params
|
|
2770
|
+
*/
|
|
2771
|
+
loadModelFromUrl(_0) {
|
|
2772
|
+
return __async(this, arguments, function* (modelSourceOrURL, params = {}) {
|
|
2773
|
+
var _a;
|
|
2774
|
+
const source = isString(modelSourceOrURL) ? { url: modelSourceOrURL } : modelSourceOrURL;
|
|
2775
|
+
const useCache = (_a = params.useCache) != null ? _a : true;
|
|
2776
|
+
const model = useCache ? yield this.modelManager.getModelOrDownload(source, params) : yield this.modelManager.downloadModel(source, params);
|
|
2777
|
+
const blobs = yield model.open();
|
|
2778
|
+
return yield this.loadModel(blobs, params);
|
|
2779
|
+
});
|
|
2780
|
+
}
|
|
2781
|
+
/**
|
|
2782
|
+
* Load model from a given Hugging Face model ID and file path.
|
|
2783
|
+
*
|
|
2784
|
+
* @param hfOptions
|
|
2785
|
+
* @param params
|
|
2786
|
+
*/
|
|
2787
|
+
loadModelFromHF(_0) {
|
|
2788
|
+
return __async(this, arguments, function* (hfOptions, params = {}) {
|
|
2789
|
+
const source = yield getHFModelSource(hfOptions);
|
|
2790
|
+
return yield this.loadModelFromUrl(source, params);
|
|
2791
|
+
});
|
|
2792
|
+
}
|
|
2793
|
+
/**
|
|
2794
|
+
* Load model from a given list of Blob.
|
|
2795
|
+
*
|
|
2796
|
+
* You can pass multiple buffers into the function (in case the model contains multiple shards).
|
|
2797
|
+
*
|
|
2798
|
+
* @param ggufBlobsOrModel Can be either list of Blobs (in case you use local file), or a Model object (in case you use ModelManager)
|
|
2799
|
+
* @param params LoadModelParams
|
|
2800
|
+
*/
|
|
2801
|
+
loadModel(_0) {
|
|
2802
|
+
return __async(this, arguments, function* (ggufBlobsOrModel, params = {}) {
|
|
2803
|
+
var _a, _b, _c, _d, _e, _f, _g, _h;
|
|
2804
|
+
const blobs = ggufBlobsOrModel instanceof Model ? yield ggufBlobsOrModel.open() : [...ggufBlobsOrModel];
|
|
2805
|
+
if (blobs.some((b) => b.size === 0)) {
|
|
2806
|
+
throw new WllamaError(
|
|
2807
|
+
"Input model (or splits) must be non-empty Blob or File",
|
|
2808
|
+
"load_error"
|
|
2809
|
+
);
|
|
2810
|
+
}
|
|
2811
|
+
if (!this.pathConfig["default"]) {
|
|
2812
|
+
throw new WllamaError(
|
|
2813
|
+
'"default" is missing from pathConfig',
|
|
2814
|
+
"load_error"
|
|
2815
|
+
);
|
|
2816
|
+
}
|
|
2817
|
+
if (this.proxy) {
|
|
2818
|
+
throw new WllamaError("Module is already initialized", "load_error");
|
|
2819
|
+
}
|
|
2820
|
+
const supportMultiThread = yield isSupportMultiThread();
|
|
2821
|
+
const hwConccurency = Math.floor((navigator.hardwareConcurrency || 1) / 2);
|
|
2822
|
+
const nbThreads = (_a = params.n_threads) != null ? _a : hwConccurency;
|
|
2823
|
+
this.nbThreads = nbThreads;
|
|
2824
|
+
this.useMultiThread = supportMultiThread && nbThreads > 1;
|
|
2825
|
+
const workerResources = this.getWorkerResources();
|
|
2826
|
+
this.proxy = new ProxyToWorker(
|
|
2827
|
+
workerResources,
|
|
2828
|
+
this.useMultiThread ? nbThreads : 0,
|
|
2829
|
+
// 0 means disable pthread
|
|
2830
|
+
(_b = this.config.suppressNativeLog) != null ? _b : false,
|
|
2831
|
+
this.logger()
|
|
2832
|
+
);
|
|
2833
|
+
let logLevel = (_c = params.log_level) != null ? _c : 2 /* INFO */;
|
|
2834
|
+
if (this.config.suppressNativeLog) {
|
|
2835
|
+
logLevel = 9999;
|
|
2836
|
+
}
|
|
2837
|
+
const modelFiles = yield prepareBlobs(blobs);
|
|
2838
|
+
yield this.proxy.moduleInit(modelFiles.all);
|
|
2839
|
+
this.logger().debug("Calling wllamaStart...");
|
|
2840
|
+
const startResult = yield this.proxy.wllamaStart();
|
|
2841
|
+
if (!startResult.success) {
|
|
2842
|
+
throw new WllamaError(
|
|
2843
|
+
`Error while calling start function, result = ${startResult}`
|
|
2844
|
+
);
|
|
2845
|
+
}
|
|
2846
|
+
this.logger().debug("Loading model...");
|
|
2847
|
+
const loadResult = yield this.proxy.wllamaAction("load", {
|
|
2848
|
+
_name: "load_req",
|
|
2849
|
+
log_level: logLevel,
|
|
2850
|
+
// if async read is not supported, use mmap; refer to README-dev.md for more details
|
|
2851
|
+
use_mmap: !canUseAsyncFileRead(workerResources.compat),
|
|
2852
|
+
use_mlock: false,
|
|
2853
|
+
n_gpu_layers: (_d = params.n_gpu_layers) != null ? _d : 99999,
|
|
2854
|
+
n_ctx: (_e = params.n_ctx) != null ? _e : 1024,
|
|
2855
|
+
n_threads: this.useMultiThread ? nbThreads : 1,
|
|
2856
|
+
n_ctx_auto: false,
|
|
2857
|
+
// not supported for now
|
|
2858
|
+
mmproj_path: modelFiles.mmproj ? `/models/${MMPROJ_FILE_NAME}` : void 0,
|
|
2859
|
+
model_paths: modelFiles.llm.map((f) => `models/${f.name}`),
|
|
2860
|
+
embeddings: params.embeddings,
|
|
2861
|
+
offload_kqv: params.offload_kqv,
|
|
2862
|
+
n_batch: params.n_batch,
|
|
2863
|
+
pooling_type: params.pooling_type,
|
|
2864
|
+
rope_scaling_type: params.rope_scaling_type,
|
|
2865
|
+
rope_freq_base: params.rope_freq_base,
|
|
2866
|
+
rope_freq_scale: params.rope_freq_scale,
|
|
2867
|
+
yarn_ext_factor: params.yarn_ext_factor,
|
|
2868
|
+
yarn_attn_factor: params.yarn_attn_factor,
|
|
2869
|
+
yarn_beta_fast: params.yarn_beta_fast,
|
|
2870
|
+
yarn_beta_slow: params.yarn_beta_slow,
|
|
2871
|
+
yarn_orig_ctx: params.yarn_orig_ctx,
|
|
2872
|
+
cache_type_k: params.cache_type_k,
|
|
2873
|
+
cache_type_v: params.cache_type_v,
|
|
2874
|
+
n_parallel: 1,
|
|
2875
|
+
// only support single sequence for now
|
|
2876
|
+
kv_unified: false,
|
|
2877
|
+
// TODO: support kv unified cache
|
|
2878
|
+
flash_attn: params.flash_attn,
|
|
2879
|
+
swa_full: params.swa_full,
|
|
2880
|
+
chat_template: params.chat_template,
|
|
2881
|
+
jinja: params.jinja,
|
|
2882
|
+
reasoning: params.reasoning,
|
|
2883
|
+
image_min_tokens: params.image_min_tokens,
|
|
2884
|
+
image_max_tokens: params.image_max_tokens,
|
|
2885
|
+
warmup: params.warmup,
|
|
2886
|
+
no_kv_offload: params.no_kv_offload,
|
|
2887
|
+
mmproj_offload: params.mmproj_offload,
|
|
2888
|
+
cont_batching: params.cont_batching,
|
|
2889
|
+
n_keep: params.n_keep,
|
|
2890
|
+
ctx_shift: params.ctx_shift,
|
|
2891
|
+
cache_idle_slots: params.cache_idle_slots,
|
|
2892
|
+
n_cache_reuse: params.n_cache_reuse,
|
|
2893
|
+
lora_paths: (_f = params.lora_adapters) == null ? void 0 : _f.map((a) => a.path),
|
|
2894
|
+
lora_scales: (_g = params.lora_adapters) == null ? void 0 : _g.map((a) => {
|
|
2895
|
+
var _a2;
|
|
2896
|
+
return (_a2 = a.scale) != null ? _a2 : 1;
|
|
2897
|
+
}),
|
|
2898
|
+
lora_init_without_apply: params.lora_init_without_apply,
|
|
2899
|
+
spec_draft_model: params.spec_draft_model,
|
|
2900
|
+
spec_draft_ngl: params.spec_draft_ngl,
|
|
2901
|
+
spec_draft_n_max: params.spec_draft_n_max,
|
|
2902
|
+
spec_draft_n_min: params.spec_draft_n_min,
|
|
2903
|
+
spec_draft_p_min: params.spec_draft_p_min,
|
|
2904
|
+
spec_draft_threads: params.spec_draft_threads,
|
|
2905
|
+
spec_draft_threads_batch: params.spec_draft_threads_batch,
|
|
2906
|
+
kv_overrides_keys: params.kv_overrides ? Object.keys(params.kv_overrides) : void 0,
|
|
2907
|
+
kv_overrides_vals: params.kv_overrides ? Object.values(params.kv_overrides) : void 0,
|
|
2908
|
+
reasoning_budget_tokens: params.reasoning_budget_tokens,
|
|
2909
|
+
reasoning_budget_message: params.reasoning_budget_message,
|
|
2910
|
+
reasoning_format: params.reasoning_format,
|
|
2911
|
+
skip_chat_parsing: params.skip_chat_parsing,
|
|
2912
|
+
prefill_assistant: params.prefill_assistant
|
|
2913
|
+
});
|
|
2914
|
+
const loadedCtxInfo = __spreadProps(__spreadValues({}, loadResult), {
|
|
2915
|
+
metadata: {}
|
|
2916
|
+
});
|
|
2917
|
+
for (let i = 0; i < loadResult.metadata_key.length; i++) {
|
|
2918
|
+
loadedCtxInfo.metadata[loadResult.metadata_key[i]] = loadResult.metadata_val[i];
|
|
2919
|
+
}
|
|
2920
|
+
this.seed = params.seed;
|
|
2921
|
+
this.bosToken = loadedCtxInfo.token_bos;
|
|
2922
|
+
this.eosToken = loadedCtxInfo.token_eos;
|
|
2923
|
+
this.eotToken = loadedCtxInfo.token_eot;
|
|
2924
|
+
this.useEmbeddings = !!params.embeddings;
|
|
2925
|
+
this.useRerank = params.pooling_type == "rank";
|
|
2926
|
+
this.metadata = {
|
|
2927
|
+
hparams: {
|
|
2928
|
+
nVocab: loadedCtxInfo.n_vocab,
|
|
2929
|
+
nCtxTrain: loadedCtxInfo.n_ctx_train,
|
|
2930
|
+
nEmbd: loadedCtxInfo.n_embd,
|
|
2931
|
+
nLayer: loadedCtxInfo.n_layer
|
|
2932
|
+
},
|
|
2933
|
+
meta: loadedCtxInfo.metadata
|
|
2934
|
+
};
|
|
2935
|
+
this.hasEncoder = !!loadedCtxInfo.has_encoder;
|
|
2936
|
+
this.decoderStartToken = loadedCtxInfo.token_decoder_start;
|
|
2937
|
+
this.addBosToken = loadedCtxInfo.add_bos_token;
|
|
2938
|
+
this.addEosToken = loadedCtxInfo.add_eos_token;
|
|
2939
|
+
this.chatTemplate = loadedCtxInfo.metadata["tokenizer.chat_template"];
|
|
2940
|
+
this.loadedContextInfo = loadedCtxInfo;
|
|
2941
|
+
this.eogTokens = new Set(loadedCtxInfo.list_tokens_eog);
|
|
2942
|
+
this.mediaMarker = loadedCtxInfo.media_marker;
|
|
2943
|
+
this.chatTemplateKwargs = (_h = params.default_template_kwargs) != null ? _h : {};
|
|
2944
|
+
this.logger().debug({ loadedCtxInfo });
|
|
2945
|
+
});
|
|
2946
|
+
}
|
|
2947
|
+
getLoadedContextInfo() {
|
|
2948
|
+
this.checkModelLoaded();
|
|
2949
|
+
if (!this.loadedContextInfo) {
|
|
2950
|
+
throw new WllamaError("Loaded context info is not available");
|
|
2951
|
+
}
|
|
2952
|
+
return __spreadValues({}, this.loadedContextInfo);
|
|
2953
|
+
}
|
|
2954
|
+
//////////////////////////////////////////////
|
|
2955
|
+
// High level API
|
|
2956
|
+
/**
|
|
2957
|
+
* Calculate embedding vector for a given text.
|
|
2958
|
+
* By default, BOS and EOS tokens will be added automatically. You can use the "skipBOS" and "skipEOS" option to disable it.
|
|
2959
|
+
* @param options OAI-compatible embedding creation options
|
|
2960
|
+
* @returns OAI-compatible embedding response
|
|
2961
|
+
*/
|
|
2962
|
+
createEmbedding(options) {
|
|
2963
|
+
return __async(this, null, function* () {
|
|
2964
|
+
this.checkModelLoaded();
|
|
2965
|
+
if (!this.useEmbeddings) {
|
|
2966
|
+
throw new WllamaError(
|
|
2967
|
+
"Embeddings is not enabled. Please set it via LoadModelParams.embeddings"
|
|
2968
|
+
);
|
|
2969
|
+
}
|
|
2970
|
+
const result = yield this.proxy.wllamaAction(
|
|
2971
|
+
"embedding",
|
|
2972
|
+
{
|
|
2973
|
+
_name: "embd_req",
|
|
2974
|
+
data_json: JSON.stringify(options),
|
|
2975
|
+
files: []
|
|
2976
|
+
// TODO: support file input
|
|
2977
|
+
}
|
|
2978
|
+
);
|
|
2979
|
+
if (!result.success) {
|
|
2980
|
+
throw new WllamaError(
|
|
2981
|
+
"Model failed to start inference",
|
|
2982
|
+
"inference_error"
|
|
2983
|
+
);
|
|
2984
|
+
}
|
|
2985
|
+
return yield this.getResponse(options, false);
|
|
2986
|
+
});
|
|
2987
|
+
}
|
|
2988
|
+
/**
|
|
2989
|
+
* Rerank a list of documents against a query.
|
|
2990
|
+
* Requires the model to be loaded with embeddings: true and pooling_type: 'rank'.
|
|
2991
|
+
* @param options Reranking options (query, documents, top_n)
|
|
2992
|
+
* @returns Reranking response with relevance scores sorted highest first
|
|
2993
|
+
*/
|
|
2994
|
+
createRerank(options) {
|
|
2995
|
+
return __async(this, null, function* () {
|
|
2996
|
+
var _a, _b;
|
|
2997
|
+
this.checkModelLoaded();
|
|
2998
|
+
if (!this.useEmbeddings || !this.useRerank) {
|
|
2999
|
+
throw new WllamaError(
|
|
3000
|
+
"Rerank is not enabled. Please set it via LoadModelParams: embeddings = true and pooling_type = rank"
|
|
3001
|
+
);
|
|
3002
|
+
}
|
|
3003
|
+
const top_n = (_a = options.top_n) != null ? _a : options.documents.length;
|
|
3004
|
+
let totalTokens = 0;
|
|
3005
|
+
const rawResults = [];
|
|
3006
|
+
for (let i = 0; i < options.documents.length; i++) {
|
|
3007
|
+
const result = yield this.proxy.wllamaAction("rerank", {
|
|
3008
|
+
_name: "rrnk_req",
|
|
3009
|
+
data_json: JSON.stringify({
|
|
3010
|
+
query: options.query,
|
|
3011
|
+
document: options.documents[i]
|
|
3012
|
+
})
|
|
3013
|
+
});
|
|
3014
|
+
if (!result.success) {
|
|
3015
|
+
throw new WllamaError(
|
|
3016
|
+
"Model failed to start reranking",
|
|
3017
|
+
"inference_error"
|
|
3018
|
+
);
|
|
3019
|
+
}
|
|
3020
|
+
const { score, tokens_evaluated } = yield this.getRerankResult();
|
|
3021
|
+
totalTokens += tokens_evaluated;
|
|
3022
|
+
rawResults.push({ index: i, score });
|
|
3023
|
+
}
|
|
3024
|
+
rawResults.sort((a, b) => b.score - a.score);
|
|
3025
|
+
return {
|
|
3026
|
+
model: (_b = this.getModelMetadata().meta["general.name"]) != null ? _b : "",
|
|
3027
|
+
object: "list",
|
|
3028
|
+
usage: { prompt_tokens: totalTokens, total_tokens: totalTokens },
|
|
3029
|
+
results: rawResults.slice(0, top_n).map(({ index, score }) => ({
|
|
3030
|
+
index,
|
|
3031
|
+
relevance_score: score
|
|
3032
|
+
}))
|
|
3033
|
+
};
|
|
3034
|
+
});
|
|
3035
|
+
}
|
|
3036
|
+
createChatCompletion(options) {
|
|
3037
|
+
return __async(this, null, function* () {
|
|
3038
|
+
var _a;
|
|
3039
|
+
if (Object.keys(this.chatTemplateKwargs).length > 0) {
|
|
3040
|
+
options = __spreadProps(__spreadValues({}, options), {
|
|
3041
|
+
chat_template_kwargs: __spreadValues(__spreadValues({}, this.chatTemplateKwargs), (_a = options.chat_template_kwargs) != null ? _a : {})
|
|
3042
|
+
});
|
|
3043
|
+
}
|
|
3044
|
+
if (options.stream && options.onData) {
|
|
3045
|
+
yield this.createCompletionImpl(options);
|
|
3046
|
+
} else if (options.stream) {
|
|
3047
|
+
return yield this.createCompletionGenerator(options);
|
|
3048
|
+
} else {
|
|
3049
|
+
return yield this.createCompletionImpl(__spreadProps(__spreadValues({}, options), { stream: false }));
|
|
3050
|
+
}
|
|
3051
|
+
});
|
|
3052
|
+
}
|
|
3053
|
+
createCompletion(options) {
|
|
3054
|
+
return __async(this, null, function* () {
|
|
3055
|
+
if (options.stream && options.onData) {
|
|
3056
|
+
yield this.createCompletionImpl(options);
|
|
3057
|
+
} else if (options.stream) {
|
|
3058
|
+
return yield this.createCompletionGenerator(options);
|
|
3059
|
+
} else {
|
|
3060
|
+
return yield this.createCompletionImpl(__spreadProps(__spreadValues({}, options), { stream: false }));
|
|
3061
|
+
}
|
|
3062
|
+
});
|
|
3063
|
+
}
|
|
3064
|
+
/**
|
|
3065
|
+
* Private implementation of createCompletion
|
|
3066
|
+
*/
|
|
3067
|
+
createCompletionImpl(options) {
|
|
3068
|
+
return __async(this, null, function* () {
|
|
3069
|
+
this.checkModelLoaded();
|
|
3070
|
+
const isStream = !!options.stream;
|
|
3071
|
+
const isChat = !!options.messages;
|
|
3072
|
+
const customOpt = {};
|
|
3073
|
+
if (this.seed !== void 0) {
|
|
3074
|
+
customOpt.seed = this.seed;
|
|
3075
|
+
}
|
|
3076
|
+
let files = [];
|
|
3077
|
+
if (isChat) {
|
|
3078
|
+
const tmp = this.prepareMultimodalInput(
|
|
3079
|
+
options
|
|
3080
|
+
);
|
|
3081
|
+
options = tmp.params;
|
|
3082
|
+
files = tmp.files;
|
|
3083
|
+
}
|
|
3084
|
+
const result = yield this.proxy.wllamaAction(
|
|
3085
|
+
"completion",
|
|
3086
|
+
{
|
|
3087
|
+
_name: "cmpl_req",
|
|
3088
|
+
is_chat: isChat,
|
|
3089
|
+
data_json: JSON.stringify(__spreadValues(__spreadValues({}, options), customOpt)),
|
|
3090
|
+
files: files.map((f) => new Uint8Array(f))
|
|
3091
|
+
}
|
|
3092
|
+
);
|
|
3093
|
+
if (!result.success) {
|
|
3094
|
+
throw new WllamaError(
|
|
3095
|
+
"Model failed to start inference",
|
|
3096
|
+
"inference_error"
|
|
3097
|
+
);
|
|
3098
|
+
}
|
|
3099
|
+
return yield this.getResponse(
|
|
3100
|
+
options,
|
|
3101
|
+
isStream
|
|
3102
|
+
);
|
|
3103
|
+
});
|
|
3104
|
+
}
|
|
3105
|
+
/**
|
|
3106
|
+
* Same with `createCompletion`, but returns an async iterator instead.
|
|
3107
|
+
* Only called when stream=true and no onData is provided.
|
|
3108
|
+
*/
|
|
3109
|
+
createCompletionGenerator(options) {
|
|
3110
|
+
return new Promise((resolve) => {
|
|
3111
|
+
const createGenerator = cbToAsyncIter(
|
|
3112
|
+
(callback) => {
|
|
3113
|
+
this.createCompletionImpl(__spreadProps(__spreadValues({}, options), {
|
|
3114
|
+
onData: (chunk) => callback(chunk)
|
|
3115
|
+
})).then(() => callback(void 0, true)).catch((err) => callback(void 0, false, err));
|
|
3116
|
+
}
|
|
3117
|
+
);
|
|
3118
|
+
resolve(createGenerator());
|
|
3119
|
+
});
|
|
3120
|
+
}
|
|
3121
|
+
/**
|
|
3122
|
+
* Whether the currently loaded model supports a specific input modality (e.g. image or audio).
|
|
3123
|
+
* @param modality
|
|
3124
|
+
* @returns
|
|
3125
|
+
*/
|
|
3126
|
+
supportInputModality(modality) {
|
|
3127
|
+
this.checkModelLoaded();
|
|
3128
|
+
if (modality === "image") {
|
|
3129
|
+
return !!this.loadedContextInfo.has_image_input;
|
|
3130
|
+
} else if (modality === "audio") {
|
|
3131
|
+
return !!this.loadedContextInfo.has_audio_input;
|
|
3132
|
+
} else {
|
|
3133
|
+
throw new WllamaError(
|
|
3134
|
+
"Unsupported modality: " + modality,
|
|
3135
|
+
"unknown_error"
|
|
3136
|
+
);
|
|
3137
|
+
}
|
|
3138
|
+
}
|
|
3139
|
+
/**
|
|
3140
|
+
* Unload the model and free all memory.
|
|
3141
|
+
*
|
|
3142
|
+
* Note: This function will NOT crash if model is not yet loaded
|
|
3143
|
+
*/
|
|
3144
|
+
exit() {
|
|
3145
|
+
return __async(this, null, function* () {
|
|
3146
|
+
var _a;
|
|
3147
|
+
yield (_a = this.proxy) == null ? void 0 : _a.wllamaExit();
|
|
3148
|
+
this.proxy = null;
|
|
3149
|
+
});
|
|
3150
|
+
}
|
|
3151
|
+
/**
|
|
3152
|
+
* [FOR DEBUGGING ONLY] Run ggml backend ops tests without loading any model.
|
|
3153
|
+
*
|
|
3154
|
+
* Initializes the wasm runtime, executes `test-backend-ops` with the given args, then shuts down.
|
|
3155
|
+
*
|
|
3156
|
+
* For more info, please refer to guides/debug.md
|
|
3157
|
+
*
|
|
3158
|
+
* @param args Arguments forwarded to test-backend-ops (e.g. ["-o", "ADD"])
|
|
3159
|
+
* @returns retcode (0 = all tests passed) and success flag
|
|
3160
|
+
*/
|
|
3161
|
+
testBackendOps() {
|
|
3162
|
+
return __async(this, arguments, function* (args = []) {
|
|
3163
|
+
var _a;
|
|
3164
|
+
if (!this.pathConfig["default"]) {
|
|
3165
|
+
throw new WllamaError(
|
|
3166
|
+
'"default" is missing from pathConfig',
|
|
3167
|
+
"load_error"
|
|
3168
|
+
);
|
|
3169
|
+
}
|
|
3170
|
+
if (!(yield isSupportMultiThread())) {
|
|
3171
|
+
throw new WllamaError(
|
|
3172
|
+
"Multi-threading is required to run backend ops tests, but it is not supported in the current environment."
|
|
3173
|
+
);
|
|
3174
|
+
}
|
|
3175
|
+
const tmpProxy = new ProxyToWorker(
|
|
3176
|
+
this.getWorkerResources(),
|
|
3177
|
+
0,
|
|
3178
|
+
// single-thread; no model needed
|
|
3179
|
+
(_a = this.config.suppressNativeLog) != null ? _a : false,
|
|
3180
|
+
this.logger()
|
|
3181
|
+
);
|
|
3182
|
+
try {
|
|
3183
|
+
yield tmpProxy.moduleInit([]);
|
|
3184
|
+
const startResult = yield tmpProxy.wllamaStart();
|
|
3185
|
+
if (!startResult.success) {
|
|
3186
|
+
throw new WllamaError(
|
|
3187
|
+
`Error while calling start function, result = ${startResult}`
|
|
3188
|
+
);
|
|
3189
|
+
}
|
|
3190
|
+
const result = yield tmpProxy.wllamaAction(
|
|
3191
|
+
"test_backend_ops",
|
|
3192
|
+
{ _name: "tbop_req", args: ["test-backend-ops", ...args] }
|
|
3193
|
+
);
|
|
3194
|
+
return { retcode: result.retcode, success: result.success };
|
|
3195
|
+
} finally {
|
|
3196
|
+
yield tmpProxy.wllamaExit();
|
|
3197
|
+
}
|
|
3198
|
+
});
|
|
3199
|
+
}
|
|
3200
|
+
//////////////////////////////////////////////
|
|
3201
|
+
// Low level API
|
|
3202
|
+
// TODO: add back
|
|
3203
|
+
/**
|
|
3204
|
+
* get debug info
|
|
3205
|
+
*/
|
|
3206
|
+
_getDebugInfo() {
|
|
3207
|
+
return __async(this, null, function* () {
|
|
3208
|
+
this.checkModelLoaded();
|
|
3209
|
+
return yield this.proxy.wllamaDebug();
|
|
3210
|
+
});
|
|
3211
|
+
}
|
|
3212
|
+
//////////////////////////////////////////////
|
|
3213
|
+
// Utils
|
|
3214
|
+
jsonDecode(data_json) {
|
|
3215
|
+
try {
|
|
3216
|
+
return JSON.parse(data_json);
|
|
3217
|
+
} catch (e) {
|
|
3218
|
+
this.logger().error("Failed to parse JSON:", data_json);
|
|
3219
|
+
throw new WllamaError("Failed to parse model output", "inference_error");
|
|
3220
|
+
}
|
|
3221
|
+
}
|
|
3222
|
+
prepareMultimodalInput(params) {
|
|
3223
|
+
const msg = params.messages;
|
|
3224
|
+
const msgNew = [];
|
|
3225
|
+
const files = [];
|
|
3226
|
+
for (const m of msg) {
|
|
3227
|
+
if (Array.isArray(m.content)) {
|
|
3228
|
+
const newContent = [];
|
|
3229
|
+
for (const c of m.content) {
|
|
3230
|
+
if (c.type === "text") {
|
|
3231
|
+
newContent.push(c);
|
|
3232
|
+
} else {
|
|
3233
|
+
if (!this.mediaMarker) {
|
|
3234
|
+
throw new WllamaError(
|
|
3235
|
+
"Media marker is undefined",
|
|
3236
|
+
"inference_error"
|
|
3237
|
+
);
|
|
3238
|
+
}
|
|
3239
|
+
files.push(c.data);
|
|
3240
|
+
newContent.push({
|
|
3241
|
+
type: "text",
|
|
3242
|
+
text: this.mediaMarker
|
|
3243
|
+
});
|
|
3244
|
+
}
|
|
3245
|
+
}
|
|
3246
|
+
msgNew.push(__spreadProps(__spreadValues({}, m), {
|
|
3247
|
+
content: newContent
|
|
3248
|
+
}));
|
|
3249
|
+
} else {
|
|
3250
|
+
msgNew.push(m);
|
|
3251
|
+
}
|
|
3252
|
+
}
|
|
3253
|
+
return {
|
|
3254
|
+
params: __spreadProps(__spreadValues({}, params), {
|
|
3255
|
+
messages: msgNew
|
|
3256
|
+
}),
|
|
3257
|
+
files
|
|
3258
|
+
};
|
|
3259
|
+
}
|
|
3260
|
+
getRerankResult() {
|
|
3261
|
+
return __async(this, null, function* () {
|
|
3262
|
+
while (true) {
|
|
3263
|
+
const chunk = yield this.proxy.wllamaAction(
|
|
3264
|
+
"get_result",
|
|
3265
|
+
{ _name: "gres_req" }
|
|
3266
|
+
);
|
|
3267
|
+
const jsonString = chunk.data_json;
|
|
3268
|
+
if (jsonString && jsonString.length > 0) {
|
|
3269
|
+
if (chunk.is_error) {
|
|
3270
|
+
const jsonData = this.jsonDecode(jsonString);
|
|
3271
|
+
throw new WllamaError(
|
|
3272
|
+
jsonData.message || "Unknown reranking error",
|
|
3273
|
+
"inference_error"
|
|
3274
|
+
);
|
|
3275
|
+
}
|
|
3276
|
+
return this.jsonDecode(jsonString);
|
|
3277
|
+
}
|
|
3278
|
+
if (!chunk.has_more) break;
|
|
3279
|
+
}
|
|
3280
|
+
throw new WllamaError("No reranking result received", "inference_error");
|
|
3281
|
+
});
|
|
3282
|
+
}
|
|
3283
|
+
getResponse(options, isStream) {
|
|
3284
|
+
return __async(this, null, function* () {
|
|
3285
|
+
var _a, _b;
|
|
3286
|
+
let finalResult = null;
|
|
3287
|
+
while (true) {
|
|
3288
|
+
if ((_a = options.abortSignal) == null ? void 0 : _a.aborted) {
|
|
3289
|
+
throw new WllamaAbortError();
|
|
3290
|
+
}
|
|
3291
|
+
const result_chunk = yield this.proxy.wllamaAction(
|
|
3292
|
+
"get_result",
|
|
3293
|
+
{
|
|
3294
|
+
_name: "gres_req"
|
|
3295
|
+
}
|
|
3296
|
+
);
|
|
3297
|
+
const jsonString = result_chunk.data_json;
|
|
3298
|
+
if (!jsonString || jsonString.length === 0) {
|
|
3299
|
+
if (!result_chunk.has_more) {
|
|
3300
|
+
break;
|
|
3301
|
+
} else {
|
|
3302
|
+
continue;
|
|
3303
|
+
}
|
|
3304
|
+
}
|
|
3305
|
+
if (jsonString == "null") {
|
|
3306
|
+
continue;
|
|
3307
|
+
}
|
|
3308
|
+
let jsonData = this.jsonDecode(jsonString);
|
|
3309
|
+
finalResult = jsonData;
|
|
3310
|
+
if (result_chunk.is_error) {
|
|
3311
|
+
this.logger().error("Model returned an error:", jsonData);
|
|
3312
|
+
throw new WllamaError(
|
|
3313
|
+
jsonData.message || "Unknown inference error",
|
|
3314
|
+
"inference_error"
|
|
3315
|
+
);
|
|
3316
|
+
}
|
|
3317
|
+
if (isStream) {
|
|
3318
|
+
if (!Array.isArray(jsonData)) {
|
|
3319
|
+
jsonData = [jsonData];
|
|
3320
|
+
}
|
|
3321
|
+
for (const chunk of jsonData) {
|
|
3322
|
+
(_b = options.onData) == null ? void 0 : _b.call(options, chunk);
|
|
3323
|
+
finalResult = chunk;
|
|
3324
|
+
}
|
|
3325
|
+
}
|
|
3326
|
+
if (!result_chunk.has_more) {
|
|
3327
|
+
break;
|
|
3328
|
+
}
|
|
3329
|
+
}
|
|
3330
|
+
return finalResult;
|
|
3331
|
+
});
|
|
3332
|
+
}
|
|
3333
|
+
getWorkerResources() {
|
|
3334
|
+
const workerResources = {
|
|
3335
|
+
wasmPath: absoluteUrl(this.pathConfig["default"]),
|
|
3336
|
+
compat: false
|
|
3337
|
+
};
|
|
3338
|
+
if (needCompat()) {
|
|
3339
|
+
if (!this.compat) {
|
|
3340
|
+
this.logger().warn(
|
|
3341
|
+
"Not using compat mode" + (isFirefox() ? " (expected on Firefox - WebGPU will be disabled)" : "")
|
|
3342
|
+
);
|
|
3343
|
+
} else {
|
|
3344
|
+
const isUsingDefault = this.compat.worker === WasmCompatFromCDN.worker && this.compat.wasm === WasmCompatFromCDN.wasm;
|
|
3345
|
+
if (isUsingDefault) {
|
|
3346
|
+
this.logger().warn(
|
|
3347
|
+
"Compatibility mode is activated, using resources from CDN. To use local resources, please refer to @wllama/wllama-compat package."
|
|
3348
|
+
);
|
|
3349
|
+
this.logger().warn(
|
|
3350
|
+
"IMPORTANT: Performance will be significantly degraded in compatibility mode."
|
|
3351
|
+
);
|
|
3352
|
+
}
|
|
3353
|
+
workerResources.wasmPath = absoluteUrl(this.compat.wasm);
|
|
3354
|
+
workerResources.jsPath = this.compat.worker;
|
|
3355
|
+
workerResources.compat = true;
|
|
3356
|
+
}
|
|
3357
|
+
}
|
|
3358
|
+
if (isFirefox()) {
|
|
3359
|
+
if (workerResources.compat) {
|
|
3360
|
+
this.logger().warn(
|
|
3361
|
+
'Using compat mode on Firefox, performance will be significantly degraded; Consider enabling "javascript.options.wasm_js_promise_integration" in "about:config".'
|
|
3362
|
+
);
|
|
3363
|
+
} else if (!isSupportJSPI()) {
|
|
3364
|
+
this.logger().warn(
|
|
3365
|
+
'WebGPU is disabled on Firefox due to missing JSPI support. Please consider enabling compat mode, or enabling "javascript.options.wasm_js_promise_integration" in "about:config".'
|
|
3366
|
+
);
|
|
3367
|
+
}
|
|
3368
|
+
}
|
|
3369
|
+
return workerResources;
|
|
3370
|
+
}
|
|
3371
|
+
};
|
|
3372
|
+
export {
|
|
3373
|
+
CacheManager,
|
|
3374
|
+
LogLevel,
|
|
3375
|
+
LoggerWithoutDebug,
|
|
3376
|
+
Model,
|
|
3377
|
+
ModelManager,
|
|
3378
|
+
ModelValidationStatus,
|
|
3379
|
+
POLYFILL_ETAG,
|
|
3380
|
+
Wllama,
|
|
3381
|
+
WllamaAbortError,
|
|
3382
|
+
WllamaError,
|
|
3383
|
+
WllamaRuntimeError,
|
|
3384
|
+
getHFFileSHA256,
|
|
3385
|
+
getHFModelSource,
|
|
3386
|
+
isValidGgufFile
|
|
3387
|
+
};
|