186 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
		
		
			
		
	
	
			186 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
| 
								 | 
							
								# User's Guide to Emscripten in Nixpkgs
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								[Emscripten](https://github.com/kripken/emscripten): An LLVM-to-JavaScript Compiler
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								This section of the manual covers how to use `emscripten` in nixpkgs.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								Minimal requirements:
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								* nix
							 | 
						||
| 
								 | 
							
								* nixpkgs
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								Modes of use of `emscripten`:
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								* **Imperative usage** (on the command line):
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								   If you want to work with `emcc`, `emconfigure` and `emmake` as you are used to from Ubuntu and similar distributions you can use these commands:
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    * `nix-env -i emscripten`
							 | 
						||
| 
								 | 
							
								    * `nix-shell -p emscripten`
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								* **Declarative usage**:
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    This mode is far more power full since this makes use of `nix` for dependency management of emscripten libraries and targets by using the `mkDerivation` which is implemented by `pkgs.emscriptenStdenv` and `pkgs.buildEmscriptenPackage`. The source for the packages is in `pkgs/top-level/emscripten-packages.nix` and the abstraction behind it in `pkgs/development/em-modules/generic/default.nix`.
							 | 
						||
| 
								 | 
							
								    * build and install all packages: 
							 | 
						||
| 
								 | 
							
								        * `nix-env -iA emscriptenPackages` 
							 | 
						||
| 
								 | 
							
								          
							 | 
						||
| 
								 | 
							
								    * dev-shell for zlib implementation hacking: 
							 | 
						||
| 
								 | 
							
								        * `nix-shell -A emscriptenPackages.zlib` 
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								## Imperative usage
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								A few things to note:
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								* `export EMCC_DEBUG=2` is nice for debugging
							 | 
						||
| 
								 | 
							
								* `~/.emscripten`, the build artifact cache sometimes creates issues and needs to be removed from time to time
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								## Declarative usage
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								Let's see two different examples from `pkgs/top-level/emscripten-packages.nix`:
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								* `pkgs.zlib.override`
							 | 
						||
| 
								 | 
							
								* `pkgs.buildEmscriptenPackage`
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								Both are interesting concepts.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								A special requirement of the `pkgs.buildEmscriptenPackage` is the `doCheck = true` is a default meaning that each emscriptenPackage requires a `checkPhase` implemented.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								* Use `export EMCC_DEBUG=2` from within a emscriptenPackage's `phase` to get more detailed debug output what is going wrong.
							 | 
						||
| 
								 | 
							
								* ~/.emscripten cache is requiring us to set `HOME=$TMPDIR` in individual phases. This makes compilation slower but also makes it more deterministic.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								### Usage 1: pkgs.zlib.override
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								This example uses `zlib` from nixpkgs but instead of compiling **C** to **ELF** it compiles **C** to **JS** since we were using `pkgs.zlib.override` and changed stdenv to `pkgs.emscriptenStdenv`. A few adaptions and hacks were set in place to make it working. One advantage is that when `pkgs.zlib` is updated, it will automatically update this package as well. However, this can also be the downside...
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								See the `zlib` example:
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    zlib = (pkgs.zlib.override {
							 | 
						||
| 
								 | 
							
								      stdenv = pkgs.emscriptenStdenv;
							 | 
						||
| 
								 | 
							
								    }).overrideDerivation
							 | 
						||
| 
								 | 
							
								    (old: rec {
							 | 
						||
| 
								 | 
							
								      buildInputs = old.buildInputs ++ [ pkgconfig ];
							 | 
						||
| 
								 | 
							
								      # we need to reset this setting!
							 | 
						||
| 
								 | 
							
								      NIX_CFLAGS_COMPILE="";
							 | 
						||
| 
								 | 
							
								      configurePhase = ''
							 | 
						||
| 
								 | 
							
								        # FIXME: Some tests require writing at $HOME
							 | 
						||
| 
								 | 
							
								        HOME=$TMPDIR
							 | 
						||
| 
								 | 
							
								        runHook preConfigure
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        #export EMCC_DEBUG=2
							 | 
						||
| 
								 | 
							
								        emconfigure ./configure --prefix=$out --shared
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        runHook postConfigure
							 | 
						||
| 
								 | 
							
								      '';
							 | 
						||
| 
								 | 
							
								      dontStrip = true;
							 | 
						||
| 
								 | 
							
								      outputs = [ "out" ];
							 | 
						||
| 
								 | 
							
								      buildPhase = ''
							 | 
						||
| 
								 | 
							
								        emmake make
							 | 
						||
| 
								 | 
							
								      '';
							 | 
						||
| 
								 | 
							
								      installPhase = ''
							 | 
						||
| 
								 | 
							
								        emmake make install
							 | 
						||
| 
								 | 
							
								      '';
							 | 
						||
| 
								 | 
							
								      checkPhase = ''
							 | 
						||
| 
								 | 
							
								        echo "================= testing zlib using node ================="
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        echo "Compiling a custom test"
							 | 
						||
| 
								 | 
							
								        set -x
							 | 
						||
| 
								 | 
							
								        emcc -O2 -s EMULATE_FUNCTION_POINTER_CASTS=1 test/example.c -DZ_SOLO \
							 | 
						||
| 
								 | 
							
								        libz.so.${old.version} -I . -o example.js
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        echo "Using node to execute the test"
							 | 
						||
| 
								 | 
							
								        ${pkgs.nodejs}/bin/node ./example.js 
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        set +x
							 | 
						||
| 
								 | 
							
								        if [ $? -ne 0 ]; then
							 | 
						||
| 
								 | 
							
								          echo "test failed for some reason"
							 | 
						||
| 
								 | 
							
								          exit 1;
							 | 
						||
| 
								 | 
							
								        else
							 | 
						||
| 
								 | 
							
								          echo "it seems to work! very good."
							 | 
						||
| 
								 | 
							
								        fi
							 | 
						||
| 
								 | 
							
								        echo "================= /testing zlib using node ================="
							 | 
						||
| 
								 | 
							
								      '';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      postPatch = pkgs.stdenv.lib.optionalString pkgs.stdenv.isDarwin ''
							 | 
						||
| 
								 | 
							
								        substituteInPlace configure \
							 | 
						||
| 
								 | 
							
								          --replace '/usr/bin/libtool' 'ar' \
							 | 
						||
| 
								 | 
							
								          --replace 'AR="libtool"' 'AR="ar"' \
							 | 
						||
| 
								 | 
							
								          --replace 'ARFLAGS="-o"' 'ARFLAGS="-r"'
							 | 
						||
| 
								 | 
							
								      '';
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								### Usage 2: pkgs.buildEmscriptenPackage
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								This `xmlmirror` example features a emscriptenPackage which is defined completely from this context and no `pkgs.zlib.override` is used. 
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    xmlmirror = pkgs.buildEmscriptenPackage rec {
							 | 
						||
| 
								 | 
							
								      name = "xmlmirror";
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      buildInputs = [ pkgconfig autoconf automake libtool gnumake libxml2 nodejs openjdk json_c ];
							 | 
						||
| 
								 | 
							
								      nativeBuildInputs = [ pkgconfig zlib ];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      src = pkgs.fetchgit {
							 | 
						||
| 
								 | 
							
								        url = "https://gitlab.com/odfplugfest/xmlmirror.git";
							 | 
						||
| 
								 | 
							
								        rev = "4fd7e86f7c9526b8f4c1733e5c8b45175860a8fd";
							 | 
						||
| 
								 | 
							
								        sha256 = "1jasdqnbdnb83wbcnyrp32f36w3xwhwp0wq8lwwmhqagxrij1r4b";
							 | 
						||
| 
								 | 
							
								      };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      configurePhase = ''
							 | 
						||
| 
								 | 
							
								        rm -f fastXmlLint.js*
							 | 
						||
| 
								 | 
							
								        # a fix for ERROR:root:For asm.js, TOTAL_MEMORY must be a multiple of 16MB, was 234217728
							 | 
						||
| 
								 | 
							
								        # https://gitlab.com/odfplugfest/xmlmirror/issues/8
							 | 
						||
| 
								 | 
							
								        sed -e "s/TOTAL_MEMORY=234217728/TOTAL_MEMORY=268435456/g" -i Makefile.emEnv
							 | 
						||
| 
								 | 
							
								        # https://github.com/kripken/emscripten/issues/6344
							 | 
						||
| 
								 | 
							
								        # https://gitlab.com/odfplugfest/xmlmirror/issues/9
							 | 
						||
| 
								 | 
							
								        sed -e "s/\$(JSONC_LDFLAGS) \$(ZLIB_LDFLAGS) \$(LIBXML20_LDFLAGS)/\$(JSONC_LDFLAGS) \$(LIBXML20_LDFLAGS) \$(ZLIB_LDFLAGS) /g" -i Makefile.emEnv
							 | 
						||
| 
								 | 
							
								        # https://gitlab.com/odfplugfest/xmlmirror/issues/11
							 | 
						||
| 
								 | 
							
								        sed -e "s/-o fastXmlLint.js/-s EXTRA_EXPORTED_RUNTIME_METHODS='[\"ccall\", \"cwrap\"]' -o fastXmlLint.js/g" -i Makefile.emEnv
							 | 
						||
| 
								 | 
							
								      '';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      buildPhase = ''
							 | 
						||
| 
								 | 
							
								        HOME=$TMPDIR
							 | 
						||
| 
								 | 
							
								        make -f Makefile.emEnv
							 | 
						||
| 
								 | 
							
								      '';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      outputs = [ "out" "doc" ];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      installPhase = ''
							 | 
						||
| 
								 | 
							
								        mkdir -p $out/share
							 | 
						||
| 
								 | 
							
								        mkdir -p $doc/share/${name}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cp Demo* $out/share
							 | 
						||
| 
								 | 
							
								        cp -R codemirror-5.12 $out/share
							 | 
						||
| 
								 | 
							
								        cp fastXmlLint.js* $out/share
							 | 
						||
| 
								 | 
							
								        cp *.xsd $out/share
							 | 
						||
| 
								 | 
							
								        cp *.js $out/share
							 | 
						||
| 
								 | 
							
								        cp *.xhtml $out/share
							 | 
						||
| 
								 | 
							
								        cp *.html $out/share
							 | 
						||
| 
								 | 
							
								        cp *.json $out/share
							 | 
						||
| 
								 | 
							
								        cp *.rng $out/share
							 | 
						||
| 
								 | 
							
								        cp README.md $doc/share/${name}
							 | 
						||
| 
								 | 
							
								      '';
							 | 
						||
| 
								 | 
							
								      checkPhase = ''
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      '';
							 | 
						||
| 
								 | 
							
								    }; 
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								### Declarative debugging
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								Use `nix-shell -I nixpkgs=/some/dir/nixpkgs -A emscriptenPackages.libz` and from there you can go trough the individual steps. This makes it easy to build a good `unit test` or list the files of the project.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								1. `nix-shell -I nixpkgs=/some/dir/nixpkgs -A emscriptenPackages.libz`
							 | 
						||
| 
								 | 
							
								2. `cd /tmp/`
							 | 
						||
| 
								 | 
							
								3. `unpackPhase`
							 | 
						||
| 
								 | 
							
								4. cd libz-1.2.3
							 | 
						||
| 
								 | 
							
								5. `configurePhase`
							 | 
						||
| 
								 | 
							
								6. `buildPhase`
							 | 
						||
| 
								 | 
							
								7. ... happy hacking...
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								## Summary
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								Using this toolchain makes it easy to leverage `nix` from NixOS, MacOSX or even Windows (WSL+ubuntu+nix). This toolchain is reproducible, behaves like the rest of the packages from nixpkgs and contains a set of well working examples to learn and adapt from.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								If in trouble, ask the maintainers.
							 | 
						||
| 
								 | 
							
								
							 |