17db96d56Sopenharmony_ci# Python WebAssembly (WASM) build
27db96d56Sopenharmony_ci
37db96d56Sopenharmony_ci**WARNING: WASM support is work-in-progress! Lots of features are not working yet.**
47db96d56Sopenharmony_ci
57db96d56Sopenharmony_ciThis directory contains configuration and helpers to facilitate cross
67db96d56Sopenharmony_cicompilation of CPython to WebAssembly (WASM). Python supports Emscripten
77db96d56Sopenharmony_ci(*wasm32-emscripten*) and WASI (*wasm32-wasi*) targets. Emscripten builds
87db96d56Sopenharmony_cirun in modern browsers and JavaScript runtimes like *Node.js*. WASI builds
97db96d56Sopenharmony_ciuse WASM runtimes such as *wasmtime*.
107db96d56Sopenharmony_ci
117db96d56Sopenharmony_ciUsers and developers are encouraged to use the script
127db96d56Sopenharmony_ci`Tools/wasm/wasm_build.py`. The tool automates the build process and provides
137db96d56Sopenharmony_ciassistance with installation of SDKs.
147db96d56Sopenharmony_ci
157db96d56Sopenharmony_ci## wasm32-emscripten build
167db96d56Sopenharmony_ci
177db96d56Sopenharmony_ciFor now the build system has two target flavors. The ``Emscripten/browser``
187db96d56Sopenharmony_citarget (``--with-emscripten-target=browser``) is optimized for browsers.
197db96d56Sopenharmony_ciIt comes with a reduced and preloaded stdlib without tests and threading
207db96d56Sopenharmony_cisupport. The ``Emscripten/node`` target has threading enabled and can
217db96d56Sopenharmony_ciaccess the file system directly.
227db96d56Sopenharmony_ci
237db96d56Sopenharmony_ciCross compiling to the wasm32-emscripten platform needs the
247db96d56Sopenharmony_ci[Emscripten](https://emscripten.org/) SDK and a build Python interpreter.
257db96d56Sopenharmony_ciEmscripten 3.1.19 or newer are recommended. All commands below are relative
267db96d56Sopenharmony_cito a repository checkout.
277db96d56Sopenharmony_ci
287db96d56Sopenharmony_ciChristian Heimes maintains a container image with Emscripten SDK, Python
297db96d56Sopenharmony_cibuild dependencies, WASI-SDK, wasmtime, and several additional tools.
307db96d56Sopenharmony_ci
317db96d56Sopenharmony_ciFrom within your local CPython repo clone, run one of the following commands:
327db96d56Sopenharmony_ci
337db96d56Sopenharmony_ci```
347db96d56Sopenharmony_ci# Fedora, RHEL, CentOS
357db96d56Sopenharmony_cipodman run --rm -ti -v $(pwd):/python-wasm/cpython:Z -w /python-wasm/cpython quay.io/tiran/cpythonbuild:emsdk3
367db96d56Sopenharmony_ci
377db96d56Sopenharmony_ci# other
387db96d56Sopenharmony_cidocker run --rm -ti -v $(pwd):/python-wasm/cpython -w /python-wasm/cpython quay.io/tiran/cpythonbuild:emsdk3
397db96d56Sopenharmony_ci```
407db96d56Sopenharmony_ci
417db96d56Sopenharmony_ci### Compile a build Python interpreter
427db96d56Sopenharmony_ci
437db96d56Sopenharmony_ciFrom within the container, run the following command:
447db96d56Sopenharmony_ci
457db96d56Sopenharmony_ci```shell
467db96d56Sopenharmony_ci./Tools/wasm/wasm_build.py build
477db96d56Sopenharmony_ci```
487db96d56Sopenharmony_ci
497db96d56Sopenharmony_ciThe command is roughly equivalent to:
507db96d56Sopenharmony_ci
517db96d56Sopenharmony_ci```shell
527db96d56Sopenharmony_cimkdir -p builddir/build
537db96d56Sopenharmony_cipushd builddir/build
547db96d56Sopenharmony_ci../../configure -C
557db96d56Sopenharmony_cimake -j$(nproc)
567db96d56Sopenharmony_cipopd
577db96d56Sopenharmony_ci```
587db96d56Sopenharmony_ci
597db96d56Sopenharmony_ci### Cross-compile to wasm32-emscripten for browser
607db96d56Sopenharmony_ci
617db96d56Sopenharmony_ci```shell
627db96d56Sopenharmony_ci./Tools/wasm/wasm_build.py emscripten-browser
637db96d56Sopenharmony_ci```
647db96d56Sopenharmony_ci
657db96d56Sopenharmony_ciThe command is roughly equivalent to:
667db96d56Sopenharmony_ci
677db96d56Sopenharmony_ci```shell
687db96d56Sopenharmony_cimkdir -p builddir/emscripten-browser
697db96d56Sopenharmony_cipushd builddir/emscripten-browser
707db96d56Sopenharmony_ci
717db96d56Sopenharmony_ciCONFIG_SITE=../../Tools/wasm/config.site-wasm32-emscripten \
727db96d56Sopenharmony_ci  emconfigure ../../configure -C \
737db96d56Sopenharmony_ci    --host=wasm32-unknown-emscripten \
747db96d56Sopenharmony_ci    --build=$(../../config.guess) \
757db96d56Sopenharmony_ci    --with-emscripten-target=browser \
767db96d56Sopenharmony_ci    --with-build-python=$(pwd)/../build/python
777db96d56Sopenharmony_ci
787db96d56Sopenharmony_ciemmake make -j$(nproc)
797db96d56Sopenharmony_cipopd
807db96d56Sopenharmony_ci```
817db96d56Sopenharmony_ci
827db96d56Sopenharmony_ciServe `python.html` with a local webserver and open the file in a browser.
837db96d56Sopenharmony_ciPython comes with a minimal web server script that sets necessary HTTP
847db96d56Sopenharmony_ciheaders like COOP, COEP, and mimetypes. Run the script outside the container
857db96d56Sopenharmony_ciand from the root of the CPython checkout.
867db96d56Sopenharmony_ci
877db96d56Sopenharmony_ci```shell
887db96d56Sopenharmony_ci./Tools/wasm/wasm_webserver.py
897db96d56Sopenharmony_ci```
907db96d56Sopenharmony_ci
917db96d56Sopenharmony_ciand open http://localhost:8000/builddir/emscripten-browser/python.html . This
927db96d56Sopenharmony_cidirectory structure enables the *C/C++ DevTools Support (DWARF)* to load C
937db96d56Sopenharmony_ciand header files with debug builds.
947db96d56Sopenharmony_ci
957db96d56Sopenharmony_ci
967db96d56Sopenharmony_ci### Cross compile to wasm32-emscripten for node
977db96d56Sopenharmony_ci
987db96d56Sopenharmony_ci```shell
997db96d56Sopenharmony_ci./Tools/wasm/wasm_build.py emscripten-browser-dl
1007db96d56Sopenharmony_ci```
1017db96d56Sopenharmony_ci
1027db96d56Sopenharmony_ciThe command is roughly equivalent to:
1037db96d56Sopenharmony_ci
1047db96d56Sopenharmony_ci```shell
1057db96d56Sopenharmony_cimkdir -p builddir/emscripten-node-dl
1067db96d56Sopenharmony_cipushd builddir/emscripten-node-dl
1077db96d56Sopenharmony_ci
1087db96d56Sopenharmony_ciCONFIG_SITE=../../Tools/wasm/config.site-wasm32-emscripten \
1097db96d56Sopenharmony_ci  emconfigure ../../configure -C \
1107db96d56Sopenharmony_ci    --host=wasm32-unknown-emscripten \
1117db96d56Sopenharmony_ci    --build=$(../../config.guess) \
1127db96d56Sopenharmony_ci    --with-emscripten-target=node \
1137db96d56Sopenharmony_ci    --enable-wasm-dynamic-linking \
1147db96d56Sopenharmony_ci    --with-build-python=$(pwd)/../build/python
1157db96d56Sopenharmony_ci
1167db96d56Sopenharmony_ciemmake make -j$(nproc)
1177db96d56Sopenharmony_cipopd
1187db96d56Sopenharmony_ci```
1197db96d56Sopenharmony_ci
1207db96d56Sopenharmony_ci```shell
1217db96d56Sopenharmony_cinode --experimental-wasm-threads --experimental-wasm-bulk-memory --experimental-wasm-bigint builddir/emscripten-node-dl/python.js
1227db96d56Sopenharmony_ci```
1237db96d56Sopenharmony_ci
1247db96d56Sopenharmony_ci(``--experimental-wasm-bigint`` is not needed with recent NodeJS versions)
1257db96d56Sopenharmony_ci
1267db96d56Sopenharmony_ci# wasm32-emscripten limitations and issues
1277db96d56Sopenharmony_ci
1287db96d56Sopenharmony_ciEmscripten before 3.1.8 has known bugs that can cause memory corruption and
1297db96d56Sopenharmony_ciresource leaks. 3.1.8 contains several fixes for bugs in date and time
1307db96d56Sopenharmony_cifunctions.
1317db96d56Sopenharmony_ci
1327db96d56Sopenharmony_ci## Network stack
1337db96d56Sopenharmony_ci
1347db96d56Sopenharmony_ci- Python's socket module does not work with Emscripten's emulated POSIX
1357db96d56Sopenharmony_ci  sockets yet. Network modules like ``asyncio``, ``urllib``, ``selectors``,
1367db96d56Sopenharmony_ci  etc. are not available.
1377db96d56Sopenharmony_ci- Only ``AF_INET`` and ``AF_INET6`` with ``SOCK_STREAM`` (TCP) or
1387db96d56Sopenharmony_ci  ``SOCK_DGRAM`` (UDP) are available. ``AF_UNIX`` is not supported.
1397db96d56Sopenharmony_ci- ``socketpair`` does not work.
1407db96d56Sopenharmony_ci- Blocking sockets are not available and non-blocking sockets don't work
1417db96d56Sopenharmony_ci  correctly, e.g. ``socket.accept`` crashes the runtime. ``gethostbyname``
1427db96d56Sopenharmony_ci  does not resolve to a real IP address. IPv6 is not available.
1437db96d56Sopenharmony_ci- The ``select`` module is limited. ``select.select()`` crashes the runtime
1447db96d56Sopenharmony_ci  due to lack of exectfd support.
1457db96d56Sopenharmony_ci
1467db96d56Sopenharmony_ci## processes, signals
1477db96d56Sopenharmony_ci
1487db96d56Sopenharmony_ci- Processes are not supported. System calls like fork, popen, and subprocess
1497db96d56Sopenharmony_ci  fail with ``ENOSYS`` or ``ENOSUP``.
1507db96d56Sopenharmony_ci- Signal support is limited. ``signal.alarm``, ``itimer``, ``sigaction``
1517db96d56Sopenharmony_ci  are not available or do not work correctly. ``SIGTERM`` exits the runtime.
1527db96d56Sopenharmony_ci- Keyboard interrupt (CTRL+C) handling is not implemented yet.
1537db96d56Sopenharmony_ci- Resource-related functions like ``os.nice`` and most functions of the
1547db96d56Sopenharmony_ci  ``resource`` module are not available.
1557db96d56Sopenharmony_ci
1567db96d56Sopenharmony_ci## threading
1577db96d56Sopenharmony_ci
1587db96d56Sopenharmony_ci- Threading is disabled by default. The ``configure`` option
1597db96d56Sopenharmony_ci  ``--enable-wasm-pthreads`` adds compiler flag ``-pthread`` and
1607db96d56Sopenharmony_ci  linker flags ``-sUSE_PTHREADS -sPROXY_TO_PTHREAD``. 
1617db96d56Sopenharmony_ci- pthread support requires WASM threads and SharedArrayBuffer (bulk memory).
1627db96d56Sopenharmony_ci  The Node.JS runtime keeps a pool of web workers around. Each web worker
1637db96d56Sopenharmony_ci  uses several file descriptors (eventfd, epoll, pipe).
1647db96d56Sopenharmony_ci- It's not advised to enable threading when building for browsers or with
1657db96d56Sopenharmony_ci  dynamic linking support; there are performance and stability issues.
1667db96d56Sopenharmony_ci
1677db96d56Sopenharmony_ci## file system
1687db96d56Sopenharmony_ci
1697db96d56Sopenharmony_ci- Most user, group, and permission related function and modules are not
1707db96d56Sopenharmony_ci  supported or don't work as expected, e.g.``pwd`` module, ``grp`` module,
1717db96d56Sopenharmony_ci  ``os.setgroups``, ``os.chown``, and so on. ``lchown`` and `lchmod`` are
1727db96d56Sopenharmony_ci  not available.
1737db96d56Sopenharmony_ci- ``umask`` is a no-op.
1747db96d56Sopenharmony_ci- hard links (``os.link``) are not supported.
1757db96d56Sopenharmony_ci- Offset and iovec I/O functions (e.g. ``os.pread``, ``os.preadv``) are not
1767db96d56Sopenharmony_ci  available.
1777db96d56Sopenharmony_ci- ``os.mknod`` and ``os.mkfifo``
1787db96d56Sopenharmony_ci  [don't work](https://github.com/emscripten-core/emscripten/issues/16158)
1797db96d56Sopenharmony_ci  and are disabled.
1807db96d56Sopenharmony_ci- Large file support crashes the runtime and is disabled.
1817db96d56Sopenharmony_ci- ``mmap`` module is unstable. flush (``msync``) can crash the runtime.
1827db96d56Sopenharmony_ci
1837db96d56Sopenharmony_ci## Misc
1847db96d56Sopenharmony_ci
1857db96d56Sopenharmony_ci- Heap memory and stack size are limited. Recursion or extensive memory
1867db96d56Sopenharmony_ci  consumption can crash Python.
1877db96d56Sopenharmony_ci- Most stdlib modules with a dependency on external libraries are missing,
1887db96d56Sopenharmony_ci  e.g. ``ctypes``, ``readline``, ``ssl``, and more.
1897db96d56Sopenharmony_ci- Shared extension modules are not implemented yet. All extension modules
1907db96d56Sopenharmony_ci  are statically linked into the main binary. The experimental configure
1917db96d56Sopenharmony_ci  option ``--enable-wasm-dynamic-linking`` enables dynamic extensions
1927db96d56Sopenharmony_ci  supports. It's currently known to crash in combination with threading.
1937db96d56Sopenharmony_ci- glibc extensions for date and time formatting are not available.
1947db96d56Sopenharmony_ci- ``locales`` module is affected by musl libc issues,
1957db96d56Sopenharmony_ci  [bpo-46390](https://bugs.python.org/issue46390).
1967db96d56Sopenharmony_ci- Python's object allocator ``obmalloc`` is disabled by default.
1977db96d56Sopenharmony_ci- ``ensurepip`` is not available.
1987db96d56Sopenharmony_ci- Some ``ctypes`` features like ``c_longlong`` and ``c_longdouble`` may need
1997db96d56Sopenharmony_ci   NodeJS option ``--experimental-wasm-bigint``.
2007db96d56Sopenharmony_ci
2017db96d56Sopenharmony_ci## wasm32-emscripten in browsers
2027db96d56Sopenharmony_ci
2037db96d56Sopenharmony_ci- The interactive shell does not handle copy 'n paste and unicode support
2047db96d56Sopenharmony_ci  well.
2057db96d56Sopenharmony_ci- The bundled stdlib is limited. Network-related modules,
2067db96d56Sopenharmony_ci  distutils, multiprocessing, dbm, tests and similar modules
2077db96d56Sopenharmony_ci  are not shipped. All other modules are bundled as pre-compiled
2087db96d56Sopenharmony_ci  ``pyc`` files.
2097db96d56Sopenharmony_ci- In-memory file system (MEMFS) is not persistent and limited.
2107db96d56Sopenharmony_ci- Test modules are disabled by default. Use ``--enable-test-modules`` build
2117db96d56Sopenharmony_ci  test modules like ``_testcapi``.
2127db96d56Sopenharmony_ci
2137db96d56Sopenharmony_ci## wasm32-emscripten in node
2147db96d56Sopenharmony_ci
2157db96d56Sopenharmony_ciNode builds use ``NODERAWFS``.
2167db96d56Sopenharmony_ci
2177db96d56Sopenharmony_ci- Node RawFS allows direct access to the host file system without need to
2187db96d56Sopenharmony_ci  perform ``FS.mount()`` call.
2197db96d56Sopenharmony_ci
2207db96d56Sopenharmony_ci## wasm64-emscripten
2217db96d56Sopenharmony_ci
2227db96d56Sopenharmony_ci- wasm64 requires recent NodeJS and ``--experimental-wasm-memory64``.
2237db96d56Sopenharmony_ci- ``EM_JS`` functions must return ``BigInt()``.
2247db96d56Sopenharmony_ci- ``Py_BuildValue()`` format strings must match size of types. Confusing 32
2257db96d56Sopenharmony_ci  and 64 bits types leads to memory corruption, see
2267db96d56Sopenharmony_ci  [gh-95876](https://github.com/python/cpython/issues/95876) and
2277db96d56Sopenharmony_ci  [gh-95878](https://github.com/python/cpython/issues/95878).
2287db96d56Sopenharmony_ci
2297db96d56Sopenharmony_ci# Hosting Python WASM builds
2307db96d56Sopenharmony_ci
2317db96d56Sopenharmony_ciThe simple REPL terminal uses SharedArrayBuffer. For security reasons
2327db96d56Sopenharmony_cibrowsers only provide the feature in secure environents with cross-origin
2337db96d56Sopenharmony_ciisolation. The webserver must send cross-origin headers and correct MIME types
2347db96d56Sopenharmony_cifor the JavaScript and WASM files. Otherwise the terminal will fail to load
2357db96d56Sopenharmony_ciwith an error message like ``Browsers disable shared array buffer``.
2367db96d56Sopenharmony_ci
2377db96d56Sopenharmony_ci## Apache HTTP .htaccess
2387db96d56Sopenharmony_ci
2397db96d56Sopenharmony_ciPlace a ``.htaccess`` file in the same directory as ``python.wasm``.
2407db96d56Sopenharmony_ci
2417db96d56Sopenharmony_ci```
2427db96d56Sopenharmony_ci# .htaccess
2437db96d56Sopenharmony_ciHeader set Cross-Origin-Opener-Policy same-origin
2447db96d56Sopenharmony_ciHeader set Cross-Origin-Embedder-Policy require-corp
2457db96d56Sopenharmony_ci
2467db96d56Sopenharmony_ciAddType application/javascript js
2477db96d56Sopenharmony_ciAddType application/wasm wasm
2487db96d56Sopenharmony_ci
2497db96d56Sopenharmony_ci<IfModule mod_deflate.c>
2507db96d56Sopenharmony_ci    AddOutputFilterByType DEFLATE text/html application/javascript application/wasm
2517db96d56Sopenharmony_ci</IfModule>
2527db96d56Sopenharmony_ci```
2537db96d56Sopenharmony_ci
2547db96d56Sopenharmony_ci# WASI (wasm32-wasi)
2557db96d56Sopenharmony_ci
2567db96d56Sopenharmony_ciWASI builds require [WASI SDK](https://github.com/WebAssembly/wasi-sdk) 16.0+.
2577db96d56Sopenharmony_ci
2587db96d56Sopenharmony_ci## Cross-compile to wasm32-wasi
2597db96d56Sopenharmony_ci
2607db96d56Sopenharmony_ciThe script ``wasi-env`` sets necessary compiler and linker flags as well as
2617db96d56Sopenharmony_ci``pkg-config`` overrides. The script assumes that WASI-SDK is installed in
2627db96d56Sopenharmony_ci``/opt/wasi-sdk`` or ``$WASI_SDK_PATH``.
2637db96d56Sopenharmony_ci
2647db96d56Sopenharmony_ci```shell
2657db96d56Sopenharmony_ci./Tools/wasm/wasm_build.py wasi
2667db96d56Sopenharmony_ci```
2677db96d56Sopenharmony_ci
2687db96d56Sopenharmony_ciThe command is roughly equivalent to:
2697db96d56Sopenharmony_ci
2707db96d56Sopenharmony_ci```shell
2717db96d56Sopenharmony_cimkdir -p builddir/wasi
2727db96d56Sopenharmony_cipushd builddir/wasi
2737db96d56Sopenharmony_ci
2747db96d56Sopenharmony_ciCONFIG_SITE=../../Tools/wasm/config.site-wasm32-wasi \
2757db96d56Sopenharmony_ci  ../../Tools/wasm/wasi-env ../../configure -C \
2767db96d56Sopenharmony_ci    --host=wasm32-unknown-wasi \
2777db96d56Sopenharmony_ci    --build=$(../../config.guess) \
2787db96d56Sopenharmony_ci    --with-build-python=$(pwd)/../build/python
2797db96d56Sopenharmony_ci
2807db96d56Sopenharmony_cimake -j$(nproc)
2817db96d56Sopenharmony_cipopd
2827db96d56Sopenharmony_ci```
2837db96d56Sopenharmony_ci
2847db96d56Sopenharmony_ci## WASI limitations and issues (WASI SDK 15.0)
2857db96d56Sopenharmony_ci
2867db96d56Sopenharmony_ciA lot of Emscripten limitations also apply to WASI. Noticable restrictions
2877db96d56Sopenharmony_ciare:
2887db96d56Sopenharmony_ci
2897db96d56Sopenharmony_ci- Call stack size is limited. Default recursion limit and parser stack size
2907db96d56Sopenharmony_ci  are smaller than in regular Python builds.
2917db96d56Sopenharmony_ci- ``socket(2)`` cannot create new socket file descriptors. WASI programs can
2927db96d56Sopenharmony_ci  call read/write/accept on a file descriptor that is passed into the process.
2937db96d56Sopenharmony_ci- ``socket.gethostname()`` and host name resolution APIs like
2947db96d56Sopenharmony_ci  ``socket.gethostbyname()`` are not implemented and always fail.
2957db96d56Sopenharmony_ci- ``open(2)`` checks flags more strictly. Caller must pass either
2967db96d56Sopenharmony_ci  ``O_RDONLY``, ``O_RDWR``, or ``O_WDONLY`` to ``os.open``. Directory file
2977db96d56Sopenharmony_ci  descriptors must be created with flags ``O_RDONLY | O_DIRECTORY``.
2987db96d56Sopenharmony_ci- ``chmod(2)`` is not available. It's not possible to modify file permissions,
2997db96d56Sopenharmony_ci  yet. A future version of WASI may provide a limited ``set_permissions`` API.
3007db96d56Sopenharmony_ci- User/group related features like ``os.chown()``, ``os.getuid``, etc. are
3017db96d56Sopenharmony_ci  stubs or fail with ``ENOTSUP``.
3027db96d56Sopenharmony_ci- File locking (``fcntl``) is not available.
3037db96d56Sopenharmony_ci- ``os.pipe()``, ``os.mkfifo()``, and ``os.mknod()`` are not supported.
3047db96d56Sopenharmony_ci- ``process_time`` does not work as expected because it's implemented using
3057db96d56Sopenharmony_ci  wall clock.
3067db96d56Sopenharmony_ci- ``os.umask()`` is a stub.
3077db96d56Sopenharmony_ci- ``sys.executable`` is empty.
3087db96d56Sopenharmony_ci- ``/dev/null`` / ``os.devnull`` may not be available.
3097db96d56Sopenharmony_ci- ``os.utime*()`` is buggy in WASM SDK 15.0, see
3107db96d56Sopenharmony_ci  [utimensat() with timespec=NULL sets wrong time](https://github.com/bytecodealliance/wasmtime/issues/4184)
3117db96d56Sopenharmony_ci- ``os.symlink()`` fails with ``PermissionError`` when attempting to create a
3127db96d56Sopenharmony_ci  symlink with an absolute path with wasmtime 0.36.0. The wasmtime runtime
3137db96d56Sopenharmony_ci  uses ``openat2(2)`` syscall with flag ``RESOLVE_BENEATH`` to open files.
3147db96d56Sopenharmony_ci  The flag causes the syscall to reject symlinks with absolute paths.
3157db96d56Sopenharmony_ci- ``os.curdir`` (aka ``.``) seems to behave differently, which breaks some
3167db96d56Sopenharmony_ci  ``importlib`` tests that add ``.`` to ``sys.path`` and indirectly
3177db96d56Sopenharmony_ci  ``sys.path_importer_cache``.
3187db96d56Sopenharmony_ci- WASI runtime environments may not provide a dedicated temp directory.
3197db96d56Sopenharmony_ci
3207db96d56Sopenharmony_ci
3217db96d56Sopenharmony_ci# Detect WebAssembly builds
3227db96d56Sopenharmony_ci
3237db96d56Sopenharmony_ci## Python code
3247db96d56Sopenharmony_ci
3257db96d56Sopenharmony_ci```python
3267db96d56Sopenharmony_ciimport os, sys
3277db96d56Sopenharmony_ci
3287db96d56Sopenharmony_ciif sys.platform == "emscripten":
3297db96d56Sopenharmony_ci    # Python on Emscripten
3307db96d56Sopenharmony_ciif sys.platform == "wasi":
3317db96d56Sopenharmony_ci    # Python on WASI
3327db96d56Sopenharmony_ci
3337db96d56Sopenharmony_ciif os.name == "posix":
3347db96d56Sopenharmony_ci    # WASM platforms identify as POSIX-like.
3357db96d56Sopenharmony_ci    # Windows does not provide os.uname().
3367db96d56Sopenharmony_ci    machine = os.uname().machine
3377db96d56Sopenharmony_ci    if machine.startswith("wasm"):
3387db96d56Sopenharmony_ci        # WebAssembly (wasm32, wasm64 in the future)
3397db96d56Sopenharmony_ci```
3407db96d56Sopenharmony_ci
3417db96d56Sopenharmony_ci```python
3427db96d56Sopenharmony_ci>>> import os, sys
3437db96d56Sopenharmony_ci>>> os.uname()
3447db96d56Sopenharmony_ciposix.uname_result(
3457db96d56Sopenharmony_ci    sysname='Emscripten',
3467db96d56Sopenharmony_ci    nodename='emscripten',
3477db96d56Sopenharmony_ci    release='3.1.19',
3487db96d56Sopenharmony_ci    version='#1',
3497db96d56Sopenharmony_ci    machine='wasm32'
3507db96d56Sopenharmony_ci)
3517db96d56Sopenharmony_ci>>> os.name
3527db96d56Sopenharmony_ci'posix'
3537db96d56Sopenharmony_ci>>> sys.platform
3547db96d56Sopenharmony_ci'emscripten'
3557db96d56Sopenharmony_ci>>> sys._emscripten_info
3567db96d56Sopenharmony_cisys._emscripten_info(
3577db96d56Sopenharmony_ci    emscripten_version=(3, 1, 10),
3587db96d56Sopenharmony_ci    runtime='Mozilla/5.0 (X11; Linux x86_64; rv:104.0) Gecko/20100101 Firefox/104.0',
3597db96d56Sopenharmony_ci    pthreads=False,
3607db96d56Sopenharmony_ci    shared_memory=False
3617db96d56Sopenharmony_ci)
3627db96d56Sopenharmony_ci```
3637db96d56Sopenharmony_ci
3647db96d56Sopenharmony_ci```python
3657db96d56Sopenharmony_ci>>> sys._emscripten_info
3667db96d56Sopenharmony_cisys._emscripten_info(
3677db96d56Sopenharmony_ci    emscripten_version=(3, 1, 19),
3687db96d56Sopenharmony_ci    runtime='Node.js v14.18.2',
3697db96d56Sopenharmony_ci    pthreads=True,
3707db96d56Sopenharmony_ci    shared_memory=True
3717db96d56Sopenharmony_ci)
3727db96d56Sopenharmony_ci```
3737db96d56Sopenharmony_ci
3747db96d56Sopenharmony_ci```python
3757db96d56Sopenharmony_ci>>> import os, sys
3767db96d56Sopenharmony_ci>>> os.uname()
3777db96d56Sopenharmony_ciposix.uname_result(
3787db96d56Sopenharmony_ci    sysname='wasi',
3797db96d56Sopenharmony_ci    nodename='(none)',
3807db96d56Sopenharmony_ci    release='0.0.0',
3817db96d56Sopenharmony_ci    version='0.0.0',
3827db96d56Sopenharmony_ci    machine='wasm32'
3837db96d56Sopenharmony_ci)
3847db96d56Sopenharmony_ci>>> os.name
3857db96d56Sopenharmony_ci'posix'
3867db96d56Sopenharmony_ci>>> sys.platform
3877db96d56Sopenharmony_ci'wasi'
3887db96d56Sopenharmony_ci```
3897db96d56Sopenharmony_ci
3907db96d56Sopenharmony_ci## C code
3917db96d56Sopenharmony_ci
3927db96d56Sopenharmony_ciEmscripten SDK and WASI SDK define several built-in macros. You can dump a
3937db96d56Sopenharmony_cifull list of built-ins with ``emcc -dM -E - < /dev/null`` and
3947db96d56Sopenharmony_ci``/path/to/wasi-sdk/bin/clang -dM -E - < /dev/null``.
3957db96d56Sopenharmony_ci
3967db96d56Sopenharmony_ci```C
3977db96d56Sopenharmony_ci#ifdef __EMSCRIPTEN__
3987db96d56Sopenharmony_ci    // Python on Emscripten
3997db96d56Sopenharmony_ci#endif
4007db96d56Sopenharmony_ci```
4017db96d56Sopenharmony_ci
4027db96d56Sopenharmony_ci* WebAssembly ``__wasm__`` (also ``__wasm``)
4037db96d56Sopenharmony_ci* wasm32 ``__wasm32__`` (also ``__wasm32``)
4047db96d56Sopenharmony_ci* wasm64 ``__wasm64__``
4057db96d56Sopenharmony_ci* Emscripten ``__EMSCRIPTEN__`` (also ``EMSCRIPTEN``)
4067db96d56Sopenharmony_ci* Emscripten version ``__EMSCRIPTEN_major__``, ``__EMSCRIPTEN_minor__``, ``__EMSCRIPTEN_tiny__``
4077db96d56Sopenharmony_ci* WASI ``__wasi__``
4087db96d56Sopenharmony_ci
4097db96d56Sopenharmony_ciFeature detection flags:
4107db96d56Sopenharmony_ci
4117db96d56Sopenharmony_ci* ``__EMSCRIPTEN_PTHREADS__``
4127db96d56Sopenharmony_ci* ``__EMSCRIPTEN_SHARED_MEMORY__``
4137db96d56Sopenharmony_ci* ``__wasm_simd128__``
4147db96d56Sopenharmony_ci* ``__wasm_sign_ext__``
4157db96d56Sopenharmony_ci* ``__wasm_bulk_memory__``
4167db96d56Sopenharmony_ci* ``__wasm_atomics__``
4177db96d56Sopenharmony_ci* ``__wasm_mutable_globals__``
4187db96d56Sopenharmony_ci
4197db96d56Sopenharmony_ci## Install SDKs and dependencies manually
4207db96d56Sopenharmony_ci
4217db96d56Sopenharmony_ciIn some cases (e.g. build bots) you may prefer to install build dependencies
4227db96d56Sopenharmony_cidirectly on the system instead of using the container image. Total disk size
4237db96d56Sopenharmony_ciof SDKs and cached libraries is about 1.6 GB.
4247db96d56Sopenharmony_ci
4257db96d56Sopenharmony_ci### Install OS dependencies
4267db96d56Sopenharmony_ci
4277db96d56Sopenharmony_ci```shell
4287db96d56Sopenharmony_ci# Debian/Ubuntu
4297db96d56Sopenharmony_ciapt update
4307db96d56Sopenharmony_ciapt install -y git make xz-utils bzip2 curl python3-minimal ccache
4317db96d56Sopenharmony_ci```
4327db96d56Sopenharmony_ci
4337db96d56Sopenharmony_ci```shell
4347db96d56Sopenharmony_ci# Fedora
4357db96d56Sopenharmony_cidnf install -y git make xz bzip2 which ccache
4367db96d56Sopenharmony_ci```
4377db96d56Sopenharmony_ci
4387db96d56Sopenharmony_ci### Install [Emscripten SDK](https://emscripten.org/docs/getting_started/downloads.html)
4397db96d56Sopenharmony_ci
4407db96d56Sopenharmony_ci**NOTE**: Follow the on-screen instructions how to add the SDK to ``PATH``.
4417db96d56Sopenharmony_ci
4427db96d56Sopenharmony_ci```shell
4437db96d56Sopenharmony_cigit clone https://github.com/emscripten-core/emsdk.git /opt/emsdk
4447db96d56Sopenharmony_ci/opt/emsdk/emsdk install latest
4457db96d56Sopenharmony_ci/opt/emsdk/emsdk activate latest
4467db96d56Sopenharmony_ci```
4477db96d56Sopenharmony_ci
4487db96d56Sopenharmony_ci### Optionally: enable ccache for EMSDK
4497db96d56Sopenharmony_ci
4507db96d56Sopenharmony_ciThe ``EM_COMPILER_WRAPPER`` must be set after the EMSDK environment is
4517db96d56Sopenharmony_cisourced. Otherwise the source script removes the environment variable.
4527db96d56Sopenharmony_ci
4537db96d56Sopenharmony_ci```
4547db96d56Sopenharmony_ci. /opt/emsdk/emsdk_env.sh
4557db96d56Sopenharmony_ciEM_COMPILER_WRAPPER=ccache
4567db96d56Sopenharmony_ci```
4577db96d56Sopenharmony_ci
4587db96d56Sopenharmony_ci### Optionally: pre-build and cache static libraries
4597db96d56Sopenharmony_ci
4607db96d56Sopenharmony_ciEmscripten SDK provides static builds of core libraries without PIC
4617db96d56Sopenharmony_ci(position-independent code). Python builds with ``dlopen`` support require
4627db96d56Sopenharmony_ciPIC. To populate the build cache, run:
4637db96d56Sopenharmony_ci
4647db96d56Sopenharmony_ci```shell
4657db96d56Sopenharmony_ci. /opt/emsdk/emsdk_env.sh
4667db96d56Sopenharmony_ciembuilder build zlib bzip2 MINIMAL_PIC
4677db96d56Sopenharmony_ciembuilder build --pic zlib bzip2 MINIMAL_PIC
4687db96d56Sopenharmony_ci```
4697db96d56Sopenharmony_ci
4707db96d56Sopenharmony_ci### Install [WASI-SDK](https://github.com/WebAssembly/wasi-sdk)
4717db96d56Sopenharmony_ci
4727db96d56Sopenharmony_ci**NOTE**: WASI-SDK's clang may show a warning on Fedora:
4737db96d56Sopenharmony_ci``/lib64/libtinfo.so.6: no version information available``,
4747db96d56Sopenharmony_ci[RHBZ#1875587](https://bugzilla.redhat.com/show_bug.cgi?id=1875587). The
4757db96d56Sopenharmony_ciwarning can be ignored.
4767db96d56Sopenharmony_ci
4777db96d56Sopenharmony_ci```shell
4787db96d56Sopenharmony_ciexport WASI_VERSION=16
4797db96d56Sopenharmony_ciexport WASI_VERSION_FULL=${WASI_VERSION}.0
4807db96d56Sopenharmony_cicurl -sSf -L -O https://github.com/WebAssembly/wasi-sdk/releases/download/wasi-sdk-${WASI_VERSION}/wasi-sdk-${WASI_VERSION_FULL}-linux.tar.gz
4817db96d56Sopenharmony_cimkdir -p /opt/wasi-sdk
4827db96d56Sopenharmony_citar --strip-components=1 -C /opt/wasi-sdk -xvf wasi-sdk-${WASI_VERSION_FULL}-linux.tar.gz
4837db96d56Sopenharmony_cirm -f wasi-sdk-${WASI_VERSION_FULL}-linux.tar.gz
4847db96d56Sopenharmony_ci```
4857db96d56Sopenharmony_ci
4867db96d56Sopenharmony_ci### Install [wasmtime](https://github.com/bytecodealliance/wasmtime) WASI runtime
4877db96d56Sopenharmony_ci
4887db96d56Sopenharmony_ciwasmtime 0.38 or newer is required.
4897db96d56Sopenharmony_ci
4907db96d56Sopenharmony_ci```shell
4917db96d56Sopenharmony_cicurl -sSf -L -o ~/install-wasmtime.sh https://wasmtime.dev/install.sh
4927db96d56Sopenharmony_cichmod +x ~/install-wasmtime.sh
4937db96d56Sopenharmony_ci~/install-wasmtime.sh --version v0.38.0
4947db96d56Sopenharmony_ciln -srf -t /usr/local/bin/ ~/.wasmtime/bin/wasmtime
4957db96d56Sopenharmony_ci```
4967db96d56Sopenharmony_ci
4977db96d56Sopenharmony_ci
4987db96d56Sopenharmony_ci### WASI debugging
4997db96d56Sopenharmony_ci
5007db96d56Sopenharmony_ci* ``wasmtime run -g`` generates debugging symbols for gdb and lldb. The
5017db96d56Sopenharmony_ci  feature is currently broken, see
5027db96d56Sopenharmony_ci  https://github.com/bytecodealliance/wasmtime/issues/4669 .
5037db96d56Sopenharmony_ci* The environment variable ``RUST_LOG=wasi_common`` enables debug and
5047db96d56Sopenharmony_ci  trace logging.
505