-
Notifications
You must be signed in to change notification settings - Fork 263
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
WebGPU as low level graphics API #53
Comments
I agree. As WASI grows, it's going to want a lot of the same functionality as the Web. And especially once we have webidl-bindings (or whatever that evolves into), there will be a straightforward way to directly use APIs like this. WASI does have its own needs and use cases outside of the Web, and I expect it won't always make sense to use Web APIs. I haven't looked at WebGPU in detail myself, however graphics APIs are an area where using the Web API would have major advantages. |
Sounds plausible. I can also imagine direct Vulkan bindings being created. Seems like it will depends on a couple of factors:
|
I think one of the benefits of using some of the web technology is that they're the only, as far as I'm aware, ones thinking about untrusted use of the GPU. There have been some nasty security vulnerabilities with WebGL and one of Vulkan's selling points, as I understand it, is that it doesn't do runtime checking in release mode. I recall a number of years ago that malformed Vulkan without the validation layer could get the GPU into a bad state or crash the operating system or something in certain circumstances. Given that, I think exposing raw Vulkan will only make sense if the Vulkan drivers are built with malicious use in mind. Vulkan seems like a good choice for building something on top of though! More generally though, it's really tricky. Certain things like denial of service by asking the GPU to do a ton of work are still an issue with WebGL as far as I know. I'm not an expert here by any means, but I am excited about getting graphical programs working with WASI! |
@sbc100 1. I don't think there will be much complexity, I don't know either API but as I understand it WebGPU is being designed with heavy influence from Vulkan/Metal, Google and Mozilla. And the mentioned native rust implementation already works using dx12, vulkan or metal depending on the platform. 2. Of course code written for WebGPU is almost non existent since it's so fresh and still being defined but is totally worth considering it as the one backend to rule them all since it will rule the web for sure 😎 |
It may be worth it to make the first WASI graphics API some kind of lowest common denominator API that works well on desktops, mobile, and the web alike, similar to OpenGL ES 3 / OpenGL 3 / WebGL 2. The point being that such an API would map relatively 1:1 onto these 3 types of platforms, and that a lot of code / game engines out there (in non-web land) already targets these APIs so would need few to no modifications to work (as opposed to WebGPU, which would require whole new engine ports). Also note that WASI doesn't need to restrict itself to one graphics API, we can spec a WASI-Vulkan later, and/or WebGPU if it makes more sense. OpenGL may seem like an old-fashioned API, but it is easy to work with, and the amount of code out there that works with it dwarfs the other API options right now. Once we have OpenGL, it may be possible to implement (a subset of) LibSDL to work on top of both the Posix and GL parts of WASI, further extending the amount of code that can run on top of this with little modification. |
If we decide to supply OpenGL, I think providing it through EGL is probably best, since engines/applications can reuse their EGL code, we can follow a spec that's already been thought out and in wide deployment (Android, Linux, and more), and it allows relatively easy customization through EGL extensions. It additionally allows binding OpenGL, OpenGL ES, OpenVG, and more. |
I'd like to make another argument for WebGPU, which is a safe Metal-like modern graphics API that is being implemented in Rust with pluggable backends (as well as getting native support in browsers). This should mean that WASI could target Vulkan, DX12, DX11, Metal, OpenGL/ES, and WebGPU itself through a WebGPU interface. It'd be lovely to see a WASI backend targeting the web and exposing a WebGPU-like interface via either native browser support or a polyfill via Edit: Something we may also want to consider is how the graphics API will interact with composition APIs (e.g. DirectComposition on Windows, Core Animation on macOS/iOS, SurfaceFlinger/Control on Android, the DOM (?) on Web, etc.). |
WebGPU is the only universal GPU API that is safe, portable, high level, high performance, and supports modern GPU features like compute shaders. Portable means: guarantees that code has the same behaviour on all platforms, backed by a conformance test suite. It works in web browsers, Windows, Mac, Linux, and mobile. Mozilla and Google are both implementing C library interfaces that target multiple platforms, and they have pledged to use a common WebGPU.h header file to promote portability. Mozilla's Rust library is layered on top of their C interface. This looks like the next big thing. Game engines are paying attention. I'm planning to dump OpenGL and switch to WebGPU when it is ready, because I don't want to create my own abstraction layer that supports multiple back ends. OpenGL won't get me compute shaders on MacOS or the web. Vulcan will never run on the web. WebGPU is the only API that gives me compute shaders on the web, so why not just use it on all platforms? |
Following up on WebGPU and OpenGL: There's an interesting argument against OpenGL in the Glium post-mortem, namely that "Drivers are still crippled with bugs" [so abstracting over OpenGL in a cross-platform way isn't practical]. I've heard that WebGPU is more like Vulkan than it is like OpenGL; can anyone with context on WebGPU speak to this? Security and being properly portable are my biggest concerns so WebGPU seems pretty appealing. |
WebGPU is the newest of the "modern" graphics APIs (e.g. Vulkan, DX12, Metal, Mantle), which I distinguish from the older APIs (OpenGL, DX11) as having explicit command queues opposed to implicit ones. As far as similarity, WebGPU compares closest to Metal (probably since Apple is the one that originally proposed it)--both don't require manual memory management while DX12 and Vulkan do. Like Metal (and OpenGL), I think WebGPU is a wonderfully friendly API that requires little boilerplate to be productive. To me, the most compelling argument for WebGPU on portability is that it seems straightforward (although not trivial) to implement some standard OpenGL on top of WebGPU as a library (there's already a handful of OpenGL on Vulkan libraries, like GLOVE and Zink), while the opposite is neither straightforward nor trivial (especially given the problems addressed in the glium post-mortem). |
That's one of the major reasons we at (libre-riscv)[https://libre-riscv.org/3d_gpu/] are just writing a Vulkan driver and not an OpenGL driver for our open-source GPU. |
If someone is willing to layout the structure of a WebGPU-WASI implementation in Rust, and at least one or two example functions, I will devote my time to fleshing out remaining functions. IOW, plant the seed & I will water it. |
@jpryne An interesting thing about WebGPU is that it has a WebIDL API, meaning that once everything is ready we can use interface types to talk to it from wasm. One of the first tasks for WASI is to evaluate that API to see if it's compatible with WASI -- that it doesn't make any assumptions about running inside a browser or having JS available, and that it fits within a Capability-based sandboxing model (roughly speaking, all functions should operate on a handle returned by the library, rather than having global impact). If someone wants to start prototyping an implementation, one possible path is to hook up Rust WebGPU and the idiomatic Rust wrappers to the wasi-common standalone implementation of WASI in Rust. To reduce the amount of boilerplate needed, it may be desirable to build a tool to auto-generate some of the interface from the WebIDL source mentioned above, possibly with help from wasm-bindgen. There's a lot in there, but feel free to ask questions! |
Thanks for your suggestions. I think, in the absence of examples to study, I should be starting with editing the documentation, (i.e. Todo: Update the IR Reference) |
Is this only about GPU or could be related on any other kind of accelerator? |
@bhack: The WebGPU standard only covers GPU. WebGPU corresponds to the intersection (common subset) of Vulkan, DX12 and Metal. Other kinds of accelerators (tensor processing units) are out of scope. |
If you have some time, @sunfishcode, help me understand what work needs to be done:
Sorry if this is a little disorganized--I'm trying to wrap my head around how all the pieces fit together. From my understanding, one needs to:
Please let me know where I'm hand-waving too much! If I can understand what needs to be done, I'm interesting in taking a shot at it! |
@sunfishcode Has anyone attempted to start a specification for a Windowing / WebGPU / Framebuffer module? I'd be interested in contributing in this area, but I don't want to do duplicate work. |
Waki is arm pit. Wakizashi is 脇差, which is 腋 "arm pit" and 差 which means like to put something through something and hold it there (like hold by sticking through a belt). The short sword known as wakizashi is just how it's called because it was a short sword designed to be fastened under your arm/arm pit. |
Here is a TypeScript program compiled to WebAssembly with AssemblyScript, using WebGL APIs outside of a browser in Node.js (the JS glue code calls the assemblyscript-webgl-node.mp4The WebGL code is written in AssemblyScript, and currently starts like this: import {WebGLRenderingContext} from '../node_modules/aswebglue/src/WebGL'
const gl = new WebGLRenderingContext(...)
// call gl.whatever as you would in JS/TS This means we at least have a WebGL interface in AssemblyScript currently working in Node or browsers. Next we would need to make the WebGL (or WebGPU) interface in WASI, but the AS code would remain the same. |
Oops, I forgot links! ASWebGLue: https://github.com/battlelinegames/ASWebGLue LUME Glas (port of Three.js to AssemblyScript that will likely fork direction): https://github.com/lume/glas (part of the LUME project, http://github.com/lume/lume) AssemblyScript's #gamedev Discord channel where we chat about ASWebGLue and AS game dev: https://discord.gg/A5n6qdeYfR LUME's #glas channel where we talk about ASWebGLue and Glas: https://discord.gg/6XvnkMb The Node-AssemblyScript-WebGL example isn't pushed up yet, but will be soon... |
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
Has somebody picked this up to make a demo? Otherwise I will make one with the rust-wasi crate as outlined here, so anything that has moved since the last comment would help me here. |
I am also interested in a webgpu submodule for WASi. But what confuses me is that there are two types of WASI notation and tools for codegen - witx (https://github.com/WebAssembly/WASI/tree/main/tools/witx) and wit (https://github.com/bytecodealliance/wit-bindgen). As I understand the second one is newer, but it is under the Bytecodealliance umbrella and I understand it is still under development. It would be great if someone from Wasm WG could clarify which tools to use and how stable they are. |
I also believe this should have someone from api design to look at, to enumerate the webgpu api module's endpoints. |
@MaxGraey witx is the tooling that wasi_snapshot_preview1 uses. It's been around a while, we've gotten a lot of feedback and experience with it. We learned a lot, including that we'd really like to do several things very differently. It's also significantly out of date at this point with where interface types and module linking have evolved to. wit is a new format being designed to be part of interface types, which is under the WebAssembly CG umbrella. As a part of interface types, it'll be kept current with interface types evolution. And, besides having its own syntax instead of S-expressions, it also avoids the need for awkward wit-bindgen is an implementation of a bindings generator that uses this wit format. It's new, and under active development, and it's already much more powerful and easier to use than current witx tools. For example, in the witx tools, "string" is only supported for arguments. In wit-bindgen, "string" is also supported as a return type, record field, tuple element, and more. At this time, I recommend people writing new interfaces to use the wit format, and the wit-bindgen tools. There are some limitations, and not everything is stable yet, but they're already far ahead of the witx tooling in almost every way, and they're being actively developed. |
@sunfishcode Thanks! That's basically what I thought. It's just that there is very little information on the topic. It wasn't even clear whether wit was the official direction of development. It is now clear what it will be. It would be nice to add a full documentation / specification for wit. It would also be nice to mention on official WASI channels / resources about witx deprecation and WIT and the tools for it should be used. |
Related: what about WebGL? I know WebGPU is superior, but there are still many apps written with WebGL, and WebGL isn't going away any time soon. Soon, it will be possible to port Three.js apps (WebGL-based) to lume/glas. It'd be sweet to be able to compile them to native too. :) |
Any update on this thread? Very interested in support for WebGPU |
I had to move to other work for multiple reasons. Feel free to pick it up. |
Sorry for going completely off-topic but: tl;dr, at least in common usage, waki on its own often refers to armpit but the waki in wakizashi often means "a subsidiary/complementary sword". 脇 does mainly mean "armpit" but it also means "something that's not the main focus". 脇役 (in which 役 means "role") means "side role/character" for example. Although the exact origin of the meaning is disputed (and your interpretation is indeed one of them), it's also argued that it referred to "a sword that was hidden/concealed under a person's clothes". And by the 16th century or so, it became commonplace for bushi (armed people) to carry two swords: 大刀 (big sword) for longer reach and a 小刀 (small sword) for indoor usage. The two were also referred to as 本差 (honzashi, main sword) and 脇差 (wakizashi, sub sword). Both swords would be strapped near the waist so it clearly didn't refer to the body part in this case. This is probably the usage most familiar to a layman. Source: My uncle used to work at Japan Annnnd... yeah, I would love WebGPU support in WASI, yeah... Yeah... |
I'm developing a map renderer in Rust called maplibre-rs based on WebGPU which can compile to WASM and run in the browser. I'd be very interested in being able to run it in a WASI runtime. I guess for me it would be a toy project though. Though, it definitely has potential in the server space. Maplibre-rs also supports a headless mode. A WASI binary deployed on a server could render PNG raster tiles :) I'm not sure where WASM is going, but that would be a cool usecase for this feature. |
Consider if Wayland is viable for windowing. It is security-minded and can be a good portable abstraction, see WSLg for Wayland under Windows. Windowing/GUI toolkits can mostly (if not completely) reuse their Wayland code for WASI, maybe requiring some changes in |
Requring WSLg to be installed would more than likely be a non-starter for WASI. For the scope of a windowing library maybe something with an api surface like winit could be considered which is a relatively new toolkit and does run on most platforms. On macOS there is Owl which does exist for Wayland on macOS, but it ultimately has to follow the rules Cocoa sets. |
This is my attempt to get this going. |
(my background is web & game dev) WebGPU has gotten a lot of focus and is a great start. At risk of stating the obvious, I wanted to expound upon the I interpret the windowing request to be: Display a 3D GL context inside of a Window (but potentially also second mode to display a Browser w/ 2D Canvas; ie. Electron). Maybe interact w/ common properties of said window (set: title, width, height, minimized/maximized state. get: isFocused, keyDown, etc.). That's it. IMO it would look like a WASI that replaces/integrates libs like SDL [1][2] as at least one implication here. It's essentially portable GPU, Audio, Windowing, and Human Interface Devices. Game devs of 30 years ago would proudly forego stdlib for custom deterministic cross-platform minimalist implementations of everything (sockets, threads, time, video, audio, math, types, etc.) By "platform" we mean to include the game consoles--but its a similar (more extreme) argument for embedded systems and particularly space. (ie. NASA[1]; maybe SpaceX will be shipping WASM satellites?) Portability being just one reason, others including trade-offs like sacrificing precision in pursuit of constant-time performance[1] (for real-time applications). Assuming the performance penalty for runtime bytecode interpretation remains low, I'm imagining WASM on XBox/PS. Compiling and debugging in a WASM-first approach. Potentially even a universally portable binary like Cosmopolitan[1] aims for; which is native machine code, JIT unpacked/compiled by a bytecode interpreter on startup/first-run. Modern UI IMO is more GPU-accelerated HTML+CSS or Dear ImGui[1], less Qt+Wayland. (e.g., React Native for mobile devices) In my view it makes more sense to take a GPU-first/default approach, and for portability instead build shims in reverse (ie. TTY emulation on GPU). I would take this farther to include GPU programming concepts and AI applications. (Thesis: GPU is the modern FPU processor; it will always be with us going forward, yet we should be able to opt-out for legacy/headless cases.) I sense this future is on multiple folks' minds. (e.g., would impact Epic, nVidia, Microsoft, etc.) Portability has been a fever dream since the earliest days of GNU, but always sabotaged by one commercial hardware vendor or another (DRM, blob drivers, etc.), precisely because it negatively impacts many proprietary licensing revenue streams. But today it seems closer than ever. Until WASI WebGPU, the workflow (for game engines) is: compile and test on PC with C and SDL, with WASM as just another target (and only for use by the browser) ... plus many proprietary + disparately-maintained C cross-platform portability libraries [1]. But after, it could be that the cross-platform hello-world of graphics & AI programming is 10 lines of .WAT. |
As an update here, the wasi-webgpu proposal mentioned above is now an official WASI proposal and has been renamed to wasi-gfx, so the repository is now at https://github.com/WebAssembly/wasi-gfx . For further discussion of WebGPU and related use cases, please file new issues in that repo! |
Soon will come the time to decide how to pain something on a screen, are there discussions about this topic in the community? didn't see any. Seeing what WebGPU is supposed to do seems like a natural fit for WASI to use this API instead of defining its own or doing something WASIVulkan/WASIMetal ... I imagine on top of, next to it, or even bellow, other APIs could be defined right? like stuff to draw 2d graphics web 2d canvas style, or create windows, or not sure if it would need to go lower level like defining a compositor, a framebuffer, etc.
I imagine Rust WebGPU implementation should make it easy for wasi runtimes like wasmtime or wasmer to come up with a working prototype since they are rust based, there are idiomatic rust wrappers, works natively on windows, mac and linux and there are a WebIDL definitions.
The text was updated successfully, but these errors were encountered: