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