Skip to content

Commit

Permalink
Merge branch 'main' into update-chained-struct-names
Browse files Browse the repository at this point in the history
  • Loading branch information
kainino0x authored Aug 26, 2024
2 parents 447dfe6 + c697989 commit c803c81
Show file tree
Hide file tree
Showing 7 changed files with 368 additions and 78 deletions.
2 changes: 1 addition & 1 deletion Doxyfile
Original file line number Diff line number Diff line change
Expand Up @@ -79,7 +79,7 @@ LAYOUT_FILE =

# When adding extra files, separate them with spaces.

INPUT = webgpu.h
INPUT = webgpu.h doc/articles/

#---------------------------------------------------------------------------
# Configuration options related to source browsing
Expand Down
71 changes: 71 additions & 0 deletions doc/articles/Asynchronous Operations.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
# Asynchronous Operations {#Asynchronous-Operations}

Asynchronous operations in webgpu.h return a @ref WGPUFuture. This is an opaque handle which applications may use to poll or wait for completion of the asynchronous operation.

On all implementations of webgpu.h, it must be possible to react to async operations without undue latency or spin-waiting.

## Creation

All asynchronous operations start when the application calls an asynchronous webgpu.h function. These functions return `WGPUFuture`, and take in their arguments a `CallbackInfo` struct like @ref WGPURequestDeviceCallbackInfo or @ref WGPUBufferMapCallbackInfo. `CallbackInfo` structs have the following members:
- a `WGPUChainedStruct const * nextInChain` pointer, for extensions.
- a @ref WGPUCallbackMode `mode` enum.
- a `callback` function pointer. For example:
```c
typedef void (*WGPUBufferMapCallback)(
WGPUMapAsyncStatus status,
char const * message,
WGPU_NULLABLE void* userdata1,
WGPU_NULLABLE void* userdata2) WGPU_FUNCTION_ATTRIBUTE;
```
- Two application-owned userdata members.<br>
`void* userdata1`<br>
`void* userdata2`
The `callback` function pointer is called when the application _observes completion_ of the asynchronous operation. The `userdata1` and `userdata2` members are passed back to the application as the last two arguments in the callback function. Callbacks **might not** be called unless the application explicitly flushes them in order to _observe completion_. The point in time a callback is called depends on the @ref WGPUCallbackMode of the operation. webgpu.h provides three callback modes: `::WGPUCallbackMode_WaitAnyOnly`, `::WGPUCallbackMode_AllowProcessEvents`, and `::WGPUCallbackMode_AllowSpontaneous`.
> @copydoc ::WGPUCallbackMode_WaitAnyOnly
> @copydoc ::WGPUCallbackMode_AllowProcessEvents
> @copydoc ::WGPUCallbackMode_AllowSpontaneous
## wgpuInstanceWaitAny {#Wait-Any}
`WGPUWaitStatus wgpuInstanceWaitAny(WGPUInstance, size_t futureCount, WGPUFutureWaitInfo * futures, uint64_t timeoutNS)`
Waits on any WGPUFuture in the list of `futures` to complete for `timeoutNS` nanoseconds. Returns when at least one `WGPUFuture` is completed or `timeoutNS` elapses, whichever is first. If `timeoutNS` is zero, all `futures` are polled once, without blocking.
Returns `::WGPUWaitStatus_Success` if at least one `WGPUFuture` completes. WGPUFutureWaitInfo::completed is set to true for all completed futures. See @ref WGPUWaitStatus for other status codes.
Within this call, for any `WGPUFuture`s that completed, their respective callbacks will fire.
### Timed Wait {#Timed-Wait}
Use of _timed waits_ (`timeoutNS > 0`), must be enabled on the WGPUInstance in `::wgpuCreateInstance` with `WGPUInstanceFeatures::timedWaitAnyEnable`, and the number of futures waited on must be less than or equal to `WGPUInstanceFeatures::timedWaitAnyMaxCount`. Supported instance features may be queried using `::wgpuGetInstanceFeatures`.
### Mixed Sources {#Mixed-Sources}
Asynchronous operations may originate from different sources. There are CPU-timeline operations and there are Queue-timeline operations. Within a _timed wait_, it is only valid to wait on `WGPUFuture`s originating from a single `WGPUQueue`. Waiting on two futures from different queues, or waiting on a Queue-timeline future and some other CPU-timeline future is an error.
#### CPU-Timeline Operations
- ::wgpuInstanceRequestAdapter
- ::wgpuAdapterRequestDevice
- ::wgpuShaderModuleGetCompilationInfo
- ::wgpuDeviceCreateRenderPipelineAsync
- ::wgpuDeviceCreateComputePipelineAsync
- ::wgpuDevicePopErrorScope
#### Queue-Timeline Operations
- ::wgpuBufferMapAsync
- ::wgpuQueueOnSubmittedWorkDone
## wgpuInstanceProcessEvents {#Process-Events}
`void wgpuInstanceProcessEvents(WGPUInstance)`
Processes asynchronous events on this `WGPUInstance`, calling any callbacks for asynchronous operations created with `::WGPUCallbackMode_AllowProcessEvents` that have completed. This is a non-blocking operation.
## Device Events
Device events are slightly different in that their callback info (`WGPUDeviceLostCallbackInfo` and `WGPUUncapturedErrorCallbackInfo`) are passed on the `WGPUDeviceDescriptor`, instead of in a function argument. There is no `WGPUFuture` returned for either callback.
@todo Add a getter for the device lost WGPUFuture. See discussion at https://github.com/webgpu-native/webgpu-headers/issues/199#issuecomment-1866850031.
The `WGPUUncapturedErrorCallbackInfo` _does not_ have a callback mode member. It is always as-if it were `::WGPUCallbackMode_AllowSpontaneous`. Note also that the uncaptured error callback is a _repeating_ callback that fires multiple times, unlike other callbacks in webgpu.h.
The uncaptured error callback is guaranteed not to fire after the device becomes lost. When the device is lost, it is an appropriate time for the application to free userdata variables for the uncaptured error callback. Note that the device becomes lost _before_ the actual device lost callback fires. First the device state transitions to lost, then the device lost callback fires. The timing of the callback depends on the device lost callback mode.
3 changes: 3 additions & 0 deletions doc/articles/index.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
\page articles Articles

