• Stars
    star
    527
  • Rank 84,091 (Top 2 %)
  • Language
    JavaScript
  • License
    MIT License
  • Created over 7 years ago
  • Updated over 6 years ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

mono-wasm

This project is a proof-of-concept aiming at building C# applications into WebAssembly, by using Mono and compiling/linking everything statically into one .wasm file that can be easily delivered to browsers.

The process does not use Emscripten (or Binaryen) but instead uses the experimental WebAssembly backend of LLVM with clang and lld to generate the final .wasm code. The goal is to use as few dependencies as possible. At the moment the only dependencies are LLVM, clang and lld trunk.

mono-wasm supports 2 build modes: one that links all the LLVM bitcode into one module then performs a WebAssembly codegen on it, and one that compiles project dependencies into WebAssembly incrementally (the runtime and the mscorlib assembly) then uses lld to link into a final .wasm file. The later is experimental but will become the default as it allows build times lesser than a second.

The .wasm file is loaded from JavaScript (see index.js), which also exposes proper callbacks for system calls that the C library will be calling into. These syscalls are responsible for heap management, I/O, etc.

This project is a work in progress. Feel free to ping me if you have questions or feedback: [email protected]

Related repositories

  • mono-wasm-mono: a fork of Mono with changes for a wasm32 target, used to build the runtime and compiler.
  • mono-wasm-libc: a fork of the WebAssembly/musl C library with tweaks for our version of Mono and our JS glue.

Current status

This is a work in progress, but you can see sample/hello/hello.cs running here: www.hipbyte.com/~lrz/mono-wasm-hello

Binary releases

Binary releases are avalable here (for testing only): https://github.com/lrz/mono-wasm/releases

How does it work?

An ASCII graph is worth a thousand words:

+----------------+-------------+  +---------------------+
|  Mono runtime  |  C library  |  |    C# assemblies    | <-------+
+----------------+-------------+  +----------+----------+         |
           clang |                           | mono               |
  -target=wasm32 |                           | -aot=llvmonly      |
                 v                           v                    |
+-------------------------------------------------------+         | load
|                       LLVM bitcode                    |         | metadata
+----------------------------+--------------------------+         | (runtime)
                             | mono-wasm                          |
                             | (bitcode -> wasm)                  | 
                             v                                    | 
+-------------------------------------------------------+         |
|                        index.wasm                     |---------+
+----------------------------------------+--------------+               
                 ^                       | libc                         
   load, compile |                       | syscalls                     
    + run main() |                       v                             
+----------------+--------------------------------------+         +-----------+ 
|                         index.js                      | <-----> |  Browser  |
+-------------------------------------------------------+         +-----------+

Build instructions

We will assume that you want to build everything in the ~/src/mono-wasm directory.

$ mkdir ~/src/mono-wasm
$ cd ~/src/mono-wasm
$ git clone [email protected]:lrz/mono-wasm.git build

LLVM+clang+lld with WebAssembly target

We need a copy of the LLVM tooling (clang and lld included) with the experimental WebAssembly target enabled. Make sure to build a Release build (as indicated below) otherwise the WASM codegen will be significantly slower.

$ cd ~/src/mono-wasm
$ svn co http://llvm.org/svn/llvm-project/llvm/trunk llvm
$ cd llvm/tools
$ svn co http://llvm.org/svn/llvm-project/cfe/trunk clang
$ svn co http://llvm.org/svn/llvm-project/lld/trunk lld
$ cd ../..
$ mkdir llvm-build
$ cd llvm-build
$ cmake -G "Unix Makefiles" -DLLVM_EXPERIMENTAL_TARGETS_TO_BUILD=WebAssembly -DCMAKE_BUILD_TYPE=Release ../llvm
$ make

After you did this you should have the LLVM static libraries for the WebAssembly target:

$ ls ~/src/mono-wasm/llvm-build/lib | grep WebAssembly
libLLVMWebAssemblyAsmPrinter.a
libLLVMWebAssemblyCodeGen.a
libLLVMWebAssemblyDesc.a
libLLVMWebAssemblyDisassembler.a
libLLVMWebAssemblyInfo.a

