mirror of
				https://github.com/sqlite/sqlite.git
				synced 2025-10-24 09:53:10 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			2523 lines
		
	
	
		
			98 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			2523 lines
		
	
	
		
			98 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /**
 | |
|   2022-07-08
 | |
| 
 | |
|   The author disclaims copyright to this source code.  In place of a
 | |
|   legal notice, here is a blessing:
 | |
| 
 | |
|   *   May you do good and not evil.
 | |
|   *   May you find forgiveness for yourself and forgive others.
 | |
|   *   May you share freely, never taking more than you give.
 | |
| 
 | |
|   ***********************************************************************
 | |
| 
 | |
|   This code is developed in conjunction with the Jaccwabyt project:
 | |
| 
 | |
|   https://fossil.wanderinghorse.net/r/jaccwabyt
 | |
| 
 | |
|   More specifically:
 | |
| 
 | |
|   https://fossil.wanderinghorse.net/r/jaccwabyt/file/common/whwasmutil.js
 | |
| 
 | |
|   and SQLite:
 | |
| 
 | |
|   https://sqlite.org
 | |
| 
 | |
|   This file is kept in sync between both of those trees.
 | |
| */
 | |
| /**
 | |
|    The primary goal of this function is to replace, where possible,
 | |
|    Emscripten-generated glue code with equivalent utility code which
 | |
|    can be used in arbitrary WASM environments built with toolchains
 | |
|    other than Emscripten. To that end, it populates the given object
 | |
|    with various WASM-specific APIs. These APIs work with both 32- and
 | |
|    64-bit WASM builds.
 | |
| 
 | |
|    Forewarning: this API explicitly targets only browser environments.
 | |
|    If a given non-browser environment has the capabilities needed for
 | |
|    a given feature (e.g. TextEncoder), great, but it does not go out
 | |
|    of its way to account for them and does not provide compatibility
 | |
|    crutches for them.
 | |
| 
 | |
|    Intended usage:
 | |
| 
 | |
|    ```
 | |
|    const target = {}; // ... some object ...
 | |
|    globalThis.WhWasmUtilInstaller(target);
 | |
|    delete globalThis.WhWasmUtilInstaller;
 | |
|    ```
 | |
| 
 | |
|    The `target` object then holds the APIs. The caller may set certain
 | |
|    properties on it, before calling this, to configure it, as
 | |
|    documented below.
 | |
| 
 | |
|    The global-scope symbol for this function is intended only to
 | |
|    provide an easy way to make it available to 3rd-party scripts and
 | |
|    "should" be deleted after calling it. That symbol is _not_ used
 | |
|    within the library.
 | |
| 
 | |
|    It currently offers alternatives to the following
 | |
|    Emscripten-generated APIs:
 | |
| 
 | |
|    - OPTIONALLY memory allocation, but how this gets imported is
 | |
|      environment-specific.  Most of the following features only work
 | |
|      if allocation is available.
 | |
| 
 | |
|    - WASM-exported "indirect function table" access and
 | |
|      manipulation. e.g.  creating new WASM-side functions using JS
 | |
|      functions, analog to Emscripten's addFunction() and
 | |
|      uninstallFunction() but slightly different and with more useful
 | |
|      lifetime semantics.
 | |
| 
 | |
|    - Get/set specific heap memory values, analog to Emscripten's
 | |
|      getValue() and setValue().
 | |
| 
 | |
|    - String length counting in UTF-8 bytes (C-style and JS strings).
 | |
| 
 | |
|    - JS string to C-string conversion and vice versa, analog to
 | |
|      Emscripten's stringToUTF8Array() and friends, but with slighter
 | |
|      different interfaces.
 | |
| 
 | |
|    - JS string to Uint8Array conversion, noting that browsers actually
 | |
|      already have this built in via TextEncoder.
 | |
| 
 | |
|    - "Scoped" allocation, such that allocations made inside of a given
 | |
|      explicit scope will be automatically cleaned up when the scope is
 | |
|      closed. This is fundamentally similar to Emscripten's
 | |
|      stackAlloc() and friends but uses the heap instead of the stack
 | |
|      because access to the stack requires C code.
 | |
| 
 | |
|    - Create JS wrappers for WASM functions, analog to Emscripten's
 | |
|      ccall() and cwrap() functions, except that the automatic
 | |
|      conversions for function arguments and return values can be
 | |
|      easily customized by the client by assigning custom function
 | |
|      signature type names to conversion functions. Essentially,
 | |
|      it's ccall() and cwrap() on steroids.
 | |
| 
 | |
|    How to install...
 | |
| 
 | |
|    Passing an object to this function will install this library's
 | |
|    functionality into that object. It returns its argument.
 | |
| 
 | |
|    After installation, client code "should" delete this function's
 | |
|    global symbol (if any).
 | |
| 
 | |
|    This code requires that the target object have the following
 | |
|    properties, though they needn't be available until the first time
 | |
|    one of the installed APIs is used (as opposed to when this function
 | |
|    is called) except where explicitly noted:
 | |
| 
 | |
|    - `exports` must be a property of the target object OR a property
 | |
|      of `target.instance` (a WebAssembly.Module instance) and it must
 | |
|      contain the symbols exported by the WASM module associated with
 | |
|      this code. In an Enscripten environment it must be set to
 | |
|      `Module['asm']` (versions <=3.1.43) or `wasmExports` (versions
 | |
|      >=3.1.44). The exports object must contain a minimum of the
 | |
|      following symbols:
 | |
| 
 | |
|      - `memory`: a WebAssembly.Memory object representing the WASM
 | |
|        memory. _Alternately_, the `memory` property can be set as
 | |
|        `target.memory`, in particular if the WASM heap memory is
 | |
|        initialized in JS an _imported_ into WASM, as opposed to being
 | |
|        initialized in WASM and exported to JS.
 | |
| 
 | |
|      - `__indirect_function_table`: the WebAssembly.Table object which
 | |
|        holds WASM-exported functions. This API does not strictly
 | |
|        require that the table be able to grow but it will throw if its
 | |
|        `installFunction()` is called and the table cannot grow.
 | |
| 
 | |
|    - `functionTable`: WebAssembly.Table object holding the indirect
 | |
|      function call table.  If not set then
 | |
|      `exports.__indirect_function_table` is assumed. Achtung: this
 | |
|      property gets replaced by a function with the same name (because
 | |
|      this API used that name before this config option was added).
 | |
| 
 | |
|    In order to simplify downstream usage, if `target.exports` is not
 | |
|    set when this is called then a property access interceptor
 | |
|    (read-only, configurable, enumerable) gets installed as `exports`
 | |
|    which resolves to `target.instance.exports`, noting that the latter
 | |
|    property need not exist until the first time `target.exports` is
 | |
|    accessed.
 | |
| 
 | |
|    Some APIs _optionally_ make use of the `bigIntEnabled` property of
 | |
|    the target object. It "should" be set to true if the WASM
 | |
|    environment is compiled with BigInt support, else it must be
 | |
|    false. If it is false, certain BigInt-related features will trigger
 | |
|    an exception if invoked. This property, if not set when this is
 | |
|    called, will get a default value of true only if the BigInt64Array
 | |
|    constructor is available, else it will default to false.  Having
 | |
|    the BigInt type is not sufficient for full int64 integration with
 | |
|    WASM: the target WASM file must also have been built with that
 | |
|    support. In Emscripten that's done using the `-sWASM_BIGINT` flag.
 | |
| 
 | |
|    Some optional APIs require that the target have the following
 | |
|    methods:
 | |
| 
 | |
|    - 'alloc()` must behave like C's `malloc()`, allocating N bytes of
 | |
|      memory and returning its pointer. In Emscripten this is
 | |
|      conventionally made available via `Module['_malloc']`. This API
 | |
|      requires that the alloc routine throw on allocation error, as
 | |
|      opposed to returning null or 0.
 | |
| 
 | |
|    - 'dealloc()` must behave like C's `free()`, accepting either a
 | |
|      pointer returned from its allocation counterpart or the values
 | |
|      null/0 (for which it must be a no-op). In Emscripten this is
 | |
|      conventionally made available via `Module['_free']`.
 | |
| 
 | |
|    APIs which require allocation routines are explicitly documented as
 | |
|    such and/or have "alloc" in their names.
 | |
| 
 | |
|    Optional configuration values which may be set on target before
 | |
|    calling this:
 | |
| 
 | |
|    - `pointerIR`: an IR-format string for the WASM environment's
 | |
|       pointer size. If set it must be either 'i32' or 'i64'. If not
 | |
|       set, it gets set to whatever this code thinks the pointer size
 | |
|       is.  Modifying it after this call has no effect. A reliable
 | |
|       way to get this value is (typeof X()), where X is a function
 | |
|       from target.exports which returns an innocuous pointer.
 | |
| 
 | |
|    - `pointerSize`: if set, it must be one of 4 or 8 and must
 | |
|       correspond to the value of `pointerIR`. If not set, it gets set
 | |
|       to whatever this code thinks the pointer size is (4 unless
 | |
|       `pointerIR` is 'i64'). If `pointerSize` is set but `pointerIR`
 | |
|       is not, `pointerIR` gets set appropriately, and vice versa.
 | |
| 
 | |
|    When building with Emscripten's -sMEMORY64=1, `pointerIR` must be
 | |
|    set to 'i64' and/or `pointerSize` must be set to 8.
 | |
| 
 | |
|    After calling this, the pointerIR and pointerSize properties are
 | |
|    replaced with a read-only Object member named target.ptr. It
 | |
|    contains the following read-only helper methods and properties to
 | |
|    assist in using WASM pointers without having to know what type they
 | |
|    are:
 | |
| 
 | |
|    - `size` = pointerSize
 | |
| 
 | |
|    - `ir` = pointerIR
 | |
| 
 | |
|    - `null` = a "null" pointer of type Number or BigInt. Equivalent to
 | |
|      one of Number(0) or BigInt(0). This value compares === to
 | |
|      WASM NULL pointers.
 | |
| 
 | |
|    - `coerce(arg)` = equivalent to one of Number(arg) or BigInt(arg||0).
 | |
| 
 | |
|    - `add(...args)` = performs "pointer arithmetic" (`wasmPtr+offset`
 | |
|      does not work in 64-bit builds unless all operands are of type
 | |
|      BigInt). Adds up all of its arguments, accounting for whether
 | |
|      each is a Number of BigInt, and returns either a Number or
 | |
|      BigInt.
 | |
| 
 | |
|    - `addn(...args)` = like `add()` but always returns its result as a
 | |
|      Number. Equivalent to Number(add(...)).
 | |
| 
 | |
|    ------------------------------------------------------------
 | |
|    Design notes:
 | |
| 
 | |
|    - This function should probably take a config object and return the
 | |
|      newly-created (or config-provided) target. The current approach
 | |
|      seemed better at the time.
 | |
| */
 | |
