From 304a8b637c1a5ba230c520485547947a74b74ed4 Mon Sep 17 00:00:00 2001 From: pavelsavara Date: Mon, 15 Nov 2021 15:35:31 +0100 Subject: [PATCH] - no imports from outer scope - move dotnet to -extern-pre-js - re-enable JS minification - rename main javaScript files to main.js and test-main.js - sample and test script cleanup - rename set_exit_code method - rewrite test start as async method - improve script loading via script element on page in test - use BINDING.bind_static_method instead of INTERNAL where possible - better .d.ts exports - formatted html files --- eng/pipelines/coreclr/templates/perf-job.yml | 2 +- eng/testing/tests.wasm.targets | 6 +- src/libraries/sendtohelixhelp.proj | 4 +- src/mono/sample/mbr/browser/WasmDelta.csproj | 2 +- src/mono/sample/mbr/browser/index.html | 62 ++++---- .../mbr/browser/{runtime.js => main.js} | 1 - src/mono/sample/wasm/Directory.Build.targets | 4 +- .../Console/Wasm.Console.Bench.Sample.csproj | 2 +- .../Wasm.Browser.Bench.Sample.csproj | 2 +- .../wasm/browser-bench/appstart-frame.html | 4 +- src/mono/sample/wasm/browser-bench/index.html | 137 +++++++++--------- .../browser-bench/{runtime.js => main.js} | 9 +- .../sample/wasm/browser-profile/README.md | 4 +- .../Wasm.BrowserProfile.Sample.csproj | 4 +- .../sample/wasm/browser-profile/index.html | 6 +- .../browser-profile/{runtime.js => main.js} | 37 ++--- .../wasm/browser/Wasm.Browser.Sample.csproj | 2 +- src/mono/sample/wasm/browser/index.html | 75 ++++------ .../wasm/browser/{runtime.js => main.js} | 13 +- .../wasm/console/Wasm.Console.Sample.csproj | 2 +- src/mono/sample/wasm/wasm.mk | 2 +- src/mono/wasm/build/README.md | 4 +- src/mono/wasm/build/WasmApp.Native.targets | 5 +- src/mono/wasm/build/WasmApp.targets | 2 +- .../aot-tests/ProxyProjectForAOTOnHelix.proj | 2 +- .../DebuggerTestSuite/BreakpointTests.cs | 20 +-- .../tests/debugger-test/debugger-driver.html | 2 +- .../tests/debugger-test/debugger-main.js | 23 +++ .../tests/debugger-test/debugger-test.csproj | 4 +- .../tests/debugger-test/runtime-debugger.js | 24 --- src/mono/wasm/runtime/CMakeLists.txt | 2 +- src/mono/wasm/runtime/export-types.ts | 7 + src/mono/wasm/runtime/exports.ts | 96 +++++++----- src/mono/wasm/runtime/library-dotnet.js | 4 +- src/mono/wasm/runtime/method-binding.ts | 8 +- src/mono/wasm/runtime/modules.ts | 14 +- src/mono/wasm/runtime/package.json | 2 +- src/mono/wasm/runtime/rollup.config.js | 27 ++-- src/mono/wasm/runtime/startup.ts | 37 ++++- src/mono/wasm/runtime/tsconfig.json | 2 +- src/mono/wasm/runtime/types.ts | 4 +- src/mono/wasm/runtime/types/emscripten.d.ts | 16 +- src/mono/wasm/runtime/web-socket.ts | 1 + .../wasm/{runtime-test.js => test-main.js} | 93 ++++++------ src/tasks/WasmAppBuilder/WasmAppBuilder.cs | 4 +- .../Wasm.Build.Tests/BuildTestBase.cs | 8 +- .../SatelliteAssembliesTests.cs | 2 +- .../Wasm.Build.Tests/Wasm.Build.Tests.csproj | 2 +- src/tests/Common/Directory.Build.targets | 2 +- .../wasm-test-runner/WasmTestRunner.proj | 2 +- src/tests/Directory.Build.targets | 2 +- .../WebAssembly.Browser.HotReload.Test.csproj | 4 +- .../WebAssembly/Browser/HotReload/index.html | 75 ++++------ .../Browser/HotReload/{runtime.js => main.js} | 11 +- ...Assembly.Browser.RuntimeConfig.Test.csproj | 2 +- .../Browser/RuntimeConfig/index.html | 75 ++++------ .../RuntimeConfig/{runtime.js => main.js} | 9 +- 57 files changed, 495 insertions(+), 482 deletions(-) rename src/mono/sample/mbr/browser/{runtime.js => main.js} (95%) rename src/mono/sample/wasm/browser-bench/{runtime.js => main.js} (74%) rename src/mono/sample/wasm/browser-profile/{runtime.js => main.js} (66%) rename src/mono/sample/wasm/browser/{runtime.js => main.js} (71%) create mode 100644 src/mono/wasm/debugger/tests/debugger-test/debugger-main.js delete mode 100644 src/mono/wasm/debugger/tests/debugger-test/runtime-debugger.js create mode 100644 src/mono/wasm/runtime/export-types.ts rename src/mono/wasm/{runtime-test.js => test-main.js} (82%) rename src/tests/FunctionalTests/WebAssembly/Browser/HotReload/{runtime.js => main.js} (72%) rename src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/{runtime.js => main.js} (71%) diff --git a/eng/pipelines/coreclr/templates/perf-job.yml b/eng/pipelines/coreclr/templates/perf-job.yml index 6802056be03a6..50941e60f1c5f 100644 --- a/eng/pipelines/coreclr/templates/perf-job.yml +++ b/eng/pipelines/coreclr/templates/perf-job.yml @@ -135,7 +135,7 @@ jobs: artifactName: BrowserWasm displayName: BrowserWasm - - script: "mkdir $(librariesDownloadDir)/bin/wasm;unzip -o $(librariesDownloadDir)/BrowserWasm/artifacts/packages/Release/Shipping/Microsoft.NETCore.App.Runtime.Mono.browser-wasm.7.0.0-ci.nupkg data/* runtimes/* -d $(librariesDownloadDir)/bin/wasm;cp src/mono/wasm/runtime-test.js $(librariesDownloadDir)/bin/wasm/runtime-test.js;find $(librariesDownloadDir)/bin/wasm -type f -exec chmod 664 {} \\;" + - script: "mkdir $(librariesDownloadDir)/bin/wasm;unzip -o $(librariesDownloadDir)/BrowserWasm/artifacts/packages/Release/Shipping/Microsoft.NETCore.App.Runtime.Mono.browser-wasm.7.0.0-ci.nupkg data/* runtimes/* -d $(librariesDownloadDir)/bin/wasm;cp src/mono/wasm/test-main.js $(librariesDownloadDir)/bin/wasm/test-main.js;find $(librariesDownloadDir)/bin/wasm -type f -exec chmod 664 {} \\;" displayName: "Create wasm directory (Linux)" # Download mono AOT diff --git a/eng/testing/tests.wasm.targets b/eng/testing/tests.wasm.targets index 00aa7b19f1701..53b0f0254148d 100644 --- a/eng/testing/tests.wasm.targets +++ b/eng/testing/tests.wasm.targets @@ -23,13 +23,13 @@ <_XHarnessArgs Condition="'$(OS)' != 'Windows_NT'">wasm $XHARNESS_COMMAND --app=. --output-directory=$XHARNESS_OUT <_XHarnessArgs Condition="'$(OS)' == 'Windows_NT'">wasm %XHARNESS_COMMAND% --app=. --output-directory=%XHARNESS_OUT% - <_XHarnessArgs Condition="'$(Scenario)' != 'WasmTestOnBrowser' and '$(Scenario)' != 'BuildWasmApps'">$(_XHarnessArgs) --engine=$(JSEngine) $(JSEngineArgs) --js-file=runtime.js + <_XHarnessArgs Condition="'$(Scenario)' != 'WasmTestOnBrowser' and '$(Scenario)' != 'BuildWasmApps'">$(_XHarnessArgs) --engine=$(JSEngine) $(JSEngineArgs) --js-file=main.js <_XHarnessArgs Condition="'$(BrowserHost)' == 'windows'">$(_XHarnessArgs) --browser=chrome --browser-path=%HELIX_CORRELATION_PAYLOAD%\chrome-win\chrome.exe <_XHarnessArgs Condition="'$(IsFunctionalTest)' == 'true'" >$(_XHarnessArgs) --expected-exit-code=$(ExpectedExitCode) <_XHarnessArgs Condition="'$(WasmXHarnessArgs)' != ''" >$(_XHarnessArgs) $(WasmXHarnessArgs) <_AppArgs Condition="'$(IsFunctionalTest)' != 'true' and '$(Scenario)' != 'BuildWasmApps'">--run WasmTestRunner.dll $(AssemblyName).dll - <_AppArgs Condition="'$(IsFunctionalTest)' == 'true'">--run $(AssemblyName).dll --testing + <_AppArgs Condition="'$(IsFunctionalTest)' == 'true'">--run $(AssemblyName).dll <_AppArgs Condition="'$(WasmTestAppArgs)' != ''">$(_AppArgs) $(WasmTestAppArgs) @@ -161,7 +161,7 @@ $(BundleDir) WasmTestRunner.dll - $(MonoProjectRoot)\wasm\runtime-test.js + $(MonoProjectRoot)\wasm\test-main.js $(InvariantGlobalization) true false diff --git a/src/libraries/sendtohelixhelp.proj b/src/libraries/sendtohelixhelp.proj index 3eac0bacd9245..83eeca33b065d 100644 --- a/src/libraries/sendtohelixhelp.proj +++ b/src/libraries/sendtohelixhelp.proj @@ -491,7 +491,7 @@ %(Identity) - $(ExecXHarnessCmd) wasm $(XHarnessCommand) --app=. --engine=V8 --engine-arg=--stack-trace-limit=1000 --js-file=runtime.js --output-directory=$(XHarnessOutput) -- --run %(FileName).dll + $(ExecXHarnessCmd) wasm $(XHarnessCommand) --app=. --engine=V8 --engine-arg=--stack-trace-limit=1000 --js-file=main.js --output-directory=$(XHarnessOutput) -- --run %(FileName).dll @@ -501,7 +501,7 @@ %(Identity) - $(ExecXHarnessCmd) wasm $(XHarnessCommand) --app=. --browser=Chrome $(XHarnessBrowserPathArg) --html-file=index.html --output-directory=$(XHarnessOutput) -- %(FileName).dll --testing + $(ExecXHarnessCmd) wasm $(XHarnessCommand) --app=. --browser=Chrome $(XHarnessBrowserPathArg) --html-file=index.html --output-directory=$(XHarnessOutput) -- %(FileName).dll diff --git a/src/mono/sample/mbr/browser/WasmDelta.csproj b/src/mono/sample/mbr/browser/WasmDelta.csproj index c150529f753c6..5cc4b15f09511 100644 --- a/src/mono/sample/mbr/browser/WasmDelta.csproj +++ b/src/mono/sample/mbr/browser/WasmDelta.csproj @@ -7,7 +7,7 @@ bin $(MSBuildProjectDirectory)\bin\$(Configuration)\AppBundle\ - runtime.js + main.js false diff --git a/src/mono/sample/mbr/browser/index.html b/src/mono/sample/mbr/browser/index.html index 6122370dd9d65..91ccdf9c1e4bc 100644 --- a/src/mono/sample/mbr/browser/index.html +++ b/src/mono/sample/mbr/browser/index.html @@ -2,37 +2,37 @@ - - TESTS - - - - - - Result from Sample.Test.TestMeaning: -
- Click here (upto 2 times): -
- - - + + Hot Reload Sample + + + + + + Result from Sample.Test.TestMeaning: +
+ Click here (upto 2 times): +
+ + + + - - + \ No newline at end of file diff --git a/src/mono/sample/mbr/browser/runtime.js b/src/mono/sample/mbr/browser/main.js similarity index 95% rename from src/mono/sample/mbr/browser/runtime.js rename to src/mono/sample/mbr/browser/main.js index 36feada2026d6..6294d62187e37 100644 --- a/src/mono/sample/mbr/browser/runtime.js +++ b/src/mono/sample/mbr/browser/main.js @@ -3,7 +3,6 @@ "use strict"; var Module = { - config: null, configSrc: "./mono-config.json", onConfigLoaded: function () { MONO.config.environment_variables["DOTNET_MODIFIABLE_ASSEMBLIES"] = "debug"; diff --git a/src/mono/sample/wasm/Directory.Build.targets b/src/mono/sample/wasm/Directory.Build.targets index f0b90a539a9c5..95415ef67fb16 100644 --- a/src/mono/sample/wasm/Directory.Build.targets +++ b/src/mono/sample/wasm/Directory.Build.targets @@ -14,12 +14,12 @@ - + - + diff --git a/src/mono/sample/wasm/browser-bench/Console/Wasm.Console.Bench.Sample.csproj b/src/mono/sample/wasm/browser-bench/Console/Wasm.Console.Bench.Sample.csproj index d60474926aa91..ab8a632046fa7 100644 --- a/src/mono/sample/wasm/browser-bench/Console/Wasm.Console.Bench.Sample.csproj +++ b/src/mono/sample/wasm/browser-bench/Console/Wasm.Console.Bench.Sample.csproj @@ -1,7 +1,7 @@ true - $(MonoProjectRoot)\wasm\runtime-test.js + $(MonoProjectRoot)\wasm\test-main.js true true diff --git a/src/mono/sample/wasm/browser-bench/Wasm.Browser.Bench.Sample.csproj b/src/mono/sample/wasm/browser-bench/Wasm.Browser.Bench.Sample.csproj index 68366a982e9ca..89c7feed07ef7 100644 --- a/src/mono/sample/wasm/browser-bench/Wasm.Browser.Bench.Sample.csproj +++ b/src/mono/sample/wasm/browser-bench/Wasm.Browser.Bench.Sample.csproj @@ -2,7 +2,7 @@ false - runtime.js + main.js true diff --git a/src/mono/sample/wasm/browser-bench/appstart-frame.html b/src/mono/sample/wasm/browser-bench/appstart-frame.html index 4e24c6e9663c1..84f0832330583 100644 --- a/src/mono/sample/wasm/browser-bench/appstart-frame.html +++ b/src/mono/sample/wasm/browser-bench/appstart-frame.html @@ -11,7 +11,7 @@ - + diff --git a/src/mono/sample/wasm/browser-bench/index.html b/src/mono/sample/wasm/browser-bench/index.html index b163465aa6278..bec077fe5324c 100644 --- a/src/mono/sample/wasm/browser-bench/index.html +++ b/src/mono/sample/wasm/browser-bench/index.html @@ -2,85 +2,78 @@ - - TESTS - - - - - - - Output:

- - - + }, - + ReachedManaged: async function () { + AppStart.Construct(); + try { + await AppStart.WaitForReached(); + } finally { + AppStart.RemoveFrame(); + } + } + }; + + + + + - - + \ No newline at end of file diff --git a/src/mono/sample/wasm/browser-bench/runtime.js b/src/mono/sample/wasm/browser-bench/main.js similarity index 74% rename from src/mono/sample/wasm/browser-bench/runtime.js rename to src/mono/sample/wasm/browser-bench/main.js index f810b524f6a33..40c3e7f673219 100644 --- a/src/mono/sample/wasm/browser-bench/runtime.js +++ b/src/mono/sample/wasm/browser-bench/main.js @@ -3,18 +3,17 @@ "use strict"; var Module = { - config: null, configSrc: "./mono-config.json", - onDotNetReady: function () { + onDotNetReady: () => { try { App.init(); } catch (error) { console.log("exception: " + error); - test_exit(1); + set_exit_code(1, error); throw (error); } }, - onAbort: function (err) { - test_exit(1); + onAbort: (error) => { + set_exit_code(1, error); }, }; diff --git a/src/mono/sample/wasm/browser-profile/README.md b/src/mono/sample/wasm/browser-profile/README.md index 26afb1386711b..909b296c5208a 100644 --- a/src/mono/sample/wasm/browser-profile/README.md +++ b/src/mono/sample/wasm/browser-profile/README.md @@ -9,11 +9,11 @@ public static void StopProfile(){} ``` -2. Initialize the profiler in the main javascript (e.g. runtime.js) +2. Initialize the profiler in the main javascript (e.g. main.js) ``` var Module = { - onRuntimeInitialized: function () { + onConfigLoaded: () => { ... if (config.enable_profiler) diff --git a/src/mono/sample/wasm/browser-profile/Wasm.BrowserProfile.Sample.csproj b/src/mono/sample/wasm/browser-profile/Wasm.BrowserProfile.Sample.csproj index 9d14836ecb4f3..be10935e8bc32 100644 --- a/src/mono/sample/wasm/browser-profile/Wasm.BrowserProfile.Sample.csproj +++ b/src/mono/sample/wasm/browser-profile/Wasm.BrowserProfile.Sample.csproj @@ -1,7 +1,7 @@ true - runtime.js + main.js aot; @@ -9,4 +9,6 @@ + + diff --git a/src/mono/sample/wasm/browser-profile/index.html b/src/mono/sample/wasm/browser-profile/index.html index de4a5599e94ad..db4344793d2ee 100644 --- a/src/mono/sample/wasm/browser-profile/index.html +++ b/src/mono/sample/wasm/browser-profile/index.html @@ -3,15 +3,15 @@ - TESTS + Profiler Sample - + Result from Sample.Test.TestMeaning: - + diff --git a/src/mono/sample/wasm/browser-profile/runtime.js b/src/mono/sample/wasm/browser-profile/main.js similarity index 66% rename from src/mono/sample/wasm/browser-profile/runtime.js rename to src/mono/sample/wasm/browser-profile/main.js index ebedaba70e29e..4f9abe18afd65 100644 --- a/src/mono/sample/wasm/browser-profile/runtime.js +++ b/src/mono/sample/wasm/browser-profile/main.js @@ -3,10 +3,8 @@ "use strict"; var Module = { - is_testing: false, - config: null, configSrc: "./mono-config.json", - onConfigLoaded: function () { + onConfigLoaded: () => { if (MONO.config.enable_profiler) { MONO.config.aot_profiler_options = { write_at: "Sample.Test::StopProfile", @@ -14,42 +12,37 @@ var Module = { } } }, - onDotNetReady: function () { + onDotNetReady: () => { try { Module.init(); } catch (error) { - test_exit(1); + set_exit_code(1, error); throw (error); } }, - onAbort: function (err) { - test_exit(1); + onAbort: (error) => { + set_exit_code(1, error); }, - init: function () { + init: () => { console.log("not ready yet") - const ret = INTERNAL.call_static_method("[Wasm.BrowserProfile.Sample] Sample.Test:TestMeaning", []); + const testMeaning = BINDING.bind_static_method("[Wasm.BrowserProfile.Sample] Sample.Test:TestMeaning"); + const stopProfile = BINDING.bind_static_method("[Wasm.BrowserProfile.Sample] Sample.Test:StopProfile"); + const ret = testMeaning(); document.getElementById("out").innerHTML = ret; console.log("ready"); - if (Module.is_testing) { - console.debug(`ret: ${ret}`); - let exit_code = ret == 42 ? 0 : 1; - Module.test_exit(exit_code); - } + console.debug(`ret: ${ret}`); + let exit_code = ret == 42 ? 0 : 1; + Module.set_exit_code(exit_code); if (MONO.config.enable_profiler) { - INTERNAL.call_static_method("[Wasm.BrowserProfile.Sample] Sample.Test:StopProfile", []); + stopProfile(); Module.saveProfile(); } }, - test_exit: function (exit_code) { - if (!Module.is_testing) { - console.log(`test_exit: ${exit_code}`); - return; - } - + set_exit_code: (exit_code, reason) => { /* Set result in a tests_done element, to be read by xharness */ const tests_done_elem = document.createElement("label"); tests_done_elem.id = "tests_done"; @@ -59,7 +52,7 @@ var Module = { console.log(`WASM EXIT ${exit_code}`); }, - saveProfile: function () { + saveProfile: () => { const a = document.createElement('a'); const blob = new Blob([INTERNAL.aot_profile_data]); a.href = URL.createObjectURL(blob); diff --git a/src/mono/sample/wasm/browser/Wasm.Browser.Sample.csproj b/src/mono/sample/wasm/browser/Wasm.Browser.Sample.csproj index 15386dece90a9..eb9934dc662ca 100644 --- a/src/mono/sample/wasm/browser/Wasm.Browser.Sample.csproj +++ b/src/mono/sample/wasm/browser/Wasm.Browser.Sample.csproj @@ -2,7 +2,7 @@ Debug true - runtime.js + main.js true embedded 1 diff --git a/src/mono/sample/wasm/browser/index.html b/src/mono/sample/wasm/browser/index.html index cedcae5715afb..2fa08c756cbf2 100644 --- a/src/mono/sample/wasm/browser/index.html +++ b/src/mono/sample/wasm/browser/index.html @@ -2,55 +2,42 @@ - - TESTS - - - - - - Result from Sample.Test.TestMeaning: - - + console.debug(`ret: ${ret}`); + let exit_code = ret == 42 ? 0 : 1; + set_exit_code(exit_code); + }, + }; + + + - + - \ No newline at end of file diff --git a/src/mono/sample/wasm/browser/runtime.js b/src/mono/sample/wasm/browser/main.js similarity index 71% rename from src/mono/sample/wasm/browser/runtime.js rename to src/mono/sample/wasm/browser/main.js index ef84c7108e8bf..9e347d4d346cd 100644 --- a/src/mono/sample/wasm/browser/runtime.js +++ b/src/mono/sample/wasm/browser/main.js @@ -2,19 +2,18 @@ // The .NET Foundation licenses this file to you under the MIT license. "use strict"; + var Module = { - config: null, configSrc: "./mono-config.json", - onDotNetReady: function () { + onDotNetReady: () => { try { App.init(); } catch (error) { - test_exit(1); + set_exit_code(1, error); throw (error); } }, - onAbort: function () { - test_exit(1); - + onAbort: (error) => { + set_exit_code(1, error); }, -}; +}; \ No newline at end of file diff --git a/src/mono/sample/wasm/console/Wasm.Console.Sample.csproj b/src/mono/sample/wasm/console/Wasm.Console.Sample.csproj index 237718ebecb44..1dabb4aa0c787 100644 --- a/src/mono/sample/wasm/console/Wasm.Console.Sample.csproj +++ b/src/mono/sample/wasm/console/Wasm.Console.Sample.csproj @@ -1,7 +1,7 @@ true - $(MonoProjectRoot)\wasm\runtime-test.js + $(MonoProjectRoot)\wasm\test-main.js true diff --git a/src/mono/sample/wasm/wasm.mk b/src/mono/sample/wasm/wasm.mk index fb11cf9757035..5a1c70133489f 100644 --- a/src/mono/sample/wasm/wasm.mk +++ b/src/mono/sample/wasm/wasm.mk @@ -30,4 +30,4 @@ run-browser: fi run-console: - cd bin/$(CONFIG)/AppBundle && ~/.jsvu/v8 --stack-trace-limit=1000 --single-threaded --expose_wasm runtime.js -- $(DOTNET_MONO_LOG_LEVEL) --run $(CONSOLE_DLL) $(ARGS) + cd bin/$(CONFIG)/AppBundle && ~/.jsvu/v8 --stack-trace-limit=1000 --single-threaded --expose_wasm main.js -- $(DOTNET_MONO_LOG_LEVEL) --run $(CONSOLE_DLL) $(ARGS) diff --git a/src/mono/wasm/build/README.md b/src/mono/wasm/build/README.md index 8d4920a7be532..79b9ef30f46dd 100644 --- a/src/mono/wasm/build/README.md +++ b/src/mono/wasm/build/README.md @@ -10,9 +10,9 @@ Wasm app build can run in two scenarios: A dotnet wasm app has some native wasm files (`dotnet.wasm`, and `dotnet.js`). How these files are obtained, or generated: 1. Build - a. with no native libraries referenced (AOT setting is ignored here) + - a. with no native libraries referenced (AOT setting is ignored here) - files from the runtime pack are used as-is - b. with native libraries referenced + - b. with native libraries referenced - dotnet.wasm is relinked with the native libraries 2. Publish - dotnet.wasm is relinked with the native libraries, and updated pinvoke/icalls from the trimmed assemblies diff --git a/src/mono/wasm/build/WasmApp.Native.targets b/src/mono/wasm/build/WasmApp.Native.targets index ca1f5041eabc2..b33b74d1611b0 100644 --- a/src/mono/wasm/build/WasmApp.Native.targets +++ b/src/mono/wasm/build/WasmApp.Native.targets @@ -223,7 +223,7 @@ <_WasmRuntimePackSrcFile ObjectFile="$(_WasmIntermediateOutputPath)%(FileName).o" /> <_DotnetJSSrcFile Include="$(_WasmRuntimePackSrcDir)\*.js" Exclude="$(_WasmRuntimePackSrcDir)\*.iffe.js"/> - <_WasmExtraJSFile Include="$(_WasmRuntimePackSrcDir)\*.iffe.js" Kind="pre-js" /> + <_WasmExtraJSFile Include="$(_WasmRuntimePackSrcDir)\*.iffe.js" Kind="extern-pre-js" /> <_WasmNativeFileForLinking Include="@(NativeFileReference)" /> @@ -353,8 +353,9 @@ <_EmccLinkStepArgs Include="--js-library "%(_WasmExtraJSFile.Identity)"" Condition="'%(_WasmExtraJSFile.Kind)' == 'js-library'" /> <_EmccLinkStepArgs Include="--pre-js "%(_WasmExtraJSFile.Identity)"" Condition="'%(_WasmExtraJSFile.Kind)' == 'pre-js'" /> + <_EmccLinkStepArgs Include="--extern-pre-js "%(_WasmExtraJSFile.Identity)"" Condition="'%(_WasmExtraJSFile.Kind)' == 'extern-pre-js'" /> <_EmccLinkStepArgs Include="--post-js "%(_WasmExtraJSFile.Identity)"" Condition="'%(_WasmExtraJSFile.Kind)' == 'post-js'" /> - <_WasmLinkDependencies Include="@(_WasmExtraJSFile)" Condition="'%(_WasmExtraJSFile.Kind)' == 'js-library' or '%(_WasmExtraJSFile.Kind)' == 'pre-js' or '%(_WasmExtraJSFile.Kind)' == 'post-js'" /> + <_WasmLinkDependencies Include="@(_WasmExtraJSFile)" Condition="'%(_WasmExtraJSFile.Kind)' == 'js-library' or '%(_WasmExtraJSFile.Kind)' == 'pre-js' or '%(_WasmExtraJSFile.Kind)' == 'post-js' or '%(_WasmExtraJSFile.Kind)' == 'extern-post-js'" /> <_EmccLinkStepArgs Include=""%(_WasmNativeFileForLinking.Identity)"" /> <_WasmLinkDependencies Include="@(_WasmNativeFileForLinking)" /> diff --git a/src/mono/wasm/build/WasmApp.targets b/src/mono/wasm/build/WasmApp.targets index 9c96d759a228a..85195c1ae637a 100644 --- a/src/mono/wasm/build/WasmApp.targets +++ b/src/mono/wasm/build/WasmApp.targets @@ -299,7 +299,7 @@ diff --git a/src/mono/wasm/data/aot-tests/ProxyProjectForAOTOnHelix.proj b/src/mono/wasm/data/aot-tests/ProxyProjectForAOTOnHelix.proj index 65a0df19fb640..51dcde0db6c09 100644 --- a/src/mono/wasm/data/aot-tests/ProxyProjectForAOTOnHelix.proj +++ b/src/mono/wasm/data/aot-tests/ProxyProjectForAOTOnHelix.proj @@ -29,7 +29,7 @@ $(TestRootDir)AppBundle\ $(OriginalPublishDir)WasmTestRunner.dll - $(OriginalPublishDir)runtime-test.js + $(OriginalPublishDir)test-main.js true diff --git a/src/mono/wasm/debugger/DebuggerTestSuite/BreakpointTests.cs b/src/mono/wasm/debugger/DebuggerTestSuite/BreakpointTests.cs index 398df51a299dd..0e7f79f7a2a7a 100644 --- a/src/mono/wasm/debugger/DebuggerTestSuite/BreakpointTests.cs +++ b/src/mono/wasm/debugger/DebuggerTestSuite/BreakpointTests.cs @@ -27,8 +27,8 @@ public async Task CreateGoodBreakpoint() Assert.NotNull(loc["scriptId"]); Assert.Equal("dotnet://debugger-test.dll/debugger-test.cs", scripts[loc["scriptId"]?.Value()]); - Assert.Equal(10, loc["lineNumber"]); - Assert.Equal(8, loc["columnNumber"]); + Assert.Equal(10, (int)loc["lineNumber"]); + Assert.Equal(8, (int)loc["columnNumber"]); } [Fact] @@ -45,8 +45,8 @@ public async Task CreateJSBreakpoint() var loc = bp1_res.Value["locations"]?.Value()[0]; Assert.NotNull(loc["scriptId"]); - Assert.Equal(13, loc["lineNumber"]); - Assert.Equal(24, loc["columnNumber"]); + Assert.Equal(13, (int)loc["lineNumber"]); + Assert.Equal(24, (int)loc["columnNumber"]); var bp2_res = await SetBreakpoint("/debugger-driver.html", 13, 33); @@ -56,8 +56,8 @@ public async Task CreateJSBreakpoint() var loc2 = bp2_res.Value["locations"]?.Value()[0]; Assert.NotNull(loc2["scriptId"]); - Assert.Equal(13, loc2["lineNumber"]); - Assert.Equal(33, loc2["columnNumber"]); + Assert.Equal(13, (int)loc2["lineNumber"]); + Assert.Equal(33, (int)loc2["columnNumber"]); } [Fact] @@ -73,8 +73,8 @@ public async Task CreateJS0Breakpoint() var loc = bp1_res.Value["locations"]?.Value()[0]; Assert.NotNull(loc["scriptId"]); - Assert.Equal(13, loc["lineNumber"]); - Assert.Equal(24, loc["columnNumber"]); + Assert.Equal(13, (int)loc["lineNumber"]); + Assert.Equal(24, (int)loc["columnNumber"]); var bp2_res = await SetBreakpoint("/debugger-driver.html", 13, 33); @@ -84,8 +84,8 @@ public async Task CreateJS0Breakpoint() var loc2 = bp2_res.Value["locations"]?.Value()[0]; Assert.NotNull(loc2["scriptId"]); - Assert.Equal(13, loc2["lineNumber"]); - Assert.Equal(33, loc2["columnNumber"]); + Assert.Equal(13, (int)loc2["lineNumber"]); + Assert.Equal(33, (int)loc2["columnNumber"]); } [Theory] diff --git a/src/mono/wasm/debugger/tests/debugger-test/debugger-driver.html b/src/mono/wasm/debugger/tests/debugger-test/debugger-driver.html index 07b06b95fe9e8..a1cce5ee2b8ca 100644 --- a/src/mono/wasm/debugger/tests/debugger-test/debugger-driver.html +++ b/src/mono/wasm/debugger/tests/debugger-test/debugger-driver.html @@ -88,7 +88,7 @@ window.location.replace("http://localhost:9400/non-wasm-page.html"); } - + Stuff goes here diff --git a/src/mono/wasm/debugger/tests/debugger-test/debugger-main.js b/src/mono/wasm/debugger/tests/debugger-test/debugger-main.js new file mode 100644 index 0000000000000..c34cdf6bae430 --- /dev/null +++ b/src/mono/wasm/debugger/tests/debugger-test/debugger-main.js @@ -0,0 +1,23 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +"use strict"; + +var Module = { + configSrc: "./mono-config.json", + onConfigLoaded: () => { + MONO.config.environment_variables["DOTNET_MODIFIABLE_ASSEMBLIES"] = "debug"; + // For custom logging patch the functions below + /* + MONO.config.environment_variables["MONO_LOG_LEVEL"] = "debug"; + MONO.config.environment_variables["MONO_LOG_MASK"] = "all"; + INTERNAL.logging = { + trace: function (domain, log_level, message, isFatal, dataPtr) { }, + debugger: function (level, message) { } + }; + */ + }, + onDotNetReady: () => { + App.init(); + }, +}; diff --git a/src/mono/wasm/debugger/tests/debugger-test/debugger-test.csproj b/src/mono/wasm/debugger/tests/debugger-test/debugger-test.csproj index 46bbe3c6d4144..c3cc1800aad87 100644 --- a/src/mono/wasm/debugger/tests/debugger-test/debugger-test.csproj +++ b/src/mono/wasm/debugger/tests/debugger-test/debugger-test.csproj @@ -12,7 +12,7 @@ - + @@ -28,7 +28,7 @@ false $(AppDir) - $(MonoProjectRoot)wasm\runtime-test.js + $(MonoProjectRoot)wasm\test-main.js -1 diff --git a/src/mono/wasm/debugger/tests/debugger-test/runtime-debugger.js b/src/mono/wasm/debugger/tests/debugger-test/runtime-debugger.js deleted file mode 100644 index 7eea4d3c51599..0000000000000 --- a/src/mono/wasm/debugger/tests/debugger-test/runtime-debugger.js +++ /dev/null @@ -1,24 +0,0 @@ -// Licensed to the .NET Foundation under one or more agreements. -// The .NET Foundation licenses this file to you under the MIT license. - -"use strict"; - -var Module = { - config: null, - configSrc: "./mono-config.json", - onConfigLoaded: function () { - MONO.config.environment_variables["DOTNET_MODIFIABLE_ASSEMBLIES"] = "debug"; - // For custom logging patch the functions below - /* - MONO.config.environment_variables["MONO_LOG_LEVEL"] = "debug"; - MONO.config.environment_variables["MONO_LOG_MASK"] = "all"; - INTERNAL.logging = { - trace: function (domain, log_level, message, isFatal, dataPtr) { }, - debugger: function (level, message) { } - }; - */ - }, - onDotNetReady: function () { - App.init(); - }, -}; diff --git a/src/mono/wasm/runtime/CMakeLists.txt b/src/mono/wasm/runtime/CMakeLists.txt index 450d8df3918b8..4a67bcfe1a816 100644 --- a/src/mono/wasm/runtime/CMakeLists.txt +++ b/src/mono/wasm/runtime/CMakeLists.txt @@ -25,7 +25,7 @@ target_link_libraries(dotnet set_target_properties(dotnet PROPERTIES LINK_DEPENDS "${NATIVE_BIN_DIR}/src/emcc-default.rsp;${NATIVE_BIN_DIR}/src/runtime.iffe.js;${SOURCE_DIR}/library-dotnet.js;${SYSTEM_NATIVE_DIR}/pal_random.js" - LINK_FLAGS "@${NATIVE_BIN_DIR}/src/emcc-default.rsp ${CONFIGURATION_LINK_FLAGS} -DENABLE_NETCORE=1 --pre-js ${NATIVE_BIN_DIR}/src/runtime.iffe.js --js-library ${SOURCE_DIR}/library-dotnet.js --js-library ${SYSTEM_NATIVE_DIR}/pal_random.js" + LINK_FLAGS "@${NATIVE_BIN_DIR}/src/emcc-default.rsp ${CONFIGURATION_LINK_FLAGS} -DENABLE_NETCORE=1 --extern-pre-js ${NATIVE_BIN_DIR}/src/runtime.iffe.js --js-library ${SOURCE_DIR}/library-dotnet.js --js-library ${SYSTEM_NATIVE_DIR}/pal_random.js" RUNTIME_OUTPUT_DIRECTORY "${NATIVE_BIN_DIR}") if(CMAKE_BUILD_TYPE STREQUAL "Release") diff --git a/src/mono/wasm/runtime/export-types.ts b/src/mono/wasm/runtime/export-types.ts new file mode 100644 index 0000000000000..7c1be00a98848 --- /dev/null +++ b/src/mono/wasm/runtime/export-types.ts @@ -0,0 +1,7 @@ +import { DotNetExports } from "./exports"; +import { EmscriptenModuleConfig } from "./types"; + +// this is the only public export from the dotnet.js module +declare function createDotnetRuntime(moduleFactory: (api: DotNetExports) => EmscriptenModuleConfig): Promise; +export default createDotnetRuntime; + diff --git a/src/mono/wasm/runtime/exports.ts b/src/mono/wasm/runtime/exports.ts index 3dd3c12362309..024276a086b7b 100644 --- a/src/mono/wasm/runtime/exports.ts +++ b/src/mono/wasm/runtime/exports.ts @@ -20,7 +20,7 @@ import { mono_wasm_raise_debug_event, mono_wasm_fire_debugger_agent_message, } from "./debug"; -import { runtimeHelpers, setLegacyModules } from "./modules"; +import { runtimeHelpers, setAPI } from "./modules"; import { EmscriptenModuleMono, MonoArray, MonoConfig, MonoConfigError, MonoObject } from "./types"; import { mono_load_runtime_and_bcl_args, mono_wasm_load_config, @@ -63,7 +63,7 @@ import { getU8, getU16, getU32, getF32, getF64, } from "./memory"; -export const MONO: MONO = { +const MONO: MONO = { // current "public" MONO API mono_wasm_setenv, mono_wasm_load_bytes_into_heap, @@ -88,7 +88,7 @@ export const MONO: MONO = { mono_wasm_new_roots, }; -export const BINDING: BINDING = { +const BINDING: BINDING = { //current "public" BINDING API mono_obj_array_new: cwraps.mono_wasm_obj_array_new, mono_obj_array_set: cwraps.mono_wasm_obj_array_set, @@ -115,23 +115,48 @@ export const BINDING: BINDING = { // this is executed early during load of emscripten runtime // it exports methods to global objects MONO, BINDING and Module in backward compatible way // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types -function export_to_emscripten(dotnet: any, mono: any, binding: any, internal: any, module: any): void { +function exportAPI(mono: any, binding: any, internal: any, module: any, isGlobal: boolean, isNode: boolean, isShell: boolean, isWeb: boolean, glocateFile: Function): void { const moduleExt = module as EmscriptenModuleMono; + const globalThisAny = globalThis as any; // we want to have same instance of MONO, BINDING and Module in dotnet iffe - setLegacyModules(dotnet, mono, binding, internal, module); + setAPI(mono, binding, internal, module, isGlobal, isNode, isShell, isWeb, glocateFile); - // here we merge methods to it from the local objects - Object.assign(dotnet, DOTNET); + // here we merge methods from the local objects into exported objects Object.assign(mono, MONO); Object.assign(binding, BINDING); Object.assign(internal, INTERNAL); + const exports: DotNetExports = { + MONO: mono, + BINDING: binding, + INTERNAL: internal, + Module: module + }; + + if (moduleExt.configSrc) { + // this could be overriden on Module + if (!module.preInit) { + module.preInit = []; + } else if (typeof module.preInit === "function") { + module.preInit = [module.preInit]; + } + module.preInit.unshift(mono_wasm_pre_init); + } // this could be overriden on Module - moduleExt.preInit = mono_wasm_pre_init; - moduleExt.onRuntimeInitialized = mono_wasm_on_runtime_initialized; + if (!moduleExt.onRuntimeInitialized) { + moduleExt.onRuntimeInitialized = mono_wasm_on_runtime_initialized; + } + if (!moduleExt.print) { + moduleExt.print = console.log; + } + if (!moduleExt.printErr) { + moduleExt.printErr = console.error; + } + + if (isGlobal || !moduleExt.disableDotNet6Compatibility) { + Object.assign(module, exports); - if (!moduleExt.disableDotNet6Compatibility) { // backward compatibility // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore @@ -141,38 +166,42 @@ function export_to_emscripten(dotnet: any, mono: any, binding: any, internal: an }; // here we expose objects used in tests to global namespace - (globalThis).Module = module; - const warnWrap = (name: string, value: any) => { - if (typeof ((globalThis)[name]) !== "undefined") { + const warnWrap = (name: string, provider: () => any) => { + if (typeof globalThisAny[name] !== "undefined") { // it already exists in the global namespace return; } - let warnOnce = true; + let value: any = undefined; Object.defineProperty(globalThis, name, { get: () => { - if (warnOnce) { + if (!value) { const stack = (new Error()).stack; const nextLine = stack ? stack.substr(stack.indexOf("\n", 8) + 1) : ""; console.warn(`global ${name} is obsolete, please use Module.${name} instead ${nextLine}`); - warnOnce = false; + value = provider(); } return value; } }); }; - warnWrap("MONO", mono); - warnWrap("BINDING", binding); + globalThisAny.MONO = mono; + globalThisAny.BINDING = binding; + globalThisAny.INTERNAL = internal; + if (!isGlobal) { + globalThisAny.Module = module; + } // Blazor back compat - warnWrap("cwrap", Module.cwrap); - warnWrap("addRunDependency", Module.addRunDependency); - warnWrap("removeRunDependency", Module.removeRunDependency); + warnWrap("cwrap", () => module.cwrap); + warnWrap("addRunDependency", () => module.addRunDependency); + warnWrap("removeRunDependency", () => module.removeRunDependency); } } +export const __exportAPI: any = exportAPI; // don't want to export the type // the methods would be visible to EMCC linker -// --- keep in sync with library-dotnet.js --- -const linker_exports = { +// --- keep in sync with dotnet.lib.js --- +export const __linker_exports: any = { // mini-wasm.c mono_set_timeout, @@ -214,16 +243,10 @@ const linker_exports = { mono_wasm_load_icu_data, mono_wasm_get_icudt_name, }; -export const DOTNET: any = { -}; -export const INTERNAL: any = { +const INTERNAL: any = { // startup BINDING_ASM: "[System.Private.Runtime.InteropServices.JavaScript]System.Runtime.InteropServices.JavaScript.Runtime", - export_to_emscripten, - - // linker - linker_exports: linker_exports, // tests call_static_method, @@ -281,7 +304,7 @@ export const INTERNAL: any = { // this represents visibility in the javascript // like https://github.com/dotnet/aspnetcore/blob/main/src/Components/Web.JS/src/Platform/Mono/MonoTypes.ts -export interface MONO { +interface MONO { mono_wasm_runtime_ready: typeof mono_wasm_runtime_ready mono_wasm_setenv: typeof mono_wasm_setenv mono_wasm_load_data_archive: typeof mono_wasm_load_data_archive; @@ -294,8 +317,8 @@ export interface MONO { mono_wasm_release_roots: typeof mono_wasm_release_roots; // for Blazor's future! - mono_wasm_add_assembly: typeof cwraps.mono_wasm_add_assembly, - mono_wasm_load_runtime: typeof cwraps.mono_wasm_load_runtime, + mono_wasm_add_assembly: (name: string, data: VoidPtr, size: number) => number, + mono_wasm_load_runtime: (unused: string, debug_level: number) => void, loaded_files: string[]; config: MonoConfig | MonoConfigError, @@ -303,7 +326,7 @@ export interface MONO { // this represents visibility in the javascript // like https://github.com/dotnet/aspnetcore/blob/main/src/Components/Web.JS/src/Platform/Mono/MonoTypes.ts -export interface BINDING { +interface BINDING { mono_obj_array_new: (size: number) => MonoArray, mono_obj_array_set: (array: MonoArray, idx: number, obj: MonoObject) => void, js_string_to_mono_string: typeof js_string_to_mono_string, @@ -314,4 +337,9 @@ export interface BINDING { bind_static_method: typeof mono_bind_static_method, call_assembly_entry_point: typeof mono_call_assembly_entry_point, unbox_mono_obj: typeof unbox_mono_obj +} +export interface DotNetExports { + MONO: MONO, + BINDING: BINDING, + Module: any } \ No newline at end of file diff --git a/src/mono/wasm/runtime/library-dotnet.js b/src/mono/wasm/runtime/library-dotnet.js index a999659b1949d..d716a7098061a 100644 --- a/src/mono/wasm/runtime/library-dotnet.js +++ b/src/mono/wasm/runtime/library-dotnet.js @@ -10,7 +10,7 @@ const DotNetSupportLib = { $BINDING: {}, $INTERNAL: {}, // this line will be executed on runtime, populating the objects with methods - $DOTNET__postset: "__dotnet_runtime.INTERNAL.export_to_emscripten (DOTNET, MONO, BINDING, INTERNAL, Module);", + $DOTNET__postset: "__dotnet_runtime.__exportAPI(MONO, BINDING, INTERNAL, Module, true, ENVIRONMENT_IS_NODE, ENVIRONMENT_IS_SHELL, ENVIRONMENT_IS_WEB, locateFile);", }; // the methods would be visible to EMCC linker @@ -61,7 +61,7 @@ const linked_functions = [ // -- this javascript file is evaluated by emcc during compilation! -- // we generate simple proxy for each exported function so that emcc will include them in the final output for (let linked_function of linked_functions) { - const fn_template = `return __dotnet_runtime.INTERNAL.linker_exports.${linked_function}.apply(__dotnet_runtime, arguments)`; + const fn_template = `return __dotnet_runtime.__linker_exports.${linked_function}.apply(__dotnet_runtime, arguments)`; DotNetSupportLib[linked_function] = new Function(fn_template); } diff --git a/src/mono/wasm/runtime/method-binding.ts b/src/mono/wasm/runtime/method-binding.ts index a9b2ebb00d35a..cc15129f87ee5 100644 --- a/src/mono/wasm/runtime/method-binding.ts +++ b/src/mono/wasm/runtime/method-binding.ts @@ -3,13 +3,13 @@ import { WasmRoot, WasmRootBuffer, mono_wasm_new_root } from "./roots"; import { MonoClass, MonoMethod, MonoObject, coerceNull, VoidPtrNull, VoidPtr, MonoType } from "./types"; -import { BINDING, runtimeHelpers } from "./modules"; +import { BINDING, Module, runtimeHelpers } from "./modules"; import { js_to_mono_enum, _js_to_mono_obj, _js_to_mono_uri } from "./js-to-cs"; import { js_string_to_mono_string, js_string_to_mono_string_interned } from "./strings"; import { MarshalType, _unbox_mono_obj_root_with_known_nonprimitive_type } from "./cs-to-js"; -import { - _create_temp_frame, - getI32, getU32, getF32, getF64, +import { + _create_temp_frame, + getI32, getU32, getF32, getF64, setI32, setU32, setF32, setF64, setI64, } from "./memory"; import { diff --git a/src/mono/wasm/runtime/modules.ts b/src/mono/wasm/runtime/modules.ts index 3dfa33312712b..9b30b8affa96f 100644 --- a/src/mono/wasm/runtime/modules.ts +++ b/src/mono/wasm/runtime/modules.ts @@ -10,16 +10,24 @@ import { EmscriptenModuleMono, MonoConfig, RuntimeHelpers } from "./types"; export let Module: EmscriptenModule & EmscriptenModuleMono; export let MONO: any; export let BINDING: any; -export let DOTNET: any; export let INTERNAL: any; +export let ENVIRONMENT_IS_GLOBAL: boolean; +export let ENVIRONMENT_IS_NODE: boolean; +export let ENVIRONMENT_IS_SHELL: boolean; +export let ENVIRONMENT_IS_WEB: boolean; +export let locateFile: Function; // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types -export function setLegacyModules(dotnet: any, mono: any, binding: any, internal: any, module: EmscriptenModule & EmscriptenModuleMono) { - DOTNET = dotnet; +export function setAPI(mono: any, binding: any, internal: any, module: EmscriptenModule & EmscriptenModuleMono, isGlobal: boolean, isNode: boolean, isShell: boolean, isWeb: boolean, glocateFile: Function) { MONO = mono; BINDING = binding; INTERNAL = internal; Module = module; + ENVIRONMENT_IS_GLOBAL = isGlobal; + ENVIRONMENT_IS_NODE = isNode; + ENVIRONMENT_IS_SHELL = isShell; + ENVIRONMENT_IS_WEB = isWeb; + locateFile = glocateFile; } let monoConfig: MonoConfig; diff --git a/src/mono/wasm/runtime/package.json b/src/mono/wasm/runtime/package.json index dae87adeaef12..2958e9d2f22b9 100644 --- a/src/mono/wasm/runtime/package.json +++ b/src/mono/wasm/runtime/package.json @@ -8,7 +8,7 @@ "version": "1.0.0", "scripts": { "rollup": "rollup -c", - "lint": "eslint --no-color --max-warnings=0 ./**/*.ts ./**/*.js" + "lint": "eslint --no-color --max-warnings=0 ./**/*.ts ./*.js" }, "keywords": [ "dotnet", diff --git a/src/mono/wasm/runtime/rollup.config.js b/src/mono/wasm/runtime/rollup.config.js index d3bca849bef4a..33bbb20e44845 100644 --- a/src/mono/wasm/runtime/rollup.config.js +++ b/src/mono/wasm/runtime/rollup.config.js @@ -11,13 +11,15 @@ const isDebug = process.env.Configuration !== "Release"; const nativeBinDir = process.env.NativeBinDir ? process.env.NativeBinDir.replace(/"/g, "") : "bin"; const terserConfig = { compress: { - defaults: false,// to agressive minification breaks subsequent emcc compilation + defaults: false,// too agressive minification breaks subsequent emcc compilation drop_debugger: false,// we invoke debugger drop_console: false,// we log to console unused: false,// this breaks stuff // below are minification features which seems to work fine collapse_vars: true, conditionals: true, + computed_props: true, + properties: true, dead_code: true, if_return: true, inline: true, @@ -32,39 +34,40 @@ const terserConfig = { // because of stack walk at src/mono/wasm/debugger/BrowserDebugProxy/MonoProxy.cs keep_fnames: /(mono_wasm_runtime_ready|mono_wasm_fire_debugger_agent_message)/, }, - // we export ES5 because emcc parser has trouble parsing it otherwise - ecma: 5, }; const plugins = isDebug ? [writeOnChangePlugin()] : [terser(terserConfig), writeOnChangePlugin()]; +const banner = "//! Licensed to the .NET Foundation under one or more agreements.\n//! The .NET Foundation licenses this file to you under the MIT license.\n"; +// emcc doesn't know how to load ES6 module, that's why we need the whole rollup.js +const format = "iife"; +const name = "__dotnet_runtime"; export default defineConfig([ { treeshake: !isDebug, input: "exports.ts", output: [{ - banner: "//! Licensed to the .NET Foundation under one or more agreements.\n//! The .NET Foundation licenses this file to you under the MIT license.\n", - name: "__dotnet_runtime", file: nativeBinDir + "/src/" + outputFileName, - - // emcc doesn't know how to load ES6 module, that's why we need the whole rollup.js - format: "iife", - plugins: plugins + name, + banner, + format, + plugins, }], plugins: [typescript()] }, { - input: "./exports.ts", + input: "./export-types.ts", output: [ + // dotnet.d.ts { format: "es", file: nativeBinDir + "/src/" + "dotnet.d.ts", } ], plugins: [dts()], - }, + } ]); -// this would create .md5 file next to the output file, so that we do not touch datetime of the file if it's same -> faster incremental build. +// this would create .sha256 file next to the output file, so that we do not touch datetime of the file if it's same -> faster incremental build. function writeOnChangePlugin() { return { name: "writeOnChange", diff --git a/src/mono/wasm/runtime/startup.ts b/src/mono/wasm/runtime/startup.ts index df61a27399d86..083945818996a 100644 --- a/src/mono/wasm/runtime/startup.ts +++ b/src/mono/wasm/runtime/startup.ts @@ -1,8 +1,8 @@ // Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the MIT license. -import { INTERNAL, Module, MONO, runtimeHelpers } from "./modules"; import { AllAssetEntryTypes, AssetEntry, CharPtr, CharPtrNull, EmscriptenModuleMono, GlobalizationMode, MonoConfig, TypedArray, VoidPtr, wasm_type_symbol } from "./types"; +import { ENVIRONMENT_IS_NODE, ENVIRONMENT_IS_SHELL, ENVIRONMENT_IS_WEB, INTERNAL, locateFile, Module, MONO, runtimeHelpers } from "./modules"; import cwraps from "./cwraps"; import { mono_wasm_raise_debug_event, mono_wasm_runtime_ready } from "./debug"; import { mono_wasm_globalization_init, mono_wasm_load_icu_data } from "./icu"; @@ -12,11 +12,25 @@ import { mono_wasm_load_bytes_into_heap } from "./buffers"; import { bind_runtime_method, get_method, _create_primitive_converters } from "./method-binding"; import { find_corlib_class } from "./class-loader"; +export let runtime_is_initialized_resolve: Function; +export let runtime_is_initialized_reject: Function; +export const mono_wasm_runtime_is_initialized = new Promise((resolve, reject) => { + runtime_is_initialized_resolve = resolve; + runtime_is_initialized_reject = reject; +}); + + export async function mono_wasm_pre_init(): Promise { const moduleExt = Module as EmscriptenModuleMono; if (moduleExt.configSrc) { - // sets MONO.config implicitly - await mono_wasm_load_config(moduleExt.configSrc); + try { + // sets MONO.config implicitly + await mono_wasm_load_config(moduleExt.configSrc); + } + catch (err: any) { + runtime_is_initialized_reject(err); + throw err; + } if (moduleExt.onConfigLoaded) { try { @@ -26,6 +40,7 @@ export async function mono_wasm_pre_init(): Promise { Module.printErr("MONO_WASM: onConfigLoaded () failed: " + err); Module.printErr("MONO_WASM: Stacktrace: \n"); Module.printErr(err.stack); + runtime_is_initialized_reject(err); throw err; } } @@ -223,6 +238,20 @@ function _finalize_startup(args: MonoConfig, ctx: MonoInitContext) { mono_wasm_setenv("TZ", tz || "UTC"); mono_wasm_runtime_ready(); + //legacy config loading + const argsAny: any = args; + if (argsAny.loaded_cb) { + try { + argsAny.loaded_cb(); + } + catch (err: any) { + Module.printErr("MONO_WASM: loaded_cb () failed: " + err); + Module.printErr("MONO_WASM: Stacktrace: \n"); + Module.printErr(err.stack); + throw err; + } + } + if (moduleExt.onDotNetReady) { try { moduleExt.onDotNetReady(); @@ -234,6 +263,8 @@ function _finalize_startup(args: MonoConfig, ctx: MonoInitContext) { throw err; } } + + runtime_is_initialized_resolve(); } export function bindings_lazy_init(): void { diff --git a/src/mono/wasm/runtime/tsconfig.json b/src/mono/wasm/runtime/tsconfig.json index a1f0c30602189..63d7619e09f9f 100644 --- a/src/mono/wasm/runtime/tsconfig.json +++ b/src/mono/wasm/runtime/tsconfig.json @@ -4,7 +4,7 @@ "noEmitOnError": true, "removeComments": false, "sourceMap": false, - "target": "ES5", + "target": "ES2018", "moduleResolution": "Node", "lib": [ "esnext", diff --git a/src/mono/wasm/runtime/types.ts b/src/mono/wasm/runtime/types.ts index c2db47aae5cc4..d452649c96b2e 100644 --- a/src/mono/wasm/runtime/types.ts +++ b/src/mono/wasm/runtime/types.ts @@ -179,7 +179,9 @@ export type CoverageProfilerOptions = { } // how we extended emscripten Module -export type EmscriptenModuleMono = EmscriptenModule & { +export type EmscriptenModuleMono = EmscriptenModule & EmscriptenModuleConfig; + +export type EmscriptenModuleConfig = { disableDotNet6Compatibility?: boolean, // backward compatibility diff --git a/src/mono/wasm/runtime/types/emscripten.d.ts b/src/mono/wasm/runtime/types/emscripten.d.ts index 5ddcea2beb8e7..39df45d862e93 100644 --- a/src/mono/wasm/runtime/types/emscripten.d.ts +++ b/src/mono/wasm/runtime/types/emscripten.d.ts @@ -22,20 +22,6 @@ declare interface CharPtrPtr extends NativePointer { __brand: "CharPtrPtr" } -declare let ENVIRONMENT_IS_WEB: boolean; -declare let ENVIRONMENT_IS_SHELL: boolean; -declare let ENVIRONMENT_IS_NODE: boolean; -declare let ENVIRONMENT_IS_WORKER: boolean; -declare let LibraryManager: any; - -declare function autoAddDeps(a: object, b: string): void; -declare function mergeInto(a: object, b: object): void; - -// TODO, what's wrong with EXPORTED_RUNTIME_METHODS ? -declare function locateFile(path: string, prefix?: string): string; - -declare let Module: EmscriptenModule; - declare interface EmscriptenModule { HEAP8: Int8Array, HEAP16: Int16Array; @@ -66,7 +52,7 @@ declare interface EmscriptenModule { removeRunDependency(id: string): void; addRunDependency(id: string): void; - preInit?: () => Promise; + preInit?: (() => Promise)[]; onRuntimeInitialized?: () => void; } diff --git a/src/mono/wasm/runtime/web-socket.ts b/src/mono/wasm/runtime/web-socket.ts index 00dac60a7f490..7f50a68c4e82e 100644 --- a/src/mono/wasm/runtime/web-socket.ts +++ b/src/mono/wasm/runtime/web-socket.ts @@ -11,6 +11,7 @@ import { _wrap_js_thenable_as_task } from "./js-to-cs"; import { wrap_error } from "./method-calls"; import { conv_string } from "./strings"; import { Int32Ptr, JSHandle, MonoArray, MonoObject, MonoObjectNull, MonoString } from "./types"; +import { Module } from "./modules"; const wasm_ws_pending_send_buffer = Symbol.for("wasm ws_pending_send_buffer"); const wasm_ws_pending_send_buffer_offset = Symbol.for("wasm ws_pending_send_buffer_offset"); diff --git a/src/mono/wasm/runtime-test.js b/src/mono/wasm/test-main.js similarity index 82% rename from src/mono/wasm/runtime-test.js rename to src/mono/wasm/test-main.js index 60ff70d065fe6..a0b2253093b7b 100644 --- a/src/mono/wasm/runtime-test.js +++ b/src/mono/wasm/test-main.js @@ -8,7 +8,7 @@ //glue code to deal with the differences between chrome, ch, d8, jsc and sm. const is_browser = typeof window != "undefined"; -const is_node = !is_browser && typeof process != 'undefined'; +const is_node = !is_browser && typeof process === 'object' && typeof process.versions === 'object' && typeof process.versions.node === 'string'; // if the engine doesn't provide a console if (typeof (console) === "undefined") { @@ -101,18 +101,12 @@ if (typeof globalThis.performance === 'undefined') { } } } - var Module = { - no_global_exports: true, - mainScriptUrlOrBlob: "dotnet.js", config: null, configSrc: "./mono-config.json", - print: console.log, - printErr: console.error, - onConfigLoaded: function () { + onConfigLoaded: () => { if (!Module.config) { - console.error("Could not find ./mono-config.json. Cancelling run"); - fail_exec(1); + set_exit_code(1, "Could not find ./mono-config.json. Cancelling run"); } // Have to set env vars here to enable setting MONO_LOG_LEVEL etc. for (let variable in processedArguments.setenv) { @@ -123,10 +117,10 @@ var Module = { INTERNAL.mono_wasm_enable_on_demand_gc(0); } }, - onDotNetReady: function () { + onDotNetReady: () => { let wds = Module.FS.stat(processedArguments.working_dir); if (wds === undefined || !Module.FS.isDir(wds.mode)) { - fail_exec(1, `Could not find working directory ${processedArguments.working_dir}`); + set_exit_code(1, `Could not find working directory ${processedArguments.working_dir}`); return; } @@ -134,26 +128,26 @@ var Module = { App.init(); }, - onAbort: function (x) { - console.log("ABORT: " + x); + onAbort: (error) => { + console.log("ABORT: " + error); const err = new Error(); console.log("Stacktrace: \n"); console.error(err.stack); - fail_exec(1); + set_exit_code(1, error); }, }; + const App = { - init: function () { + init: async function () { console.info("Initializing....."); - for (let i = 0; i < processedArguments.profilers.length; ++i) { const init = Module.cwrap('mono_wasm_load_profiler_' + processedArguments.profilers[i], 'void', ['string']); init(""); } if (processedArguments.applicationArgs.length == 0) { - fail_exec(1, "Missing required --run argument"); + set_exit_code(1, "Missing required --run argument"); return; } @@ -171,7 +165,7 @@ const App = { } if (res) - fail_exec(1, "REGRESSION TEST FAILED"); + set_exit_code(1, "REGRESSION TEST FAILED"); return; } @@ -182,31 +176,23 @@ const App = { if (processedArguments.applicationArgs[0] == "--run") { // Run an exe if (processedArguments.applicationArgs.length == 1) { - fail_exec(1, "Error: Missing main executable argument."); + set_exit_code(1, "Error: Missing main executable argument."); return; } + try { - const main_assembly_name = processedArguments.applicationArgs[1]; - const app_args = processedArguments.applicationArgs.slice(2); - INTERNAL.mono_wasm_set_main_args(processedArguments.applicationArgs[1], app_args); - - // Automatic signature isn't working correctly - const result = BINDING.call_assembly_entry_point(main_assembly_name, [app_args], "m"); - const onError = function (error) { - console.error(error); - if (error.stack) - console.error(error.stack); + const main_assembly_name = processedArguments.applicationArgs[1]; + const app_args = processedArguments.applicationArgs.slice(2); + INTERNAL.mono_wasm_set_main_args(processedArguments.applicationArgs[1], app_args); - fail_exec(1); - } - try { - result.then(fail_exec).catch(onError); + // Automatic signature isn't working correctly + const result = await BINDING.call_assembly_entry_point(main_assembly_name, [app_args], "m"); + set_exit_code(result); } catch (error) { - onError(error); + set_exit_code(1, error); } - } else { - fail_exec(1, "Unhandled argument: " + processedArguments.applicationArgs[0]); + set_exit_code(1, "Unhandled argument: " + processedArguments.applicationArgs[0]); } }, @@ -229,19 +215,24 @@ const App = { }; globalThis.App = App; // Necessary as System.Runtime.InteropServices.JavaScript.Tests.MarshalTests (among others) call the App.call_test_method directly -function fail_exec(exit_code, reason) { +function set_exit_code(exit_code, reason) { if (reason) { - console.error(reason); + console.error(reason.toString()); + if (reason.stack) { + console.error(reason.stack); + } } if (is_browser) { // Notify the selenium script Module.exit_code = exit_code; - originalConsole.log("WASM EXIT " + exit_code); const tests_done_elem = document.createElement("label"); tests_done_elem.id = "tests_done"; tests_done_elem.innerHTML = exit_code.toString(); document.body.appendChild(tests_done_elem); - } else { // shell or node + + // tell xharness stream procesor we are done + console.log("WASM EXIT " + exit_code); + } else if (INTERNAL) { INTERNAL.mono_wasm_exit(exit_code); } } @@ -262,7 +253,7 @@ function processArguments(incomingArguments) { const arg = currentArg.substring("--setenv=".length); const parts = arg.split('='); if (parts.length != 2) - fail_exec(1, "Error: malformed argument: '" + currentArg); + set_exit_code(1, "Error: malformed argument: '" + currentArg); setenv[parts[0]] = parts[1]; } else if (currentArg.startsWith("--runtime-arg=")) { const arg = currentArg.substring("--runtime-arg=".length); @@ -322,8 +313,10 @@ try { async function loadDotnet(file) { let loadScript = undefined; if (typeof WScript !== "undefined") { // Chakra - loadScript = WScript.LoadScriptFile; - return globalThis.Module; + loadScript = function (file) { + WScript.LoadScriptFile(file); + return globalThis.Module; + }; } else if (is_node) { // NodeJS loadScript = async function (file) { return require(file); @@ -333,7 +326,17 @@ async function loadDotnet(file) { const script = document.createElement("script"); script.src = file; document.head.appendChild(script); - return globalThis.Module; + let timeout = 100; + // bysy spin waiting for script to load into global namespace + while (timeout > 0) { + if (globalThis.Module) { + return globalThis.Module; + } + // delay 10ms + await new Promise(resolve => setTimeout(resolve, 10)); + timeout--; + } + throw new Error("Can't load " + file); } } else if (typeof globalThis.load !== 'undefined') { @@ -351,6 +354,6 @@ async function loadDotnet(file) { loadDotnet("./dotnet.js").catch(function (err) { console.error(err); - fail_exec(1, "failed to load the dotnet.js file"); + set_exit_code(1, "failed to load the dotnet.js file"); throw err; }); \ No newline at end of file diff --git a/src/tasks/WasmAppBuilder/WasmAppBuilder.cs b/src/tasks/WasmAppBuilder/WasmAppBuilder.cs index bf6708d4268f8..f592f488327cf 100644 --- a/src/tasks/WasmAppBuilder/WasmAppBuilder.cs +++ b/src/tasks/WasmAppBuilder/WasmAppBuilder.cs @@ -174,12 +174,12 @@ private bool ExecuteInternal () if (!FileCopyChecked(item.ItemSpec, dest, "NativeAssets")) return false; } - FileCopyChecked(MainJS!, Path.Combine(AppDir, "runtime.js"), string.Empty); + FileCopyChecked(MainJS!, Path.Combine(AppDir, "main.js"), string.Empty); string indexHtmlPath = Path.Combine(AppDir, "index.html"); if (!File.Exists(indexHtmlPath)) { - var html = @""; + var html = @""; File.WriteAllText(indexHtmlPath, html); } diff --git a/src/tests/BuildWasmApps/Wasm.Build.Tests/BuildTestBase.cs b/src/tests/BuildWasmApps/Wasm.Build.Tests/BuildTestBase.cs index b6fab5c6c434b..cc4aa85a2c39d 100644 --- a/src/tests/BuildWasmApps/Wasm.Build.Tests/BuildTestBase.cs +++ b/src/tests/BuildWasmApps/Wasm.Build.Tests/BuildTestBase.cs @@ -147,7 +147,7 @@ protected string RunAndTestWasmApp(BuildArgs buildArgs, string bundleDir = Path.Combine(GetBinDir(baseDir: buildDir, config: buildArgs.Config), "AppBundle"); (string testCommand, string extraXHarnessArgs) = host switch { - RunHost.V8 => ("wasm test", "--js-file=runtime.js --engine=V8 -v trace"), + RunHost.V8 => ("wasm test", "--js-file=main.js --engine=V8 -v trace"), _ => ("wasm test-browser", $"-v trace -b {host}") }; @@ -254,7 +254,7 @@ protected static void InitProjectDir(string dir) {s_targetFramework} Exe true - runtime-test.js + test-main.js ##EXTRA_PROPERTIES## @@ -310,7 +310,7 @@ protected static BuildArgs ExpandBuildArgs(BuildArgs buildArgs, string extraProp initProject?.Invoke(); File.WriteAllText(Path.Combine(_projectDir, $"{buildArgs.ProjectName}.csproj"), buildArgs.ProjectFileContents); - File.Copy(Path.Combine(AppContext.BaseDirectory, "runtime-test.js"), Path.Combine(_projectDir, "runtime-test.js")); + File.Copy(Path.Combine(AppContext.BaseDirectory, "test-main.js"), Path.Combine(_projectDir, "test-main.js")); } else if (_projectDir is null) { @@ -486,7 +486,7 @@ protected static void AssertBasicAppBundle(string bundleDir, string projectName, AssertFilesExist(bundleDir, new [] { "index.html", - "runtime.js", + "main.js", "dotnet.timezones.blat", "dotnet.wasm", "mono-config.json", diff --git a/src/tests/BuildWasmApps/Wasm.Build.Tests/SatelliteAssembliesTests.cs b/src/tests/BuildWasmApps/Wasm.Build.Tests/SatelliteAssembliesTests.cs index c3a40b470200a..3532028dcc491 100644 --- a/src/tests/BuildWasmApps/Wasm.Build.Tests/SatelliteAssembliesTests.cs +++ b/src/tests/BuildWasmApps/Wasm.Build.Tests/SatelliteAssembliesTests.cs @@ -148,7 +148,7 @@ private void CreateProgramForCultureTest(string dir, string resourceName, string {s_targetFramework} Exe true - runtime-test.js + test-main.js ##EXTRA_PROPERTIES## diff --git a/src/tests/BuildWasmApps/Wasm.Build.Tests/Wasm.Build.Tests.csproj b/src/tests/BuildWasmApps/Wasm.Build.Tests/Wasm.Build.Tests.csproj index 0b05047a21fec..7250c9e029dcd 100644 --- a/src/tests/BuildWasmApps/Wasm.Build.Tests/Wasm.Build.Tests.csproj +++ b/src/tests/BuildWasmApps/Wasm.Build.Tests/Wasm.Build.Tests.csproj @@ -27,7 +27,7 @@
- + diff --git a/src/tests/Common/Directory.Build.targets b/src/tests/Common/Directory.Build.targets index 009a19e92c775..e003e2dc6ec86 100644 --- a/src/tests/Common/Directory.Build.targets +++ b/src/tests/Common/Directory.Build.targets @@ -155,7 +155,7 @@ TargetDir="wasm-test-runner/"/> $(TestAssemblyFileName) $(AppDir) - $(CORE_ROOT)\runtime-test\runtime-test.js + $(CORE_ROOT)\runtime-test\test-main.js true true true diff --git a/src/tests/Directory.Build.targets b/src/tests/Directory.Build.targets index 2866110fb4c36..383ec39277934 100644 --- a/src/tests/Directory.Build.targets +++ b/src/tests/Directory.Build.targets @@ -387,7 +387,7 @@ TargetDir="wasm-test-runner/"/> true WasmTestOnBrowser 42 - runtime.js + main.js false - + diff --git a/src/tests/FunctionalTests/WebAssembly/Browser/HotReload/index.html b/src/tests/FunctionalTests/WebAssembly/Browser/HotReload/index.html index 7832e37351d5b..8f85c8d7d8d48 100644 --- a/src/tests/FunctionalTests/WebAssembly/Browser/HotReload/index.html +++ b/src/tests/FunctionalTests/WebAssembly/Browser/HotReload/index.html @@ -2,54 +2,41 @@ - - TESTS - - - - - - Result from Sample.Test.TestMeaning: - - + console.debug(`exit_code: ${exit_code}`); + set_exit_code(exit_code); + }, + }; + + + - + - - + \ No newline at end of file diff --git a/src/tests/FunctionalTests/WebAssembly/Browser/HotReload/runtime.js b/src/tests/FunctionalTests/WebAssembly/Browser/HotReload/main.js similarity index 72% rename from src/tests/FunctionalTests/WebAssembly/Browser/HotReload/runtime.js rename to src/tests/FunctionalTests/WebAssembly/Browser/HotReload/main.js index 065061d86b22d..861ddd263cd84 100644 --- a/src/tests/FunctionalTests/WebAssembly/Browser/HotReload/runtime.js +++ b/src/tests/FunctionalTests/WebAssembly/Browser/HotReload/main.js @@ -4,20 +4,19 @@ "use strict"; var Module = { - config: null, configSrc: "./mono-config.json", - onConfigLoaded: function () { + onConfigLoaded: () => { MONO.config.environment_variables["DOTNET_MODIFIABLE_ASSEMBLIES"] = "debug"; }, - onDotNetReady: function () { + onDotNetReady: () => { try { App.init(); } catch (error) { - test_exit(1); + set_exit_code(1, error); throw (error); } }, - onAbort: function () { - test_exit(1); + onAbort: (error) => { + set_exit_code(1, error); }, }; diff --git a/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/WebAssembly.Browser.RuntimeConfig.Test.csproj b/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/WebAssembly.Browser.RuntimeConfig.Test.csproj index 35e9eec2cae87..d9c7984051cd3 100644 --- a/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/WebAssembly.Browser.RuntimeConfig.Test.csproj +++ b/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/WebAssembly.Browser.RuntimeConfig.Test.csproj @@ -3,7 +3,7 @@ true WasmTestOnBrowser 42 - runtime.js + main.js diff --git a/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/index.html b/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/index.html index 2b618eaf31cbd..b8640040e73ac 100644 --- a/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/index.html +++ b/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/index.html @@ -2,54 +2,41 @@ - - TESTS - - - - - - Result from Sample.Test.TestMeaning: - - + console.debug(`exit_code: ${exit_code}`); + set_exit_code(exit_code); + }, + }; + + + - + - - + \ No newline at end of file diff --git a/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/runtime.js b/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/main.js similarity index 71% rename from src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/runtime.js rename to src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/main.js index 5c388258c859a..88db9c67c1dd9 100644 --- a/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/runtime.js +++ b/src/tests/FunctionalTests/WebAssembly/Browser/RuntimeConfig/main.js @@ -4,17 +4,16 @@ "use strict"; var Module = { - config: null, configSrc: "./mono-config.json", - onDotNetReady: function () { + onDotNetReady: () => { try { App.init(); } catch (error) { - test_exit(1); + set_exit_code(1, error); throw (error); } }, - onAbort: function () { - test_exit(1); + onAbort: (error) => { + set_exit_code(1, error); }, };