You should also have the ~/src/mono-wasm/llvm-build/bin/clang program built with the wasm32 target:

$ ~/src/mono-wasm/llvm-build/bin/clang --version
clang version 5.0.0 (trunk 306818)
Target: x86_64-apple-darwin15.6.0
Thread model: posix
InstalledDir: /Users/lrz/src/mono-wasm/llvm-build/bin

  Registered Targets:
[...]
    wasm32     - WebAssembly 32-bit
    wasm64     - WebAssembly 64-bit

You should also have the wasm lld (linker) library:

$ ls ~/src/mono-wasm/llvm-build/lib/liblldWasm.a
/Users/lrz/src/mono-wasm/llvm-build/lib/liblldWasm.a

Mono compiler

We need a build a copy of the Mono compiler that we will use to generate LLVM bitcode from assemblies. We are building this for 32-bit Intel (i386) because the Mono compiler assumes way too many things from the host environment when generating the bitcode, so we want to match the target architecture (which is also 32-bit).

First, you need to build a copy of the Mono fork of LLVM. We are building it for both 32-bit and 64-bit Intel so that we can easily switch the Mono compiler back to 64-bit later, and we manually have to copy the headers to the build directory as the Mono build system doesn't support external LLVM builds.

$ cd ~/src/mono-wasm
$ git clone [email protected]:mono/llvm.git llvm-mono
$ mkdir llvm-mono-build
$ cd llvm-mono-build
$ cmake -G "Unix Makefiles" -DCMAKE_OSX_ARCHITECTURES="i386;x86_64" ../llvm-mono
$ ditto ../llvm-mono/include include
$ make

Now, we can now build the Mono compiler itself.

$ cd ~/src/mono-wasm
$ git clone [email protected]:lrz/mono-wasm-mono.git mono-compiler
$ cd mono-compiler
$ ./autogen.sh --host=i386-darwin --with-cross-offsets=offsets-wasm32.h CFLAGS="-DCOMPILE_WASM32 -DMONO_CROSS_COMPILE" CXXFLAGS="-DCOMPILE_WASM32 -DMONO_CROSS_COMPILE" --disable-boehm --with-sigaltstack=no --enable-llvm --enable-llvm-runtime --with-llvm=../llvm-mono-build --disable-btls --with-runtime_preset=testing_aot_full
$ cd eglib
$ make
$ cd ../mono
$ make

At the end of this process you should have a mono executable installed as ~/src/mono-wasm/mono-compiler/mono/mini/mono built for the i386 architecture.

$ file ~/src/mono-wasm/mono-compiler/mono/mini/mono
mono/mini/mono: Mach-O executable i386

Now let's build the mscorlib.dll assembly for the WebAssembly profile. We can't use the mono runtime we just built as it's full AOT, so we use assume you have a normal mono runtime in your PATH that we can use. Clearly a hack, but in the meantime it works.

$ cd ~/src/mono-wasm/mono-compiler/mcs/class/corlib
$ make V=1 PROFILE=wasm RUNTIME=mono STRING_REPLACER=true SN=true

After this you should have the assembly file created in the proper location:

$ file ~/src/mono-wasm/mono-compiler/mcs/class/lib/wasm/mscorlib.dll 
/Users/lrz/src/mono-wasm/mono-compiler/mcs/class/lib/wasm/mscorlib.dll: PE32 executable (DLL) (console) Intel 80386 Mono/.Net assembly, for MS Windows

Mono runtime

Now we can prepare the Mono runtime. We have to clone a new copy of the source code. We are not building the runtime code using the Mono autotools system, so we have to copy header files that are normally generated.

$ cd ~/src/mono-wasm
$ git clone [email protected]:lrz/mono-wasm-mono.git mono-runtime
$ cd mono-runtime
$ cp config-wasm32.h config.h
$ cp eglib/src/eglib-config-wasm32.h eglib/src/eglib-config.h

