The generated "c" code for cabi_realloc looks like this:
__attribute__((__weak__, __export_name__("cabi_realloc"))) void *cabi_realloc(void *ptr, size_t old_size, size_t align, size_t new_size)
{
if (new_size == 0)
return (void *)align;
void *ret = realloc(ptr, new_size);
if (!ret)
abort();
return ret;
}
My instinct was for something more like:
__attribute__((__weak__, __export_name__("cabi_realloc"))) void *cabi_realloc(void *ptr, size_t old_size, size_t align, size_t new_size)
{
if (new_size == 0) {
free(ptr);
return 0;
}
void *ret = realloc(ptr, new_size);
if (!ret)
abort();
return ret;
}
This would allow the host to cleanup any wasm memory it had previously allocated via calls to cabi_realloc?
POSIX realloc doesn't free on size = 0; thats a non-portable glibc behavior. (Which isn't to say that cabi_realloc couldn't adopt that behavior, just probably why it doesn't.)
IMO the glibc behavior is more confusing/surprising.
I think technically this would be ok, but when the canonical ABI wants to free something it will always do so through cabi_free
as opposed to cabi_realloc
, so I don't think it's necessary to make the change as well
Ahhh - so the issue is more that my wit generated stubs didn't create an export for cabi_free
?
FWIW I am testing a simple wit file:
world test {
export echo: func(msg: string) -> string
}
And I create several wasm files that implement it (in various languages).
I am now writing a host which wants to call that function so I:
echo
functioncabi_post_echo
to free the memory that the result was usingHad there been a cabi_free exported from the wasm I would / should have called that?
This is what my exports look like from the c++ guest wasm:
(export "memory" (memory 0))
(export "cabi_post_echo" (func 150))
(export "cabi_realloc" (func 151))
(export "add" (func 152))
(export "add3" (func 153))
(export "sub" (func 154))
(export "echo" (func 155))
oh sorry I'm thiking about the past where cabi_free
was a thing, which it no longer is. In any case though your stub's cabi_post_echo
is the "free" function, and it's specialized for the echo
export
Which is fine for the "returned" string - but its not going to clean up the memory I allocated in step 1. ?
Memory allocated for params becomes owned by the guest function. e.g. if your guest is Rust it will be wrapped in an owned String
.
@Lann Martin - is that convention documented anywhere? I had a good look around the wit-bindgen docs and https://github.com/WebAssembly/component-model but didn't spot anything?
Also if that is the case, should it be taken care of in the generated stub:
__attribute__((__export_name__("echo")))
int32_t __wasm_export_test_echo(int32_t arg, int32_t arg0) {
test_string_t arg1 = (test_string_t) { (char*)(arg), (size_t)(arg0) };
test_string_t ret;
test_echo(&arg1, &ret);
-------->free(arg)<-------
int32_t ptr = (int32_t) &RET_AREA;
*((int32_t*)(ptr + 4)) = (int32_t) (ret).len;
*((int32_t*)(ptr + 0)) = (int32_t) (ret).ptr;
return ptr;
}
Type of thing?
I don't know if it is explicitly described anywhere.
Generated C would not free parameters because the guest code may want to keep them around. It would be up to the user to free parameters.
This sort of thing would typically be documented with the function declaration; I'm not sure how best to document generated C bindings like that.
Ok as long as I know what the official intention is I am happy enough to have:
world test {
// guests dispose all params as needed
// hosts call cabi_post_XXX to dispose "results" as needed
export echo: func(msg: string) -> string
...
}
(and it solves my initial issue with cabi_realloc not freeing!)
Gordon Smith has marked this topic as resolved.
@Lann Martin - can I assume that the opposite would be true for host functions that are imported into the wasm?
(If so then we would still need a cabi_free or equivalent)...
Gordon Smith has marked this topic as unresolved.
For calling an import I believe the caller is responsible for cleaning up both parameters and results. The difference with imports is that the caller is only touching its own memory, so it doesn't need to call a cabi_realloc
.
The callee (e.g. the host) may call cabi_realloc
to allocate the result
If the callee wants access to the parameters after the function returns it has to copy them out of the caller's memory.
All that said, I am most familiar with older versions of the ABI so hopefully someone will confirm my answer there :smile:
Getting this written down in a less-formal way (than the canonical ABI's Python definition) so that everyone can understand is a pretty high priority for many of us. @Till Schneidereit
agreed! Relatedly, I was just in a meeting where we discussed how specifying some details of the ABI that Component tooling otherwise abstracts away, because we want to enable runtimes to support WASI Preview2+ without requiring the whole component model
Great info - I did spend too much time looking for this info (but did learn a lot on the way). tldr:
// Imports ---
// guests dispose params as needed
// guests dispose "results" as needed
import dbglog: func(msg: string)
// Exports ---
// guests dispose params as needed
// hosts call cabi_post_XXX to dispose "results" as needed
export echo: func(msg: string) -> string
Gordon Smith has marked this topic as resolved.
Thanks, I saw a PR that fixed up the imports for threads (moving the ret aret to the stack), so just wondered. (https://github.com/bytecodealliance/wit-bindgen/pull/326)
Last updated: Jan 24 2025 at 00:11 UTC