diff --git a/cabi/README.md b/cabi/README.md index d9e5f0878..3efed1547 100644 --- a/cabi/README.md +++ b/cabi/README.md @@ -1,6 +1,12 @@ -### Usage +# Canonical ABI -The `cabi` package contains a single exported WebAssembly function `cabi_realloc` ([Canonical ABI] realloc). To use, import this package with `_`: +## Types + +Package `cabi` declares a number of types, including [Component Model](https://component-model.bytecodealliance.org/) [primitive types](https://component-model.bytecodealliance.org/design/wit.html#primitive-types), along with resource and handle types. + +## cabi_realloc + +The `cabi` package contains an exported WebAssembly function `cabi_realloc` ([Canonical ABI] realloc). To use, import this package with `_`: ``` import _ "github.com/ydnar/wasm-tools-go/cabi" diff --git a/cabi/memory.go b/cabi/memory.go new file mode 100644 index 000000000..f7f880d69 --- /dev/null +++ b/cabi/memory.go @@ -0,0 +1,35 @@ +package cabi + +import ( + "sync" + "unsafe" +) + +// TODO: remove this or move it to package cm? + +var ( + mu sync.Mutex + pointers = make(map[unsafe.Pointer]int) +) + +// KeepAlive reference counts a pointer. +// TODO: prove this works. +func KeepAlive(ptr unsafe.Pointer) { + mu.Lock() + n := pointers[ptr] + pointers[ptr] = n + 1 + mu.Unlock() +} + +// Drop drops a reference to ptr. +// TODO: prove this works. +func Drop(ptr unsafe.Pointer) { + mu.Lock() + n := pointers[ptr] + n -= 1 + // TODO: panic if n < 0? + if n <= 0 { + delete(pointers, ptr) + } + mu.Unlock() +} diff --git a/cabi/primitive.go b/cabi/primitive.go new file mode 100644 index 000000000..b4ab2b962 --- /dev/null +++ b/cabi/primitive.go @@ -0,0 +1,17 @@ +package cabi + +// TODO: remove this or move it to package cm. + +type Bool bool +type S8 int8 +type U8 uint8 +type S16 int16 +type U16 uint16 +type S32 int32 +type U32 uint32 +type S64 int64 +type U64 uint64 +type F32 float32 +type F64 float64 +type Char rune +type String string diff --git a/cabi/resource.go b/cabi/resource.go new file mode 100644 index 000000000..e240b8dcf --- /dev/null +++ b/cabi/resource.go @@ -0,0 +1,36 @@ +package cabi + +// TODO: remove this or move it to package cm. + +// Resource is the interface implemented by all [resource] types. +type Resource[T any] interface { + ResourceHandle() Handle[T] + // BorrowResource() Borrow[T] + // OwnResource() Own[T] +} + +// Handle is an opaque handle to a [resource]. +type Handle[T any] uint32 + +// Own is a handle to an owned [resource]. +type Own[T any] Handle[T] + +func (o Own[T]) Rep() T { + return Rep(Handle[T](o)) +} + +// Borrow is a handle to a borrowed [resource]. +type Borrow[T any] Handle[T] + +func (b Borrow[T]) Rep() T { + return Rep(Handle[T](b)) +} + +// TODO: can we use finalizers for dropping handles? + +// Rep returns the representation of handle, if any. +func Rep[T any, H Handle[T]](handle H) T { + // TODO: extract the actual representation from a table + var v T + return v +} diff --git a/cmd/wit-bindgen-go/cmd/syntax/syntax.go b/cmd/wit-bindgen-go/cmd/syntax/syntax.go deleted file mode 100644 index 8397adb09..000000000 --- a/cmd/wit-bindgen-go/cmd/syntax/syntax.go +++ /dev/null @@ -1,25 +0,0 @@ -package syntax - -import ( - "context" - "fmt" - - "github.com/urfave/cli/v3" - "github.com/ydnar/wasm-tools-go/internal/witcli" -) - -// Command is the CLI command for wit. -var Command = &cli.Command{ - Name: "wit", - Usage: "reverses a WIT JSON file into WIT syntax", - Action: action, -} - -func action(ctx context.Context, cmd *cli.Command) error { - res, err := witcli.LoadOne(cmd.Args().Slice()...) - if err != nil { - return err - } - fmt.Println(res.WIT(nil, "")) - return nil -} diff --git a/design/README.md b/design/README.md new file mode 100644 index 000000000..68359f79b --- /dev/null +++ b/design/README.md @@ -0,0 +1,3 @@ +# Design + +This directory contains hand-written Go packages that correspond to WIT definitions for the purposes of fleshing out a viable design. None of these packages should be imported or used. diff --git a/design/import-and-export/README.md b/design/import-and-export/README.md new file mode 100644 index 000000000..928289e1b --- /dev/null +++ b/design/import-and-export/README.md @@ -0,0 +1,5 @@ +# import-and-export + +This design example contains a subset of the `wasi:clocks` package with a world that both *imports* and *exports* the `wasi:clocks/wall-clock` interface. The purpose of this example is to sketch out a mechanism for generating Go code whereby an interface (and types) can simultaneously be imported and exported. + +The source WIT file is [world.wit](./world.wit). diff --git a/design/import-and-export/wasi/clocks/wallclock/exports.go b/design/import-and-export/wasi/clocks/wallclock/exports.go new file mode 100644 index 000000000..4edd78439 --- /dev/null +++ b/design/import-and-export/wasi/clocks/wallclock/exports.go @@ -0,0 +1,22 @@ +//go:build wasm + +package wallclock + +// Instance is the sole global instance of the +// WIT interface "wasi:clocks/wall-clock". +// Assign it to accept calls to this interface. +var Instance Interface + +// FIXME: correct type for struct return values +// +//go:wasmexport wasi:clocks/wall-clock now +func wasmexport_now() DateTime { + return Instance.Now() +} + +// FIXME: correct type for struct return values +// +//go:wasmexport wasi:clocks/wall-clock resolution +func wasmexport_resolution() DateTime { + return Instance.Resolution() +} diff --git a/design/import-and-export/wasi/clocks/wallclock/imports.go b/design/import-and-export/wasi/clocks/wallclock/imports.go new file mode 100644 index 000000000..0de970c76 --- /dev/null +++ b/design/import-and-export/wasi/clocks/wallclock/imports.go @@ -0,0 +1,25 @@ +//go:build wasm + +package wallclock + +// Now returns a DateTime for the current wall clock, corresponding to +// the Component Model function "wasi:clocks/wall-clock.now". +func Now() DateTime { + return wasmimport_now() +} + +// FIXME: correct type for struct return values +// +//go:wasmimport wasi:clocks/wall-clock now +func wasmimport_now() DateTime + +// Resolution returns the resolution of the current wall clock, corresponding to +// the Component Model function "wasi:clocks/wall-clock.resolution". +func Resolution() DateTime { + return wasmimport_resolution() +} + +// FIXME: correct type for struct return values +// +//go:wasmimport wasi:clocks/wall-clock resolution +func wasmimport_resolution() DateTime diff --git a/design/import-and-export/wasi/clocks/wallclock/types.go b/design/import-and-export/wasi/clocks/wallclock/types.go new file mode 100644 index 000000000..11def3665 --- /dev/null +++ b/design/import-and-export/wasi/clocks/wallclock/types.go @@ -0,0 +1,13 @@ +package wallclock + +// Interface is the Go implementation of WIT interface "wasi:clocks/wall-clock". +type Interface interface { + Now() DateTime + Resolution() DateTime +} + +// DateTime is a Go implementation of WIT type "wasi:clocks/wall-clock.datetime". +type DateTime struct { + Seconds uint64 + Nanonseconds uint32 +} diff --git a/design/import-and-export/world.wit b/design/import-and-export/world.wit new file mode 100644 index 000000000..5cadfa88a --- /dev/null +++ b/design/import-and-export/world.wit @@ -0,0 +1,15 @@ +package wasi:clocks; + +interface wall-clock { + record datetime { + seconds: u64, + nanoseconds: u32 + } + now: func() -> datetime; + resolution: func() -> datetime; +} + +world import-and-export { + import wall-clock; + export wall-clock; +} diff --git a/design/import-and-export/world.wit.json b/design/import-and-export/world.wit.json new file mode 100644 index 000000000..7be5ca114 --- /dev/null +++ b/design/import-and-export/world.wit.json @@ -0,0 +1,82 @@ +{ + "worlds": [ + { + "name": "import-and-export", + "imports": { + "interface-0": { + "interface": 0 + } + }, + "exports": { + "interface-0": { + "interface": 0 + } + }, + "package": 0 + } + ], + "interfaces": [ + { + "name": "wall-clock", + "types": { + "datetime": 0 + }, + "functions": { + "now": { + "name": "now", + "kind": "freestanding", + "params": [], + "results": [ + { + "type": 0 + } + ] + }, + "resolution": { + "name": "resolution", + "kind": "freestanding", + "params": [], + "results": [ + { + "type": 0 + } + ] + } + }, + "package": 0 + } + ], + "types": [ + { + "name": "datetime", + "kind": { + "record": { + "fields": [ + { + "name": "seconds", + "type": "u64" + }, + { + "name": "nanoseconds", + "type": "u32" + } + ] + } + }, + "owner": { + "interface": 0 + } + } + ], + "packages": [ + { + "name": "wasi:clocks", + "interfaces": { + "wall-clock": 0 + }, + "worlds": { + "import-and-export": 0 + } + } + ] +} \ No newline at end of file diff --git a/design/wasi/io/poll/exports/exports.go b/design/wasi/io/poll/exports/exports.go new file mode 100644 index 000000000..81a954866 --- /dev/null +++ b/design/wasi/io/poll/exports/exports.go @@ -0,0 +1,54 @@ +package exports + +import ( + "unsafe" + + "github.com/ydnar/wasm-tools-go/cabi" + "github.com/ydnar/wasm-tools-go/cm" +) + +// Interface implements the Component Model interface "wasi:io/poll". +type Interface interface { + Poll(in cm.List[cabi.Borrow[Pollable]]) cm.List[uint32] + Pollable() interface { + // constructor, static functions would go here + } +} + +// Export registers a concrete implementation of "wasi:io/poll". +func Export(i Interface) { + impl = i +} + +// TODO: make a default implementation that panics with a helpful message on all function calls. +var impl Interface + +//go:wasmexport wasi:io/poll#poll +func poll(in cm.List[cabi.Borrow[Pollable]], result *cm.List[uint32]) { + *result = impl.Poll(in) + // sData := unsafe.SliceData(s.Slice()) + // cabi.KeepAlive(unsafe.Pointer(sData)) +} + +//go:wasmexport wasi:io/poll#cabi_post_poll +func cabi_post_poll(result *cm.List[uint32]) { + // Is this necessary if the Go GC runs after the wasmexport call? + cabi.Drop(unsafe.Pointer(result.Data())) +} + +//go:wasmexport wasi:io/poll#[method]pollable.block +func method_pollable_block(self cabi.Borrow[Pollable]) { + self.Rep().Block() +} + +//go:wasmexport wasi:io/poll#[method]pollable.ready +func method_pollable_ready(self cabi.Borrow[Pollable]) bool { + return self.Rep().Ready() +} + +// Pollable represents the Component Model type "wasi:io/poll.pollable". +type Pollable interface { + Block() + Ready() bool + cabi.Resource[Pollable] +} diff --git a/design/wasi/io/poll/imports/imports.go b/design/wasi/io/poll/imports/imports.go new file mode 100644 index 000000000..9ad20a3f8 --- /dev/null +++ b/design/wasi/io/poll/imports/imports.go @@ -0,0 +1,48 @@ +//go:build wasm + +package imports + +import ( + "unsafe" + + "github.com/ydnar/wasm-tools-go/cabi" + "github.com/ydnar/wasm-tools-go/design/wasi/io/poll" +) + +// Pollable represents the Component Model type "wasi:io/poll.pollable". +type Pollable cabi.Handle[poll.Pollable] + +var _ poll.Pollable = Pollable(0) + +// Poll imports Component Model func "wasi:io/poll.poll". +func Poll(in []poll.Pollable) []uint32 { + in_ := make([]Pollable, len(in)) + for i := range in { + in_[i] = Pollable(in[i].ResourceHandle()) + } + ptr, size := wasmimport_poll(unsafe.SliceData(in_), uint32(len(in_))) + return unsafe.Slice(ptr, size) +} + +func (self Pollable) ResourceHandle() cabi.Handle[poll.Pollable] { + return cabi.Handle[poll.Pollable](self) +} + +//go:wasmimport wasi:io/poll poll +func wasmimport_poll(data *Pollable, size uint32) (*uint32, int32) + +// Block imports Component Model method "wasi:io/poll.pollable.block". +func (self Pollable) Block() { + wasmimport_pollable_block(self) +} + +//go:wasmimport wasi:io/poll [method]pollable.block +func wasmimport_pollable_block(self Pollable) + +// Ready imports Component Model method "wasi:io/poll.pollable.ready". +func (self Pollable) Ready() bool { + return wasmimport_pollable_ready(self) +} + +//go:wasmimport wasi:io/poll [method]pollable.ready +func wasmimport_pollable_ready(self Pollable) bool diff --git a/design/wasi/io/poll/types.go b/design/wasi/io/poll/types.go new file mode 100644 index 000000000..b1539df91 --- /dev/null +++ b/design/wasi/io/poll/types.go @@ -0,0 +1,10 @@ +package poll + +import "github.com/ydnar/wasm-tools-go/cabi" + +// Pollable represents the Component Model type "wasi:io/poll.pollable". +type Pollable interface { + Block() + Ready() bool + cabi.Resource[Pollable] +} diff --git a/design/wasi/io/streams/v0.2.0/example.go b/design/wasi/io/streams/v0.2.0/example.go new file mode 100644 index 000000000..8aecaafdc --- /dev/null +++ b/design/wasi/io/streams/v0.2.0/example.go @@ -0,0 +1,3 @@ +package streams + +const Version = "0.2.0" diff --git a/design/wasi/io/world.wit b/design/wasi/io/world.wit new file mode 100644 index 000000000..6ee210c8d --- /dev/null +++ b/design/wasi/io/world.wit @@ -0,0 +1,49 @@ +package wasi:io; + +interface poll { + resource pollable { + block: func(); + ready: func() -> bool; + } + poll: func(in: list>) -> list; +} + +interface streams { + resource error { + to-debug-string: func() -> string; + } + resource input-stream { + blocking-read: func(len: u64) -> result, stream-error>; + blocking-skip: func(len: u64) -> result; + read: func(len: u64) -> result, stream-error>; + skip: func(len: u64) -> result; + subscribe: func() -> own; + } + resource output-stream { + blocking-flush: func() -> result<_, stream-error>; + blocking-splice: func(src: borrow, len: u64) -> result; + blocking-write-and-flush: func(contents: list) -> result<_, stream-error>; + blocking-write-zeroes-and-flush: func(len: u64) -> result<_, stream-error>; + check-write: func() -> result; + flush: func() -> result<_, stream-error>; + splice: func(src: borrow, len: u64) -> result; + subscribe: func() -> own; + write: func(contents: list) -> result<_, stream-error>; + write-zeroes: func(len: u64) -> result<_, stream-error>; + } + use poll.{pollable}; + variant stream-error { + last-operation-failed(own), + closed + } +} + +world imports { + import poll; + import streams; +} + +world exports { + export poll; + export streams; +} diff --git a/design/wasi/io/world.wit.json b/design/wasi/io/world.wit.json new file mode 100644 index 000000000..1f514cc6a --- /dev/null +++ b/design/wasi/io/world.wit.json @@ -0,0 +1,596 @@ +{ + "worlds": [ + { + "name": "imports", + "imports": { + "interface-0": { + "interface": 0 + }, + "interface-1": { + "interface": 1 + } + }, + "exports": {}, + "package": 0 + }, + { + "name": "exports", + "imports": {}, + "exports": { + "interface-0": { + "interface": 0 + }, + "interface-1": { + "interface": 1 + } + }, + "package": 0 + } + ], + "interfaces": [ + { + "name": "poll", + "types": { + "pollable": 0 + }, + "functions": { + "[method]pollable.block": { + "name": "[method]pollable.block", + "kind": { + "method": 0 + }, + "params": [ + { + "name": "self", + "type": 1 + } + ], + "results": [] + }, + "[method]pollable.ready": { + "name": "[method]pollable.ready", + "kind": { + "method": 0 + }, + "params": [ + { + "name": "self", + "type": 1 + } + ], + "results": [ + { + "type": "bool" + } + ] + }, + "poll": { + "name": "poll", + "kind": "freestanding", + "params": [ + { + "name": "in", + "type": 2 + } + ], + "results": [ + { + "type": 3 + } + ] + } + }, + "package": 0 + }, + { + "name": "streams", + "types": { + "pollable": 4, + "error": 5, + "input-stream": 6, + "output-stream": 7, + "stream-error": 9 + }, + "functions": { + "[method]error.to-debug-string": { + "name": "[method]error.to-debug-string", + "kind": { + "method": 5 + }, + "params": [ + { + "name": "self", + "type": 10 + } + ], + "results": [ + { + "type": "string" + } + ] + }, + "[method]input-stream.blocking-read": { + "name": "[method]input-stream.blocking-read", + "kind": { + "method": 6 + }, + "params": [ + { + "name": "self", + "type": 11 + }, + { + "name": "len", + "type": "u64" + } + ], + "results": [ + { + "type": 13 + } + ] + }, + "[method]input-stream.blocking-skip": { + "name": "[method]input-stream.blocking-skip", + "kind": { + "method": 6 + }, + "params": [ + { + "name": "self", + "type": 11 + }, + { + "name": "len", + "type": "u64" + } + ], + "results": [ + { + "type": 14 + } + ] + }, + "[method]input-stream.read": { + "name": "[method]input-stream.read", + "kind": { + "method": 6 + }, + "params": [ + { + "name": "self", + "type": 11 + }, + { + "name": "len", + "type": "u64" + } + ], + "results": [ + { + "type": 13 + } + ] + }, + "[method]input-stream.skip": { + "name": "[method]input-stream.skip", + "kind": { + "method": 6 + }, + "params": [ + { + "name": "self", + "type": 11 + }, + { + "name": "len", + "type": "u64" + } + ], + "results": [ + { + "type": 14 + } + ] + }, + "[method]input-stream.subscribe": { + "name": "[method]input-stream.subscribe", + "kind": { + "method": 6 + }, + "params": [ + { + "name": "self", + "type": 11 + } + ], + "results": [ + { + "type": 17 + } + ] + }, + "[method]output-stream.blocking-flush": { + "name": "[method]output-stream.blocking-flush", + "kind": { + "method": 7 + }, + "params": [ + { + "name": "self", + "type": 15 + } + ], + "results": [ + { + "type": 16 + } + ] + }, + "[method]output-stream.blocking-splice": { + "name": "[method]output-stream.blocking-splice", + "kind": { + "method": 7 + }, + "params": [ + { + "name": "self", + "type": 15 + }, + { + "name": "src", + "type": 11 + }, + { + "name": "len", + "type": "u64" + } + ], + "results": [ + { + "type": 14 + } + ] + }, + "[method]output-stream.blocking-write-and-flush": { + "name": "[method]output-stream.blocking-write-and-flush", + "kind": { + "method": 7 + }, + "params": [ + { + "name": "self", + "type": 15 + }, + { + "name": "contents", + "type": 12 + } + ], + "results": [ + { + "type": 16 + } + ] + }, + "[method]output-stream.blocking-write-zeroes-and-flush": { + "name": "[method]output-stream.blocking-write-zeroes-and-flush", + "kind": { + "method": 7 + }, + "params": [ + { + "name": "self", + "type": 15 + }, + { + "name": "len", + "type": "u64" + } + ], + "results": [ + { + "type": 16 + } + ] + }, + "[method]output-stream.check-write": { + "name": "[method]output-stream.check-write", + "kind": { + "method": 7 + }, + "params": [ + { + "name": "self", + "type": 15 + } + ], + "results": [ + { + "type": 14 + } + ] + }, + "[method]output-stream.flush": { + "name": "[method]output-stream.flush", + "kind": { + "method": 7 + }, + "params": [ + { + "name": "self", + "type": 15 + } + ], + "results": [ + { + "type": 16 + } + ] + }, + "[method]output-stream.splice": { + "name": "[method]output-stream.splice", + "kind": { + "method": 7 + }, + "params": [ + { + "name": "self", + "type": 15 + }, + { + "name": "src", + "type": 11 + }, + { + "name": "len", + "type": "u64" + } + ], + "results": [ + { + "type": 14 + } + ] + }, + "[method]output-stream.subscribe": { + "name": "[method]output-stream.subscribe", + "kind": { + "method": 7 + }, + "params": [ + { + "name": "self", + "type": 15 + } + ], + "results": [ + { + "type": 17 + } + ] + }, + "[method]output-stream.write": { + "name": "[method]output-stream.write", + "kind": { + "method": 7 + }, + "params": [ + { + "name": "self", + "type": 15 + }, + { + "name": "contents", + "type": 12 + } + ], + "results": [ + { + "type": 16 + } + ] + }, + "[method]output-stream.write-zeroes": { + "name": "[method]output-stream.write-zeroes", + "kind": { + "method": 7 + }, + "params": [ + { + "name": "self", + "type": 15 + }, + { + "name": "len", + "type": "u64" + } + ], + "results": [ + { + "type": 16 + } + ] + } + }, + "package": 0 + } + ], + "types": [ + { + "name": "pollable", + "kind": "resource", + "owner": { + "interface": 0 + } + }, + { + "name": null, + "kind": { + "handle": { + "borrow": 0 + } + }, + "owner": null + }, + { + "name": null, + "kind": { + "list": 1 + }, + "owner": null + }, + { + "name": null, + "kind": { + "list": "u32" + }, + "owner": null + }, + { + "name": "pollable", + "kind": { + "type": 0 + }, + "owner": { + "interface": 1 + } + }, + { + "name": "error", + "kind": "resource", + "owner": { + "interface": 1 + } + }, + { + "name": "input-stream", + "kind": "resource", + "owner": { + "interface": 1 + } + }, + { + "name": "output-stream", + "kind": "resource", + "owner": { + "interface": 1 + } + }, + { + "name": null, + "kind": { + "handle": { + "own": 5 + } + }, + "owner": null + }, + { + "name": "stream-error", + "kind": { + "variant": { + "cases": [ + { + "name": "last-operation-failed", + "type": 8 + }, + { + "name": "closed", + "type": null + } + ] + } + }, + "owner": { + "interface": 1 + } + }, + { + "name": null, + "kind": { + "handle": { + "borrow": 5 + } + }, + "owner": null + }, + { + "name": null, + "kind": { + "handle": { + "borrow": 6 + } + }, + "owner": null + }, + { + "name": null, + "kind": { + "list": "u8" + }, + "owner": null + }, + { + "name": null, + "kind": { + "result": { + "ok": 12, + "err": 9 + } + }, + "owner": null + }, + { + "name": null, + "kind": { + "result": { + "ok": "u64", + "err": 9 + } + }, + "owner": null + }, + { + "name": null, + "kind": { + "handle": { + "borrow": 7 + } + }, + "owner": null + }, + { + "name": null, + "kind": { + "result": { + "ok": null, + "err": 9 + } + }, + "owner": null + }, + { + "name": null, + "kind": { + "handle": { + "own": 4 + } + }, + "owner": null + } + ], + "packages": [ + { + "name": "wasi:io", + "interfaces": { + "poll": 0, + "streams": 1 + }, + "worlds": { + "imports": 0, + "exports": 1 + } + } + ] +} \ No newline at end of file