C library

Similarly as above, we clone a copy of the C library that we will be using.

$ cd ~/src/mono-wasm
$ git clone [email protected]:lrz/mono-wasm-libc.git libc

OK ready!

We are ready to build our Hello World.

First, we need to build everything into the dist directory:

$ cd ~/src/mono-wasm/build
$ vi Makefile               # make sure the *_PATH variables point to proper locations, should be the case if you followed these instructions
$ make

This will build the mono runtime and the libc as LLVM bitcode using our version of clang, then link everything into a runtime.bc file. This will also build the mono-wasm tool which links against the LLVM and lld libraries. Finally, we will copy the Mono compiler and its mscorlib.dll file.

$ find dist -type f
dist/bin/monoc
dist/bin/mono-wasm
dist/lib/runtime.bc
dist/lib/index.js
dist/lib/mscorlib.dll
dist/lib/mscorlib.xml

Once done, you can build the Hello World sample:

$ cd sample/hello
$ make

TODO

TODO (now):

  • fix garbage collection (need to figure out how to scan the stack)
  • ship a first 'alpha' release

TODO (later):

  • put mscorlib on a diet (currently 'hello world' is 10MB) by removing more functionality within the wasm.make profile and doing more aggressive IL linking
  • work on patches for mono based on the changes made in the fork
  • merge the WebAssembly LLVM code into the mono/llvm fork so that the Mono compiler can target wasm32 directly, and that we can merge the code into mono-wasm (we won't have to ship the Mono compiler separately as monoc)
  • improve the C# -> JS interop by doing a full C# API replication in JS like embeddinator 4000
  • investigate: threads, sockets, debugger, stack unwinding, simd and atomic operations, etc.

License

This work is distributed under the terms of the MIT license. See the LICENSE.txt file for more information.

More Repositories

1

TensorFlowSharp

TensorFlow API for .NET languages
C#
3,138
star
2

SwiftGodot

New Godot bindings for Swift
Swift
1,020
star
3

SwiftTerm

Xterm/VT100 Terminal emulator in Swift
Swift
938
star
4

TermKit

Terminal Kit - Console UI toolkit for Swift applications
Swift
455
star
5

MonoTouch.Dialog

Tools to simplify creating dialogs with the user using MonoTouch
C#
430
star
6

WasmerSharp

.NET Bindings for the Wasmer Runtime
C#
317
star
7

SwiftTermApp

Swift
304
star
8

SwiftGodotKit

Embed Godot into Swift apps
C
217
star
9

PlaticaBot

MacOS, iOS and watchOS ChatGPT client using your own OpenAI key
Swift
189
star
10

TweetStation

MonoTouch based Twitter client
C#
180
star
11

XtermSharp

XTerm emulator as a .NET library
C#
161
star
12

redis-sharp

A C#/.NET binding for the Redis server.
C#
152
star
13

SkiaKit

Swift Bindings to the Skia 2D graphics Library
Swift
134
star
14

TextBufferKit

Swift TextBuffer
Swift
104
star
15

CovidGraphs

Covid Graphs app
Swift
93
star
16

SwiftChatGPT

Simple ChatGPT API
Swift
74
star
17

GodotSwift

Swift bindings for the Godot Game Engine
C
56
star
18

mc

C# based curses file manager
C#
45
star
19

FuchsiaSharp

Bindings to Fuchsia
C#
36
star
20

OpenFlowSharp

Sample CoverFlow implementation for MonoTouch, based on OpenFlow
C#
33
star
21

muget

Command line front end to NuGet that is not overengineered
C#
31
star
22

RealityActions

RealityActions
Swift
27
star
23

monotouch-libs

Various MonoTouch bindings and libraries
C#
25
star
24

TldExtract

.NET Library to extracts the root domain, subdomain name, and top level domain from a host name using the Public Suffix List
C#
22
star
25

SwiftNavigation

C++
19
star
26

SwiftGodotBinary

SwiftGodot - binaries for ease of consumption
Swift
18
star
27

libgodot

Godot modified to be embeddable as a library, for use in SwiftGodotKit
C++
18
star
28

mono-wasm-libc

C
15
star
29

TurboSwift

Swift
14
star
30

mono-wasm-mono

C#
14
star
31

FluidInterfaces

Port of Nathan Gitter's Fluid Interfaces code
C#
14
star
32

MaterialSharp

Material design components for iOS applications written in C#
C#
13
star
33

NetCatNetwork

NetCat implementation using the new Apple Network framework, C# port of the C sample
C#
11
star
34

CovidExtractor

My Covid Extractor
Swift
9
star
35

SwiftGodotDocs

9
star
36

CocoaDriver

Xamarin.Mac-based Cocoa driver for Mono's System.Windows.Forms
C#
8
star
37

Darwin.CopyFile

Bindings for Darwin's CopyFile API
C#
8
star
38

SwiftNavigationDemo

Demo of the SwiftNavigation API
Swift
8
star
39

MonoTouch.Bindings

Some bindings for third party libraries
6
star
40

migueldeicaza

Repository for my README
6
star
41

gtk

Fork of Gtk branch 2-24 with all Xamarin patches from github.com/mono/bockbuild applied
C
5
star
42

KeyboardButton

Simple keyboard button in SwiftUI eventually for use in the Terminal Emulator
Swift
5
star
43

GodotSwiftLink

Binding module, temporary, while I move it to Godot.
C
4
star
44

Mac-SceneKit-Globe-Test

Sample F# Mac app showing SceneKit in action.
F#
4
star
45

PhotoPanner

C# version of Facebooks Photo Panner
C#
4
star
46

Mono.VFS

C# based Virtual File System API.
4
star
47

SimpleCollage

Simple picture collage with SkiaSharp
C#
4
star
48

til

4
star
49

MapWrapper

SwiftUI wrapper around MapKit
Swift
4
star
50

SwiftApiExtractIssue

Shows a bug with swift-api-extract
Swift
3
star
51

RosettaMaker

Creates the Rosetta HTML page that shows the C#/Objective-C/C mappings
C#
3
star
52

KitCore.NET

.NET bindings to KitCore - the ultimate core application framework
3
star
53

SplashView

C# Port of SplashView https://github.com/callumboddy/CBZSplashView
C#
3
star
54

NailedIt

My Google Glasses app to "Capture Moments" and "Nailed it"
Java
3
star
55

FileProvider-ios-crashing

Swift
2
star
56

monotouch-samples

MonoTouch Sample Programs
2
star
57

SwiftSH.binaries

Binary for the SwiftSH framework
C
2
star
58

SkiaKit.binaries

Precompiled binaries for various platforms of the runtime used by SkiaKit (SkiaSharp)
2
star
59

CovidExtractorPM

SwiftPM version of CovidExtractor
Swift
2
star
60

paint-mono

Automatically exported from code.google.com/p/paint-mono
C#
2
star
61

SkiaKitPayloads

Binary payloads directory for the SkiaKit library
Shell
1
star
62

SwiftOnnxMnist

Mnist sample using OnnxRuntime in Swift
C++
1
star
63

TreemapWeb

C#
1
star
64

iptoloc

IP to Location Tools
C#
1
star
65

mlist

Gtk# Mlist widget - managed implementation of a list view (2005 era)
C#
1
star
66

mono-soc-2007

Automatically exported from code.google.com/p/mono-soc-2007
C#
1
star
67

blog-monomac

MonoMac Blog
1
star
68

SwiftDocCPlusPlusBug

Sample of how Swift Docc fails if you use the new Swift C++ interop
Swift
1
star
69

GodotSnippetEditor

A tool to edit Godot documentation code snippets
Swift
1
star
70

SwiftTermDocs

Placeholder to publish the API docs for SwiftTerm
1
star
71

esctest

Mirror from https://gitlab.freedesktop.org/terminal-wg/esctest.git
Python
1
star
72

CoreKitSharp

Bindings to the Ultimate Framework Platform
1
star