| // Copyright (c) 2020, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| // This file has been automatically generated. Please do not edit it manually. |
| // To regenerate the file, use the following command |
| // "generate_ffi_boilerplate.py". |
| |
| import 'dart:async'; |
| import 'dart:convert'; |
| import 'dart:ffi'; |
| import 'dart:io'; |
| import 'dart:typed_data'; |
| import 'package:ffi/ffi.dart'; |
| import 'package:path/path.dart' as path; |
| import 'wasmer_api.dart'; |
| |
| class WasmImportDescriptor { |
| int kind; |
| String moduleName; |
| String name; |
| Pointer<WasmerFunctype> funcType; |
| WasmImportDescriptor(this.kind, this.moduleName, this.name, this.funcType); |
| |
| String toString() { |
| var kindName = wasmerExternKindName(kind); |
| if (kind == WasmerExternKindFunction) { |
| var runtime = WasmRuntime(); |
| var sig = WasmRuntime.getSignatureString("${moduleName}::${name}", |
| runtime.getArgTypes(funcType), runtime.getReturnType(funcType)); |
| return "$kindName: $sig"; |
| } else { |
| return "$kindName: ${moduleName}::${name}"; |
| } |
| } |
| } |
| |
| class WasmExportDescriptor { |
| int kind; |
| String name; |
| Pointer<WasmerFunctype> funcType; |
| WasmExportDescriptor(this.kind, this.name, this.funcType); |
| |
| String toString() { |
| var kindName = wasmerExternKindName(kind); |
| if (kind == WasmerExternKindFunction) { |
| var runtime = WasmRuntime(); |
| var sig = WasmRuntime.getSignatureString( |
| name, runtime.getArgTypes(funcType), runtime.getReturnType(funcType)); |
| return "$kindName: $sig"; |
| } else { |
| return "$kindName: ${name}"; |
| } |
| } |
| } |
| |
| class _WasmTrapsEntry { |
| dynamic exception; |
| _WasmTrapsEntry(this.exception); |
| } |
| |
| class WasmRuntime { |
| static WasmRuntime? _inst; |
| |
| DynamicLibrary _lib; |
| late Pointer<WasmerEngine> _engine; |
| Map<int, _WasmTrapsEntry> traps = {}; |
| late WasmerDartInitializeApiDLFn _Dart_InitializeApiDL; |
| late WasmerSetFinalizerForEngineFn _set_finalizer_for_engine; |
| late WasmerSetFinalizerForFuncFn _set_finalizer_for_func; |
| late WasmerSetFinalizerForInstanceFn _set_finalizer_for_instance; |
| late WasmerSetFinalizerForMemoryFn _set_finalizer_for_memory; |
| late WasmerSetFinalizerForMemorytypeFn _set_finalizer_for_memorytype; |
| late WasmerSetFinalizerForModuleFn _set_finalizer_for_module; |
| late WasmerSetFinalizerForStoreFn _set_finalizer_for_store; |
| late WasmerSetFinalizerForTrapFn _set_finalizer_for_trap; |
| late WasmerWasiConfigInheritStderrFn _wasi_config_inherit_stderr; |
| late WasmerWasiConfigInheritStdoutFn _wasi_config_inherit_stdout; |
| late WasmerWasiConfigNewFn _wasi_config_new; |
| late WasmerWasiEnvDeleteFn _wasi_env_delete; |
| late WasmerWasiEnvNewFn _wasi_env_new; |
| late WasmerWasiEnvReadStderrFn _wasi_env_read_stderr; |
| late WasmerWasiEnvReadStdoutFn _wasi_env_read_stdout; |
| late WasmerWasiEnvSetMemoryFn _wasi_env_set_memory; |
| late WasmerWasiGetImportsFn _wasi_get_imports; |
| late WasmerByteVecDeleteFn _byte_vec_delete; |
| late WasmerByteVecNewFn _byte_vec_new; |
| late WasmerByteVecNewEmptyFn _byte_vec_new_empty; |
| late WasmerByteVecNewUninitializedFn _byte_vec_new_uninitialized; |
| late WasmerEngineDeleteFn _engine_delete; |
| late WasmerEngineNewFn _engine_new; |
| late WasmerExporttypeNameFn _exporttype_name; |
| late WasmerExporttypeTypeFn _exporttype_type; |
| late WasmerExporttypeVecDeleteFn _exporttype_vec_delete; |
| late WasmerExporttypeVecNewFn _exporttype_vec_new; |
| late WasmerExporttypeVecNewEmptyFn _exporttype_vec_new_empty; |
| late WasmerExporttypeVecNewUninitializedFn _exporttype_vec_new_uninitialized; |
| late WasmerExternAsFuncFn _extern_as_func; |
| late WasmerExternAsMemoryFn _extern_as_memory; |
| late WasmerExternDeleteFn _extern_delete; |
| late WasmerExternKindFn _extern_kind; |
| late WasmerExternVecDeleteFn _extern_vec_delete; |
| late WasmerExternVecNewFn _extern_vec_new; |
| late WasmerExternVecNewEmptyFn _extern_vec_new_empty; |
| late WasmerExternVecNewUninitializedFn _extern_vec_new_uninitialized; |
| late WasmerExterntypeAsFunctypeFn _externtype_as_functype; |
| late WasmerExterntypeDeleteFn _externtype_delete; |
| late WasmerExterntypeKindFn _externtype_kind; |
| late WasmerFuncAsExternFn _func_as_extern; |
| late WasmerFuncCallFn _func_call; |
| late WasmerFuncDeleteFn _func_delete; |
| late WasmerFuncNewWithEnvFn _func_new_with_env; |
| late WasmerFunctypeDeleteFn _functype_delete; |
| late WasmerFunctypeParamsFn _functype_params; |
| late WasmerFunctypeResultsFn _functype_results; |
| late WasmerImporttypeModuleFn _importtype_module; |
| late WasmerImporttypeNameFn _importtype_name; |
| late WasmerImporttypeTypeFn _importtype_type; |
| late WasmerImporttypeVecDeleteFn _importtype_vec_delete; |
| late WasmerImporttypeVecNewFn _importtype_vec_new; |
| late WasmerImporttypeVecNewEmptyFn _importtype_vec_new_empty; |
| late WasmerImporttypeVecNewUninitializedFn _importtype_vec_new_uninitialized; |
| late WasmerInstanceDeleteFn _instance_delete; |
| late WasmerInstanceExportsFn _instance_exports; |
| late WasmerInstanceNewFn _instance_new; |
| late WasmerMemoryAsExternFn _memory_as_extern; |
| late WasmerMemoryDataFn _memory_data; |
| late WasmerMemoryDataSizeFn _memory_data_size; |
| late WasmerMemoryDeleteFn _memory_delete; |
| late WasmerMemoryGrowFn _memory_grow; |
| late WasmerMemoryNewFn _memory_new; |
| late WasmerMemorySizeFn _memory_size; |
| late WasmerMemorytypeDeleteFn _memorytype_delete; |
| late WasmerMemorytypeNewFn _memorytype_new; |
| late WasmerModuleDeleteFn _module_delete; |
| late WasmerModuleExportsFn _module_exports; |
| late WasmerModuleImportsFn _module_imports; |
| late WasmerModuleNewFn _module_new; |
| late WasmerStoreDeleteFn _store_delete; |
| late WasmerStoreNewFn _store_new; |
| late WasmerTrapDeleteFn _trap_delete; |
| late WasmerTrapMessageFn _trap_message; |
| late WasmerTrapNewFn _trap_new; |
| late WasmerValtypeDeleteFn _valtype_delete; |
| late WasmerValtypeKindFn _valtype_kind; |
| late WasmerValtypeVecDeleteFn _valtype_vec_delete; |
| late WasmerValtypeVecNewFn _valtype_vec_new; |
| late WasmerValtypeVecNewEmptyFn _valtype_vec_new_empty; |
| late WasmerValtypeVecNewUninitializedFn _valtype_vec_new_uninitialized; |
| late WasmerWasmerLastErrorLengthFn _wasmer_last_error_length; |
| late WasmerWasmerLastErrorMessageFn _wasmer_last_error_message; |
| |
| factory WasmRuntime() { |
| return _inst ??= WasmRuntime._init(); |
| } |
| |
| static String _getLibName() { |
| if (Platform.isMacOS) return "libwasmer.dylib"; |
| if (Platform.isLinux) return "libwasmer.so"; |
| // TODO(dartbug.com/37882): Support more platforms. |
| throw Exception("Wasm not currently supported on this platform"); |
| } |
| |
| static String _getLibDir() { |
| // The common case, and how cli_util.dart computes the Dart SDK directory, |
| // path.dirname called twice on Platform.resolvedExecutable. |
| var commonLibDir = path.join( |
| path.absolute(path.dirname(path.dirname(Platform.resolvedExecutable))), |
| 'bin', |
| 'third_party', |
| 'wasmer'); |
| if (Directory(commonLibDir).existsSync()) { |
| return commonLibDir; |
| } |
| |
| // This is the less common case where the user is in the checked out Dart |
| // SDK, and is executing dart via: |
| // ./out/ReleaseX64/dart ... |
| var checkedOutLibDir = path.join( |
| path.absolute(path.dirname(Platform.resolvedExecutable)), |
| 'dart-sdk', |
| 'bin', |
| 'third_party', |
| 'wasmer'); |
| if (Directory(checkedOutLibDir).existsSync()) { |
| return checkedOutLibDir; |
| } |
| |
| // If neither returned above, we return the common case: |
| return commonLibDir; |
| } |
| |
| WasmRuntime._init() |
| : _lib = DynamicLibrary.open(path.join(_getLibDir(), _getLibName())) { |
| _Dart_InitializeApiDL = _lib.lookupFunction< |
| NativeWasmerDartInitializeApiDLFn, |
| WasmerDartInitializeApiDLFn>('Dart_InitializeApiDL'); |
| _set_finalizer_for_engine = _lib.lookupFunction< |
| NativeWasmerSetFinalizerForEngineFn, |
| WasmerSetFinalizerForEngineFn>('set_finalizer_for_engine'); |
| _set_finalizer_for_func = _lib.lookupFunction< |
| NativeWasmerSetFinalizerForFuncFn, |
| WasmerSetFinalizerForFuncFn>('set_finalizer_for_func'); |
| _set_finalizer_for_instance = _lib.lookupFunction< |
| NativeWasmerSetFinalizerForInstanceFn, |
| WasmerSetFinalizerForInstanceFn>('set_finalizer_for_instance'); |
| _set_finalizer_for_memory = _lib.lookupFunction< |
| NativeWasmerSetFinalizerForMemoryFn, |
| WasmerSetFinalizerForMemoryFn>('set_finalizer_for_memory'); |
| _set_finalizer_for_memorytype = _lib.lookupFunction< |
| NativeWasmerSetFinalizerForMemorytypeFn, |
| WasmerSetFinalizerForMemorytypeFn>('set_finalizer_for_memorytype'); |
| _set_finalizer_for_module = _lib.lookupFunction< |
| NativeWasmerSetFinalizerForModuleFn, |
| WasmerSetFinalizerForModuleFn>('set_finalizer_for_module'); |
| _set_finalizer_for_store = _lib.lookupFunction< |
| NativeWasmerSetFinalizerForStoreFn, |
| WasmerSetFinalizerForStoreFn>('set_finalizer_for_store'); |
| _set_finalizer_for_trap = _lib.lookupFunction< |
| NativeWasmerSetFinalizerForTrapFn, |
| WasmerSetFinalizerForTrapFn>('set_finalizer_for_trap'); |
| _wasi_config_inherit_stderr = _lib.lookupFunction< |
| NativeWasmerWasiConfigInheritStderrFn, |
| WasmerWasiConfigInheritStderrFn>('wasi_config_inherit_stderr'); |
| _wasi_config_inherit_stdout = _lib.lookupFunction< |
| NativeWasmerWasiConfigInheritStdoutFn, |
| WasmerWasiConfigInheritStdoutFn>('wasi_config_inherit_stdout'); |
| _wasi_config_new = |
| _lib.lookupFunction<NativeWasmerWasiConfigNewFn, WasmerWasiConfigNewFn>( |
| 'wasi_config_new'); |
| _wasi_env_delete = |
| _lib.lookupFunction<NativeWasmerWasiEnvDeleteFn, WasmerWasiEnvDeleteFn>( |
| 'wasi_env_delete'); |
| _wasi_env_new = |
| _lib.lookupFunction<NativeWasmerWasiEnvNewFn, WasmerWasiEnvNewFn>( |
| 'wasi_env_new'); |
| _wasi_env_read_stderr = _lib.lookupFunction<NativeWasmerWasiEnvReadStderrFn, |
| WasmerWasiEnvReadStderrFn>('wasi_env_read_stderr'); |
| _wasi_env_read_stdout = _lib.lookupFunction<NativeWasmerWasiEnvReadStdoutFn, |
| WasmerWasiEnvReadStdoutFn>('wasi_env_read_stdout'); |
| _wasi_env_set_memory = _lib.lookupFunction<NativeWasmerWasiEnvSetMemoryFn, |
| WasmerWasiEnvSetMemoryFn>('wasi_env_set_memory'); |
| _wasi_get_imports = _lib.lookupFunction<NativeWasmerWasiGetImportsFn, |
| WasmerWasiGetImportsFn>('wasi_get_imports'); |
| _byte_vec_delete = |
| _lib.lookupFunction<NativeWasmerByteVecDeleteFn, WasmerByteVecDeleteFn>( |
| 'wasm_byte_vec_delete'); |
| _byte_vec_new = |
| _lib.lookupFunction<NativeWasmerByteVecNewFn, WasmerByteVecNewFn>( |
| 'wasm_byte_vec_new'); |
| _byte_vec_new_empty = _lib.lookupFunction<NativeWasmerByteVecNewEmptyFn, |
| WasmerByteVecNewEmptyFn>('wasm_byte_vec_new_empty'); |
| _byte_vec_new_uninitialized = _lib.lookupFunction< |
| NativeWasmerByteVecNewUninitializedFn, |
| WasmerByteVecNewUninitializedFn>('wasm_byte_vec_new_uninitialized'); |
| _engine_delete = |
| _lib.lookupFunction<NativeWasmerEngineDeleteFn, WasmerEngineDeleteFn>( |
| 'wasm_engine_delete'); |
| _engine_new = |
| _lib.lookupFunction<NativeWasmerEngineNewFn, WasmerEngineNewFn>( |
| 'wasm_engine_new'); |
| _exporttype_name = _lib.lookupFunction<NativeWasmerExporttypeNameFn, |
| WasmerExporttypeNameFn>('wasm_exporttype_name'); |
| _exporttype_type = _lib.lookupFunction<NativeWasmerExporttypeTypeFn, |
| WasmerExporttypeTypeFn>('wasm_exporttype_type'); |
| _exporttype_vec_delete = _lib.lookupFunction< |
| NativeWasmerExporttypeVecDeleteFn, |
| WasmerExporttypeVecDeleteFn>('wasm_exporttype_vec_delete'); |
| _exporttype_vec_new = _lib.lookupFunction<NativeWasmerExporttypeVecNewFn, |
| WasmerExporttypeVecNewFn>('wasm_exporttype_vec_new'); |
| _exporttype_vec_new_empty = _lib.lookupFunction< |
| NativeWasmerExporttypeVecNewEmptyFn, |
| WasmerExporttypeVecNewEmptyFn>('wasm_exporttype_vec_new_empty'); |
| _exporttype_vec_new_uninitialized = _lib.lookupFunction< |
| NativeWasmerExporttypeVecNewUninitializedFn, |
| WasmerExporttypeVecNewUninitializedFn>( |
| 'wasm_exporttype_vec_new_uninitialized'); |
| _extern_as_func = |
| _lib.lookupFunction<NativeWasmerExternAsFuncFn, WasmerExternAsFuncFn>( |
| 'wasm_extern_as_func'); |
| _extern_as_memory = _lib.lookupFunction<NativeWasmerExternAsMemoryFn, |
| WasmerExternAsMemoryFn>('wasm_extern_as_memory'); |
| _extern_delete = |
| _lib.lookupFunction<NativeWasmerExternDeleteFn, WasmerExternDeleteFn>( |
| 'wasm_extern_delete'); |
| _extern_kind = |
| _lib.lookupFunction<NativeWasmerExternKindFn, WasmerExternKindFn>( |
| 'wasm_extern_kind'); |
| _extern_vec_delete = _lib.lookupFunction<NativeWasmerExternVecDeleteFn, |
| WasmerExternVecDeleteFn>('wasm_extern_vec_delete'); |
| _extern_vec_new = |
| _lib.lookupFunction<NativeWasmerExternVecNewFn, WasmerExternVecNewFn>( |
| 'wasm_extern_vec_new'); |
| _extern_vec_new_empty = _lib.lookupFunction<NativeWasmerExternVecNewEmptyFn, |
| WasmerExternVecNewEmptyFn>('wasm_extern_vec_new_empty'); |
| _extern_vec_new_uninitialized = _lib.lookupFunction< |
| NativeWasmerExternVecNewUninitializedFn, |
| WasmerExternVecNewUninitializedFn>('wasm_extern_vec_new_uninitialized'); |
| _externtype_as_functype = _lib.lookupFunction< |
| NativeWasmerExterntypeAsFunctypeFn, |
| WasmerExterntypeAsFunctypeFn>('wasm_externtype_as_functype'); |
| _externtype_delete = _lib.lookupFunction<NativeWasmerExterntypeDeleteFn, |
| WasmerExterntypeDeleteFn>('wasm_externtype_delete'); |
| _externtype_kind = _lib.lookupFunction<NativeWasmerExterntypeKindFn, |
| WasmerExterntypeKindFn>('wasm_externtype_kind'); |
| _func_as_extern = |
| _lib.lookupFunction<NativeWasmerFuncAsExternFn, WasmerFuncAsExternFn>( |
| 'wasm_func_as_extern'); |
| _func_call = _lib.lookupFunction<NativeWasmerFuncCallFn, WasmerFuncCallFn>( |
| 'wasm_func_call'); |
| _func_delete = |
| _lib.lookupFunction<NativeWasmerFuncDeleteFn, WasmerFuncDeleteFn>( |
| 'wasm_func_delete'); |
| _func_new_with_env = _lib.lookupFunction<NativeWasmerFuncNewWithEnvFn, |
| WasmerFuncNewWithEnvFn>('wasm_func_new_with_env'); |
| _functype_delete = _lib.lookupFunction<NativeWasmerFunctypeDeleteFn, |
| WasmerFunctypeDeleteFn>('wasm_functype_delete'); |
| _functype_params = _lib.lookupFunction<NativeWasmerFunctypeParamsFn, |
| WasmerFunctypeParamsFn>('wasm_functype_params'); |
| _functype_results = _lib.lookupFunction<NativeWasmerFunctypeResultsFn, |
| WasmerFunctypeResultsFn>('wasm_functype_results'); |
| _importtype_module = _lib.lookupFunction<NativeWasmerImporttypeModuleFn, |
| WasmerImporttypeModuleFn>('wasm_importtype_module'); |
| _importtype_name = _lib.lookupFunction<NativeWasmerImporttypeNameFn, |
| WasmerImporttypeNameFn>('wasm_importtype_name'); |
| _importtype_type = _lib.lookupFunction<NativeWasmerImporttypeTypeFn, |
| WasmerImporttypeTypeFn>('wasm_importtype_type'); |
| _importtype_vec_delete = _lib.lookupFunction< |
| NativeWasmerImporttypeVecDeleteFn, |
| WasmerImporttypeVecDeleteFn>('wasm_importtype_vec_delete'); |
| _importtype_vec_new = _lib.lookupFunction<NativeWasmerImporttypeVecNewFn, |
| WasmerImporttypeVecNewFn>('wasm_importtype_vec_new'); |
| _importtype_vec_new_empty = _lib.lookupFunction< |
| NativeWasmerImporttypeVecNewEmptyFn, |
| WasmerImporttypeVecNewEmptyFn>('wasm_importtype_vec_new_empty'); |
| _importtype_vec_new_uninitialized = _lib.lookupFunction< |
| NativeWasmerImporttypeVecNewUninitializedFn, |
| WasmerImporttypeVecNewUninitializedFn>( |
| 'wasm_importtype_vec_new_uninitialized'); |
| _instance_delete = _lib.lookupFunction<NativeWasmerInstanceDeleteFn, |
| WasmerInstanceDeleteFn>('wasm_instance_delete'); |
| _instance_exports = _lib.lookupFunction<NativeWasmerInstanceExportsFn, |
| WasmerInstanceExportsFn>('wasm_instance_exports'); |
| _instance_new = |
| _lib.lookupFunction<NativeWasmerInstanceNewFn, WasmerInstanceNewFn>( |
| 'wasm_instance_new'); |
| _memory_as_extern = _lib.lookupFunction<NativeWasmerMemoryAsExternFn, |
| WasmerMemoryAsExternFn>('wasm_memory_as_extern'); |
| _memory_data = |
| _lib.lookupFunction<NativeWasmerMemoryDataFn, WasmerMemoryDataFn>( |
| 'wasm_memory_data'); |
| _memory_data_size = _lib.lookupFunction<NativeWasmerMemoryDataSizeFn, |
| WasmerMemoryDataSizeFn>('wasm_memory_data_size'); |
| _memory_delete = |
| _lib.lookupFunction<NativeWasmerMemoryDeleteFn, WasmerMemoryDeleteFn>( |
| 'wasm_memory_delete'); |
| _memory_grow = |
| _lib.lookupFunction<NativeWasmerMemoryGrowFn, WasmerMemoryGrowFn>( |
| 'wasm_memory_grow'); |
| _memory_new = |
| _lib.lookupFunction<NativeWasmerMemoryNewFn, WasmerMemoryNewFn>( |
| 'wasm_memory_new'); |
| _memory_size = |
| _lib.lookupFunction<NativeWasmerMemorySizeFn, WasmerMemorySizeFn>( |
| 'wasm_memory_size'); |
| _memorytype_delete = _lib.lookupFunction<NativeWasmerMemorytypeDeleteFn, |
| WasmerMemorytypeDeleteFn>('wasm_memorytype_delete'); |
| _memorytype_new = |
| _lib.lookupFunction<NativeWasmerMemorytypeNewFn, WasmerMemorytypeNewFn>( |
| 'wasm_memorytype_new'); |
| _module_delete = |
| _lib.lookupFunction<NativeWasmerModuleDeleteFn, WasmerModuleDeleteFn>( |
| 'wasm_module_delete'); |
| _module_exports = |
| _lib.lookupFunction<NativeWasmerModuleExportsFn, WasmerModuleExportsFn>( |
| 'wasm_module_exports'); |
| _module_imports = |
| _lib.lookupFunction<NativeWasmerModuleImportsFn, WasmerModuleImportsFn>( |
| 'wasm_module_imports'); |
| _module_new = |
| _lib.lookupFunction<NativeWasmerModuleNewFn, WasmerModuleNewFn>( |
| 'wasm_module_new'); |
| _store_delete = |
| _lib.lookupFunction<NativeWasmerStoreDeleteFn, WasmerStoreDeleteFn>( |
| 'wasm_store_delete'); |
| _store_new = _lib.lookupFunction<NativeWasmerStoreNewFn, WasmerStoreNewFn>( |
| 'wasm_store_new'); |
| _trap_delete = |
| _lib.lookupFunction<NativeWasmerTrapDeleteFn, WasmerTrapDeleteFn>( |
| 'wasm_trap_delete'); |
| _trap_message = |
| _lib.lookupFunction<NativeWasmerTrapMessageFn, WasmerTrapMessageFn>( |
| 'wasm_trap_message'); |
| _trap_new = _lib.lookupFunction<NativeWasmerTrapNewFn, WasmerTrapNewFn>( |
| 'wasm_trap_new'); |
| _valtype_delete = |
| _lib.lookupFunction<NativeWasmerValtypeDeleteFn, WasmerValtypeDeleteFn>( |
| 'wasm_valtype_delete'); |
| _valtype_kind = |
| _lib.lookupFunction<NativeWasmerValtypeKindFn, WasmerValtypeKindFn>( |
| 'wasm_valtype_kind'); |
| _valtype_vec_delete = _lib.lookupFunction<NativeWasmerValtypeVecDeleteFn, |
| WasmerValtypeVecDeleteFn>('wasm_valtype_vec_delete'); |
| _valtype_vec_new = |
| _lib.lookupFunction<NativeWasmerValtypeVecNewFn, WasmerValtypeVecNewFn>( |
| 'wasm_valtype_vec_new'); |
| _valtype_vec_new_empty = _lib.lookupFunction< |
| NativeWasmerValtypeVecNewEmptyFn, |
| WasmerValtypeVecNewEmptyFn>('wasm_valtype_vec_new_empty'); |
| _valtype_vec_new_uninitialized = _lib.lookupFunction< |
| NativeWasmerValtypeVecNewUninitializedFn, |
| WasmerValtypeVecNewUninitializedFn>( |
| 'wasm_valtype_vec_new_uninitialized'); |
| _wasmer_last_error_length = _lib.lookupFunction< |
| NativeWasmerWasmerLastErrorLengthFn, |
| WasmerWasmerLastErrorLengthFn>('wasmer_last_error_length'); |
| _wasmer_last_error_message = _lib.lookupFunction< |
| NativeWasmerWasmerLastErrorMessageFn, |
| WasmerWasmerLastErrorMessageFn>('wasmer_last_error_message'); |
| |
| if (_Dart_InitializeApiDL(NativeApi.initializeApiDLData) != 0) { |
| throw Exception("Failed to initialize Dart API"); |
| } |
| _engine = _engine_new(); |
| _checkNotEqual(_engine, nullptr, "Failed to initialize Wasm engine."); |
| _set_finalizer_for_engine(this, _engine); |
| } |
| |
| Pointer<WasmerStore> newStore(Object owner) { |
| Pointer<WasmerStore> store = _checkNotEqual( |
| _store_new(_engine), nullptr, "Failed to create Wasm store."); |
| _set_finalizer_for_store(owner, store); |
| return store; |
| } |
| |
| Pointer<WasmerModule> compile( |
| Object owner, Pointer<WasmerStore> store, Uint8List data) { |
| var dataPtr = allocate<Uint8>(count: data.length); |
| for (int i = 0; i < data.length; ++i) { |
| dataPtr[i] = data[i]; |
| } |
| var dataVec = allocate<WasmerByteVec>(); |
| dataVec.ref.data = dataPtr; |
| dataVec.ref.length = data.length; |
| |
| var modulePtr = _module_new(store, dataVec); |
| |
| free(dataPtr); |
| free(dataVec); |
| |
| _checkNotEqual(modulePtr, nullptr, "Wasm module compile failed."); |
| _set_finalizer_for_module(owner, modulePtr); |
| return modulePtr; |
| } |
| |
| List<WasmExportDescriptor> exportDescriptors(Pointer<WasmerModule> module) { |
| var exportsVec = allocate<WasmerExporttypeVec>(); |
| _module_exports(module, exportsVec); |
| var exps = <WasmExportDescriptor>[]; |
| for (var i = 0; i < exportsVec.ref.length; ++i) { |
| var exp = exportsVec.ref.data[i]; |
| var extern = _exporttype_type(exp); |
| var kind = _externtype_kind(extern); |
| var fnType = kind == WasmerExternKindFunction |
| ? _externtype_as_functype(extern) |
| : nullptr; |
| exps.add(WasmExportDescriptor( |
| kind, _exporttype_name(exp).ref.toString(), fnType)); |
| } |
| free(exportsVec); |
| return exps; |
| } |
| |
| List<WasmImportDescriptor> importDescriptors(Pointer<WasmerModule> module) { |
| var importsVec = allocate<WasmerImporttypeVec>(); |
| _module_imports(module, importsVec); |
| var imps = <WasmImportDescriptor>[]; |
| for (var i = 0; i < importsVec.ref.length; ++i) { |
| var imp = importsVec.ref.data[i]; |
| var extern = _importtype_type(imp); |
| var kind = _externtype_kind(extern); |
| var fnType = kind == WasmerExternKindFunction |
| ? _externtype_as_functype(extern) |
| : nullptr; |
| imps.add(WasmImportDescriptor( |
| kind, |
| _importtype_module(imp).ref.toString(), |
| _importtype_name(imp).ref.toString(), |
| fnType)); |
| } |
| free(importsVec); |
| return imps; |
| } |
| |
| void maybeThrowTrap(Pointer<WasmerTrap> trap, String source) { |
| if (trap != nullptr) { |
| // There are 2 kinds of trap, and their memory is managed differently. |
| // Traps created in the newTrap method below are stored in the traps map |
| // with a corresponding exception, and their memory is managed using a |
| // finalizer on the _WasmTrapsEntry. Traps can also be created by WASM |
| // code, and in that case we delete them in this function. |
| var entry = traps[trap.address]; |
| if (entry != null) { |
| traps.remove(entry); |
| throw entry.exception; |
| } else { |
| var trapMessage = allocate<WasmerByteVec>(); |
| _trap_message(trap, trapMessage); |
| var message = "Wasm trap when calling $source: ${trapMessage.ref}"; |
| _byte_vec_delete(trapMessage); |
| free(trapMessage); |
| _trap_delete(trap); |
| throw Exception(message); |
| } |
| } |
| } |
| |
| Pointer<WasmerInstance> instantiate(Object owner, Pointer<WasmerStore> store, |
| Pointer<WasmerModule> module, Pointer<WasmerExternVec> imports) { |
| var trap = allocate<Pointer<WasmerTrap>>(); |
| trap.value = nullptr; |
| var inst = _instance_new(store, module, imports, trap); |
| maybeThrowTrap(trap.value, "module initialization function"); |
| free(trap); |
| _checkNotEqual(inst, nullptr, "Wasm module instantiation failed."); |
| _set_finalizer_for_instance(owner, inst); |
| return inst; |
| } |
| |
| // Clean up the exports after use, with deleteExports. |
| Pointer<WasmerExternVec> exports(Pointer<WasmerInstance> instancePtr) { |
| var exports = allocate<WasmerExternVec>(); |
| _instance_exports(instancePtr, exports); |
| return exports; |
| } |
| |
| void deleteExports(Pointer<WasmerExternVec> exports) { |
| _extern_vec_delete(exports); |
| free(exports); |
| } |
| |
| int externKind(Pointer<WasmerExtern> extern) { |
| return _extern_kind(extern); |
| } |
| |
| Pointer<WasmerFunc> externToFunction(Pointer<WasmerExtern> extern) { |
| return _extern_as_func(extern); |
| } |
| |
| Pointer<WasmerExtern> functionToExtern(Pointer<WasmerFunc> func) { |
| return _func_as_extern(func); |
| } |
| |
| List<int> getArgTypes(Pointer<WasmerFunctype> funcType) { |
| var types = <int>[]; |
| var args = _functype_params(funcType); |
| for (var i = 0; i < args.ref.length; ++i) { |
| types.add(_valtype_kind(args.ref.data[i])); |
| } |
| return types; |
| } |
| |
| int getReturnType(Pointer<WasmerFunctype> funcType) { |
| var rets = _functype_results(funcType); |
| if (rets.ref.length == 0) { |
| return WasmerValKindVoid; |
| } else if (rets.ref.length > 1) { |
| throw Exception("Multiple return values are not supported"); |
| } |
| return _valtype_kind(rets.ref.data[0]); |
| } |
| |
| void call(Pointer<WasmerFunc> func, Pointer<WasmerValVec> args, |
| Pointer<WasmerValVec> results, String source) { |
| maybeThrowTrap(_func_call(func, args, results), source); |
| } |
| |
| Pointer<WasmerMemory> externToMemory(Pointer<WasmerExtern> extern) { |
| return _extern_as_memory(extern); |
| } |
| |
| Pointer<WasmerExtern> memoryToExtern(Pointer<WasmerMemory> memory) { |
| return _memory_as_extern(memory); |
| } |
| |
| Pointer<WasmerMemory> newMemory( |
| Object owner, Pointer<WasmerStore> store, int pages, int? maxPages) { |
| var limPtr = allocate<WasmerLimits>(); |
| limPtr.ref.min = pages; |
| limPtr.ref.max = maxPages ?? wasm_limits_max_default; |
| var memType = _memorytype_new(limPtr); |
| free(limPtr); |
| _checkNotEqual(memType, nullptr, "Failed to create memory type."); |
| _set_finalizer_for_memorytype(owner, memType); |
| var memory = _checkNotEqual( |
| _memory_new(store, memType), nullptr, "Failed to create memory."); |
| _set_finalizer_for_memory(owner, memory); |
| return memory; |
| } |
| |
| void growMemory(Pointer<WasmerMemory> memory, int deltaPages) { |
| _checkNotEqual( |
| _memory_grow(memory, deltaPages), 0, "Failed to grow memory."); |
| } |
| |
| int memoryLength(Pointer<WasmerMemory> memory) { |
| return _memory_size(memory); |
| } |
| |
| Uint8List memoryView(Pointer<WasmerMemory> memory) { |
| return _memory_data(memory).asTypedList(_memory_data_size(memory)); |
| } |
| |
| Pointer<WasmerFunc> newFunc( |
| Object owner, |
| Pointer<WasmerStore> store, |
| Pointer<WasmerFunctype> funcType, |
| Pointer func, |
| Pointer env, |
| Pointer finalizer) { |
| var f = _func_new_with_env( |
| store, funcType, func.cast(), env.cast(), finalizer.cast()); |
| _checkNotEqual(f, nullptr, "Failed to create function."); |
| _set_finalizer_for_func(owner, f); |
| return f; |
| } |
| |
| Pointer<WasmerTrap> newTrap(Pointer<WasmerStore> store, dynamic exception) { |
| var msg = allocate<WasmerByteVec>(); |
| msg.ref.data = allocate<Uint8>(); |
| msg.ref.data[0] = 0; |
| msg.ref.length = 0; |
| var trap = _trap_new(store, msg); |
| free(msg.ref.data); |
| free(msg); |
| _checkNotEqual(trap, nullptr, "Failed to create trap."); |
| var entry = _WasmTrapsEntry(exception); |
| _set_finalizer_for_trap(entry, trap); |
| traps[trap.address] = entry; |
| return trap; |
| } |
| |
| Pointer<WasmerWasiConfig> newWasiConfig() { |
| var name = allocate<Uint8>(); |
| name[0] = 0; |
| var config = _wasi_config_new(name); |
| free(name); |
| return _checkNotEqual(config, nullptr, "Failed to create WASI config."); |
| } |
| |
| void captureWasiStdout(Pointer<WasmerWasiConfig> config) { |
| _wasi_config_inherit_stdout(config); |
| } |
| |
| void captureWasiStderr(Pointer<WasmerWasiConfig> config) { |
| _wasi_config_inherit_stderr(config); |
| } |
| |
| Pointer<WasmerWasiEnv> newWasiEnv(Pointer<WasmerWasiConfig> config) { |
| return _checkNotEqual( |
| _wasi_env_new(config), nullptr, "Failed to create WASI environment."); |
| } |
| |
| void wasiEnvSetMemory( |
| Pointer<WasmerWasiEnv> env, Pointer<WasmerMemory> memory) { |
| _wasi_env_set_memory(env, memory); |
| } |
| |
| void getWasiImports(Pointer<WasmerStore> store, Pointer<WasmerModule> mod, |
| Pointer<WasmerWasiEnv> env, Pointer<WasmerExternVec> imports) { |
| _checkNotEqual(_wasi_get_imports(store, mod, env, imports), 0, |
| "Failed to fill WASI imports."); |
| } |
| |
| Stream<List<int>> getWasiStdoutStream(Pointer<WasmerWasiEnv> env) { |
| return Stream.fromIterable(_WasiStreamIterable(env, _wasi_env_read_stdout)); |
| } |
| |
| Stream<List<int>> getWasiStderrStream(Pointer<WasmerWasiEnv> env) { |
| return Stream.fromIterable(_WasiStreamIterable(env, _wasi_env_read_stderr)); |
| } |
| |
| String _getLastError() { |
| var length = _wasmer_last_error_length(); |
| var buf = allocate<Uint8>(count: length); |
| _wasmer_last_error_message(buf, length); |
| String message = utf8.decode(buf.asTypedList(length)); |
| free(buf); |
| return message; |
| } |
| |
| T _checkNotEqual<T>(T x, T y, String errorMessage) { |
| if (x == y) { |
| throw Exception("$errorMessage\n${_getLastError()}"); |
| } |
| return x; |
| } |
| |
| static String getSignatureString( |
| String name, List<int> argTypes, int returnType) { |
| return "${wasmerValKindName(returnType)} $name" + |
| "(${argTypes.map(wasmerValKindName).join(", ")})"; |
| } |
| } |
| |
| class _WasiStreamIterator implements Iterator<List<int>> { |
| static final int _bufferLength = 1024; |
| Pointer<WasmerWasiEnv> _env; |
| Function _reader; |
| Pointer<Uint8> _buf = allocate<Uint8>(count: _bufferLength); |
| int _length = 0; |
| _WasiStreamIterator(this._env, this._reader) {} |
| |
| bool moveNext() { |
| _length = _reader(_env, _buf, _bufferLength); |
| return true; |
| } |
| |
| List<int> get current => _buf.asTypedList(_length); |
| } |
| |
| class _WasiStreamIterable extends Iterable<List<int>> { |
| Pointer<WasmerWasiEnv> _env; |
| Function _reader; |
| _WasiStreamIterable(this._env, this._reader) {} |
| @override |
| Iterator<List<int>> get iterator => _WasiStreamIterator(_env, _reader); |
| } |