| globalThis.WhWasmUtilInstaller = function(target){
 | |
|   'use strict';
 | |
|   if(undefined===target.bigIntEnabled){
 | |
|     target.bigIntEnabled = !!globalThis['BigInt64Array'];
 | |
|   }
 | |
| 
 | |
|   /** Throws a new Error, the message of which is the concatenation of
 | |
|       all args with a space between each. */
 | |
|   const toss = (...args)=>{throw new Error(args.join(' '))};
 | |
| 
 | |
|   /**
 | |
|      As of 2025-09-21, this library works with 64-bit WASM modules
 | |
|      built with Emscripten's -sMEMORY64=1.
 | |
|   */
 | |
|   if( target.pointerSize && !target.pointerIR ){
 | |
|     target.pointerIR = (4===target.pointerSize ? 'i32' : 'i64');
 | |
|   }
 | |
|   const __ptrIR = (target.pointerIR ??= 'i32');
 | |
|   const __ptrSize = (target.pointerSize ??=
 | |
|                      ('i32'===__ptrIR ? 4 : ('i64'===__ptrIR ? 8 : 0)));
 | |
|   delete target.pointerSize;
 | |
|   delete target.pointerIR;
 | |
| 
 | |
|   if( 'i32'!==__ptrIR && 'i64'!==__ptrIR ){
 | |
|     toss("Invalid pointerIR:",__ptrIR);
 | |
|   }else if( 8!==__ptrSize && 4!==__ptrSize ){
 | |
|     toss("Invalid pointerSize:",__ptrSize);
 | |
|   }
 | |
| 
 | |
|   /** Either BigInt or, if !target.bigIntEnabled, a function which
 | |
|       throws complaining that BigInt is not enabled. */
 | |
|   const __BigInt = target.bigIntEnabled
 | |
|         ? (v)=>BigInt(v || 0)
 | |
|         : (v)=>toss("BigInt support is disabled in this build.");
 | |
| 
 | |
|   const __Number = (v)=>Number(v||0)/*treat undefined the same as null*/;
 | |
| 
 | |
|   /**
 | |
|      If target.ptr.ir==='i32' then this is equivalent to
 | |
|      Number(v||0) else it's equivalent to BigInt(v||0), throwing
 | |
|      if BigInt support is disabled.
 | |
| 
 | |
|      Why? Because Number(null)===0, but BigInt(null) throws.  We
 | |
|      perform the same for Number to allow the undefined value to be
 | |
|      treated as a NULL WASM pointer, primarily to reduce friction in
 | |
|      many SQLite3 bindings which have long relied on that.
 | |
|   */
 | |
|   const __asPtrType = (4===__ptrSize) ? __Number : __BigInt;
 | |
| 
 | |
|   /**
 | |
|      The number 0 as either type Number or BigInt, depending on
 | |
|      target.ptr.ir.
 | |
|   */
 | |
|   const __NullPtr = __asPtrType(0);
 | |
| 
 | |
|   /**
 | |
|      Expects any number of numeric arguments, each one of either type
 | |
|      Number or BigInt. It sums them up (from an implicit starting
 | |
|      point of 0 or 0n) and returns them as a number of the same type
 | |
|      which target.ptr.coerce() uses.
 | |
| 
 | |
|      This is a workaround for not being able to mix Number/BigInt in
 | |
|      addition/subtraction expressions (which we frequently need for
 | |
|      calculating pointer offsets).
 | |
|   */
 | |
|   const __ptrAdd = function(...args){
 | |
|     let rc = __asPtrType(0);
 | |
|     for(const v of args) rc += __asPtrType(v);
 | |
|     return rc;
 | |
|   };
 | |
| 
 | |
|   /** Set up target.ptr... */
 | |
|   {
 | |
|     const __ptr = Object.create(null);
 | |
|     Object.defineProperty(target, 'ptr', {
 | |
|       enumerable: true,
 | |
|       get: ()=>__ptr,
 | |
|       set: ()=>toss("The ptr property is read-only.")
 | |
|     });
 | |
|     (function f(name, val){
 | |
|       Object.defineProperty(__ptr, name, {
 | |
|         enumerable: true,
 | |
|         get: ()=>val,
 | |
|         set: ()=>toss("ptr["+name+"] is read-only.")
 | |
|       });
 | |
|       return f;
 | |
|     })(
 | |
|       'null', __NullPtr
 | |
|     )(
 | |
|       'size', __ptrSize
 | |
|     )(
 | |
|       'ir', __ptrIR
 | |
|     )(
 | |
|       'coerce', __asPtrType
 | |
|     )(
 | |
|       'add', __ptrAdd
 | |
|     )(
 | |
|       'addn', (4===__ptrIR) ? __ptrAdd : (...args)=>Number(__ptrAdd(...args))
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   if(!target.exports){
 | |
|     Object.defineProperty(target, 'exports', {
 | |
|       enumerable: true, configurable: true,
 | |
|       get: ()=>(target.instance?.exports)
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   /** Stores various cached state. */
 | |
|   const cache = Object.create(null);
 | |
|   /** Previously-recorded size of cache.memory.buffer, noted so that
 | |
|       we can recreate the view objects if the heap grows. */
 | |
|   cache.heapSize = 0;
 | |
|   /** WebAssembly.Memory object extracted from target.memory or
 | |
|       target.exports.memory the first time heapWrappers() is
 | |
|       called. */
 | |
|   cache.memory = null;
 | |
|   /** uninstallFunction() puts table indexes in here for reuse and
 | |
|       installFunction() extracts them. */
 | |
|   cache.freeFuncIndexes = [];
 | |
|   /**
 | |
|      List-of-lists used by scopedAlloc() and friends.
 | |
|   */
 | |
|   cache.scopedAlloc = [];
 | |
| 
 | |
|   /** Push the pointer ptr to the current cache.scopedAlloc list
 | |
|       (which must already exist) and return ptr. */
 | |
|   cache.scopedAlloc.pushPtr = (ptr)=>{
 | |
|     cache.scopedAlloc[cache.scopedAlloc.length-1].push(ptr);
 | |
|     return ptr;
 | |
|   };
 | |
| 
 | |
|   cache.utf8Decoder = new TextDecoder();
 | |
|   cache.utf8Encoder = new TextEncoder('utf-8');
 | |
| 
 | |
|   /**
 | |
|      For the given IR-like string in the set ('i8', 'i16', 'i32',
 | |
|      'f32', 'float', 'i64', 'f64', 'double', '*'), or any string value
 | |
|      ending in '*', returns the sizeof for that value
 | |
|      (target.ptr.size in the latter case). For any other value, it
 | |
|      returns the undefined value.
 | |
|   */
 | |
|   target.sizeofIR = (n)=>{
 | |
|     switch(n){
 | |
|         case 'i8': return 1;
 | |
|         case 'i16': return 2;
 | |
|         case 'i32': case 'f32': case 'float': return 4;
 | |
|         case 'i64': case 'f64': case 'double': return 8;
 | |
|         case '*': return __ptrSize;
 | |
|         default:
 | |
|           return (''+n).endsWith('*') ? __ptrSize : undefined;
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      If (cache.heapSize !== cache.memory.buffer.byteLength), i.e. if
 | |
|      the heap has grown since the last call, updates cache.HEAPxyz.
 | |
|      Returns the cache object.
 | |
|   */
 | |
|   const heapWrappers = function(){
 | |
|     if(!cache.memory){
 | |
|       cache.memory = (target.memory instanceof WebAssembly.Memory)
 | |
|         ? target.memory : target.exports.memory;
 | |
|     }else if(cache.heapSize === cache.memory.buffer.byteLength){
 | |
|       return cache;
 | |
|     }
 | |
|     // heap is newly-acquired or has been resized....
 | |
|     const b = cache.memory.buffer;
 | |
|     cache.HEAP8 = new Int8Array(b); cache.HEAP8U = new Uint8Array(b);
 | |
|     cache.HEAP16 = new Int16Array(b); cache.HEAP16U = new Uint16Array(b);
 | |
|     cache.HEAP32 = new Int32Array(b); cache.HEAP32U = new Uint32Array(b);
 | |
|     cache.HEAP32F = new Float32Array(b); cache.HEAP64F = new Float64Array(b);
 | |
|     if(target.bigIntEnabled){
 | |
|       if( 'undefined'!==typeof BigInt64Array ){
 | |
|         cache.HEAP64 = new BigInt64Array(b); cache.HEAP64U = new BigUint64Array(b);
 | |
|       }else{
 | |
|         toss("BigInt support is enabled, but the BigInt64Array type is missing.");
 | |
|       }
 | |
|     }
 | |
|     cache.heapSize = b.byteLength;
 | |
|     return cache;
 | |
|   };
 | |
| 
 | |
|   /** Convenience equivalent of this.heapForSize(8,false). */
 | |
|   target.heap8 = ()=>heapWrappers().HEAP8;
 | |
| 
 | |
|   /** Convenience equivalent of this.heapForSize(8,true). */
 | |
|   target.heap8u = ()=>heapWrappers().HEAP8U;
 | |
| 
 | |
|   /** Convenience equivalent of this.heapForSize(16,false). */
 | |
|   target.heap16 = ()=>heapWrappers().HEAP16;
 | |
| 
 | |
|   /** Convenience equivalent of this.heapForSize(16,true). */
 | |
|   target.heap16u = ()=>heapWrappers().HEAP16U;
 | |
| 
 | |
|   /** Convenience equivalent of this.heapForSize(32,false). */
 | |
|   target.heap32 = ()=>heapWrappers().HEAP32;
 | |
| 
 | |
|   /** Convenience equivalent of this.heapForSize(32,true). */
 | |
|   target.heap32u = ()=>heapWrappers().HEAP32U;
 | |
| 
 | |
|   /**
 | |
|      Requires n to be one of:
 | |
| 
 | |
|      - integer 8, 16, or 32.
 | |
|      - A integer-type TypedArray constructor: Int8Array, Int16Array,
 | |
|      Int32Array, or their Uint counterparts.
 | |
| 
 | |
|      If this.bigIntEnabled is true, it also accepts the value 64 or a
 | |
|      BigInt64Array/BigUint64Array, else it throws if passed 64 or one
 | |
|      of those constructors.
 | |
| 
 | |
|      Returns an integer-based TypedArray view of the WASM heap memory
 | |
|      buffer associated with the given block size. If passed an integer
 | |
|      as the first argument and unsigned is truthy then the "U"
 | |
|      (unsigned) variant of that view is returned, else the signed
 | |
|      variant is returned. If passed a TypedArray value, the 2nd
 | |
|      argument is ignored. Float32Array and Float64Array views are not
 | |
|      supported by this function.
 | |
| 
 | |
|      Growth of the heap will invalidate any references to this heap,
 | |
|      so do not hold a reference longer than needed and do not use a
 | |
|      reference after any operation which may allocate. Instead,
 | |
|      re-fetch the reference by calling this function again.
 | |
| 
 | |
|      Throws if passed an invalid n.
 | |
|   */
 | |
|   target.heapForSize = function(n,unsigned = true){
 | |
|     let ctor;
 | |
|     const c = (cache.memory && cache.heapSize === cache.memory.buffer.byteLength)
 | |
|           ? cache : heapWrappers();
 | |
|     switch(n){
 | |
|         case Int8Array: return c.HEAP8; case Uint8Array: return c.HEAP8U;
 | |
|         case Int16Array: return c.HEAP16; case Uint16Array: return c.HEAP16U;
 | |
|         case Int32Array: return c.HEAP32; case Uint32Array: return c.HEAP32U;
 | |
|         case 8:  return unsigned ? c.HEAP8U : c.HEAP8;
 | |
|         case 16: return unsigned ? c.HEAP16U : c.HEAP16;
 | |
|         case 32: return unsigned ? c.HEAP32U : c.HEAP32;
 | |
|         case 64:
 | |
|           if(c.HEAP64) return unsigned ? c.HEAP64U : c.HEAP64;
 | |
|           break;
 | |
|         default:
 | |
|           if(target.bigIntEnabled){
 | |
|             if(n===globalThis['BigUint64Array']) return c.HEAP64U;
 | |
|             else if(n===globalThis['BigInt64Array']) return c.HEAP64;
 | |
|             break;
 | |
|           }
 | |
|     }
 | |
|     toss("Invalid heapForSize() size: expecting 8, 16, 32,",
 | |
|          "or (if BigInt is enabled) 64.");
 | |
|   };
 | |
| 
 | |
|   const __funcTable = target.functionTable;
 | |
|   delete target.functionTable;
 | |
| 
 | |
|   /**
 | |
|      Returns the WASM-exported "indirect function table".
 | |
|   */
 | |
|   target.functionTable = __funcTable
 | |
|     ? ()=>__funcTable
 | |
|     : ()=>target.exports.__indirect_function_table
 | |
|     /** -----------------^^^^^ "seems" to be a standardized export name.
 | |
|         From Emscripten release notes from 2020-09-10:
 | |
|         - Use `__indirect_function_table` as the import name for the
 | |
|         table, which is what LLVM does.
 | |
| 
 | |
|         We must delay access to target.exports until after the library
 | |
|         is bootstrapped.
 | |
|     */;
 | |
| 
 | |
|   /**
 | |
|      Given a function pointer, returns the WASM function table entry
 | |
|      if found, else returns a falsy value: undefined if fptr is out of
 | |
|      range or null if it's in range but the table entry is empty.
 | |
|   */
 | |
|   target.functionEntry = function(fptr){
 | |
|     const ft = target.functionTable();
 | |
|     //console.debug("functionEntry(",arguments,")", __asPtrType(fptr));
 | |
|     //-sMEMORY64=1: we get a BigInt fptr and ft.get() wants a BigInt.
 | |
|     //-sMEMORY64=2: we get a Number fptr and ft.get() wants a Number.
 | |
|     return fptr < ft.length ? ft.get(__asPtrType(fptr)) : undefined;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Creates a WASM function which wraps the given JS function and
 | |
|      returns the JS binding of that WASM function. The signature
 | |
|      string must be the Jaccwabyt-format or Emscripten
 | |
|      addFunction()-format function signature string. In short: in may
 | |
|      have one of the following formats:
 | |
| 
 | |
|      - Emscripten: `"x..."`, where the first x is a letter representing
 | |
|        the result type and subsequent letters represent the argument
 | |
|        types. Functions with no arguments have only a single
 | |
|        letter.
 | |
| 
 | |
|      - Jaccwabyt: `"x(...)"` where `x` is the letter representing the
 | |
|        result type and letters in the parens (if any) represent the
 | |
|        argument types. Functions with no arguments use `x()`.
 | |
| 
 | |
|      Supported letters:
 | |
| 
 | |
|      - `i` = int32
 | |
|      - `p` = int32 or int64 ("pointer"), depending on target.ptr.size
 | |
|      - `j` = int64
 | |
|      - `f` = float32
 | |
|      - `d` = float64
 | |
|      - `v` = void, only legal for use as the result type
 | |
| 
 | |
|      It throws if an invalid signature letter is used.
 | |
| 
 | |
|      Jaccwabyt-format signatures support some additional letters which
 | |
|      have no special meaning here but (in this context) act as aliases
 | |
|      for other letters:
 | |
| 
 | |
|      - `s`, `P`: same as `p`
 | |
| 
 | |
|      Sidebar: this code is developed together with Jaccwabyt, thus the
 | |
|      support for its signature format.
 | |
| 
 | |
|      The arguments may be supplied in either order: (func,sig) or
 | |
|      (sig,func).
 | |
|   */
 | |
|   target.jsFuncToWasm = function f(func, sig){
 | |
|     /** Attribution: adapted up from Emscripten-generated glue code,
 | |
|         refactored primarily for efficiency's sake, eliminating
 | |
|         call-local functions and superfluous temporary arrays. */
 | |
|     if(!f._){/*static init...*/
 | |
|       f._ = {
 | |
|         /* Map of signature letters to type IR values */
 | |
|         sigTypes: Object.assign(Object.create(null),{
 | |
|           i: 'i32', p: __ptrIR, P: __ptrIR, s: __ptrIR,
 | |
|           j: 'i64', f: 'f32', d: 'f64'
 | |
|         }),
 | |
| 
 | |
|         /* Map of type IR values to WASM type code values */
 | |
|         typeCodes: Object.assign(Object.create(null),{
 | |
|           f64: 0x7c, f32: 0x7d, i64: 0x7e, i32: 0x7f
 | |
|         }),
 | |
| 
 | |
|         /** Encodes n, which must be <2^14 (16384), into target array
 | |
|             tgt, as a little-endian value, using the given method
 | |
|             ('push' or 'unshift'). */
 | |
|         uleb128Encode: (tgt, method, n)=>{
 | |
|           if(n<128) tgt[method](n);
 | |
|           else tgt[method]( (n % 128) | 128, n>>7);
 | |
|         },
 | |
| 
 | |
|         /** Intentionally-lax pattern for Jaccwabyt-format function
 | |
|             pointer signatures, the intent of which is simply to
 | |
|             distinguish them from Emscripten-format signatures. The
 | |
|             downstream checks are less lax. */
 | |
|         rxJSig: /^(\w)\((\w*)\)$/,
 | |
| 
 | |
|         /** Returns the parameter-value part of the given signature
 | |
|             string. */
 | |
|         sigParams: (sig)=>{
 | |
|           const m = f._.rxJSig.exec(sig);
 | |
|           return m ? m[2] : sig.substr(1);
 | |
|         },
 | |
| 
 | |
|         /** Returns the IR value for the given letter or throws
 | |
|             if the letter is invalid. */
 | |
|         letterType: (x)=>f._.sigTypes[x] || toss("Invalid signature letter:",x),
 | |
| 
 | |
|         /** Pushes the WASM data type code for the given signature
 | |
|             letter to the given target array. Throws if letter is
 | |
|             invalid. */
 | |
|         pushSigType: (dest, letter)=>dest.push(f._.typeCodes[f._.letterType(letter)])
 | |
| 
 | |
|         /** Returns an object describing the result type and parameter
 | |
|             type(s) of the given function signature, or throws if the
 | |
|             signature is invalid. */
 | |
|         /******** // only valid for use with the WebAssembly.Function ctor, which
 | |
|                   // is not yet documented on MDN.
 | |
|         sigToWasm: function(sig){
 | |
|           const rc = {parameters:[], results: []};
 | |
|           if('v'!==sig[0]) rc.results.push(f.sigTypes(sig[0]));
 | |
|           for(const x of f._.sigParams(sig)){
 | |
|             rc.parameters.push(f._.typeCodes(x));
 | |
|           }
 | |
|           return rc;
 | |
|         },************/
 | |
|       };
 | |
|     }/*static init*/
 | |
|     if('string'===typeof func){
 | |
|       const x = sig;
 | |
|       sig = func;
 | |
|       func = x;
 | |
|     }
 | |
|     const _ = f._;
 | |
|     const sigParams = _.sigParams(sig);
 | |
|     const wasmCode = [0x01/*count: 1*/, 0x60/*function*/];
 | |
|     _.uleb128Encode(wasmCode, 'push', sigParams.length);
 | |
|     for(const x of sigParams) _.pushSigType(wasmCode, x);
 | |
|     if('v'===sig[0]) wasmCode.push(0);
 | |
|     else{
 | |
|       wasmCode.push(1);
 | |
|       _.pushSigType(wasmCode, sig[0]);
 | |
|     }
 | |
|     _.uleb128Encode(wasmCode, 'unshift', wasmCode.length)/* type section length */;
 | |
|     wasmCode.unshift(
 | |
|       0x00, 0x61, 0x73, 0x6d, /* magic: "\0asm" */
 | |
|       0x01, 0x00, 0x00, 0x00, /* version: 1 */
 | |
|       0x01 /* type section code */
 | |
|     );
 | |
|     wasmCode.push(
 | |
|       /* import section: */ 0x02, 0x07,
 | |
|       /* (import "e" "f" (func 0 (type 0))): */
 | |
|       0x01, 0x01, 0x65, 0x01, 0x66, 0x00, 0x00,
 | |
|       /* export section: */ 0x07, 0x05,
 | |
|       /* (export "f" (func 0 (type 0))): */
 | |
|       0x01, 0x01, 0x66, 0x00, 0x00
 | |
|     );
 | |
|     return (new WebAssembly.Instance(
 | |
|       new WebAssembly.Module(new Uint8Array(wasmCode)), {
 | |
|         e: { f: func }
 | |
|       })).exports['f'];
 | |
|   }/*jsFuncToWasm()*/;
 | |
| 
 | |
|   /**
 | |
|      Documented as target.installFunction() except for the 3rd
 | |
|      argument: if truthy, the newly-created function pointer
 | |
|      is stashed in the current scoped-alloc scope and will be
 | |
|      cleaned up at the matching scopedAllocPop(), else it
 | |
|      is not stashed there.
 | |
|   */
 | |
|   const __installFunction = function f(func, sig, scoped){
 | |
|     if(scoped && !cache.scopedAlloc.length){
 | |
|       toss("No scopedAllocPush() scope is active.");
 | |
|     }
 | |
|     if('string'===typeof func){
 | |
|       const x = sig;
 | |
|       sig = func;
 | |
|       func = x;
 | |
|     }
 | |
|     if('string'!==typeof sig || !(func instanceof Function)){
 | |
|       toss("Invalid arguments: expecting (function,signature) "+
 | |
|            "or (signature,function).");
 | |
|     }
 | |
|     const ft = target.functionTable();
 | |
|     const oldLen = __asPtrType(ft.length);
 | |
|     let ptr;
 | |
|     while(cache.freeFuncIndexes.length){
 | |
|       ptr = cache.freeFuncIndexes.pop();
 | |
|       if(ft.get(ptr)){ /* Table was modified via a different API */
 | |
|         ptr = null;
 | |
|         continue;
 | |
|       }else{
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|     if(!ptr){
 | |
|       ptr = __asPtrType(oldLen);
 | |
|       ft.grow(__asPtrType(1));
 | |
|     }
 | |
|     try{
 | |
|       /*this will only work if func is a WASM-exported function*/
 | |
|       ft.set(ptr, func);
 | |
|       if(scoped){
 | |
|         cache.scopedAlloc.pushPtr(ptr);
 | |
|       }
 | |
|       return ptr;
 | |
|     }catch(e){
 | |
|       if(!(e instanceof TypeError)){
 | |
|         if(ptr===oldLen) cache.freeFuncIndexes.push(oldLen);
 | |
|         throw e;
 | |
|       }
 | |
|     }
 | |
|     // It's not a WASM-exported function, so compile one...
 | |
|     try {
 | |
|       const fptr = target.jsFuncToWasm(func, sig);
 | |
|       ft.set(ptr, fptr);
 | |
|       if(scoped){
 | |
|         cache.scopedAlloc.pushPtr(ptr);
 | |
|       }
 | |
|     }catch(e){
 | |
|       if(ptr===oldLen) cache.freeFuncIndexes.push(oldLen);
 | |
|       throw e;
 | |
|     }
 | |
|     return ptr;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Expects a JS function and signature, exactly as for
 | |
|      this.jsFuncToWasm(). It uses that function to create a
 | |
|      WASM-exported function, installs that function to the next
 | |
|      available slot of this.functionTable(), and returns the
 | |
|      function's index in that table (which acts as a pointer to that
 | |
|      function). The returned pointer can be passed to
 | |
|      uninstallFunction() to uninstall it and free up the table slot
 | |
|      for reuse.
 | |
| 
 | |
|      If passed (string,function) arguments then it treats the first
 | |
|      argument as the signature and second as the function.
 | |
| 
 | |
|      As a special case, if the passed-in function is a WASM-exported
 | |
|      function then the signature argument is ignored and func is
 | |
|      installed as-is, without requiring re-compilation/re-wrapping.
 | |
| 
 | |
|      This function will propagate an exception if
 | |
|      WebAssembly.Table.grow() throws or this.jsFuncToWasm() throws.
 | |
|      The former case can happen in an Emscripten-compiled environment
 | |
|      when building without Emscripten's `-sALLOW_TABLE_GROWTH` flag.
 | |
| 
 | |
|      Sidebar: this function differs from Emscripten's addFunction()
 | |
|      _primarily_ in that it does not share that function's
 | |
|      undocumented behavior of reusing a function if it's passed to
 | |
|      addFunction() more than once, which leads to uninstallFunction()
 | |
|      breaking clients which do not take care to avoid that case:
 | |
| 
 | |
|      https://github.com/emscripten-core/emscripten/issues/17323
 | |
|   */
 | |
|   target.installFunction = (func, sig)=>__installFunction(func, sig, false);
 | |
| 
 | |
|   /**
 | |
|      Works exactly like installFunction() but requires that a
 | |
|      scopedAllocPush() is active and uninstalls the given function
 | |
|      when that alloc scope is popped via scopedAllocPop().
 | |
|      This is used for implementing JS/WASM function bindings which
 | |
|      should only persist for the life of a call into a single
 | |
|      C-side function.
 | |
|   */
 | |
|   target.scopedInstallFunction = (func, sig)=>__installFunction(func, sig, true);
 | |
| 
 | |
|   /**
 | |
|      Requires a pointer value previously returned from
 | |
|      this.installFunction(). Removes that function from the WASM
 | |
|      function table, marks its table slot as free for re-use, and
 | |
|      returns that function. It is illegal to call this before
 | |
|      installFunction() has been called and results are undefined if
 | |
|      ptr was not returned by that function. The returned function
 | |
|      may be passed back to installFunction() to reinstall it.
 | |
| 
 | |
|      To simplify certain use cases, if passed a falsy non-0 value
 | |
|      (noting that 0 is a valid function table index), this function
 | |
|      has no side effects and returns undefined.
 | |
|   */
 | |
|   target.uninstallFunction = function(ptr){
 | |
|     if(!ptr && 0!==ptr) return undefined;
 | |
|     const fi = cache.freeFuncIndexes;
 | |
|     const ft = target.functionTable();
 | |
|     fi.push(ptr);
 | |
|     const rc = ft.get(ptr);
 | |
|     ft.set(ptr, null);
 | |
|     return rc;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Given a WASM heap memory address and a data type name in the form
 | |
|      (i8, i16, i32, i64, float (or f32), double (or f64)), this
 | |
|      fetches the numeric value from that address and returns it as a
 | |
|      number or, for the case of type='i64', a BigInt (with the caveat
 | |
|      BigInt will trigger an exception if this.bigIntEnabled is
 | |
|      falsy). Throws if given an invalid type.
 | |
| 
 | |
|      If the first argument is an array, it is treated as an array of
 | |
|      addresses and the result is an array of the values from each of
 | |
|      those address, using the same 2nd argument for determining the
 | |
|      value type to fetch.
 | |
| 
 | |
|      As a special case, if type ends with a `*`, it is considered to
 | |
|      be a pointer type and is treated as the WASM numeric type
 | |
|      appropriate for the pointer size (==this.ptr.ir).
 | |
| 
 | |
|      While possibly not obvious, this routine and its poke()
 | |
|      counterpart are how pointer-to-value _output_ parameters in
 | |
|      WASM-compiled C code can be interacted with:
 | |
| 
 | |
|      ```
 | |
|      const ptr = alloc(4);
 | |
|      poke32(ptr, 0); // clear the ptr's value
 | |
|      aCFuncWithOutputPtrToInt32Arg(ptr); // e.g. void foo(int *x);
 | |
|      const result = peek32(ptr); // fetch ptr's value
 | |
|      dealloc(ptr);
 | |
|      ```
 | |
| 
 | |
|      scopedAlloc() and friends can be used to make handling of
 | |
|      `ptr` safe against leaks in the case of an exception:
 | |
| 
 | |
|      ```
 | |
|      let result;
 | |
|      const scope = scopedAllocPush();
 | |
|      try{
 | |
|        const ptr = scopedAlloc(4);
 | |
|        poke32(ptr, 0);
 | |
|        aCFuncWithOutputPtrArg(ptr);
 | |
|        result = peek32(ptr);
 | |
|      }finally{
 | |
|        scopedAllocPop(scope);
 | |
|      }
 | |
|      ```
 | |
| 
 | |
|      As a rule poke() must be called to set (typically zero out) the
 | |
|      pointer's value, else it will contain an essentially random
 | |
|      value.
 | |
| 
 | |
|      ACHTUNG: calling this often, e.g. in a loop, can have a noticably
 | |
|      painful impact on performance. Rather than doing so, use
 | |
|      heapForSize() to fetch the heap object and read directly from it.
 | |
| 
 | |
|      ACHTUNG #2: ptr may be a BigInt (and generally will be in 64-bit
 | |
|      builds) but this function must coerce it into a Number in order
 | |
|      to access the heap's contents. Ergo: BitInts outside of the
 | |
|      (extrardinarily genereous) address range exposed to browser-side
 | |
|      WASM may cause misbehavior.
 | |
| 
 | |
|      See also: poke()
 | |
|   */
 | |
|   target.peek = function f(ptr, type='i8'){
 | |
|     if(type.endsWith('*')) type = __ptrIR;
 | |
|     const c = (cache.memory && cache.heapSize === cache.memory.buffer.byteLength)
 | |
|           ? cache : heapWrappers();
 | |
|     const list = Array.isArray(ptr) ? [] : undefined;
 | |
|     let rc;
 | |
|     do{
 | |
|       if(list) ptr = arguments[0].shift();
 | |
|       switch(type){
 | |
|           case 'i1':
 | |
|           case 'i8': rc = c.HEAP8[Number(ptr/*tag:64bit*/)>>0]; break;
 | |
|           case 'i16': rc = c.HEAP16[Number(ptr/*tag:64bit*/)>>1]; break;
 | |
|           case 'i32': rc = c.HEAP32[Number(ptr/*tag:64bit*/)>>2]; break;
 | |
|           case 'float': case 'f32': rc = c.HEAP32F[Number(ptr/*tag:64bit*/)>>2]; break;
 | |
|           case 'double': case 'f64': rc = Number(c.HEAP64F[Number(ptr/*tag:64bit*/)>>3]); break;
 | |
|           case 'i64':
 | |
|             if(c.HEAP64){
 | |
|               rc = __BigInt(c.HEAP64[Number(ptr/*tag:64bit*/)>>3]);
 | |
|               break;
 | |
|             }
 | |
|             /* fallthru */
 | |
|           default:
 | |
|             toss('Invalid type for peek():',type);
 | |
|       }
 | |
|       if(list) list.push(rc);
 | |
|     }while(list && arguments[0].length);
 | |
|     return list || rc;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      The counterpart of peek(), this sets a numeric value at the given
 | |
|      WASM heap address, using the 3rd argument to define how many
 | |
|      bytes are written. Throws if given an invalid type. See peek()
 | |
|      for details about the `type` argument. If the 3rd argument ends
 | |
|      with `*` then it is treated as a pointer type and this function
 | |
|      behaves as if the 3rd argument were this.ptr.ir.
 | |
| 
 | |
|      If the first argument is an array, it is treated like a list
 | |
|      of pointers and the given value is written to each one.
 | |
| 
 | |
|      Returns `this`. (Prior to 2022-12-09 it returned this function.)
 | |
| 
 | |
|      ACHTUNG #1: see peek()'s ACHTUNG #1.
 | |
| 
 | |
|      ACHTUNG #2: see peek()'s ACHTUNG #2.
 | |
|   */
 | |
|   target.poke = function(ptr, value, type='i8'){
 | |
|     if (type.endsWith('*')) type = __ptrIR;
 | |
|     const c = (cache.memory && cache.heapSize === cache.memory.buffer.byteLength)
 | |
|           ? cache : heapWrappers();
 | |
|     for(const p of (Array.isArray(ptr) ? ptr : [ptr])){
 | |
|       switch (type) {
 | |
|           case 'i1':
 | |
|           case 'i8': c.HEAP8[Number(p/*tag:64bit*/)>>0] = value; continue;
 | |
|           case 'i16': c.HEAP16[Number(p/*tag:64bit*/)>>1] = value; continue;
 | |
|           case 'i32': c.HEAP32[Number(p/*tag:64bit*/)>>2] = value; continue;
 | |
|           case 'float': case 'f32': c.HEAP32F[Number(p/*tag:64bit*/)>>2] = value; continue;
 | |
|           case 'double': case 'f64': c.HEAP64F[Number(p/*tag:64bit*/)>>3] = value; continue;
 | |
|           case 'i64':
 | |
|             if(c.HEAP64){
 | |
|               c.HEAP64[Number(p/*tag:64bit*/)>>3] = __BigInt(value);
 | |
|               continue;
 | |
|             }
 | |
|             /* fallthru */
 | |
|           default:
 | |
|             toss('Invalid type for poke(): ' + type);
 | |
|       }
 | |
|     }
 | |
|     return this;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Convenience form of peek() intended for fetching
 | |
|      pointer-to-pointer values. If passed a single non-array argument
 | |
|      it returns the value of that one pointer address. If passed
 | |
|      multiple arguments, or a single array of arguments, it returns an
 | |
|      array of their values.
 | |
|   */
 | |
|   target.peekPtr = (...ptr)=>target.peek( (1===ptr.length ? ptr[0] : ptr), __ptrIR );
 | |
| 
 | |
|   /**
 | |
|      A variant of poke() intended for setting pointer-to-pointer
 | |
|      values. Its differences from poke() are that (1) it defaults to a
 | |
|      value of 0 and (2) it always writes to the pointer-sized heap
 | |
|      view.
 | |
|   */
 | |
|   target.pokePtr = (ptr, value=0)=>target.poke(ptr, value, __ptrIR);
 | |
| 
 | |
|   /**
 | |
|      Convenience form of peek() intended for fetching i8 values. If
 | |
|      passed a single non-array argument it returns the value of that
 | |
|      one pointer address. If passed multiple arguments, or a single
 | |
|      array of arguments, it returns an array of their values.
 | |
|   */
 | |
|   target.peek8 = (...ptr)=>target.peek( (1===ptr.length ? ptr[0] : ptr), 'i8' );
 | |
|   /**
 | |
|      Convenience form of poke() intended for setting individual bytes.
 | |
|      Its difference from poke() is that it always writes to the
 | |
|      i8-sized heap view.
 | |
|   */
 | |
|   target.poke8 = (ptr, value)=>target.poke(ptr, value, 'i8');
 | |
|   /** i16 variant of peek8(). */
 | |
|   target.peek16 = (...ptr)=>target.peek( (1===ptr.length ? ptr[0] : ptr), 'i16' );
 | |
|   /** i16 variant of poke8(). */
 | |
|   target.poke16 = (ptr, value)=>target.poke(ptr, value, 'i16');
 | |
|   /** i32 variant of peek8(). */
 | |
|   target.peek32 = (...ptr)=>target.peek( (1===ptr.length ? ptr[0] : ptr), 'i32' );
 | |
|   /** i32 variant of poke8(). */
 | |
|   target.poke32 = (ptr, value)=>target.poke(ptr, value, 'i32');
 | |
|   /** i64 variant of peek8(). Will throw if this build is not
 | |
|       configured for BigInt support. */
 | |
|   target.peek64 = (...ptr)=>target.peek( (1===ptr.length ? ptr[0] : ptr), 'i64' );
 | |
|   /** i64 variant of poke8(). Will throw if this build is not
 | |
|       configured for BigInt support. Note that this returns
 | |
|       a BigInt-type value, not a Number-type value. */
 | |
|   target.poke64 = (ptr, value)=>target.poke(ptr, value, 'i64');
 | |
|   /** f32 variant of peek8(). */
 | |
|   target.peek32f = (...ptr)=>target.peek( (1===ptr.length ? ptr[0] : ptr), 'f32' );
 | |
|   /** f32 variant of poke8(). */
 | |
|   target.poke32f = (ptr, value)=>target.poke(ptr, value, 'f32');
 | |
|   /** f64 variant of peek8(). */
 | |
|   target.peek64f = (...ptr)=>target.peek( (1===ptr.length ? ptr[0] : ptr), 'f64' );
 | |
|   /** f64 variant of poke8(). */
 | |
|   target.poke64f = (ptr, value)=>target.poke(ptr, value, 'f64');
 | |
| 
 | |
|   /** Deprecated alias for getMemValue() */
 | |
|   target.getMemValue = target.peek;
 | |
|   /** Deprecated alias for peekPtr() */
 | |
|   target.getPtrValue = target.peekPtr;
 | |
|   /** Deprecated alias for poke() */
 | |
|   target.setMemValue = target.poke;
 | |
|   /** Deprecated alias for pokePtr() */
 | |
|   target.setPtrValue = target.pokePtr;
 | |
| 
 | |
|   /**
 | |
|      Returns true if the given value appears to be legal for use as
 | |
|      a WASM pointer value. Its _range_ of values is not (cannot be)
 | |
|      validated except to ensure that it is a 32-bit integer with a
 | |
|      value of 0 or greater. Likewise, it cannot verify whether the
 | |
|      value actually refers to allocated memory in the WASM heap.
 | |
| 
 | |
|      Whether or not null or undefined are legal are context-dependent.
 | |
|      They generally are legal but this function does not treat them as
 | |
|      such because they're not strictly legal for passing as-is as WASM
 | |
|      integer arguments.
 | |
|   */
 | |
|   target.isPtr32 = (ptr)=>(
 | |
|     'number'===typeof ptr && ptr>=0 && ptr===(ptr|0)
 | |
|   );
 | |
| 
 | |
|   /** 64-bit counterpart of isPtr32() and falls back to that function
 | |
|       if ptr is not a BigInt. */
 | |
|   target.isPtr64 = (ptr)=>(
 | |
|     ('bigint'===typeof ptr) ? ptr >= 0 : target.isPtr32(ptr)
 | |
|   );
 | |
| 
 | |
|   /**
 | |
|      isPtr() is an alias for isPtr32() or isPtr64(), depending on the
 | |
|      value of target.ptr.size.
 | |
|   */
 | |
|   target.isPtr = (4===__ptrSize) ? target.isPtr32 : target.isPtr64;
 | |
| 
 | |
|   /**
 | |
|      Expects ptr to be a pointer into the WASM heap memory which
 | |
|      refers to a NUL-terminated C-style string encoded as UTF-8.
 | |
|      Returns the length, in bytes, of the string, as for `strlen(3)`.
 | |
|      As a special case, if !ptr or if it's not a pointer then it
 | |
|      returns `null`. Throws if ptr is out of range for
 | |
|      target.heap8u().
 | |
|   */
 | |
|   target.cstrlen = function(ptr){
 | |
|     if(!ptr || !target.isPtr(ptr)) return null;
 | |
|     ptr = Number(ptr) /*tag:64bit*/;
 | |
|     const h = heapWrappers().HEAP8U;
 | |
|     let pos = ptr;
 | |
|     for( ; h[pos] !== 0; ++pos ){}
 | |
|     return Number(pos - ptr);
 | |
|   };
 | |
| 
 | |
|   /** Internal helper to use in operations which need to distinguish
 | |
|       between TypedArrays which are backed by a SharedArrayBuffer
 | |
|       from those which are not. */
 | |
|   const __SAB = ('undefined'===typeof SharedArrayBuffer)
 | |
|         ? function(){/*dummy class*/} : SharedArrayBuffer;
 | |
|   /** Returns true if the given TypedArray object is backed by a
 | |
|       SharedArrayBuffer, else false. */
 | |
|   const isSharedTypedArray = (aTypedArray)=>(aTypedArray.buffer instanceof __SAB);
 | |
| 
 | |
|   target.isSharedTypedArray = isSharedTypedArray;
 | |
| 
 | |
|   /**
 | |
|      Returns either aTypedArray.slice(begin,end) (if
 | |
|      aTypedArray.buffer is a SharedArrayBuffer) or
 | |
|      aTypedArray.subarray(begin,end) (if it's not).
 | |
| 
 | |
|      This distinction is important for APIs which don't like to
 | |
|      work on SABs, e.g. TextDecoder, and possibly for our
 | |
|      own APIs which work on memory ranges which "might" be
 | |
|      modified by other threads while they're working.
 | |
| 
 | |
|      begin and end may be of type Number or (in 64-bit builds) BigInt
 | |
|      (which get coerced to Numbers).
 | |
|   */
 | |
|   const typedArrayPart = (aTypedArray, begin, end)=>{
 | |
|     if( 8===__ptrSize ){
 | |
|       // slice() and subarray() do not like BigInt args.
 | |
|       if( 'bigint'===typeof begin ) begin = Number(begin);
 | |
|       if( 'bigint'===typeof end ) end = Number(end);
 | |
|     }
 | |
|     return isSharedTypedArray(aTypedArray)
 | |
|       ? aTypedArray.slice(begin, end)
 | |
|       : aTypedArray.subarray(begin, end);
 | |
|   };
 | |
| 
 | |
|   target.typedArrayPart = typedArrayPart;
 | |
| 
 | |
|   /**
 | |
|      Uses TextDecoder to decode the given half-open range of the given
 | |
|      TypedArray to a string. This differs from a simple call to
 | |
|      TextDecoder in that it accounts for whether the first argument is
 | |
|      backed by a SharedArrayBuffer or not, and can work more
 | |
|      efficiently if it's not (TextDecoder refuses to act upon an SAB).
 | |
| 
 | |
|      If begin/end are not provided or are falsy then each defaults to
 | |
|      the start/end of the array.
 | |
|   */
 | |
|   const typedArrayToString = (typedArray, begin, end)=>
 | |
|         cache.utf8Decoder.decode(
 | |
|           typedArrayPart(typedArray, begin, end)
 | |
|         );
 | |
| 
 | |
|   target.typedArrayToString = typedArrayToString;
 | |
| 
 | |
|   /**
 | |
|      Expects ptr to be a pointer into the WASM heap memory which
 | |
|      refers to a NUL-terminated C-style string encoded as UTF-8. This
 | |
|      function counts its byte length using cstrlen() then returns a
 | |
|      JS-format string representing its contents. As a special case, if
 | |
|      ptr is falsy or not a pointer, `null` is returned.
 | |
|   */
 | |
|   target.cstrToJs = function(ptr){
 | |
|     const n = target.cstrlen(ptr);
 | |
|     return n
 | |
|       ? typedArrayToString(heapWrappers().HEAP8U, Number(ptr), Number(ptr)+n)
 | |
|       : (null===n ? n : "");
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Given a JS string, this function returns its UTF-8 length in
 | |
|      bytes. Returns null if str is not a string.
 | |
|   */
 | |
|   target.jstrlen = function(str){
 | |
|     /** Attribution: derived from Emscripten's lengthBytesUTF8() */
 | |
|     if('string'!==typeof str) return null;
 | |
|     const n = str.length;
 | |
|     let len = 0;
 | |
|     for(let i = 0; i < n; ++i){
 | |
|       let u = str.charCodeAt(i);
 | |
|       if(u>=0xd800 && u<=0xdfff){
 | |
|         u = 0x10000 + ((u & 0x3FF) << 10) | (str.charCodeAt(++i) & 0x3FF);
 | |
|       }
 | |
|       if(u<=0x7f) ++len;
 | |
|       else if(u<=0x7ff) len += 2;
 | |
|       else if(u<=0xffff) len += 3;
 | |
|       else len += 4;
 | |
|     }
 | |
|     return len;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Encodes the given JS string as UTF8 into the given TypedArray
 | |
|      tgt, starting at the given offset and writing, at most, maxBytes
 | |
|      bytes (including the NUL terminator if addNul is true, else no
 | |
|      NUL is added). If it writes any bytes at all and addNul is true,
 | |
|      it always NUL-terminates the output, even if doing so means that
 | |
|      the NUL byte is all that it writes.
 | |
| 
 | |
|      If maxBytes is negative (the default) then it is treated as the
 | |
|      remaining length of tgt, starting at the given offset.
 | |
| 
 | |
|      If writing the last character would surpass the maxBytes count
 | |
|      because the character is multi-byte, that character will not be
 | |
|      written (as opposed to writing a truncated multi-byte character).
 | |
|      This can lead to it writing as many as 3 fewer bytes than
 | |
|      maxBytes specifies.
 | |
| 
 | |
|      Returns the number of bytes written to the target, _including_
 | |
|      the NUL terminator (if any). If it returns 0, it wrote nothing at
 | |
|      all, which can happen if:
 | |
| 
 | |
|      - str is empty and addNul is false.
 | |
|      - offset < 0.
 | |
|      - maxBytes == 0.
 | |
|      - maxBytes is less than the byte length of a multi-byte str[0].
 | |
| 
 | |
|      Throws if tgt is not an Int8Array or Uint8Array.
 | |
| 
 | |
|      Design notes:
 | |
| 
 | |
|      - In C's strcpy(), the destination pointer is the first
 | |
|        argument. That is not the case here primarily because the 3rd+
 | |
|        arguments are all referring to the destination, so it seems to
 | |
|        make sense to have them grouped with it.
 | |
| 
 | |
|      - Emscripten's counterpart of this function (stringToUTF8Array())
 | |
|        returns the number of bytes written sans NUL terminator. That
 | |
|        is, however, ambiguous: str.length===0 or maxBytes===(0 or 1)
 | |
|        all cause 0 to be returned.
 | |
|   */
 | |
|   target.jstrcpy = function(jstr, tgt, offset = 0, maxBytes = -1, addNul = true){
 | |
|     /** Attribution: the encoding bits are taken from Emscripten's
 | |
|         stringToUTF8Array(). */
 | |
|     if(!tgt || (!(tgt instanceof Int8Array) && !(tgt instanceof Uint8Array))){
 | |
|       toss("jstrcpy() target must be an Int8Array or Uint8Array.");
 | |
|     }
 | |
|     maxBytes = Number(maxBytes)/*tag:64bit*/;
 | |
|     offset = Number(offset)/*tag:64bit*/;
 | |
|     if(maxBytes<0) maxBytes = tgt.length - offset;
 | |
|     if(!(maxBytes>0) || !(offset>=0)) return 0;
 | |
|     let i = 0, max = jstr.length;
 | |
|     const begin = offset, end = offset + maxBytes - (addNul ? 1 : 0);
 | |
|     for(; i < max && offset < end; ++i){
 | |
|       let u = jstr.charCodeAt(i);
 | |
|       if(u>=0xd800 && u<=0xdfff){
 | |
|         u = 0x10000 + ((u & 0x3FF) << 10) | (jstr.charCodeAt(++i) & 0x3FF);
 | |
|       }
 | |
|       if(u<=0x7f){
 | |
|         if(offset >= end) break;
 | |
|         tgt[offset++] = u;
 | |
|       }else if(u<=0x7ff){
 | |
|         if(offset + 1 >= end) break;
 | |
|         tgt[offset++] = 0xC0 | (u >> 6);
 | |
|         tgt[offset++] = 0x80 | (u & 0x3f);
 | |
|       }else if(u<=0xffff){
 | |
|         if(offset + 2 >= end) break;
 | |
|         tgt[offset++] = 0xe0 | (u >> 12);
 | |
|         tgt[offset++] = 0x80 | ((u >> 6) & 0x3f);
 | |
|         tgt[offset++] = 0x80 | (u & 0x3f);
 | |
|       }else{
 | |
|         if(offset + 3 >= end) break;
 | |
|         tgt[offset++] = 0xf0 | (u >> 18);
 | |
|         tgt[offset++] = 0x80 | ((u >> 12) & 0x3f);
 | |
|         tgt[offset++] = 0x80 | ((u >> 6) & 0x3f);
 | |
|         tgt[offset++] = 0x80 | (u & 0x3f);
 | |
|       }
 | |
|     }
 | |
|     if(addNul) tgt[offset++] = 0;
 | |
|     return offset - begin;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Works similarly to C's strncpy(), copying, at most, n bytes (not
 | |
|      characters) from srcPtr to tgtPtr. It copies until n bytes have
 | |
|      been copied or a 0 byte is reached in src. _Unlike_ strncpy(), it
 | |
|      returns the number of bytes it assigns in tgtPtr, _including_ the
 | |
|      NUL byte (if any). If n is reached before a NUL byte in srcPtr,
 | |
|      tgtPtr will _not_ be NULL-terminated. If a NUL byte is reached
 | |
|      before n bytes are copied, tgtPtr will be NUL-terminated.
 | |
| 
 | |
|      If n is negative, cstrlen(srcPtr)+1 is used to calculate it, the
 | |
|      +1 being for the NUL byte.
 | |
| 
 | |
|      Throws if tgtPtr or srcPtr are falsy. Results are undefined if:
 | |
| 
 | |
|      - either is not a pointer into the WASM heap or
 | |
| 
 | |
|      - srcPtr is not NUL-terminated AND n is less than srcPtr's
 | |
|        logical length.
 | |
| 
 | |
|      ACHTUNG: it is possible to copy partial multi-byte characters
 | |
|      this way, and converting such strings back to JS strings will
 | |
|      have undefined results.
 | |
|   */
 | |
|   target.cstrncpy = function(tgtPtr, srcPtr, n){
 | |
|     if(!tgtPtr || !srcPtr) toss("cstrncpy() does not accept NULL strings.");
 | |
|     if(n<0) n = target.cstrlen(strPtr)+1;
 | |
|     else if(!(n>0)) return 0;
 | |
|     const heap = target.heap8u();
 | |
|     let i = 0, ch;
 | |
|     const tgtNumber = Number(tgtPtr), srcNumber = Number(srcPtr)/*tag:64bit*/;
 | |
|     for(; i < n && (ch = heap[srcNumber+i]); ++i){
 | |
|       heap[tgtNumber+i] = ch;
 | |
|     }
 | |
|     if(i<n) heap[tgtNumber + i++] = 0;
 | |
|     return i;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      For the given JS string, returns a Uint8Array of its contents
 | |
|      encoded as UTF-8. If addNul is true, the returned array will have
 | |
|      a trailing 0 entry, else it will not.
 | |
|   */
 | |
|   target.jstrToUintArray = (str, addNul=false)=>{
 | |
|     return cache.utf8Encoder.encode(addNul ? (str+"\0") : str);
 | |
|   };
 | |
| 
 | |
|   const __affirmAlloc = (obj,funcName)=>{
 | |
|     if(!(obj.alloc instanceof Function) ||
 | |
|        !(obj.dealloc instanceof Function)){
 | |
|       toss("Object is missing alloc() and/or dealloc() function(s)",
 | |
|            "required by",funcName+"().");
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   const __allocCStr = function(jstr, returnWithLength, allocator, funcName){
 | |
|     __affirmAlloc(target, funcName);
 | |
|     if('string'!==typeof jstr) return null;
 | |
|     const u = cache.utf8Encoder.encode(jstr),
 | |
|           ptr = allocator(u.length+1);
 | |
|     let toFree = ptr;
 | |
|     try{
 | |
|       const heap = heapWrappers().HEAP8U;
 | |
|       heap.set(u, Number(ptr));
 | |
|       heap[__ptrAdd(ptr, u.length)] = 0;
 | |
|       toFree = __NullPtr;
 | |
|       return returnWithLength ? [ptr, u.length] : ptr;
 | |
|     }finally{
 | |
|       if( toFree ) target.dealloc(toFree);
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Uses target.alloc() to allocate enough memory for jstrlen(jstr)+1
 | |
|      bytes of memory, copies jstr to that memory using jstrcpy(),
 | |
|      NUL-terminates it, and returns the pointer to that C-string.
 | |
|      Ownership of the pointer is transfered to the caller, who must
 | |
|      eventually pass the pointer to dealloc() to free it.
 | |
| 
 | |
|      If passed a truthy 2nd argument then its return semantics change:
 | |
|      it returns [ptr,n], where ptr is the C-string's pointer and n is
 | |
|      its cstrlen().
 | |
| 
 | |
|      Throws if `target.alloc` or `target.dealloc` are not functions.
 | |
|   */
 | |
|   target.allocCString =
 | |
|     (jstr, returnWithLength=false)=>__allocCStr(jstr, returnWithLength,
 | |
|                                                 target.alloc, 'allocCString()');
 | |
| 
 | |
|   /**
 | |
|      Starts an "allocation scope." All allocations made using
 | |
|      scopedAlloc() are recorded in this scope and are freed when the
 | |
|      value returned from this function is passed to
 | |
|      scopedAllocPop().
 | |
| 
 | |
|      This family of functions requires that the API's object have both
 | |
|      `alloc()` and `dealloc()` methods, else this function will throw.
 | |
| 
 | |
|      Intended usage:
 | |
| 
 | |
|      ```
 | |
|      const scope = scopedAllocPush();
 | |
|      try {
 | |
|        const ptr1 = scopedAlloc(100);
 | |
|        const ptr2 = scopedAlloc(200);
 | |
|        const ptr3 = scopedAlloc(300);
 | |
|        ...
 | |
|        // Note that only allocations made via scopedAlloc()
 | |
|        // are managed by this allocation scope.
 | |
|      }finally{
 | |
|        scopedAllocPop(scope);
 | |
|      }
 | |
|      ```
 | |
| 
 | |
|      The value returned by this function must be treated as opaque by
 | |
|      the caller, suitable _only_ for passing to scopedAllocPop().
 | |
|      Its type and value are not part of this function's API and may
 | |
|      change in any given version of this code.
 | |
| 
 | |
|      `scopedAlloc.level` can be used to determine how many scoped
 | |
|      alloc levels are currently active.
 | |
|    */
 | |
|   target.scopedAllocPush = function(){
 | |
|     __affirmAlloc(target, 'scopedAllocPush');
 | |
|     const a = [];
 | |
|     cache.scopedAlloc.push(a);
 | |
|     return a;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Cleans up all allocations made using scopedAlloc() in the context
 | |
|      of the given opaque state object, which must be a value returned
 | |
|      by scopedAllocPush(). See that function for an example of how to
 | |
|      use this function. It also uninstalls any WASM functions
 | |
|      installed with scopedInstallFunction().
 | |
| 
 | |
|      Though scoped allocations are managed like a stack, this API
 | |
|      behaves properly if allocation scopes are popped in an order
 | |
|      other than the order they were pushed. The intent is that it
 | |
|      _always_ be used in a stack-like manner.
 | |
| 
 | |
|      If called with no arguments, it pops the most recent
 | |
|      scopedAllocPush() result:
 | |
| 
 | |
|      ```
 | |
|      scopedAllocPush();
 | |
|      try{ ... } finally { scopedAllocPop(); }
 | |
|      ```
 | |
| 
 | |
|      It's generally recommended that it be passed an explicit argument
 | |
|      to help ensure that push/push are used in matching pairs, but in
 | |
|      trivial code that may be a non-issue.
 | |
|   */
 | |
|   target.scopedAllocPop = function(state){
 | |
|     __affirmAlloc(target, 'scopedAllocPop');
 | |
|     const n = arguments.length
 | |
|           ? cache.scopedAlloc.indexOf(state)
 | |
|           : cache.scopedAlloc.length-1;
 | |
|     if(n<0) toss("Invalid state object for scopedAllocPop().");
 | |
|     if(0===arguments.length) state = cache.scopedAlloc[n];
 | |
|     cache.scopedAlloc.splice(n,1);
 | |
|     for(let p; (p = state.pop()); ){
 | |
|       if(target.functionEntry(p)){
 | |
|         //console.warn("scopedAllocPop() uninstalling function",p);
 | |
|         target.uninstallFunction(p);
 | |
|       }else{
 | |
|         target.dealloc(p);
 | |
|       }
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Allocates n bytes of memory using this.alloc() and records that
 | |
|      fact in the state for the most recent call of scopedAllocPush().
 | |
|      Ownership of the memory is given to scopedAllocPop(), which
 | |
|      will clean it up when it is called. The memory _must not_ be
 | |
|      passed to this.dealloc(). Throws if this API object is missing
 | |
|      the required `alloc()` or `dealloc()` functions or no scoped
 | |
|      alloc is active.
 | |
| 
 | |
|      See scopedAllocPush() for an example of how to use this function.
 | |
| 
 | |
|      The `level` property of this function can be queried to query how
 | |
|      many scoped allocation levels are currently active.
 | |
| 
 | |
|      See also: scopedAllocPtr(), scopedAllocCString()
 | |
|   */
 | |
|   target.scopedAlloc = function(n){
 | |
|     if(!cache.scopedAlloc.length){
 | |
|       toss("No scopedAllocPush() scope is active.");
 | |
|     }
 | |
|     const p = __asPtrType(target.alloc(n));
 | |
|     return cache.scopedAlloc.pushPtr(p);
 | |
|   };
 | |
| 
 | |
|   Object.defineProperty(target.scopedAlloc, 'level', {
 | |
|     configurable: false, enumerable: false,
 | |
|     get: ()=>cache.scopedAlloc.length,
 | |
|     set: ()=>toss("The 'active' property is read-only.")
 | |
|   });
 | |
| 
 | |
|   /**
 | |
|      Works identically to allocCString() except that it allocates the
 | |
|      memory using scopedAlloc().
 | |
| 
 | |
|      Will throw if no scopedAllocPush() call is active.
 | |
|   */
 | |
|   target.scopedAllocCString =
 | |
|     (jstr, returnWithLength=false)=>__allocCStr(jstr, returnWithLength,
 | |
|                                                 target.scopedAlloc,
 | |
|                                                 'scopedAllocCString()');
 | |
| 
 | |
|   // impl for allocMainArgv() and scopedAllocMainArgv().
 | |
|   const __allocMainArgv = function(isScoped, list){
 | |
|     const pList = target[
 | |
|       isScoped ? 'scopedAlloc' : 'alloc'
 | |
|     ]((list.length + 1) * target.ptr.size);
 | |
|     let i = 0;
 | |
|     list.forEach((e)=>{
 | |
|       target.pokePtr(__ptrAdd(pList, target.ptr.size * i++),
 | |
|                      target[
 | |
|                        isScoped ? 'scopedAllocCString' : 'allocCString'
 | |
|                      ](""+e));
 | |
|     });
 | |
|     target.pokePtr(__ptrAdd(pList, target.ptr.size * i), 0);
 | |
|     return pList;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Creates an array, using scopedAlloc(), suitable for passing to a
 | |
|      C-level main() routine. The input is a collection with a length
 | |
|      property and a forEach() method. A block of memory
 | |
|      (list.length+1) entries long is allocated and each pointer-sized
 | |
|      block of that memory is populated with a scopedAllocCString()
 | |
|      conversion of the (""+value) of each element, with the exception
 | |
|      that the final entry is a NULL pointer. Returns a pointer to the
 | |
|      start of the list, suitable for passing as the 2nd argument to a
 | |
|      C-style main() function.
 | |
| 
 | |
|      Throws if scopedAllocPush() is not active.
 | |
| 
 | |
|      Design note: the returned array is allocated with an extra NULL
 | |
|      pointer entry to accommodate certain APIs, but client code which
 | |
|      does not need that functionality should treat the returned array
 | |
|      as list.length entries long.
 | |
|   */
 | |
|   target.scopedAllocMainArgv = (list)=>__allocMainArgv(true, list);
 | |
| 
 | |
|   /**
 | |
|      Identical to scopedAllocMainArgv() but uses alloc() instead of
 | |
|      scopedAlloc().
 | |
|   */
 | |
|   target.allocMainArgv = (list)=>__allocMainArgv(false, list);
 | |
| 
 | |
|   /**
 | |
|      Expects to be given a C-style string array and its length. It
 | |
|      returns a JS array of strings and/or nulls: any entry in the
 | |
|      pArgv array which is NULL results in a null entry in the result
 | |
|      array. If argc is 0 then an empty array is returned.
 | |
| 
 | |
|      Results are undefined if any entry in the first argc entries of
 | |
|      pArgv are neither 0 (NULL) nor legal UTF-format C strings.
 | |
| 
 | |
|      To be clear, the expected C-style arguments to be passed to this
 | |
|      function are `(int, char **)` (optionally const-qualified).
 | |
|   */
 | |
|   target.cArgvToJs = (argc, pArgv)=>{
 | |
|     const list = [];
 | |
|     for(let i = 0; i < argc; ++i){
 | |
|       const arg = target.peekPtr(__ptrAdd(pArgv, target.ptr.size * i));
 | |
|       list.push( arg ? target.cstrToJs(arg) : null );
 | |
|     }
 | |
|     return list;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Wraps function call func() in a scopedAllocPush() and
 | |
|      scopedAllocPop() block, such that all calls to scopedAlloc() and
 | |
|      friends from within that call will have their memory freed
 | |
|      automatically when func() returns. If func throws or propagates
 | |
|      an exception, the scope is still popped, otherwise it returns the
 | |
|      result of calling func().
 | |
|   */
 | |
|   target.scopedAllocCall = function(func){
 | |
|     target.scopedAllocPush();
 | |
|     try{ return func() } finally{ target.scopedAllocPop() }
 | |
|   };
 | |
| 
 | |
|   /** Internal impl for allocPtr() and scopedAllocPtr(). */
 | |
|   const __allocPtr = function(howMany, safePtrSize, method){
 | |
|     __affirmAlloc(target, method);
 | |
|     const pIr = safePtrSize ? 'i64' : __ptrIR;
 | |
|     let m = target[method](howMany * (safePtrSize ? 8 : __ptrSize));
 | |
|     target.poke(m, 0, pIr)
 | |
|     if(1===howMany){
 | |
|       return m;
 | |
|     }
 | |
|     const a = [m];
 | |
|     for(let i = 1; i < howMany; ++i){
 | |
|       m = __ptrAdd(m, (safePtrSize ? 8 : __ptrSize));
 | |
|       a[i] = m;
 | |
|       target.poke(m, 0, pIr);
 | |
|     }
 | |
|     return a;
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Allocates one or more pointers as a single chunk of memory and
 | |
|      zeroes them out.
 | |
| 
 | |
|      The first argument is the number of pointers to allocate. The
 | |
|      second specifies whether they should use a "safe" pointer size (8
 | |
|      bytes) or whether they may use the default pointer size
 | |
|      (typically 4 but also possibly 8).
 | |
| 
 | |
|      How the result is returned depends on its first argument: if
 | |
|      passed 1, it returns the allocated memory address. If passed more
 | |
|      than one then an array of pointer addresses is returned, which
 | |
|      can optionally be used with "destructuring assignment" like this:
 | |
| 
 | |
|      ```
 | |
|      const [p1, p2, p3] = allocPtr(3);
 | |
|      ```
 | |
| 
 | |
|      ACHTUNG: when freeing the memory, pass only the _first_ result
 | |
|      value to dealloc(). The others are part of the same memory chunk
 | |
|      and must not be freed separately.
 | |
| 
 | |
|      The reason for the 2nd argument is...
 | |
| 
 | |
|      When one of the returned pointers will refer to a 64-bit value,
 | |
|      e.g. a double or int64, and that value must be written or fetched,
 | |
|      e.g. using poke() or peek(), it is important that
 | |
|      the pointer in question be aligned to an 8-byte boundary or else
 | |
|      it will not be fetched or written properly and will corrupt or
 | |
|      read neighboring memory. It is only safe to pass false when the
 | |
|      client code is certain that it will only get/fetch 4-byte values
 | |
|      (or smaller).
 | |
|   */
 | |
|   target.allocPtr =
 | |
|     (howMany=1, safePtrSize=true)=>__allocPtr(howMany, safePtrSize, 'alloc');
 | |
| 
 | |
|   /**
 | |
|      Identical to allocPtr() except that it allocates using scopedAlloc()
 | |
|      instead of alloc().
 | |
|   */
 | |
|   target.scopedAllocPtr =
 | |
|     (howMany=1, safePtrSize=true)=>__allocPtr(howMany, safePtrSize, 'scopedAlloc');
 | |
| 
 | |
|   /**
 | |
|      If target.exports[name] exists, it is returned, else an
 | |
|      exception is thrown.
 | |
|   */
 | |
|   target.xGet = function(name){
 | |
|     return target.exports[name] || toss("Cannot find exported symbol:",name);
 | |
|   };
 | |
| 
 | |
|   const __argcMismatch =
 | |
|         (f,n)=>toss(f+"() requires",n,"argument(s).");
 | |
| 
 | |
|   /**
 | |
|      Looks up a WASM-exported function named fname from
 | |
|      target.exports. If found, it is called, passed all remaining
 | |
|      arguments, and its return value is returned to xCall's caller. If
 | |
|      not found, an exception is thrown. This function does no
 | |
|      conversion of argument or return types, but see xWrap() and
 | |
|      xCallWrapped() for variants which do.
 | |
| 
 | |
|      If the first argument is a function is is assumed to be
 | |
|      a WASM-bound function and is used as-is instead of looking up
 | |
|      the function via xGet().
 | |
| 
 | |
|      As a special case, if passed only 1 argument after the name and
 | |
|      that argument in an Array, that array's entries become the
 | |
|      function arguments. (This is not an ambiguous case because it's
 | |
|      not legal to pass an Array object to a WASM function.)
 | |
|   */
 | |
|   target.xCall = function(fname, ...args){
 | |
|     const f = (fname instanceof Function) ? fname : target.xGet(fname);
 | |
|     if(!(f instanceof Function)) toss("Exported symbol",fname,"is not a function.");
 | |
|     if(f.length!==args.length){
 | |
|       __argcMismatch(((f===fname) ? f.name : fname),f.length)
 | |
|       /* This is arguably over-pedantic but we want to help clients keep
 | |
|          from shooting themselves in the foot when calling C APIs. */;
 | |
|     }
 | |
|     return (2===arguments.length && Array.isArray(arguments[1]))
 | |
|       ? f.apply(null, arguments[1])
 | |
|       : f.apply(null, args);
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      State for use with xWrap().
 | |
|   */
 | |
|   cache.xWrap = Object.create(null);
 | |
|   cache.xWrap.convert = Object.create(null);
 | |
|   /** Map of type names to argument conversion functions. */
 | |
|   cache.xWrap.convert.arg = new Map;
 | |
|   /** Map of type names to return result conversion functions. */
 | |
|   cache.xWrap.convert.result = new Map;
 | |
|   /** Scope-local convenience aliases. */
 | |
|   const xArg = cache.xWrap.convert.arg, xResult = cache.xWrap.convert.result;
 | |
| 
 | |
|   const __xArgPtr = __asPtrType;
 | |
|   xArg
 | |
|     .set('i64', __BigInt)
 | |
|     .set('i32', (i)=>i|0)
 | |
|     .set('i16', (i)=>((i | 0) & 0xFFFF))
 | |
|     .set('i8', (i)=>((i | 0) & 0xFF))
 | |
|     .set('f32', (i)=>Number(i).valueOf())
 | |
|     .set('float', xArg.get('f32'))
 | |
|     .set('f64', xArg.get('f32'))
 | |
|     .set('double', xArg.get('f64'))
 | |
|     .set('int', xArg.get('i32'))
 | |
|     .set('null', (i)=>i)
 | |
|     .set(null, xArg.get('null'))
 | |
|     .set('**', __xArgPtr)
 | |
|     .set('*', __xArgPtr)
 | |
|   ;
 | |
|   xResult.set('*', __xArgPtr)
 | |
|     .set('pointer', __xArgPtr)
 | |
|     .set('number', (v)=>Number(v))
 | |
|     .set('void', (v)=>undefined)
 | |
|     .set(undefined, xResult.get('void'))
 | |
|     .set('null', (v)=>v)
 | |
|     .set(null, xResult.get('null'))
 | |
|   ;
 | |
| 
 | |
|   /* Copy xArg[...] handlers to xResult[...] for cases which have
 | |
|      identical semantics. Also add pointer-style variants of those
 | |
|      xArg entries to both xArg and xResult. */
 | |
|   for(const t of [
 | |
|     'i8', 'i16', 'i32', 'i64', 'int',
 | |
|     'f32', 'float', 'f64', 'double'
 | |
|   ]){
 | |
|     xArg.set(t+'*', __xArgPtr);
 | |
|     xResult.set(t+'*', __xArgPtr);
 | |
|     xResult.set(
 | |
|       t, xArg.get(t)
 | |
|         || toss("Maintenance required: missing arg converter for",t)
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|      In order for args of type string to work in various contexts in
 | |
|      the sqlite3 API, we need to pass them on as, variably, a C-string
 | |
|      or a pointer value. Thus for ARGs of type 'string' and
 | |
|      '*'/'pointer' we behave differently depending on whether the
 | |
|      argument is a string or not:
 | |
| 
 | |
|      - If v is a string, scopeAlloc() a new C-string from it and return
 | |
|        that temp string's pointer.
 | |
| 
 | |
|      - Else return the value from the arg adapter defined for
 | |
|        target.ptr.ir.
 | |
| 
 | |
|      TODO? Permit an Int8Array/Uint8Array and convert it to a string?
 | |
|      Would that be too much magic concentrated in one place, ready to
 | |
|      backfire? We handle that at the client level in sqlite3 with a
 | |
|      custom argument converter.
 | |
|   */
 | |
|   const __xArgString = (v)=>{
 | |
|     return ('string'===typeof v)
 | |
|       ? target.scopedAllocCString(v)
 | |
|       : __asPtrType(v);
 | |
|   };
 | |
| 
 | |
|   xArg.set('string', __xArgString)
 | |
|     .set('utf8', __xArgString)
 | |
|     // (much later: why did we do this?) .set('pointer', __xArgString)
 | |
|   ;
 | |
| 
 | |
|   xResult
 | |
|     .set('string', (i)=>target.cstrToJs(i))
 | |
|     .set('utf8', xResult.get('string'))
 | |
|     .set('string:dealloc', (i)=>{
 | |
|       try { return i ? target.cstrToJs(i) : null }
 | |
|       finally{ target.dealloc(i) }
 | |
|     })
 | |
|     .set('utf8:dealloc', xResult.get('string:dealloc'))
 | |
|     .set('json', (i)=>JSON.parse(target.cstrToJs(i)))
 | |
|     .set('json:dealloc', (i)=>{
 | |
|       try{ return i ? JSON.parse(target.cstrToJs(i)) : null }
 | |
|       finally{ target.dealloc(i) }
 | |
|     });
 | |
| 
 | |
|   /**
 | |
|      Internal-use-only base class for FuncPtrAdapter and potentially
 | |
|      additional stateful argument adapter classes.
 | |
| 
 | |
|      Its main interface (convertArg()) is strictly internal, not to be
 | |
|      exposed to client code, as it may still need re-shaping. Only the
 | |
|      constructors of concrete subclasses should be exposed to clients,
 | |
|      and those in such a way that does not hinder internal redesign of
 | |
|      the convertArg() interface.
 | |
|   */
 | |
|   const AbstractArgAdapter = class {
 | |
|     constructor(opt){
 | |
|       this.name = opt.name || 'unnamed adapter';
 | |
|     }
 | |
|     /**
 | |
|        Gets called via xWrap() to "convert" v to whatever type
 | |
|        this specific class supports.
 | |
| 
 | |
|        argIndex is the argv index of _this_ argument in the
 | |
|        being-xWrap()'d call. argv is the current argument list
 | |
|        undergoing xWrap() argument conversion. argv entries to the
 | |
|        left of argIndex will have already undergone transformation and
 | |
|        those to the right will not have (they will have the values the
 | |
|        client-level code passed in, awaiting conversion). The RHS
 | |
|        indexes must never be relied upon for anything because their
 | |
|        types are indeterminate, whereas the LHS values will be
 | |
|        WASM-compatible values by the time this is called.
 | |
| 
 | |
|        The reason for the argv and argIndex arguments is that we
 | |
|        frequently need more context than v for a specific conversion,
 | |
|        and that context invariably lies in the LHS arguments of v.
 | |
|        Examples of how this is useful can be found in FuncPtrAdapter.
 | |
|     */
 | |
|     convertArg(v,argv,argIndex){
 | |
|       toss("AbstractArgAdapter must be subclassed.");
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      This type is recognized by xWrap() as a proxy for converting a JS
 | |
|      function to a C-side function, either permanently, for the
 | |
|      duration of a single call into the C layer, or semi-contextual,
 | |
|      where it may keep track of a single binding for a given context
 | |
|      and uninstall the binding if it's replaced.
 | |
| 
 | |
|      The constructor requires an options object with these properties:
 | |
| 
 | |
|      - name (optional): string describing the function binding. This
 | |
|        is solely for debugging and error-reporting purposes. If not
 | |
|        provided, an empty string is assumed.
 | |
| 
 | |
|      - signature: a function signature string compatible with
 | |
|        jsFuncToWasm().
 | |
| 
 | |
|      - bindScope (string): one of ('transient', 'context',
 | |
|        'singleton', 'permanent'). Bind scopes are:
 | |
| 
 | |
|        - 'transient': it will convert JS functions to WASM only for
 | |
|          the duration of the xWrap()'d function call, using
 | |
|          scopedInstallFunction(). Before that call returns, the
 | |
|          WASM-side binding will be uninstalled.
 | |
| 
 | |
|        - 'singleton': holds one function-pointer binding for this
 | |
|          instance. If it's called with a different function pointer,
 | |
|          it uninstalls the previous one after converting the new
 | |
|          value. This is only useful for use with "global" functions
 | |
|          which do not rely on any state other than this function
 | |
|          pointer. If the being-converted function pointer is intended
 | |
|          to be mapped to some sort of state object (e.g. an
 | |
|          `sqlite3*`) then "context" (see below) is the proper mode.
 | |
| 
 | |
|        - 'context': similar to singleton mode but for a given
 | |
|          "context", where the context is a key provided by the user
 | |
|          and possibly dependent on a small amount of call-time
 | |
|          context. This mode is the default if bindScope is _not_ set
 | |
|          but a property named contextKey (described below) is.
 | |
| 
 | |
|        - 'permanent': the function is installed and left there
 | |
|          forever. There is no way to recover its pointer address
 | |
|          later on for cleanup purposes. i.e. it effectively leaks.
 | |
| 
 | |
|      - callProxy (function): if set, this must be a function which
 | |
|        will act as a proxy for any "converted" JS function. It is
 | |
|        passed the being-converted function value and must return
 | |
|        either that function or a function which acts on its
 | |
|        behalf. The returned function will be the one which gets
 | |
|        installed into the WASM function table. The proxy must perform
 | |
|        any required argument conversion (it will be called from C
 | |
|        code, so will receive C-format arguments) before passing them
 | |
|        on to the being-converted function. Whether or not the proxy
 | |
|        itself must return a value depends on the context. If it does,
 | |
|        it must be a WASM-friendly value, as it will be returning from
 | |
|        a call made from WASM code.
 | |
| 
 | |
|      - contextKey (function): is only used if bindScope is 'context'
 | |
|        or if bindScope is not set and this function is, in which case
 | |
|        a bindScope of 'context' is assumed. This function gets bound
 | |
|        to this object, so its "this" is this object. It gets passed
 | |
|        (argv,argIndex), where argIndex is the index of _this_ function
 | |
|        in its _wrapping_ function's arguments, and argv is the
 | |
|        _current_ still-being-xWrap()-processed args array. (Got all
 | |
|        that?) When thisFunc(argv,argIndex) is called by xWrap(), all
 | |
|        arguments in argv to the left of argIndex will have been
 | |
|        processed by xWrap() by the time this is called. argv[argIndex]
 | |
|        will be the value the user passed in to the xWrap()'d function
 | |
|        for the argument this FuncPtrAdapter is mapped to. Arguments to
 | |
|        the right of argv[argIndex] will not yet have been converted
 | |
|        before this is called. The function must return a key which
 | |
|        uniquely identifies this function mapping context for _this_
 | |
|        FuncPtrAdapter instance (other instances are not considered),
 | |
|        taking into account that C functions often take some sort of
 | |
|        state object as one or more of their arguments. As an example,
 | |
|        if the xWrap()'d function takes `(int,T*,functionPtr,X*)` then
 | |
|        this FuncPtrAdapter instance is argv[2], and contextKey(argv,2)
 | |
|        might return 'T@'+argv[1], or even just argv[1].  Note,
 | |
|        however, that the (`X*`) argument will not yet have been
 | |
|        processed by the time this is called and should not be used as
 | |
|        part of that key because its pre-conversion data type might be
 | |
|        unpredictable. Similarly, care must be taken with C-string-type
 | |
|        arguments: those to the left in argv will, when this is called,
 | |
|        be WASM pointers, whereas those to the right might (and likely
 | |
|        do) have another data type. When using C-strings in keys, never
 | |
|        use their pointers in the key because most C-strings in this
 | |
|        constellation are transient. Conversely, the pointer address
 | |
|        makes an ideal key for longer-lived native pointer types.
 | |
| 
 | |
|      Yes, that ^^^ is quite awkward, but it's what we have. In
 | |
|      context, as it were, it actually makes some sense, but one must
 | |
|      look under its hook a bit to understand why it's shaped the
 | |
|      way it is.
 | |
| 
 | |
|      The constructor only saves the above state for later, and does
 | |
|      not actually bind any functions. The conversion, if any, is
 | |
|      performed when its convertArg() method is called via xWrap().
 | |
| 
 | |
|      Shortcomings:
 | |
| 
 | |
|      - These "reverse" bindings, i.e. calling into a JS-defined
 | |
|        function from a WASM-defined function (the generated proxy
 | |
|        wrapper), lack all type conversion support. That means, for
 | |
|        example, that...
 | |
| 
 | |
|      - Function pointers which include C-string arguments may still
 | |
|        need a level of hand-written wrappers around them, depending on
 | |
|        how they're used, in order to provide the client with JS
 | |
|        strings. Alternately, clients will need to perform such
 | |
|        conversions on their own, e.g. using cstrToJs(). The purpose of
 | |
|        the callProxy() method is to account for such cases.
 | |
|   */
 | |
|   xArg.FuncPtrAdapter = class FuncPtrAdapter extends AbstractArgAdapter {
 | |
|     constructor(opt) {
 | |
|       super(opt);
 | |
|       if(xArg.FuncPtrAdapter.warnOnUse){
 | |
|         console.warn('xArg.FuncPtrAdapter is an internal-only API',
 | |
|                      'and is not intended to be invoked from',
 | |
|                      'client-level code. Invoked with:',opt);
 | |
|       }
 | |
|       this.name = opt.name || "unnamed";
 | |
|       this.signature = opt.signature;
 | |
|       if(opt.contextKey instanceof Function){
 | |
|         this.contextKey = opt.contextKey;
 | |
|         if(!opt.bindScope) opt.bindScope = 'context';
 | |
|       }
 | |
|       this.bindScope = opt.bindScope
 | |
|         || toss("FuncPtrAdapter options requires a bindScope (explicit or implied).");
 | |
|       if(FuncPtrAdapter.bindScopes.indexOf(opt.bindScope)<0){
 | |
|         toss("Invalid options.bindScope ("+opt.bindMod+") for FuncPtrAdapter. "+
 | |
|              "Expecting one of: ("+FuncPtrAdapter.bindScopes.join(', ')+')');
 | |
|       }
 | |
|       this.isTransient = 'transient'===this.bindScope;
 | |
|       this.isContext = 'context'===this.bindScope;
 | |
|       this.isPermanent = 'permanent'===this.bindScope;
 | |
|       this.singleton = ('singleton'===this.bindScope) ? [] : undefined;
 | |
|       //console.warn("FuncPtrAdapter()",opt,this);
 | |
|       this.callProxy = (opt.callProxy instanceof Function)
 | |
|         ? opt.callProxy : undefined;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|        The static class members are defined outside of the class to
 | |
|        work around an emcc toolchain build problem: one of the tools
 | |
|        in emsdk v3.1.42 does not support the static keyword.
 | |
|     */
 | |
| 
 | |
|     /* Dummy impl. Overwritten per-instance as needed. */
 | |
|     contextKey(argv,argIndex){
 | |
|       return this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|        Returns this object's mapping for the given context key, in the
 | |
|        form of an an array, creating the mapping if needed. The key
 | |
|        may be anything suitable for use in a Map.
 | |
| 
 | |
|        The returned array is intended to be used as a pair of
 | |
|        [JSValue, WasmFuncPtr], where the first element is one passed
 | |
|        to this.convertArg() and the second is its WASM form.
 | |
|     */
 | |
|     contextMap(key){
 | |
|       const cm = (this.__cmap || (this.__cmap = new Map));
 | |
|       let rc = cm.get(key);
 | |
|       if(undefined===rc) cm.set(key, (rc = []));
 | |
|       return rc;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|        Gets called via xWrap() to "convert" v to a WASM-bound function
 | |
|        pointer. If v is one of (a WASM pointer, null, undefined) then
 | |
|        (v||0) is returned and any earlier function installed by this
 | |
|        mapping _might_, depending on how it's bound, be uninstalled.
 | |
|        If v is not one of those types, it must be a Function, for
 | |
|        which this method creates (if needed) a WASM function binding
 | |
|        and returns the WASM pointer to that binding.
 | |
| 
 | |
|        If this instance is not in 'transient' mode, it will remember
 | |
|        the binding for at least the next call, to avoid recreating the
 | |
|        function binding unnecessarily.
 | |
| 
 | |
|        If it's passed a pointer(ish) value for v, it assumes it's a
 | |
|        WASM function pointer and does _not_ perform any function
 | |
|        binding, so this object's bindMode is irrelevant/ignored for
 | |
|        such cases.
 | |
| 
 | |
|        See the parent class's convertArg() docs for details on what
 | |
|        exactly the 2nd and 3rd arguments are.
 | |
|     */
 | |
|     convertArg(v,argv,argIndex){
 | |
|       let pair = this.singleton;
 | |
|       if(!pair && this.isContext){
 | |
|         pair = this.contextMap(this.contextKey(argv,argIndex));
 | |
|         //FuncPtrAdapter.debugOut(this.name, this.signature, "contextKey() =",this.contextKey(argv,argIndex), pair);
 | |
|       }
 | |
|       if( 0 ){
 | |
|         FuncPtrAdapter.debugOut("FuncPtrAdapter.convertArg()",this.name,
 | |
|                                 'signature =',this.signature,
 | |
|                                 'transient ?=',this.transient,
 | |
|                                 'pair =',pair,
 | |
|                                 'v =',v);
 | |
|       }
 | |
|       if(pair && 2===pair.length && pair[0]===v){
 | |
|         /* We have already handled this function. */
 | |
|         return pair[1];
 | |
|       }
 | |
|       if(v instanceof Function){
 | |
|         /* Install a WASM binding and return its pointer. */
 | |
|         //FuncPtrAdapter.debugOut("FuncPtrAdapter.convertArg()",this.name,this.signature,this.transient,v,pair);
 | |
|         if(this.callProxy){
 | |
|           v = this.callProxy(v);
 | |
|         }
 | |
|         const fp = __installFunction(v, this.signature, this.isTransient);
 | |
|         if(FuncPtrAdapter.debugFuncInstall){
 | |
|           FuncPtrAdapter.debugOut("FuncPtrAdapter installed", this,
 | |
|                                   this.contextKey(argv,argIndex), '@'+fp, v);
 | |
|         }
 | |
|         if(pair){
 | |
|           /* Replace existing stashed mapping */
 | |
|           if(pair[1]){
 | |
|             if(FuncPtrAdapter.debugFuncInstall){
 | |
|               FuncPtrAdapter.debugOut("FuncPtrAdapter uninstalling", this,
 | |
|                                       this.contextKey(argv,argIndex), '@'+pair[1], v);
 | |
|             }
 | |
|             try{
 | |
|               /* Because the pending native call might rely on the
 | |
|                  pointer we're replacing, e.g. as is normally the case
 | |
|                  with sqlite3's xDestroy() methods, we don't
 | |
|                  immediately uninstall but instead add its pointer to
 | |
|                  the scopedAlloc stack, which will be cleared when the
 | |
|                  xWrap() mechanism is done calling the native
 | |
|                  function. We're relying very much here on xWrap()
 | |
|                  having pushed an alloc scope.
 | |
|               */
 | |
|               cache.scopedAlloc.pushPtr(pair[1]);
 | |
|             }
 | |
|             catch(e){/*ignored*/}
 | |
|           }
 | |
|           pair[0] = arguments[0] || __NullPtr/*the original v*/;
 | |
|           pair[1] = fp;
 | |
|         }
 | |
|         return fp;
 | |
|       }else if(target.isPtr(v) || null===v || undefined===v){
 | |
|         if(pair && pair[1] && pair[1]!==v){
 | |
|           /* uninstall stashed mapping and replace stashed mapping with v. */
 | |
|           if(FuncPtrAdapter.debugFuncInstall){
 | |
|             FuncPtrAdapter.debugOut("FuncPtrAdapter uninstalling", this,
 | |
|                                     this.contextKey(argv,argIndex), '@'+pair[1], v);
 | |
|           }
 | |
|           try{cache.scopedAlloc.pushPtr(pair[1]);/*see notes above*/}
 | |
|           catch(e){/*ignored*/}
 | |
|           pair[0] = pair[1] = (v || __NullPtr);
 | |
|         }
 | |
|         return v || __NullPtr;
 | |
|       }else{
 | |
|         throw new TypeError("Invalid FuncPtrAdapter argument type. "+
 | |
|                             "Expecting a function pointer or a "+
 | |
|                             (this.name ? this.name+' ' : '')+
 | |
|                             "function matching signature "+
 | |
|                             this.signature+".");
 | |
|       }
 | |
|     }/*convertArg()*/
 | |
|   }/*FuncPtrAdapter*/;
 | |
| 
 | |
|   /** If true, the constructor emits a warning. The intent is that
 | |
|       this be set to true after bootstrapping of the higher-level
 | |
|       client library is complete, to warn downstream clients that
 | |
|       they shouldn't be relying on this implementation detail which
 | |
|       does not have a stable interface. */
 | |
|   xArg.FuncPtrAdapter.warnOnUse = false;
 | |
| 
 | |
|   /** If true, convertArg() will call FuncPtrAdapter.debugOut() when
 | |
|       it (un)installs a function binding to/from WASM. Note that
 | |
|       deinstallation of bindScope=transient bindings happens via
 | |
|       scopedAllocPop() so will not be output. */
 | |
|   xArg.FuncPtrAdapter.debugFuncInstall = false;
 | |
| 
 | |
|   /** Function used for debug output. */
 | |
|   xArg.FuncPtrAdapter.debugOut = console.debug.bind(console);
 | |
| 
 | |
|   /**
 | |
|      List of legal values for the FuncPtrAdapter bindScope config
 | |
|      option.
 | |
|   */
 | |
|   xArg.FuncPtrAdapter.bindScopes = [
 | |
|     'transient', 'context', 'singleton', 'permanent'
 | |
|   ];
 | |
| 
 | |
|   /** Throws if xArg.get(t) returns falsy. */
 | |
|   const __xArgAdapterCheck =
 | |
|         (t)=>xArg.get(t) || toss("Argument adapter not found:",t);
 | |
| 
 | |
|   /** Throws if xResult.get(t) returns falsy. */
 | |
|   const __xResultAdapterCheck =
 | |
|         (t)=>xResult.get(t) || toss("Result adapter not found:",t);
 | |
| 
 | |
|   /**
 | |
|      Fetches the xWrap() argument adapter mapped to t, calls it,
 | |
|      passing in all remaining arguments, and returns the result.
 | |
|      Throws if t is not mapped to an argument converter.
 | |
|   */
 | |
|   cache.xWrap.convertArg = (t,...args)=>__xArgAdapterCheck(t)(...args);
 | |
|   /**
 | |
|      Identical to convertArg() except that it does not perform
 | |
|      an is-defined check on the mapping to t before invoking it.
 | |
|   */
 | |
|   cache.xWrap.convertArgNoCheck = (t,...args)=>xArg.get(t)(...args);
 | |
| 
 | |
|   /**
 | |
|      Fetches the xWrap() result adapter mapped to t, calls it, passing
 | |
|      it v, and returns the result.  Throws if t is not mapped to an
 | |
|      argument converter.
 | |
|   */
 | |
|   cache.xWrap.convertResult =
 | |
|     (t,v)=>(null===t ? v : (t ? __xResultAdapterCheck(t)(v) : undefined));
 | |
|   /**
 | |
|      Identical to convertResult() except that it does not perform an
 | |
|      is-defined check on the mapping to t before invoking it.
 | |
|   */
 | |
|   cache.xWrap.convertResultNoCheck =
 | |
|     (t,v)=>(null===t ? v : (t ? xResult.get(t)(v) : undefined));
 | |
| 
 | |
|   /**
 | |
|      Creates a wrapper for another function which converts the arguments
 | |
|      of the wrapper to argument types accepted by the wrapped function,
 | |
|      then converts the wrapped function's result to another form
 | |
|      for the wrapper.
 | |
| 
 | |
|      The first argument must be one of:
 | |
| 
 | |
|      - A JavaScript function.
 | |
|      - The name of a WASM-exported function. xGet() is used to fetch
 | |
|        the exported function, which throws if it's not found.
 | |
|      - A pointer into the indirect function table. e.g. a pointer
 | |
|        returned from target.installFunction().
 | |
| 
 | |
|      It returns either the passed-in function or a wrapper for that
 | |
|      function which converts the JS-side argument types into WASM-side
 | |
|      types and converts the result type.
 | |
| 
 | |
|      The second argument, `resultType`, describes the conversion for
 | |
|      the wrapped functions result. A literal `null` or the string
 | |
|      `'null'` both mean to return the original function's value as-is
 | |
|      (mnemonic: there is "null" conversion going on). Literal
 | |
|      `undefined` or the string `"void"` both mean to ignore the
 | |
|      function's result and return `undefined`. Aside from those two
 | |
|      special cases, the `resultType` value may be one of the values
 | |
|      described below or any mapping installed by the client using
 | |
|      xWrap.resultAdapter().
 | |
| 
 | |
|      If passed 3 arguments and the final one is an array, that array
 | |
|      must contain a list of type names (see below) for adapting the
 | |
|      arguments from JS to WASM.  If passed 2 arguments, more than 3,
 | |
|      or the 3rd is not an array, all arguments after the 2nd (if any)
 | |
|      are treated as type names. i.e.:
 | |
| 
 | |
|      ```
 | |
|      xWrap('funcname', 'i32', 'string', 'f64');
 | |
|      // is equivalent to:
 | |
|      xWrap('funcname', 'i32', ['string', 'f64']);
 | |
|      ```
 | |
| 
 | |
|      This function enforces that the given list of arguments has the
 | |
|      same arity as the being-wrapped function (as defined by its
 | |
|      `length` property) and it will throw if that is not the case.
 | |
|      Similarly, the created wrapper will throw if passed a differing
 | |
|      argument count. The intent of that strictness is to help catch
 | |
|      coding errors in using JS-bound WASM functions earlier rather
 | |
|      than laer.
 | |
| 
 | |
|      Type names are symbolic names which map the arguments to an
 | |
|      adapter function to convert, if needed, the value before passing
 | |
|      it on to WASM or to convert a return result from WASM. The list
 | |
|      of pre-defined names:
 | |
| 
 | |
|      - `i8`, `i16`, `i32` (args and results): all integer conversions
 | |
|        which convert their argument to an integer and truncate it to
 | |
|        the given bit length.
 | |
| 
 | |
|      - `*`, `**`, and `pointer` (args): are assumed to be WASM pointer
 | |
|        values and are returned coerced to an appropriately-sized
 | |
|        pointer value (i32 or i64). Non-numeric values will coerce to 0
 | |
|        and out-of-range values will have undefined results (just as
 | |
|        with any pointer misuse).
 | |
| 
 | |
|      - `*` and `pointer` (results): aliases for the current
 | |
|        WASM pointer numeric type.
 | |
| 
 | |
|      - `**` (args): is simply a descriptive alias for the WASM pointer
 | |
|        type. It's primarily intended to mark output-pointer arguments,
 | |
|        noting that JS's view of WASM does not distinguish between
 | |
|        pointers and pointers-to-pointers, so all such interpretation
 | |
|        of `**`, as distinct from `*`, necessarily happens at the
 | |
|        client level.
 | |
| 
 | |
|      - `NumType*` (args): a type name in this form, where T is
 | |
|        the name of a numeric mapping, e.g. 'int16' or 'double',
 | |
|        is treated like `*`.
 | |
| 
 | |
|      - `i64` (args and results): passes the value to BigInt() to
 | |
|        convert it to an int64. This conversion will if bigIntEnabled
 | |
|        is falsy.
 | |
| 
 | |
|      - `f32` (`float`), `f64` (`double`) (args and results): pass
 | |
|        their argument to Number(). i.e. the adapter does not currently
 | |
|        distinguish between the two types of floating-point numbers.
 | |
| 
 | |
|      - `number` (results): converts the result to a JS Number using
 | |
|        Number(theValue). This is for result conversions only, as it's
 | |
|        not possible to generically know which type of number to
 | |
|        convert arguments to.
 | |
| 
 | |
|      Non-numeric conversions include:
 | |
| 
 | |
|      - `null` literal or `"null"` string (args and results): perform
 | |
|        no translation and pass the arg on as-is. This is primarily
 | |
|        useful for results but may have a use or two for arguments.
 | |
| 
 | |
|      - `string` or `utf8` (args): has two different semantics in order
 | |
|        to accommodate various uses of certain C APIs
 | |
|        (e.g. output-style strings)...
 | |
| 
 | |
|        - If the arg is a JS string, it creates a _temporary_
 | |
|          UTF-8-encoded C-string to pass to the exported function,
 | |
|          cleaning it up before the wrapper returns. If a long-lived
 | |
|          C-string pointer is required, that requires client-side code
 | |
|          to create the string then pass its pointer to the function.
 | |
| 
 | |
|        - Else the arg is assumed to be a pointer to a string the
 | |
|          client has already allocated and it's passed on as
 | |
|          a WASM pointer.
 | |
| 
 | |
|      - `string` or `utf8` (results): treats the result value as a
 | |
|        const C-string, encoded as UTF-8, copies it to a JS string,
 | |
|        and returns that JS string.
 | |
| 
 | |
|      - `string:dealloc` or `utf8:dealloc` (results): treats the result
 | |
|        value as a non-const UTF-8 C-string, ownership of which has
 | |
|        just been transfered to the caller. It copies the C-string to a
 | |
|        JS string, frees the C-string using dealloc(), and returns the
 | |
|        JS string. If such a result value is NULL, the JS result is
 | |
|        `null`. Achtung: when using an API which returns results from a
 | |
|        specific allocator, e.g. `my_malloc()`, this conversion _is not
 | |
|        legal_. Instead, an equivalent conversion which uses the
 | |
|        appropriate deallocator is required. For example:
 | |
| 
 | |
| ```js
 | |
|    target.xWrap.resultAdapter('string:my_free',(i)=>{
 | |
|      try { return i ? target.cstrToJs(i) : null; }
 | |
|      finally{ target.exports.my_free(i); }
 | |
|    };
 | |
| ```
 | |
| 
 | |
|      - `json` (results): treats the result as a const C-string and
 | |
|        returns the result of passing the converted-to-JS string to
 | |
|        JSON.parse(). Returns `null` if the C-string is a NULL pointer.
 | |
| 
 | |
|      - `json:dealloc` (results): works exactly like `string:dealloc` but
 | |
|        returns the same thing as the `json` adapter. Note the
 | |
|        warning in `string:dealloc` regarding matching allocators and
 | |
|        deallocators.
 | |
| 
 | |
|      The type names for results and arguments are validated when
 | |
|      xWrap() is called and any unknown names will trigger an
 | |
|      exception.
 | |
| 
 | |
|      Clients may map their own result and argument adapters using
 | |
|      xWrap.resultAdapter() and xWrap.argAdapter(), noting that not all
 | |
|      type conversions are valid for both arguments _and_ result types
 | |
|      as they often have different memory ownership requirements.
 | |
| 
 | |
|      Design note: the ability to pass in a JS function as the first
 | |
|      argument is of relatively limited use, primarily for testing
 | |
|      argument and result converters. JS functions, by and large, will
 | |
|      not want to deal with C-type arguments.
 | |
| 
 | |
|      TODOs:
 | |
| 
 | |
|      - Figure out how/whether we can (semi-)transparently handle
 | |
|        pointer-type _output_ arguments. Those currently require
 | |
|        explicit handling by allocating pointers, assigning them before
 | |
|        the call using poke(), and fetching them with
 | |
|        peek() after the call. We may be able to automate some
 | |
|        or all of that.
 | |
| 
 | |
|      - Figure out whether it makes sense to extend the arg adapter
 | |
|        interface such that each arg adapter gets an array containing
 | |
|        the results of the previous arguments in the current call. That
 | |
|        might allow some interesting type-conversion feature. Use case:
 | |
|        handling of the final argument to sqlite3_prepare_v2() depends
 | |
|        on the type (pointer vs JS string) of its 2nd
 | |
|        argument. Currently that distinction requires hand-writing a
 | |
|        wrapper for that function. That case is unusual enough that
 | |
|        abstracting it into this API (and taking on the associated
 | |
|        costs) may well not make good sense.
 | |
|   */
 | |
|   target.xWrap = function callee(fArg, resultType, ...argTypes){
 | |
|     if(3===arguments.length && Array.isArray(arguments[2])){
 | |
|       argTypes = arguments[2];
 | |
|     }
 | |
|     if(target.isPtr(fArg)){
 | |
|       fArg = target.functionEntry(fArg)
 | |
|         || toss("Function pointer not found in WASM function table.");
 | |
|     }
 | |
|     const fIsFunc = (fArg instanceof Function);
 | |
|     const xf = fIsFunc ? fArg : target.xGet(fArg);
 | |
|     if(fIsFunc) fArg = xf.name || 'unnamed function';
 | |
|     if(argTypes.length!==xf.length) __argcMismatch(fArg, xf.length);
 | |
|     if( 0===xf.length
 | |
|         && (null===resultType || 'null'===resultType) ){
 | |
|       /* Func taking no args with an as-is return. We don't need a wrapper. */
 | |
|       return xf;
 | |
|     }
 | |
|     /*Verify the arg type conversions are valid...*/;
 | |
|     __xResultAdapterCheck(resultType);
 | |
|     for(const t of argTypes){
 | |
|       if(t instanceof AbstractArgAdapter) xArg.set(t, (...args)=>t.convertArg(...args));
 | |
|       else __xArgAdapterCheck(t);
 | |
|     }
 | |
|     const cxw = cache.xWrap;
 | |
|     if(0===xf.length){
 | |
|       // No args to convert, so we can create a simpler wrapper...
 | |
|       return (...args)=>(args.length
 | |
|                          ? __argcMismatch(fArg, xf.length)
 | |
|                          : cxw.convertResult(resultType, xf.call(null)));
 | |
|     }
 | |
|     return function(...args){
 | |
|       if(args.length!==xf.length) __argcMismatch(fArg, xf.length);
 | |
|       const scope = target.scopedAllocPush();
 | |
|       try{
 | |
|         /*
 | |
|           Maintenance reminder re. arguments passed to convertArg():
 | |
|           The public interface of argument adapters is that they take
 | |
|           ONE argument and return a (possibly) converted result for
 | |
|           it. The passing-on of arguments after the first is an
 | |
|           internal implementation detail for the sake of
 | |
|           AbstractArgAdapter, and not to be relied on or documented
 | |
|           for other cases. The fact that this is how
 | |
|           AbstractArgAdapter.convertArgs() gets its 2nd+ arguments,
 | |
|           and how FuncPtrAdapter.contextKey() gets its args, is also
 | |
|           an implementation detail and subject to change. i.e. the
 | |
|           public interface of 1 argument is stable.  The fact that any
 | |
|           arguments may be passed in after that one, and what those
 | |
|           arguments are, is _not_ part of the public interface and is
 | |
|           _not_ stable.
 | |
| 
 | |
|           Maintenance reminder: the Ember framework modifies the core
 | |
|           Array type, breaking for-in loops:
 | |
| 
 | |
|           https://sqlite.org/forum/forumpost/b549992634b55104
 | |
|         */
 | |
|         let i = 0;
 | |
|         if( callee.debug ){
 | |
|           console.debug("xWrap() preparing: resultType ",resultType, 'xf',xf,"argTypes",argTypes,"args",args);
 | |
|         }
 | |
|         for(; i < args.length; ++i) args[i] = cxw.convertArgNoCheck(
 | |
|           argTypes[i], args[i], args, i
 | |
|         );
 | |
|         if( callee.debug ){
 | |
|           console.debug("xWrap() calling: resultType ",resultType, 'xf',xf,"argTypes",argTypes,"args",args);
 | |
|         }
 | |
|         return cxw.convertResultNoCheck(resultType, xf.apply(null,args));
 | |
|       }finally{
 | |
|         target.scopedAllocPop(scope);
 | |
|       }
 | |
|     };
 | |
|   }/*xWrap()*/;
 | |
| 
 | |
|   /**
 | |
|      Internal impl for xWrap.resultAdapter() and argAdapter().
 | |
| 
 | |
|      func = one of xWrap.resultAdapter or xWrap.argAdapter.
 | |
| 
 | |
|      argc = the number of args in the wrapping call to this
 | |
|      function.
 | |
| 
 | |
|      typeName = the first arg to the wrapping function.
 | |
| 
 | |
|      adapter = the second arg to the wrapping function.
 | |
| 
 | |
|      modeName = a descriptive name of the wrapping function for
 | |
|      error-reporting purposes.
 | |
| 
 | |
|      xcvPart = one of xResult or xArg.
 | |
| 
 | |
|      This acts as either a getter (if 1===argc) or setter (if
 | |
|      2===argc) for the given adapter. Returns func on success or
 | |
|      throws if (A) called with 2 args but adapter is-not-a Function or
 | |
|      (B) typeName is not a string or (C) argc is not one of (1, 2).
 | |
|   */
 | |
|   const __xAdapter = function(func, argc, typeName, adapter, modeName, xcvPart){
 | |
|     if('string'===typeof typeName){
 | |
|       if(1===argc) return xcvPart.get(typeName);
 | |
|       else if(2===argc){
 | |
|         if(!adapter){
 | |
|           xcvPart.delete(typeName);
 | |
|           return func;
 | |
|         }else if(!(adapter instanceof Function)){
 | |
|           toss(modeName,"requires a function argument.");
 | |
|         }
 | |
|         xcvPart.set(typeName, adapter);
 | |
|         return func;
 | |
|       }
 | |
|     }
 | |
|     toss("Invalid arguments to",modeName);
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Gets, sets, or removes a result value adapter for use with
 | |
|      xWrap(). If passed only 1 argument, the adapter function for the
 | |
|      given type name is returned.  If the second argument is explicit
 | |
|      falsy (as opposed to defaulted), the adapter named by the first
 | |
|      argument is removed. If the 2nd argument is not falsy, it must be
 | |
|      a function which takes one value and returns a value appropriate
 | |
|      for the given type name. The adapter may throw if its argument is
 | |
|      not of a type it can work with. This function throws for invalid
 | |
|      arguments.
 | |
| 
 | |
|      Example:
 | |
| 
 | |
|      ```
 | |
|      xWrap.resultAdapter('twice',(v)=>v+v);
 | |
|      ```
 | |
| 
 | |
|      Result adapters MUST NOT use the scopedAlloc() family of APIs to
 | |
|      allocate a result value. xWrap()-generated wrappers run in the
 | |
|      context of scopedAllocPush() so that argument adapters can easily
 | |
|      convert, e.g., to C-strings, and have them cleaned up
 | |
|      automatically before the wrapper returns to the caller. Likewise,
 | |
|      if a _result_ adapter uses scoped allocation, the result will be
 | |
|      freed before the wrapper returns, leading to chaos and undefined
 | |
|      behavior.
 | |
| 
 | |
|      When called as a setter, this function returns itself.
 | |
|   */
 | |
|   target.xWrap.resultAdapter = function f(typeName, adapter){
 | |
|     return __xAdapter(f, arguments.length, typeName, adapter,
 | |
|                       'resultAdapter()', xResult);
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      Functions identically to xWrap.resultAdapter() but applies to
 | |
|      call argument conversions instead of result value conversions.
 | |
| 
 | |
|      xWrap()-generated wrappers perform argument conversion in the
 | |
|      context of a scopedAllocPush(), so any memory allocation
 | |
|      performed by argument adapters really, really, really should be
 | |
|      made using the scopedAlloc() family of functions unless
 | |
|      specifically necessary. For example:
 | |
| 
 | |
|      ```
 | |
|      xWrap.argAdapter('my-string', function(v){
 | |
|        return ('string'===typeof v)
 | |
|          ? myWasmObj.scopedAllocCString(v) : null;
 | |
|      };
 | |
|      ```
 | |
| 
 | |
|      Contrariwise, _result_ adapters _must not_ use scopedAlloc() to
 | |
|      allocate results because they would be freed before the
 | |
|      xWrap()-created wrapper returns.
 | |
| 
 | |
|      It is perfectly legitimate to use these adapters to perform
 | |
|      argument validation, as opposed (or in addition) to conversion.
 | |
|      When used that way, they should throw for invalid arguments.
 | |
|   */
 | |
|   target.xWrap.argAdapter = function f(typeName, adapter){
 | |
|     return __xAdapter(f, arguments.length, typeName, adapter,
 | |
|                       'argAdapter()', xArg);
 | |
|   };
 | |
| 
 | |
|   target.xWrap.FuncPtrAdapter = xArg.FuncPtrAdapter;
 | |
| 
 | |
|   /**
 | |
|      Functions like xCall() but performs argument and result type
 | |
|      conversions as for xWrap(). The first, second, and third
 | |
|      arguments are as documented for xWrap(), except that the 3rd
 | |
|      argument may be either a falsy value or empty array to represent
 | |
|      nullary functions. The 4th+ arguments are arguments for the call,
 | |
|      with the special case that if the 4th argument is an array, it is
 | |
|      used as the arguments for the call. Returns the converted result
 | |
|      of the call.
 | |
| 
 | |
|      This is just a thin wrapper around xWrap(). If the given function
 | |
|      is to be called more than once, it's more efficient to use
 | |
|      xWrap() to create a wrapper, then to call that wrapper as many
 | |
|      times as needed. For one-shot calls, however, this variant is
 | |
|      simpler.
 | |
|   */
 | |
|   target.xCallWrapped = function(fArg, resultType, argTypes, ...args){
 | |
|     if(Array.isArray(arguments[3])) args = arguments[3];
 | |
|     return target.xWrap(fArg, resultType, argTypes||[]).apply(null, args||[]);
 | |
|   };
 | |
| 
 | |
|   /**
 | |
|      This function is ONLY exposed in the public API to facilitate
 | |
|      testing. It should not be used in application-level code, only
 | |
|      in test code.
 | |
| 
 | |
|      Expects to be given (typeName, value) and returns a conversion
 | |
|      of that value as has been registered using argAdapter().
 | |
|      It throws if no adapter is found.
 | |
| 
 | |
|      ACHTUNG: the adapter may require that a scopedAllocPush() is
 | |
|      active and it may allocate memory within that scope. It may also
 | |
|      require additional arguments, depending on the type of
 | |
|      conversion.
 | |
|   */
 | |
|   target.xWrap.testConvertArg = cache.xWrap.convertArg;
 | |
| 
 | |
|   /**
 | |
|      This function is ONLY exposed in the public API to facilitate
 | |
|      testing. It should not be used in application-level code, only
 | |
|      in test code.
 | |
| 
 | |
|      Expects to be given (typeName, value) and returns a conversion
 | |
|      of that value as has been registered using resultAdapter().
 | |
|      It throws if no adapter is found.
 | |
| 
 | |
|      ACHTUNG: the adapter may allocate memory which the caller may need
 | |
|      to know how to free.
 | |
|   */
 | |
|   target.xWrap.testConvertResult = cache.xWrap.convertResult;
 | |
| 
 | |
|   return target;
 | |
| };
 | |
| 
 | |
| /**
 | |
|    yawl (Yet Another Wasm Loader) provides very basic wasm loader.
 | |
|    It requires a config object:
 | |
| 
 | |
|    - `uri`: required URI of the WASM file to load.
 | |
| 
 | |
|    - `onload(loadResult)`: optional callback. Its argument is an
 | |
|      object described in more detail below.
 | |
| 
 | |
|    - `imports`: optional imports object for
 | |
|      WebAssembly.instantiate[Streaming]().  The default is an empty
 | |
|      set of imports. If the module requires any imports, this object
 | |
|      must include them.
 | |
| 
 | |
|    - `wasmUtilTarget`: optional object suitable for passing to
 | |
|      WhWasmUtilInstaller(). If set, it gets passed to that function
 | |
|      before the returned promise resolves. This function sets several
 | |
|      properties on it before passing it on to that function (which
 | |
|      sets many more):
 | |
| 
 | |
|      - `module`, `instance`: the properties from the
 | |
|        instantiate[Streaming]() result.
 | |
| 
 | |
|      - If `instance.exports.memory` is _not_ set then it requires that
 | |
|        `config.imports.env.memory` be set (else it throws), and
 | |
|        assigns that to `wasmUtilTarget.memory`.
 | |
| 
 | |
|      - If `wasmUtilTarget.alloc` is not set and
 | |
|        `instance.exports.malloc` is, it installs
 | |
|        `wasmUtilTarget.alloc()` and `wasmUtilTarget.dealloc()`
 | |
|        wrappers for the exports `malloc` and `free` functions.
 | |
| 
 | |
|    It returns a function which, when called, initiates loading of the
 | |
|    module and returns a Promise. When that Promise resolves, it calls
 | |
|    the `config.onload` callback (if set) and passes it `(loadResult)`,
 | |
|    where `loadResult` is derived from the result of
 | |
|    WebAssembly.instantiate[Streaming](), an object in the form:
 | |
| 
 | |
|    ```
 | |
|    {
 | |
|      module: a WebAssembly.Module,
 | |
|      instance: a WebAssembly.Instance,
 | |
|      config: the config arg to this function
 | |
|    }
 | |
|    ```
 | |
| 
 | |
|    (The initial `then()` attached to the promise gets only that
 | |
|    object, and not the `config` object, thus the potential need for a
 | |
|    `config.onload` handler.)
 | |
| 
 | |
|    Error handling is up to the caller, who may attach a `catch()` call
 | |
|    to the promise.
 | |
| */
 | |
| globalThis.WhWasmUtilInstaller.yawl = function(config){
 | |
|   const wfetch = ()=>fetch(config.uri, {credentials: 'same-origin'});
 | |
|   const wui = this;
 | |
|   const finalThen = function(arg){
 | |
|     //log("finalThen()",arg);
 | |
|     if(config.wasmUtilTarget){
 | |
|       const toss = (...args)=>{throw new Error(args.join(' '))};
 | |
|       const tgt = config.wasmUtilTarget;
 | |
|       tgt.module = arg.module;
 | |
|       tgt.instance = arg.instance;
 | |
|       //tgt.exports = tgt.instance.exports;
 | |
|       if(!tgt.instance.exports.memory){
 | |
|         /**
 | |
|            WhWasmUtilInstaller requires either tgt.exports.memory
 | |
|            (exported from WASM) or tgt.memory (JS-provided memory
 | |
|            imported into WASM).
 | |
|         */
 | |
|         tgt.memory = config?.imports?.env?.memory
 | |
|           || toss("Missing 'memory' object!");
 | |
|       }
 | |
|       if(!tgt.alloc && arg.instance.exports.malloc){
 | |
|         const exports = arg.instance.exports;
 | |
|         tgt.alloc = function(n){
 | |
|           return exports.malloc(n) || toss("Allocation of",n,"bytes failed.");
 | |
|         };
 | |
|         tgt.dealloc = function(m){exports.free(m)};
 | |
|       }
 | |
|       wui(tgt);
 | |
|     }
 | |
|     arg.config = config;
 | |
|     if(config.onload) config.onload(arg);
 | |
|     return arg /* for any then() handler attached to
 | |
|                   yetAnotherWasmLoader()'s return value */;
 | |
|   };
 | |
|   const loadWasm = WebAssembly.instantiateStreaming
 | |
|         ? ()=>WebAssembly
 | |
|         .instantiateStreaming(wfetch(), config.imports||{})
 | |
|         .then(finalThen)
 | |
|         : ()=> wfetch()// Safari < v15
 | |
|         .then(response => response.arrayBuffer())
 | |
|         .then(bytes => WebAssembly.instantiate(bytes, config.imports||{}))
 | |
|         .then(finalThen)
 | |
|   ;
 | |
|   return loadWasm;
 | |
| }.bind(globalThis.WhWasmUtilInstaller)/*yawl()*/;
 |