porffor 0.2.0-50b82f8 → 0.2.0-5ac7ea0
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/README.md +82 -46
- package/asur/README.md +2 -0
- package/asur/index.js +978 -0
- package/compiler/2c.js +316 -71
- package/compiler/builtins/base64.ts +153 -0
- package/compiler/builtins/crypto.ts +132 -0
- package/compiler/builtins/porffor.d.ts +26 -0
- package/compiler/builtins.js +590 -255
- package/compiler/codeGen.js +520 -184
- package/compiler/decompile.js +3 -3
- package/compiler/generated_builtins.js +25 -0
- package/compiler/index.js +22 -22
- package/compiler/log.js +4 -1
- package/compiler/opt.js +52 -27
- package/compiler/parse.js +4 -2
- package/compiler/precompile.js +129 -0
- package/compiler/prefs.js +26 -0
- package/compiler/prototype.js +177 -21
- package/compiler/sections.js +8 -7
- package/compiler/wasmSpec.js +20 -6
- package/compiler/wrap.js +112 -11
- package/package.json +1 -1
- package/porf +2 -0
- package/rhemyn/compile.js +2 -1
- package/runner/index.js +26 -3
- package/runner/profiler.js +83 -0
- package/compiler/builtins/base64.js +0 -92
- package/runner/profile.js +0 -46
- package/runner/results.json +0 -1
package/compiler/builtins.js
CHANGED
@@ -1,5 +1,7 @@
|
|
1
|
-
import { Blocktype, Opcodes, Valtype } from "./wasmSpec.js";
|
1
|
+
import { Blocktype, Opcodes, Valtype, ValtypeSize } from "./wasmSpec.js";
|
2
2
|
import { number, i32x4 } from "./embedding.js";
|
3
|
+
import Prefs from './prefs.js';
|
4
|
+
import * as GeneratedBuiltins from './generated_builtins.js';
|
3
5
|
|
4
6
|
export const importedFuncs = [
|
5
7
|
{
|
@@ -19,8 +21,14 @@ export const importedFuncs = [
|
|
19
21
|
import: 't',
|
20
22
|
params: 0,
|
21
23
|
returns: 1
|
24
|
+
},
|
25
|
+
{
|
26
|
+
name: 'profile',
|
27
|
+
import: 'z',
|
28
|
+
params: 1,
|
29
|
+
returns: 0
|
22
30
|
}
|
23
|
-
];
|
31
|
+
].filter(x => x);
|
24
32
|
|
25
33
|
for (let i = 0; i < importedFuncs.length; i++) {
|
26
34
|
const f = importedFuncs[i];
|
@@ -29,6 +37,21 @@ for (let i = 0; i < importedFuncs.length; i++) {
|
|
29
37
|
|
30
38
|
const char = c => number(c.charCodeAt(0));
|
31
39
|
|
40
|
+
const printStaticStr = str => {
|
41
|
+
const out = [];
|
42
|
+
|
43
|
+
for (let i = 0; i < str.length; i++) {
|
44
|
+
out.push(
|
45
|
+
// ...number(str.charCodeAt(i)),
|
46
|
+
...number(str.charCodeAt(i), Valtype.i32),
|
47
|
+
Opcodes.i32_from_u,
|
48
|
+
[ Opcodes.call, importedFuncs.printChar ]
|
49
|
+
);
|
50
|
+
}
|
51
|
+
|
52
|
+
return out;
|
53
|
+
};
|
54
|
+
|
32
55
|
// todo: somehow diff between these (undefined != null) while remaining falsey in wasm as a number value
|
33
56
|
export const UNDEFINED = 0;
|
34
57
|
export const NULL = 0;
|
@@ -102,6 +125,10 @@ export const BuiltinVars = function() {
|
|
102
125
|
|
103
126
|
// stubs just so that parent objects exist
|
104
127
|
this.Math = number(1);
|
128
|
+
|
129
|
+
// wintercg(tm)
|
130
|
+
this.__navigator_userAgent = (scope, { makeString }) => makeString(scope, `Porffor/0.2.0`, false, '__navigator_userAgent');
|
131
|
+
this.__navigator_userAgent.type = Prefs.bytestring ? '_bytestring' : 'string';
|
105
132
|
};
|
106
133
|
|
107
134
|
export const BuiltinFuncs = function() {
|
@@ -155,25 +182,6 @@ export const BuiltinFuncs = function() {
|
|
155
182
|
]
|
156
183
|
};
|
157
184
|
|
158
|
-
// todo: return false for NaN
|
159
|
-
this.Boolean = {
|
160
|
-
params: [ valtypeBinary ],
|
161
|
-
locals: [],
|
162
|
-
returns: [ valtypeBinary ],
|
163
|
-
returnType: 'boolean',
|
164
|
-
wasm: [
|
165
|
-
[ Opcodes.local_get, 0 ],
|
166
|
-
...(valtype === 'f64' ? [
|
167
|
-
...number(0),
|
168
|
-
[ Opcodes.f64_ne ]
|
169
|
-
] : [
|
170
|
-
...Opcodes.eqz,
|
171
|
-
[ Opcodes.i32_eqz ]
|
172
|
-
]),
|
173
|
-
Opcodes.i32_from
|
174
|
-
]
|
175
|
-
};
|
176
|
-
|
177
185
|
// just return given (default 0) for (new) Object() as we somewhat supports object just not constructor
|
178
186
|
this.Object = {
|
179
187
|
params: [ valtypeBinary ],
|
@@ -187,12 +195,159 @@ export const BuiltinFuncs = function() {
|
|
187
195
|
|
188
196
|
|
189
197
|
this.__console_log = {
|
190
|
-
params: [ valtypeBinary ],
|
191
|
-
|
198
|
+
params: [ valtypeBinary, Valtype.i32 ],
|
199
|
+
typedParams: true,
|
200
|
+
locals: [ Valtype.i32, Valtype.i32 ],
|
192
201
|
returns: [],
|
193
|
-
wasm: [
|
194
|
-
[ Opcodes.local_get,
|
195
|
-
|
202
|
+
wasm: (scope, { TYPES, typeSwitch }) => [
|
203
|
+
...typeSwitch(scope, [ [ Opcodes.local_get, 1 ] ], {
|
204
|
+
[TYPES.number]: [
|
205
|
+
[ Opcodes.local_get, 0 ],
|
206
|
+
[ Opcodes.call, importedFuncs.print ],
|
207
|
+
],
|
208
|
+
[TYPES.boolean]: [
|
209
|
+
[ Opcodes.local_get, 0 ],
|
210
|
+
Opcodes.i32_to_u,
|
211
|
+
[ Opcodes.if, Blocktype.void ],
|
212
|
+
...printStaticStr('true'),
|
213
|
+
[ Opcodes.else ],
|
214
|
+
...printStaticStr('false'),
|
215
|
+
[ Opcodes.end ]
|
216
|
+
],
|
217
|
+
[TYPES.string]: [
|
218
|
+
// simply print a string :))
|
219
|
+
// cache input pointer as i32
|
220
|
+
[ Opcodes.local_get, 0 ],
|
221
|
+
Opcodes.i32_to_u,
|
222
|
+
[ Opcodes.local_tee, 2 ],
|
223
|
+
|
224
|
+
// make end pointer
|
225
|
+
[ Opcodes.i32_load, Math.log2(ValtypeSize.i32) - 1, 0 ],
|
226
|
+
...number(ValtypeSize.i16, Valtype.i32),
|
227
|
+
[ Opcodes.i32_mul ],
|
228
|
+
|
229
|
+
[ Opcodes.local_get, 2 ],
|
230
|
+
[ Opcodes.i32_add ],
|
231
|
+
[ Opcodes.local_set, 3 ],
|
232
|
+
|
233
|
+
[ Opcodes.loop, Blocktype.void ],
|
234
|
+
|
235
|
+
// print current char
|
236
|
+
[ Opcodes.local_get, 2 ],
|
237
|
+
[ Opcodes.i32_load16_u, Math.log2(ValtypeSize.i16) - 1, ValtypeSize.i32 ],
|
238
|
+
Opcodes.i32_from_u,
|
239
|
+
[ Opcodes.call, importedFuncs.printChar ],
|
240
|
+
|
241
|
+
// increment pointer by sizeof i16
|
242
|
+
[ Opcodes.local_get, 2 ],
|
243
|
+
...number(ValtypeSize.i16, Valtype.i32),
|
244
|
+
[ Opcodes.i32_add ],
|
245
|
+
[ Opcodes.local_tee, 2 ],
|
246
|
+
|
247
|
+
// if pointer != end pointer, loop
|
248
|
+
[ Opcodes.local_get, 3 ],
|
249
|
+
[ Opcodes.i32_ne ],
|
250
|
+
[ Opcodes.br_if, 0 ],
|
251
|
+
|
252
|
+
[ Opcodes.end ]
|
253
|
+
],
|
254
|
+
[TYPES._bytestring]: [
|
255
|
+
// simply print a (byte)string :))
|
256
|
+
// cache input pointer as i32
|
257
|
+
[ Opcodes.local_get, 0 ],
|
258
|
+
Opcodes.i32_to_u,
|
259
|
+
[ Opcodes.local_tee, 2 ],
|
260
|
+
|
261
|
+
// make end pointer
|
262
|
+
[ Opcodes.i32_load, Math.log2(ValtypeSize.i32) - 1, 0 ],
|
263
|
+
[ Opcodes.local_get, 2 ],
|
264
|
+
[ Opcodes.i32_add ],
|
265
|
+
[ Opcodes.local_set, 3 ],
|
266
|
+
|
267
|
+
[ Opcodes.loop, Blocktype.void ],
|
268
|
+
|
269
|
+
// print current char
|
270
|
+
[ Opcodes.local_get, 2 ],
|
271
|
+
[ Opcodes.i32_load8_u, Math.log2(ValtypeSize.i16) - 1, ValtypeSize.i32 ],
|
272
|
+
Opcodes.i32_from_u,
|
273
|
+
[ Opcodes.call, importedFuncs.printChar ],
|
274
|
+
|
275
|
+
// increment pointer
|
276
|
+
[ Opcodes.local_get, 2 ],
|
277
|
+
[ Opcodes.i32_const, 1 ],
|
278
|
+
[ Opcodes.i32_add ],
|
279
|
+
[ Opcodes.local_tee, 2 ],
|
280
|
+
|
281
|
+
// if pointer != end pointer, loop
|
282
|
+
[ Opcodes.local_get, 3 ],
|
283
|
+
[ Opcodes.i32_ne ],
|
284
|
+
[ Opcodes.br_if, 0 ],
|
285
|
+
|
286
|
+
[ Opcodes.end ]
|
287
|
+
],
|
288
|
+
[TYPES._array]: [
|
289
|
+
...printStaticStr('[ '),
|
290
|
+
|
291
|
+
// cache input pointer as i32
|
292
|
+
[ Opcodes.local_get, 0 ],
|
293
|
+
Opcodes.i32_to_u,
|
294
|
+
[ Opcodes.local_tee, 2 ],
|
295
|
+
|
296
|
+
// make end pointer
|
297
|
+
[ Opcodes.i32_load, Math.log2(ValtypeSize.i32) - 1, 0 ],
|
298
|
+
...number(ValtypeSize[valtype], Valtype.i32),
|
299
|
+
[ Opcodes.i32_mul ],
|
300
|
+
|
301
|
+
[ Opcodes.local_get, 2 ],
|
302
|
+
[ Opcodes.i32_add ],
|
303
|
+
[ Opcodes.local_set, 3 ],
|
304
|
+
|
305
|
+
[ Opcodes.loop, Blocktype.void ],
|
306
|
+
|
307
|
+
// print current char
|
308
|
+
[ Opcodes.local_get, 2 ],
|
309
|
+
[ Opcodes.load, Math.log2(ValtypeSize.i16) - 1, ValtypeSize.i32 ],
|
310
|
+
[ Opcodes.call, importedFuncs.print ],
|
311
|
+
|
312
|
+
// increment pointer by sizeof valtype
|
313
|
+
[ Opcodes.local_get, 2 ],
|
314
|
+
...number(ValtypeSize[valtype], Valtype.i32),
|
315
|
+
[ Opcodes.i32_add ],
|
316
|
+
[ Opcodes.local_tee, 2 ],
|
317
|
+
|
318
|
+
// if pointer != end pointer, print separator and loop
|
319
|
+
[ Opcodes.local_get, 3 ],
|
320
|
+
[ Opcodes.i32_ne ],
|
321
|
+
[ Opcodes.if, Blocktype.void ],
|
322
|
+
...printStaticStr(', '),
|
323
|
+
[ Opcodes.br, 1 ],
|
324
|
+
[ Opcodes.end ],
|
325
|
+
|
326
|
+
[ Opcodes.end ],
|
327
|
+
|
328
|
+
...printStaticStr(' ]'),
|
329
|
+
],
|
330
|
+
[TYPES.undefined]: [
|
331
|
+
...printStaticStr('undefined')
|
332
|
+
],
|
333
|
+
[TYPES.function]: [
|
334
|
+
...printStaticStr('function () {}')
|
335
|
+
],
|
336
|
+
[TYPES.object]: [
|
337
|
+
[ Opcodes.local_get, 0 ],
|
338
|
+
Opcodes.i32_to_u,
|
339
|
+
[ Opcodes.if, Blocktype.void ],
|
340
|
+
...printStaticStr('{}'),
|
341
|
+
[ Opcodes.else ],
|
342
|
+
...printStaticStr('null'),
|
343
|
+
[ Opcodes.end ]
|
344
|
+
],
|
345
|
+
default: [
|
346
|
+
[ Opcodes.local_get, 0 ],
|
347
|
+
[ Opcodes.call, importedFuncs.print ],
|
348
|
+
]
|
349
|
+
}, Blocktype.void),
|
350
|
+
|
196
351
|
...char('\n'),
|
197
352
|
[ Opcodes.call, importedFuncs.printChar ]
|
198
353
|
]
|
@@ -395,61 +550,314 @@ export const BuiltinFuncs = function() {
|
|
395
550
|
|
396
551
|
// this is an implementation of xorshift128+ (in wasm bytecode)
|
397
552
|
// fun fact: v8, SM, JSC also use this (you will need this fun fact to maintain your sanity reading this code)
|
398
|
-
const prngSeed0 = Math.floor(Math.random() * Number.MAX_SAFE_INTEGER), prngSeed1 = Math.floor(Math.random() * Number.MAX_SAFE_INTEGER);
|
553
|
+
// const prngSeed0 = Math.floor(Math.random() * Number.MAX_SAFE_INTEGER), prngSeed1 = Math.floor(Math.random() * Number.MAX_SAFE_INTEGER);
|
554
|
+
const prngSeed0 = (Math.random() * (2 ** 30)) | 0, prngSeed1 = (Math.random() * (2 ** 30)) | 0;
|
555
|
+
|
556
|
+
const prng = ({
|
557
|
+
'lcg32_glibc': {
|
558
|
+
globals: [ Valtype.i32 ],
|
559
|
+
locals: [],
|
560
|
+
returns: Valtype.i32,
|
561
|
+
wasm: [
|
562
|
+
// seed = (MULTIPLIER * seed + INCREMENT) % MODULUS
|
563
|
+
// MULTIPLIER * state0
|
564
|
+
[ Opcodes.global_get, 0 ],
|
565
|
+
...number(1103515245, Valtype.i32),
|
566
|
+
[ Opcodes.i32_mul ],
|
567
|
+
|
568
|
+
// + INCREMENT
|
569
|
+
...number(12345, Valtype.i32),
|
570
|
+
[ Opcodes.i32_add ],
|
571
|
+
|
572
|
+
// % MODULUS
|
573
|
+
...number(2 ** 31, Valtype.i32),
|
574
|
+
[ Opcodes.i32_rem_s ],
|
575
|
+
|
576
|
+
// state0 =
|
577
|
+
[ Opcodes.global_set, 0 ],
|
578
|
+
|
579
|
+
// state0
|
580
|
+
[ Opcodes.global_get, 0 ],
|
581
|
+
],
|
582
|
+
},
|
583
|
+
'lcg32_minstd': {
|
584
|
+
globals: [ Valtype.i32 ],
|
585
|
+
locals: [],
|
586
|
+
returns: Valtype.i32,
|
587
|
+
wasm: [
|
588
|
+
// seed = (MULTIPLIER * seed + INCREMENT) % MODULUS
|
589
|
+
// MULTIPLIER * state0
|
590
|
+
[ Opcodes.global_get, 0 ],
|
591
|
+
...number(48271, Valtype.i32),
|
592
|
+
[ Opcodes.i32_mul ],
|
593
|
+
|
594
|
+
// % MODULUS
|
595
|
+
...number((2 ** 31) - 1, Valtype.i32),
|
596
|
+
[ Opcodes.i32_rem_s ],
|
597
|
+
|
598
|
+
// state0 =
|
599
|
+
[ Opcodes.global_set, 0 ],
|
600
|
+
|
601
|
+
// state0
|
602
|
+
[ Opcodes.global_get, 0 ],
|
603
|
+
],
|
604
|
+
},
|
605
|
+
'lcg64_musl': 0, // todo
|
606
|
+
|
607
|
+
'xorshift32+': {
|
608
|
+
globals: [ Valtype.i32 ],
|
609
|
+
locals: [ Valtype.i32 ],
|
610
|
+
returns: Valtype.i32,
|
611
|
+
wasm: [
|
612
|
+
// setup: s1 = state0
|
613
|
+
[ Opcodes.global_get, 0 ], // state0
|
614
|
+
[ Opcodes.local_tee, 0 ], // s1
|
615
|
+
|
616
|
+
// s1 ^= s1 << 13
|
617
|
+
[ Opcodes.local_get, 0 ], // s1
|
618
|
+
[ Opcodes.i32_const, 13 ],
|
619
|
+
[ Opcodes.i32_shl ], // <<
|
620
|
+
[ Opcodes.i32_xor ], // ^
|
621
|
+
[ Opcodes.local_tee, 0 ], // s1
|
622
|
+
|
623
|
+
// s1 ^= s1 >> 17
|
624
|
+
[ Opcodes.local_get, 0 ], // s1
|
625
|
+
[ Opcodes.i32_const, 17 ],
|
626
|
+
[ Opcodes.i32_shr_s ], // >>
|
627
|
+
[ Opcodes.i32_xor ], // ^
|
628
|
+
[ Opcodes.local_tee, 0 ], // s1
|
629
|
+
|
630
|
+
// s1 ^= s1 << 5
|
631
|
+
[ Opcodes.local_get, 0 ], // s1
|
632
|
+
[ Opcodes.i32_const, 5 ],
|
633
|
+
[ Opcodes.i32_shl ], // <<
|
634
|
+
[ Opcodes.i32_xor ], // ^
|
635
|
+
[ Opcodes.local_tee, 0 ], // s1
|
636
|
+
|
637
|
+
// state0 = s1
|
638
|
+
[ Opcodes.global_set, 0 ],
|
639
|
+
|
640
|
+
// s1
|
641
|
+
[ Opcodes.local_get, 0 ],
|
642
|
+
],
|
643
|
+
},
|
644
|
+
|
645
|
+
'xorshift64+': {
|
646
|
+
globals: [ Valtype.i64 ],
|
647
|
+
locals: [ Valtype.i64 ],
|
648
|
+
returns: Valtype.i64,
|
649
|
+
wasm: [
|
650
|
+
// setup: s1 = state0
|
651
|
+
[ Opcodes.global_get, 0 ], // state0
|
652
|
+
[ Opcodes.local_tee, 0 ], // s1
|
653
|
+
|
654
|
+
// s1 ^= s1 >> 12
|
655
|
+
[ Opcodes.local_get, 0 ], // s1
|
656
|
+
[ Opcodes.i64_const, 12 ],
|
657
|
+
[ Opcodes.i64_shr_s ], // >>
|
658
|
+
[ Opcodes.i64_xor ], // ^
|
659
|
+
[ Opcodes.local_tee, 0 ], // s1
|
660
|
+
|
661
|
+
// s1 ^= s1 << 25
|
662
|
+
[ Opcodes.local_get, 0 ], // s1
|
663
|
+
[ Opcodes.i64_const, 25 ],
|
664
|
+
[ Opcodes.i64_shl ], // <<
|
665
|
+
[ Opcodes.i64_xor ], // ^
|
666
|
+
[ Opcodes.local_tee, 0 ], // s1
|
667
|
+
|
668
|
+
// s1 ^= s1 >> 27
|
669
|
+
[ Opcodes.local_get, 0 ], // s1
|
670
|
+
[ Opcodes.i64_const, 27 ],
|
671
|
+
[ Opcodes.i64_shr_s ], // >>
|
672
|
+
[ Opcodes.i64_xor ], // ^
|
673
|
+
[ Opcodes.local_tee, 0 ], // s1
|
674
|
+
|
675
|
+
// state0 = s1
|
676
|
+
[ Opcodes.global_set, 0 ],
|
677
|
+
|
678
|
+
// // s1 * 0x2545F4914F6CDD1D
|
679
|
+
// [ Opcodes.local_get, 0 ],
|
680
|
+
// [ Opcodes.i64_const, 0x9d, 0xba, 0xb3, 0xfb, 0x94, 0x92, 0xfd, 0xa2, 0x25 ],
|
681
|
+
// [ Opcodes.i64_mul ]
|
682
|
+
|
683
|
+
// s1
|
684
|
+
[ Opcodes.local_get, 0 ],
|
685
|
+
],
|
686
|
+
},
|
687
|
+
|
688
|
+
'xorshift128+': {
|
689
|
+
globals: [ Valtype.i64, Valtype.i64 ],
|
690
|
+
locals: [ Valtype.i64, Valtype.i64 ],
|
691
|
+
returns: Valtype.i64,
|
692
|
+
wasm: [
|
693
|
+
// setup: s1 = state0, s0 = state1, state0 = s0
|
694
|
+
[ Opcodes.global_get, 0 ], // state0
|
695
|
+
[ Opcodes.local_tee, 0 ], // s1
|
696
|
+
[ Opcodes.global_get, 1 ], // state1
|
697
|
+
[ Opcodes.local_tee, 1, ], // s0
|
698
|
+
[ Opcodes.global_set, 0 ], // state0
|
699
|
+
|
700
|
+
// s1 ^= s1 << 23
|
701
|
+
// [ Opcodes.local_get, 0 ], // s1
|
702
|
+
[ Opcodes.local_get, 0 ], // s1
|
703
|
+
[ Opcodes.i64_const, 23 ],
|
704
|
+
[ Opcodes.i64_shl ], // <<
|
705
|
+
[ Opcodes.i64_xor ], // ^
|
706
|
+
[ Opcodes.local_set, 0 ], // s1
|
707
|
+
|
708
|
+
// state1 = s1 ^ s0 ^ (s1 >> 17) ^ (s0 >> 26)
|
709
|
+
// s1 ^ s0
|
710
|
+
[ Opcodes.local_get, 0 ], // s1
|
711
|
+
[ Opcodes.local_get, 1 ], // s0
|
712
|
+
[ Opcodes.i64_xor ], // ^
|
713
|
+
|
714
|
+
// ^ (s1 >> 17)
|
715
|
+
[ Opcodes.local_get, 0 ], // s1
|
716
|
+
[ Opcodes.i64_const, 17 ],
|
717
|
+
[ Opcodes.i64_shr_u ], // >>
|
718
|
+
[ Opcodes.i64_xor ], // ^
|
719
|
+
|
720
|
+
// ^ (s0 >> 26)
|
721
|
+
[ Opcodes.local_get, 1 ], // s0
|
722
|
+
[ Opcodes.i64_const, 26 ],
|
723
|
+
[ Opcodes.i64_shr_u ], // >>
|
724
|
+
[ Opcodes.i64_xor ], // ^
|
725
|
+
|
726
|
+
// state1 =
|
727
|
+
[ Opcodes.global_set, 1 ],
|
728
|
+
|
729
|
+
// state1 + s0
|
730
|
+
[ Opcodes.global_get, 1 ], // state1
|
731
|
+
[ Opcodes.local_get, 1 ], // s0
|
732
|
+
[ Opcodes.i64_add ]
|
733
|
+
]
|
734
|
+
},
|
735
|
+
|
736
|
+
'xoroshiro128+': {
|
737
|
+
globals: [ Valtype.i64, Valtype.i64 ],
|
738
|
+
locals: [ Valtype.i64, Valtype.i64, Valtype.i64 ],
|
739
|
+
returns: Valtype.i64,
|
740
|
+
wasm: [
|
741
|
+
// setup: s1 = state1, s0 = state0
|
742
|
+
[ Opcodes.global_get, 1 ], // state0
|
743
|
+
[ Opcodes.local_tee, 0 ], // s1
|
744
|
+
[ Opcodes.global_get, 0 ], // state1
|
745
|
+
[ Opcodes.local_tee, 1, ], // s0
|
746
|
+
|
747
|
+
// result = s0 + s1
|
748
|
+
[ Opcodes.i64_add ],
|
749
|
+
[ Opcodes.local_set, 2 ], // result
|
750
|
+
|
751
|
+
// s1 ^= s0
|
752
|
+
[ Opcodes.local_get, 0 ], // s1
|
753
|
+
[ Opcodes.local_get, 1 ], // s0
|
754
|
+
[ Opcodes.i64_xor ],
|
755
|
+
[ Opcodes.local_set, 0 ], // s1
|
756
|
+
|
757
|
+
// state0 = rotl(s0, 24) ^ s1 ^ (s1 << 16)
|
758
|
+
|
759
|
+
// rotl(s0, 24) ^ s1
|
760
|
+
[ Opcodes.local_get, 1 ], // s0
|
761
|
+
...number(24, Valtype.i64),
|
762
|
+
[ Opcodes.i64_rotl ],
|
763
|
+
[ Opcodes.local_get, 0 ], // s1
|
764
|
+
[ Opcodes.i64_xor ],
|
765
|
+
|
766
|
+
// ^ (s1 << 16)
|
767
|
+
[ Opcodes.local_get, 0 ], // s1
|
768
|
+
...number(16, Valtype.i64),
|
769
|
+
[ Opcodes.i64_shl ],
|
770
|
+
[ Opcodes.i64_xor ],
|
771
|
+
|
772
|
+
// state0 =
|
773
|
+
[ Opcodes.global_set, 0 ], // state0
|
774
|
+
|
775
|
+
// state1 = rotl(s1, 37)
|
776
|
+
[ Opcodes.local_get, 0 ], // s1
|
777
|
+
...number(37, Valtype.i64),
|
778
|
+
[ Opcodes.i64_rotl ],
|
779
|
+
[ Opcodes.global_set, 1 ], // state1
|
780
|
+
|
781
|
+
// result
|
782
|
+
[ Opcodes.local_get, 2 ],
|
783
|
+
]
|
784
|
+
},
|
785
|
+
|
786
|
+
'xoshiro128+': {
|
787
|
+
globals: [ Valtype.i32, Valtype.i32, Valtype.i32, Valtype.i32 ],
|
788
|
+
locals: [ Valtype.i32, Valtype.i32 ],
|
789
|
+
returns: Valtype.i32,
|
790
|
+
wasm: [
|
791
|
+
// result = state0 + state3
|
792
|
+
[ Opcodes.global_get, 0 ], // state0
|
793
|
+
[ Opcodes.global_get, 3 ], // state0
|
794
|
+
[ Opcodes.i32_add ],
|
795
|
+
[ Opcodes.local_set, 0 ], // result
|
796
|
+
|
797
|
+
// t = state1 << 9
|
798
|
+
[ Opcodes.global_get, 1 ], // state1
|
799
|
+
...number(9, Valtype.i32),
|
800
|
+
[ Opcodes.i32_shl ],
|
801
|
+
[ Opcodes.local_set, 1 ], // t
|
802
|
+
|
803
|
+
// state2 ^= state0
|
804
|
+
[ Opcodes.global_get, 2 ], // state2
|
805
|
+
[ Opcodes.global_get, 0 ], // state0
|
806
|
+
[ Opcodes.i32_xor ],
|
807
|
+
[ Opcodes.global_set, 2 ], // state2
|
808
|
+
|
809
|
+
// state3 ^= state1
|
810
|
+
[ Opcodes.global_get, 3 ], // state3
|
811
|
+
[ Opcodes.global_get, 1 ], // state1
|
812
|
+
[ Opcodes.i32_xor ],
|
813
|
+
[ Opcodes.global_set, 3 ], // state3
|
814
|
+
|
815
|
+
// state1 ^= state2
|
816
|
+
[ Opcodes.global_get, 1 ], // state1
|
817
|
+
[ Opcodes.global_get, 2 ], // state2
|
818
|
+
[ Opcodes.i32_xor ],
|
819
|
+
[ Opcodes.global_set, 1 ], // state1
|
820
|
+
|
821
|
+
// state0 ^= state3
|
822
|
+
[ Opcodes.global_get, 0 ], // state2
|
823
|
+
[ Opcodes.global_get, 3 ], // state0
|
824
|
+
[ Opcodes.i32_xor ],
|
825
|
+
[ Opcodes.global_set, 0 ], // state2
|
826
|
+
|
827
|
+
// state2 ^= t
|
828
|
+
[ Opcodes.global_get, 2 ], // state2
|
829
|
+
[ Opcodes.local_get, 1 ], // t
|
830
|
+
[ Opcodes.i32_xor ],
|
831
|
+
[ Opcodes.global_set, 2 ], // state2
|
832
|
+
|
833
|
+
// state3 = rotl(state3, 11)
|
834
|
+
[ Opcodes.global_get, 3 ], // state3
|
835
|
+
...number(11, Valtype.i32),
|
836
|
+
[ Opcodes.i32_rotl ],
|
837
|
+
[ Opcodes.global_set, 3 ], // state3
|
838
|
+
|
839
|
+
// result
|
840
|
+
[ Opcodes.local_get, 0 ],
|
841
|
+
]
|
842
|
+
}
|
843
|
+
})[Prefs.prng ?? 'xorshift128+'];
|
844
|
+
|
845
|
+
if (!prng) throw new Error(`unknown prng algo: ${Prefs.prng}`);
|
399
846
|
|
400
847
|
this.__Math_random = {
|
401
848
|
floatOnly: true,
|
402
849
|
params: [],
|
403
|
-
locals:
|
850
|
+
locals: prng.locals,
|
404
851
|
localNames: [ 's1', 's0' ],
|
405
|
-
globals:
|
852
|
+
globals: prng.globals,
|
406
853
|
globalNames: [ 'state0', 'state1' ],
|
407
854
|
globalInits: [ prngSeed0, prngSeed1 ],
|
408
855
|
returns: [ Valtype.f64 ],
|
409
856
|
wasm: [
|
410
|
-
|
411
|
-
[ Opcodes.global_get, 0 ], // state0
|
412
|
-
[ Opcodes.local_tee, 0 ], // s1
|
413
|
-
[ Opcodes.global_get, 1 ], // state1
|
414
|
-
[ Opcodes.local_tee, 1, ], // s0
|
415
|
-
[ Opcodes.global_set, 0 ], // state0
|
416
|
-
|
417
|
-
// s1 ^= s1 << 23
|
418
|
-
// [ Opcodes.local_get, 0 ], // s1
|
419
|
-
[ Opcodes.local_get, 0 ], // s1
|
420
|
-
[ Opcodes.i64_const, 23 ],
|
421
|
-
[ Opcodes.i64_shl ], // <<
|
422
|
-
[ Opcodes.i64_xor ], // ^
|
423
|
-
[ Opcodes.local_set, 0 ], // s1
|
424
|
-
|
425
|
-
// state1 = s1 ^ s0 ^ (s1 >> 17) ^ (s0 >> 26)
|
426
|
-
// s1 ^ s0
|
427
|
-
[ Opcodes.local_get, 0 ], // s1
|
428
|
-
[ Opcodes.local_get, 1 ], // s0
|
429
|
-
[ Opcodes.i64_xor ], // ^
|
430
|
-
|
431
|
-
// ^ (s1 >> 17)
|
432
|
-
[ Opcodes.local_get, 0 ], // s1
|
433
|
-
[ Opcodes.i64_const, 17 ],
|
434
|
-
[ Opcodes.i64_shr_u ], // >>
|
435
|
-
[ Opcodes.i64_xor ], // ^
|
436
|
-
|
437
|
-
// ^ (s0 >> 26)
|
438
|
-
[ Opcodes.local_get, 1 ], // s0
|
439
|
-
[ Opcodes.i64_const, 26 ],
|
440
|
-
[ Opcodes.i64_shr_u ], // >>
|
441
|
-
[ Opcodes.i64_xor ], // ^
|
442
|
-
|
443
|
-
// state1 =
|
444
|
-
[ Opcodes.global_set, 1 ],
|
857
|
+
...prng.wasm,
|
445
858
|
|
446
859
|
// you thought it was over? now we need the result as a f64 between 0-1 :)
|
447
860
|
|
448
|
-
// state1 + s0
|
449
|
-
[ Opcodes.global_get, 1 ], // state1
|
450
|
-
[ Opcodes.local_get, 1 ], // s0
|
451
|
-
[ Opcodes.i64_add ],
|
452
|
-
|
453
861
|
// should we >> 12 here?
|
454
862
|
// it feels like it but it breaks values
|
455
863
|
|
@@ -464,15 +872,72 @@ export const BuiltinFuncs = function() {
|
|
464
872
|
// ...number(1),
|
465
873
|
// [ Opcodes.f64_sub ],
|
466
874
|
|
467
|
-
...
|
468
|
-
|
875
|
+
...(prng.returns === Valtype.i64 ? [
|
876
|
+
...number((1 << 53) - 1, Valtype.i64),
|
877
|
+
[ Opcodes.i64_and ],
|
878
|
+
|
879
|
+
// double(mantissa)
|
880
|
+
[ Opcodes.f64_convert_i64_u ],
|
881
|
+
|
882
|
+
// / (1 << 53)
|
883
|
+
...number(1 << 53),
|
884
|
+
[ Opcodes.f64_div ]
|
885
|
+
] : [
|
886
|
+
...number((1 << 21) - 1, Valtype.i32),
|
887
|
+
[ Opcodes.i32_and ],
|
888
|
+
|
889
|
+
// double(mantissa)
|
890
|
+
[ Opcodes.f64_convert_i32_u ],
|
891
|
+
|
892
|
+
// / (1 << 21)
|
893
|
+
...number(1 << 21),
|
894
|
+
[ Opcodes.f64_div ]
|
895
|
+
])
|
896
|
+
]
|
897
|
+
};
|
898
|
+
|
899
|
+
this.__Porffor_i32_random = {
|
900
|
+
params: [],
|
901
|
+
locals: prng.locals,
|
902
|
+
localNames: [ 's1', 's0' ],
|
903
|
+
globals: prng.globals,
|
904
|
+
globalNames: [ 'state0', 'state1' ],
|
905
|
+
globalInits: [ prngSeed0, prngSeed1 ],
|
906
|
+
returns: [ Valtype.i32 ],
|
907
|
+
wasm: [
|
908
|
+
...prng.wasm,
|
909
|
+
|
910
|
+
...(prng.returns === Valtype.i64 ? [
|
911
|
+
// the lowest bits of the output generated by xorshift128+ have low quality
|
912
|
+
...number(56, Valtype.i64),
|
913
|
+
[ Opcodes.i64_shr_u ],
|
914
|
+
|
915
|
+
[ Opcodes.i32_wrap_i64 ],
|
916
|
+
] : []),
|
917
|
+
]
|
918
|
+
};
|
919
|
+
|
920
|
+
this.__Porffor_i32_randomByte = {
|
921
|
+
params: [],
|
922
|
+
locals: prng.locals,
|
923
|
+
localNames: [ 's1', 's0' ],
|
924
|
+
globals: prng.globals,
|
925
|
+
globalNames: [ 'state0', 'state1' ],
|
926
|
+
globalInits: [ prngSeed0, prngSeed1 ],
|
927
|
+
returns: [ Valtype.i32 ],
|
928
|
+
wasm: [
|
929
|
+
...prng.wasm,
|
930
|
+
|
931
|
+
...(prng.returns === Valtype.i64 ? [
|
932
|
+
// the lowest bits of the output generated by xorshift128+ have low quality
|
933
|
+
...number(56, Valtype.i64),
|
934
|
+
[ Opcodes.i64_shr_u ],
|
469
935
|
|
470
|
-
|
471
|
-
|
936
|
+
[ Opcodes.i32_wrap_i64 ],
|
937
|
+
] : []),
|
472
938
|
|
473
|
-
|
474
|
-
|
475
|
-
[ Opcodes.f64_div ]
|
939
|
+
...number(0xff, Valtype.i32),
|
940
|
+
[ Opcodes.i32_and ],
|
476
941
|
]
|
477
942
|
};
|
478
943
|
|
@@ -571,206 +1036,76 @@ export const BuiltinFuncs = function() {
|
|
571
1036
|
};
|
572
1037
|
|
573
1038
|
|
574
|
-
this.
|
575
|
-
params: [ Valtype.i32 ],
|
576
|
-
|
577
|
-
|
578
|
-
|
579
|
-
|
580
|
-
|
581
|
-
|
582
|
-
|
583
|
-
|
584
|
-
|
585
|
-
|
586
|
-
|
587
|
-
|
588
|
-
wasm: [
|
589
|
-
[ Opcodes.local_get, 0 ],
|
590
|
-
[ ...Opcodes.i32x4_splat ],
|
591
|
-
]
|
592
|
-
};
|
593
|
-
|
594
|
-
this.__SIMD_i16x8_create = {
|
595
|
-
params: [ Valtype.i32, Valtype.i32, Valtype.i32, Valtype.i32, Valtype.i32, Valtype.i32, Valtype.i32, Valtype.i32 ],
|
596
|
-
locals: [],
|
597
|
-
returns: [ Valtype.v128 ],
|
598
|
-
wasm: [
|
599
|
-
...i32x4(0, 0, 0, 0),
|
600
|
-
[ Opcodes.local_get, 0 ],
|
601
|
-
[ ...Opcodes.i16x8_replace_lane, 0 ],
|
602
|
-
[ Opcodes.local_get, 1 ],
|
603
|
-
[ ...Opcodes.i16x8_replace_lane, 1 ],
|
604
|
-
[ Opcodes.local_get, 2 ],
|
605
|
-
[ ...Opcodes.i16x8_replace_lane, 2 ],
|
606
|
-
[ Opcodes.local_get, 3 ],
|
607
|
-
[ ...Opcodes.i16x8_replace_lane, 3 ],
|
608
|
-
[ Opcodes.local_get, 4 ],
|
609
|
-
[ ...Opcodes.i16x8_replace_lane, 4 ],
|
610
|
-
[ Opcodes.local_get, 5 ],
|
611
|
-
[ ...Opcodes.i16x8_replace_lane, 5 ],
|
612
|
-
[ Opcodes.local_get, 6 ],
|
613
|
-
[ ...Opcodes.i16x8_replace_lane, 6 ],
|
614
|
-
[ Opcodes.local_get, 7 ],
|
615
|
-
[ ...Opcodes.i16x8_replace_lane, 7 ],
|
616
|
-
]
|
617
|
-
};
|
618
|
-
|
619
|
-
this.__SIMD_i32x4_dot_i16x8 = {
|
620
|
-
params: [ Valtype.v128, Valtype.v128 ],
|
621
|
-
locals: [],
|
622
|
-
returns: [ Valtype.v128 ],
|
623
|
-
wasm: [
|
624
|
-
[ Opcodes.local_get, 0 ],
|
625
|
-
[ Opcodes.local_get, 1 ],
|
626
|
-
[ ...Opcodes.i32x4_dot_i16x8_s ]
|
627
|
-
]
|
1039
|
+
this.__Porffor_type = {
|
1040
|
+
params: [ valtypeBinary, Valtype.i32 ],
|
1041
|
+
typedParams: true,
|
1042
|
+
locals: [ Valtype.i32, Valtype.i32 ],
|
1043
|
+
returns: [ valtypeBinary ],
|
1044
|
+
returnType: Prefs.bytestring ? '_bytestring' : 'string',
|
1045
|
+
wasm: (scope, { TYPE_NAMES, typeSwitch, makeString }) => {
|
1046
|
+
const bc = {};
|
1047
|
+
for (const x in TYPE_NAMES) {
|
1048
|
+
bc[x] = makeString(scope, TYPE_NAMES[x], false, '#Porffor_type_result');
|
1049
|
+
}
|
1050
|
+
|
1051
|
+
return typeSwitch(scope, [ [ Opcodes.local_get, 1 ] ], bc);
|
1052
|
+
}
|
628
1053
|
};
|
629
1054
|
|
630
|
-
|
631
|
-
|
632
|
-
locals: [],
|
633
|
-
returns: [ Valtype.v128 ],
|
634
|
-
wasm: [
|
635
|
-
...i32x4(0, 0, 0, 0),
|
636
|
-
[ Opcodes.local_get, 0 ],
|
637
|
-
[ ...Opcodes.i32x4_replace_lane, 0 ],
|
638
|
-
[ Opcodes.local_get, 1 ],
|
639
|
-
[ ...Opcodes.i32x4_replace_lane, 1 ],
|
640
|
-
[ Opcodes.local_get, 2 ],
|
641
|
-
[ ...Opcodes.i32x4_replace_lane, 2 ],
|
642
|
-
[ Opcodes.local_get, 3 ],
|
643
|
-
[ ...Opcodes.i32x4_replace_lane, 3 ],
|
644
|
-
]
|
645
|
-
};
|
1055
|
+
const localIsOneOf = (getter, arr, valtype = valtypeBinary) => {
|
1056
|
+
const out = [];
|
646
1057
|
|
647
|
-
|
648
|
-
|
649
|
-
|
650
|
-
|
651
|
-
wasm: [
|
652
|
-
[ Opcodes.local_get, 0 ],
|
653
|
-
[ Opcodes.local_get, 1 ],
|
654
|
-
[ ...Opcodes.i32x4_add ]
|
655
|
-
]
|
656
|
-
};
|
1058
|
+
for (let i = 0; i < arr.length; i++) {
|
1059
|
+
out.push(...getter, ...number(arr[i], valtype), valtype === Valtype.f64 ? [ Opcodes.f64_eq ] : [ Opcodes.i32_eq ]);
|
1060
|
+
if (i !== 0) out.push([ Opcodes.i32_or ]);
|
1061
|
+
}
|
657
1062
|
|
658
|
-
|
659
|
-
params: [ Valtype.v128, Valtype.v128 ],
|
660
|
-
locals: [],
|
661
|
-
returns: [ Valtype.v128 ],
|
662
|
-
wasm: [
|
663
|
-
[ Opcodes.local_get, 0 ],
|
664
|
-
[ Opcodes.local_get, 1 ],
|
665
|
-
[ ...Opcodes.i32x4_sub ]
|
666
|
-
]
|
1063
|
+
return out;
|
667
1064
|
};
|
668
1065
|
|
669
|
-
this.
|
670
|
-
params: [
|
671
|
-
|
672
|
-
|
673
|
-
|
1066
|
+
this.__Porffor_ptr = {
|
1067
|
+
params: [ valtypeBinary, Valtype.i32 ],
|
1068
|
+
typedParams: true,
|
1069
|
+
locals: [ Valtype.i32, Valtype.i32 ],
|
1070
|
+
returns: [ valtypeBinary ],
|
1071
|
+
wasm: (scope, { TYPES }) => [
|
1072
|
+
...localIsOneOf([ [ Opcodes.local_get, 1 ] ], [ TYPES.string, TYPES._array, TYPES._bytestring ], Valtype.i32),
|
1073
|
+
[ Opcodes.if, valtypeBinary ],
|
674
1074
|
[ Opcodes.local_get, 0 ],
|
675
|
-
[ Opcodes.
|
676
|
-
|
1075
|
+
[ Opcodes.else ],
|
1076
|
+
...number(NaN),
|
1077
|
+
[ Opcodes.end ]
|
677
1078
|
]
|
678
1079
|
};
|
679
1080
|
|
680
|
-
this.
|
681
|
-
params: [ Valtype.
|
682
|
-
|
683
|
-
|
684
|
-
wasm: [
|
685
|
-
[ Opcodes.local_get, 0 ],
|
686
|
-
[ ...Opcodes.i32x4_extract_lane, 0 ],
|
687
|
-
],
|
688
|
-
},
|
689
|
-
|
690
|
-
this.__SIMD_i32x4_get1 = {
|
691
|
-
params: [ Valtype.v128 ],
|
692
|
-
locals: [],
|
693
|
-
returns: [ Valtype.i32 ],
|
694
|
-
wasm: [
|
695
|
-
[ Opcodes.local_get, 0 ],
|
696
|
-
[ ...Opcodes.i32x4_extract_lane, 1 ],
|
697
|
-
],
|
698
|
-
};
|
699
|
-
|
700
|
-
this.__SIMD_i32x4_get2 = {
|
701
|
-
params: [ Valtype.v128 ],
|
702
|
-
locals: [],
|
703
|
-
returns: [ Valtype.i32 ],
|
704
|
-
wasm: [
|
705
|
-
[ Opcodes.local_get, 0 ],
|
706
|
-
[ ...Opcodes.i32x4_extract_lane, 2 ],
|
707
|
-
],
|
708
|
-
};
|
709
|
-
|
710
|
-
this.__SIMD_i32x4_get3 = {
|
711
|
-
params: [ Valtype.v128 ],
|
712
|
-
locals: [],
|
1081
|
+
this.__Porffor_i32_ptr = {
|
1082
|
+
params: [ Valtype.i32, Valtype.i32 ],
|
1083
|
+
typedParams: true,
|
1084
|
+
locals: [ Valtype.i32, Valtype.i32 ],
|
713
1085
|
returns: [ Valtype.i32 ],
|
714
|
-
wasm: [
|
1086
|
+
wasm: (_scope, { TYPES }) => [
|
1087
|
+
...localIsOneOf([ [ Opcodes.local_get, 1 ] ], [ TYPES.string, TYPES._array, TYPES._bytestring ], Valtype.i32),
|
1088
|
+
[ Opcodes.if, Valtype.i32 ],
|
715
1089
|
[ Opcodes.local_get, 0 ],
|
716
|
-
[
|
717
|
-
|
718
|
-
|
719
|
-
|
720
|
-
this.__SIMD_i32x4_shuffle_000c = {
|
721
|
-
params: [ Valtype.v128 ],
|
722
|
-
locals: [],
|
723
|
-
returns: [ Valtype.v128 ],
|
724
|
-
wasm: [
|
725
|
-
[ Opcodes.local_get, 0 ],
|
726
|
-
...i32x4(0, 0, 0, 0),
|
727
|
-
[ ...Opcodes.i8x16_shuffle, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 8, 9, 10, 11 ], // i32x4 (a, b, c, d) -> i32x4 (0, 0, 0, c)
|
1090
|
+
[ Opcodes.else ],
|
1091
|
+
...number(-1, Valtype.i32),
|
1092
|
+
[ Opcodes.end ]
|
728
1093
|
]
|
729
1094
|
};
|
730
1095
|
|
731
|
-
|
732
|
-
|
1096
|
+
// unsafe: does not check type just ~casts to number
|
1097
|
+
this.__Porffor_i32_ptrUnsafe = {
|
1098
|
+
params: [ Valtype.i32 ],
|
733
1099
|
locals: [],
|
734
|
-
returns: [ Valtype.
|
1100
|
+
returns: [ Valtype.i32 ],
|
735
1101
|
wasm: [
|
736
1102
|
[ Opcodes.local_get, 0 ],
|
737
|
-
...i32x4(0, 0, 0, 0),
|
738
|
-
[ ...Opcodes.i8x16_shuffle, 16, 16, 16, 16, 16, 16, 16, 16, 0, 1, 2, 3, 4, 5, 6, 7 ], // i32x4 (a, b, c, d) -> i32x4 (0, 0, a, b)
|
739
1103
|
]
|
740
1104
|
};
|
741
|
-
};
|
742
1105
|
|
743
|
-
export const BuiltinPreludes = {
|
744
|
-
btoa: `var keyStr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
|
745
|
-
var btoa = function (input) {
|
746
|
-
// todo: throw invalid character for unicode
|
747
|
-
|
748
|
-
let output = "";
|
749
|
-
let chr1, chr2, chr3, enc1, enc2, enc3, enc4;
|
750
|
-
let i = 0;
|
751
|
-
|
752
|
-
while (i < input.length) {
|
753
|
-
chr1 = input.charCodeAt(i++);
|
754
|
-
chr2 = input.charCodeAt(i++);
|
755
|
-
chr3 = input.charCodeAt(i++);
|
756
|
-
|
757
|
-
enc1 = chr1 >> 2;
|
758
|
-
enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
|
759
|
-
enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
|
760
|
-
enc4 = chr3 & 63;
|
761
|
-
|
762
|
-
if (isNaN(chr2)) {
|
763
|
-
enc3 = enc4 = 64;
|
764
|
-
} else if (isNaN(chr3)) {
|
765
|
-
enc4 = 64;
|
766
|
-
}
|
767
1106
|
|
768
|
-
|
769
|
-
|
770
|
-
|
771
|
-
output += keyStr.charAt(enc4);
|
1107
|
+
const generated = new GeneratedBuiltins.BuiltinFuncs();
|
1108
|
+
for (const x in generated) {
|
1109
|
+
this[x] = generated[x];
|
772
1110
|
}
|
773
|
-
|
774
|
-
return output;
|
775
|
-
};`
|
776
1111
|
};
|