This is a sample showing how with the current .NET SDK (8 at time of writing), it
is possible to use the browser-wasm
template (in wasm-tools
workload) to
show direct graphics via WebGL (using SDL2 as a library) without using Blazor.
There are still a number of problems with .NET's WebAssembly support for this, which have to be worked around in each program/library, but it is now feasible to do this without requiring a custom build of the runtime/SDK to allow it to work. See Remaining .NET WebAssembly Issues for more info on this.
To see the difference from the templates, look at the list of commits in the
initial-net8
branch, which contains just the basic setup described below.
For other functionality, look at subsequent branches and their commits (e.g.
full AOT compilation).
If you're trying to build your project that uses a library that uses SDL internally (like MonoGame, FNA, etc), while SDL itself is easy to use effectively as-is, these other libraries are not as simple to use: see MonoGame/Library Usage of SDL.
Requirements to build/follow along with this repository:
- Install .NET 8 (or newer, but some commands/config options may change!)
- Install the
wasm-tools
workload:dotnet workload install wasm-tools
This project is effectively just:
- New project with
browser-wasm
template, - Import SDL C# binding (see Native Library Name section below),
- Import SDL native library built with Emscripten (see Emscripten Frozen Cache/Ports section below),
- Set up main loop binding for Emscripten (see Emscripten Main Loop section below),
- Set up a basic window class that just clears the screen,
With other branches containing follow-up commits that add various bits of functionality (AOT compilation, etc).
While a lot of the setup now works out of the box, there are a number of issues that are either stuck waiting on the dotnet team to change in the runtime/SDK, or they've decided against fixing and are probably going to be long-term issues.
This is not so much an issue as a consequence of the WebAssembly support still being early in development. The MSBuild parameters used to configure the Emscripten environment/build process are still in flux and likely to change either in name or behaviour over time.
Hopefully, most changes will be for the better, but it is possible some changes will break existing functionality without offering alternatives, if the dotnet team consider the functionality irrelevant or problematic.
Emscripten is used internally by the dotnet SDK as part of the build
process for projects of SDK type Microsoft.NET.Sdk.WebAssembly
. When building
the project, MSBuild will use a package called Microsoft.NET.Runtime.Emscripten
(on Windows, downloaded by default to C:\Program Files\dotnet\packs\
) to
download and store a copy of Emscripten specifically for compiling these projects.
At time of writing, this uses version 3.1.34
.
Emscripten has functionality called "ports" which are libraries provided automatically to Emscripten-built projects, to cover common library requirements (and provide functional WebAssembly versions of them). This includes both SDL 1.x and SDL 2.x (depending on build flags). So there's no need for your project to include a binary of SDL, because Emscripten will provide it.
As part of the Emscripten build process, Emscripten would download and store a
cache of these ports in its install folder... which breaks on Windows because
the Emscripten install is in C:\Program Files\
(which acts oddly when non-Admin).
To avoid this, dotnet team grabbed only the ports they thought would be required
(which does not include SDL 1 or 2), set up the cache within the
Microsoft.NET.Runtime.Emscripten.Cache
package, and made MSBuild forcibly set
a FROZEN_CACHE
environment variable as part of the build process, to
tell Emscripten that it's not allowed to download any new ports.
Therefore, if you try to send Emscripten the build flags to add SDL to the
build (as part of ports), which is -lSDL -s USE_SDL=2
for 3.1.34
, it will
try to add the port, realise the cache is frozen, and fail with the error:
Exception: Attempt to lock the cache but FROZEN_CACHE is set
.
Dotnet team could fix this going forwards, or at least offer a way to set a cache folder as part of MSBuild parameters, but until then there are two valid workarounds:
- Hook into the MSBuild process with a new target to overwrite the
FROZEN_CACHE
environment variable, and provide a new cache location for it to use. (This was the approach taken here by a Silk.NET developer). However, this is likely brittle long-term as the process will break when updating the local .NET runtime/SDK, not any packages or project files. - Don't rely on Emscripten's ports and instead include your own copy of
the SDL binary built for WebAssembly via Emscripten, using
<NativeFileReference>
MSBuild parameter. This is stable long-term, and the only downside is needing to update SDL deliberately when needed. This is the approach this project used.
One of the few code changes you'd make to allow your project to build via
Emscripten is that you need to change your main program entry-point to
not run all the code in a blocking way (like while(1) { doThings(); }
),
but instead define a function that should be called on a regular basis.
That function is then your "main loop" that is called at either a regular
interval, or called when the browser thinks animation should happen. For
graphical programs, you almost always want your main loop to run as the latter.
Unfortunately, there is no direct/obvious way to tell Emscripten this baked
into the project template, even though it would be pretty easy to add. Hopefully
in the future the dotnet developers add something to the template to describe
this logic (the way they add the JSImport
/JSExport
logic), if not make it
easier to add.
For now, these workarounds exist:
- Use some arcane
[DllImport]
attributes pointing at__Internal_emscripten
to get references to theemscripten_set_main_loop
and similar functions, which the C# code can then run, but also runs into issues with marshalling function pointers. (This was the approach taken here by a Silk.NET developer). This was previously considered the only way, however it is completely obsoleted by the next approach. - Add
EmccExportedRuntimeMethod
parameters to the.csproj
file to exposesetMainLoop
and similar methods in the Javascript side of Emscripten. Inmain.js
you can then pass that function to C# (via[JsImport]
with working function marshalling) and call it, or in reverse pass the main loop function from C# tomain.js
(via[JsExport]
) and call it there. To see the correct name to export for any Emscripten function, the Javascript can be debugged anddotnet.instance.Module
can be inspected, where all methods are listed (whether exported or not). This is the approach this project used.
This project uses the SDL2-CS
library as C# binding to SDL. This library targets older .NET versions
and therefore uses [DllImport]
and traditional marshalling (especially for
strings), which is not a major issue however is not ideal for something
known to be on .NET 8.
However there is an actual issue within the dotnet runtime that required
a custom fork of the SDL2-CS library with a very minor change.
This change is just changing the name of the native library that the [DllImport]
attributes use
to libSDL2
(the Linux version) instead of SDL2
.
The issue within the dotnet runtime is that while [DllImport]
and [LibraryImport]
try to change the name of the library to match the platform (e.g. adding lib
when building for Linux), this logic does not work correctly for WebAssembly
builds, where the library is likely to require the Linux name.
There is a function NativeLibrary.SetDllImportResolver
which attempts to
allow changing the name used for [DllImport]
at run-time, which works
only on platforms that are not AOT-compiled or WebAssembly-based. Using this,
when building for WebAssembly, the behaviour is even more broken: resolving
the library itself succeeds, but using any method fails because the symbols fail
to resolve.
Dotnet team could potentially look at fixing this so the SetDllImportResolver
function works as expected with WebAssembly, however it is unclear if full
AOT builds would still work correctly. This would require changes to the
P/Invoke table generation in dotnet's WebAssembly build tasks. In the meantime
these workarounds exist:
- Duplicate the binding class for this platform and change the library name used. Not ideal because duplication, but it works. Simplest solution, this is the approach this project used.
- Flip the setup: have the default name be
libSDL2
, then useNativeLibrary.SetDllImportResolver
to actively change it back toSDL2
for non-WebAssembly platforms. It is unclear if full AOT works correctly on all platforms for this, but it seems to work at least with[LibraryImport]
.
Technically with this work, it should be possible for someone to take their C# project and via minor changes re-build it for WebAssembly. Because all input/ output/system interaction is via SDL, as long as SDL works, then technically getting the project to run should be simple. Unfortunately, there is one problem that most of these libraries have in common: dynamic linking.
Basically, for multiple reasons including portability and ease of maintenance, these libraries don't directly bind to SDL via a static class like the one in this project. If they did, then yes outside of a few similar changes to this project, it would be straightforward to build them for WebAssembly.
Using MonoGame as the example, what the repository does at time of writing is this:
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
public delegate void d_sdl_disablescreensaver();
public static d_sdl_disablescreensaver DisableScreenSaver = FuncLoader.LoadFunction<d_sdl_disablescreensaver>(NativeLibrary, "SDL_DisableScreenSaver");
Where FuncLoader
links into a class like this:
private class Windows
{
[DllImport("kernel32", CharSet = CharSet.Ansi, ExactSpelling = true, SetLastError = true)]
public static extern IntPtr GetProcAddress(IntPtr hModule, string procName);
[DllImport("kernel32", SetLastError = true, CharSet = CharSet.Unicode)]
public static extern IntPtr LoadLibraryW(string lpszLib);
}
What this basically does is use dynamic linking at run-time, instead of statically linking a library. So instead of your C# binding directly listing the SDL functions and then tying it to a DLL that's included, it will use the OS-level functions to load that DLL/each method and call it.
This method of dynamic linking is actually supported by Emscripten
in principle, but it needs the linking to happen slightly differently. Specifically, it expects
the linked library that needs to be dynamically loaded to also be in the build process (which is the opposite
of the point of this), but be built with emcc
and have the parameter -sSIDE_MODULE
. What this means
is that you would build SDL for Emscripten, marked as a "side module", and allow Emscripten to load it
asynchronously as a separate WebAssembly module, then link it together. However because SDL's entire point
is that it uses system library calls (graphics, etc), you would also have to ensure that you use
the correct includes (EMCC_FORCE_STDLIBS
) when building SDL to include the relevant system libraries.
The two choices then are load-time linking (which basically means passing the new SDL library to emcc
when building your application, and isn't relevant here), and runtime dynamic linking. For runtime
dynamic linking, it just expects you to access it via dlopen()
and dlsym()
which are, for example,
what MonoGame does to load SDL functions at time of writing.
So I believe there is no technical reason why this would not work, but it requires the SDL library to be built in a more complex way, and would automatically remove the ability to use the built-in Emscripten "ports" functionality to load SDL automatically (since it is not a "side module"). It would also mean that failures happen at run-time always, which can make some issues hard to diagnose on top of potentially introducing issues in the future.
(There are also minor limitations around the size of the compiled WASM file, thread support, etc; but those are tooling issues and overall solvable.)
Or these libraries (MonoGame/etc) could choose to instead make WebAssembly a dedicated "platform" (e.g. for MonoGame, not like Android/iOS that still use dynamic linking, but instead its own non-DesktopGL platform that uses an SDL C# binding instead). This might make most sense because allowing WebAssembly basically also means setting a minimum .NET version for the project, at which point you can also lose a lot of cruft/slow-down that's only necessary to maintain compatibility with very old .NET.
Or in the worst case, a wrapper library that is basically a fake dlopen()
and dlsym()
implementation
could be written in C, whose only purpose is to non-dynamically override calls so that they go to
the relevant symbol in the SDL library, but at that point there's multiple weird overrides in a chain.