examples(allocation): free memory after unmarshalling a result from the guest (#1368)

Signed-off-by: Luca Burgazzoli <lburgazzoli@gmail.com>
This commit is contained in:
Luca Burgazzoli
2023-04-18 07:38:10 +02:00
committed by GitHub
parent 6a4bdd31db
commit 5aafcc4836
3 changed files with 22 additions and 6 deletions

View File

@@ -90,9 +90,15 @@ func main() {
if err != nil {
log.Panicln(err)
}
// Note: This pointer is still owned by TinyGo, so don't try to free it!
greetingPtr := uint32(ptrSize[0] >> 32)
greetingSize := uint32(ptrSize[0])
// We don't need the memory after deserialization: make sure it is freed.
if greetingPtr != 0 {
defer free.Call(ctx, uint64(greetingPtr))
}
// The pointer is a linear memory offset, which is where we write the name.
if bytes, ok := mod.Memory().Read(greetingPtr, greetingSize); !ok {
log.Panicf("Memory.Read(%d, %d) out of range of memory size %d",

View File

@@ -6,6 +6,9 @@ import (
"unsafe"
)
// #include <stdlib.h>
import "C"
// main is required for TinyGo to compile to Wasm.
func main() {}
@@ -70,10 +73,17 @@ func ptrToString(ptr uint32, size uint32) string {
}
// stringToPtr returns a pointer and size pair for the given string in a way
// compatible with WebAssembly numeric types.
// compatible with WebAssembly numeric types. The pointer is not automatically
// managed by tinygo but must be freed by the host.
func stringToPtr(s string) (uint32, uint32) {
buf := []byte(s)
ptr := &buf[0]
unsafePtr := uintptr(unsafe.Pointer(ptr))
return uint32(unsafePtr), uint32(len(buf))
if len(s) == 0 {
return 0, 0
}
size := C.ulong(len(s))
ptr := unsafe.Pointer(C.malloc(size))
copy(unsafe.Slice((*byte)(ptr), size), []byte(s))
return uint32(uintptr(ptr)), uint32(len(s))
}

Binary file not shown.