- \subpage Asynchronous-Operations
27 changes: 14 additions & 13 deletions gen/cheader.tmpl
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@

/**
* \mainpage
*
*
* **Important:** *This documentation is a Work In Progress.*
*
* This is the home of WebGPU C API specification. We define here the standard
Expand Down Expand Up @@ -68,7 +68,7 @@
/**
* \defgroup Constants
* \brief Constants.
*
*
* @{
*/

Expand All @@ -84,7 +84,7 @@
/**
* \defgroup Typedefs
* \brief Utility typedefs.
*
*
* @{
*/

Expand All @@ -102,7 +102,7 @@ typedef {{CType .Type "" ""}} WGPU{{.Name | PascalCase}}{{$.ExtSuffix}};
/**
* \defgroup Objects
* \brief Opaque, non-dispatchable handles to WebGPU objects.
*
*
* @{
*/

Expand Down Expand Up @@ -134,7 +134,7 @@ struct WGPU{{.Name | PascalCase}}CallbackInfo{{$.ExtSuffix}};
/**
* \defgroup Enumerations
* \brief Enums.
*
*
* @{
*/

Expand All @@ -160,7 +160,7 @@ typedef enum WGPU{{.Name | PascalCase}}{{$.ExtSuffix}} {
/**
* \defgroup Bitflags
* \brief Enum used as bit flags.
*
*
* @{
*/

Expand Down Expand Up @@ -191,7 +191,7 @@ typedef void (*WGPUProc)(void) WGPU_FUNCTION_ATTRIBUTE;
/**
* \defgroup Callbacks
* \brief Callbacks through which asynchronous functions return.
*
*
* @{
*/

Expand All @@ -206,7 +206,7 @@ typedef void (*WGPU{{.Name | PascalCase}}Callback{{$.ExtSuffix}})({{CallbackArgs
/**
* \defgroup ChainedStructures Chained Structures
* \brief Structures used to extend descriptors.
*
*
* @{
*/

Expand All @@ -226,14 +226,14 @@ typedef struct WGPUChainedStructOut {
/**
* \defgroup Structures
* \brief Descriptors and other transparent structures.
*
*
* @{
*/

/**
* \defgroup WGPUCallbackInfo
* \brief Callback info structures that are used in asynchronous functions.
*
*
* @{
*/

Expand Down Expand Up @@ -266,6 +266,7 @@ typedef struct WGPU{{.Name | PascalCase}}{{$.ExtSuffix}} {
WGPUChainedStructOut chain;
{{- end}}
{{- range $memberIndex, $_ := .Members}}
{{- MComment .Doc 4 }}
{{ StructMember $struct $memberIndex}}
{{- end}}
} WGPU{{.Name | PascalCase}}{{$.ExtSuffix}} WGPU_STRUCTURE_ATTRIBUTE;
Expand Down Expand Up @@ -306,7 +307,7 @@ typedef void (*WGPUProc{{.Name | PascalCase}}Release{{$.ExtSuffix}})(WGPU{{.Name
/**
* \defgroup GlobalFunctions Global Functions
* \brief Functions that are not specific to an object.
*
*
* @{
*/

Expand All @@ -323,7 +324,7 @@ WGPU_EXPORT WGPUProc wgpuGetProcAddress(WGPUDevice device, char const * procName
/**
* \defgroup Methods
* \brief Functions that are relative to a specific object.
*
*
* @{
*/

Expand All @@ -332,7 +333,7 @@ WGPU_EXPORT WGPUProc wgpuGetProcAddress(WGPUDevice device, char const * procName
/**
* \defgroup WGPU{{$object.Name | PascalCase}}Methods WGPU{{$object.Name | PascalCase}} methods
* \brief Functions whose first argument has type WGPU{{$object.Name | PascalCase}}.
*
*
* @{
*/
{{- range $object.Methods}}
Expand Down
12 changes: 6 additions & 6 deletions gen/gen.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,23 +20,23 @@ func (g *Generator) Gen(dst io.Writer) error {
t := template.
New("").
Funcs(template.FuncMap{
"SComment": func(v string, indent int) string { return Comment(v, CommentTypeSingleLine, indent, true) },
"MComment": func(v string, indent int) string { return Comment(v, CommentTypeMultiLine, indent, true) },
"SCommentN": func(v string, indent int) string { return Comment(v, CommentTypeSingleLine, indent, false) },
"MCommentN": func(v string, indent int) string { return Comment(v, CommentTypeMultiLine, indent, false) },
"SComment": func(v string, indent int) string { return Comment(v, CommentTypeSingleLine, indent, true) },
"MComment": func(v string, indent int) string { return Comment(v, CommentTypeMultiLine, indent, true) },
"SCommentN": func(v string, indent int) string { return Comment(v, CommentTypeSingleLine, indent, false) },
"MCommentN": func(v string, indent int) string { return Comment(v, CommentTypeMultiLine, indent, false) },
"MCommentEnum": func(v string, indent int, prefix string, e Enum, entryIndex int) string {
if v == "" || strings.TrimSpace(v) == "TODO" {
return ""
}
value, _ := g.EnumValue(prefix, e, entryIndex)
return Comment("`" + value + "` " + v, CommentTypeMultiLine, indent, true)
return Comment("`"+value+"` \n"+v, CommentTypeMultiLine, indent, true)
},
"MCommentBitflag": func(v string, indent int, b Bitflag, entryIndex int) string {
if v == "" || strings.TrimSpace(v) == "TODO" {
return ""
}
value, _ := g.BitflagValue(b, entryIndex)
return Comment("`" + value + "` " + v, CommentTypeMultiLine, indent, true)
return Comment("`"+value+"` "+v, CommentTypeMultiLine, indent, true)
},
"ConstantCase": ConstantCase,
"PascalCase": PascalCase,
Expand Down
Loading

0 comments on commit c803c81

Please sign in to comment.