Return string form rust in Wasm

I’m currently learning the interaction between wasm and elixir. I am writing wasm in rust language, but it can only return the starting position of the address of the string, which is *const u8. I can’t get the length. What is the solution? Thanks again.

Thanks! :smiley:

TL;DR: At least on x86_64 a &str is two u64 (16 byte) long. The first u64 is a pointer and the second u64 is the length of the string.
Rust insight: memory layout of "&str" – PHIPS BLOG

So with binary patten matching you could extract the string lenght like this:
<<txt::64, length::64>> = string_binary

Be mindfull though Rust also has a String struct which is layed out differently in memory.

Hope this helps.

Hello, I don’t know what should be returned, sorry to ask you again, wasm seems to only return i32/i64 and f32/f64. I still don’t know how to combine the address and length of the memory together

Thanks, I’ve found a way to store the address and length using the high and low bits in binary. Then use bit operation to extract it, also based on your blog, thank you very much :grinning: :grin:

YW. It is not my blog though just something I found on google. The author of the blog is Philipp Schuster.

We know wasm is unable to return the string directly, of course we can use some libraries, but with elixir wasmex is unable to get directly, we usually pass in the address and length of the string through the memory function to get the string. Currently some languages already support multi-value return, but rust is currently unable to do multi-value return, so I used binary bit manipulation to solve this problem this is my github repository, I hope to get everyone’s advice

What’s stopping you from getting a tuple, or a dedicated struct even?

Furthermore, why must you poke in the guts of a static string slice? Can’t you just return a copied String?

Not sure what’s the problem that you seek to solve here. You only mention your dig at the problem, and not the root problem itself.

In rust, Consider the following function:

pub extern "C" fn pair(a: u32, b: u32) -> [u32; 2] {
    [a, b]

He will compile the return value as a parameter
LLVM will by default compile this down into the following Wasm:

(func $pair (param i32 i32 i32)
  local.get 0
  local.get 2 offset=4
  local.get 0
  local.get 1

wasm only seems to support i32/i64 and f32/f64 at the moment, which is why I can’t return the strings directly. I probably didn’t answer very well, thanks for your comments. If it’s not the answer you want, I hope you can ask again, thank you very much

My question is: what are you trying to achieve exactly? Why do you want a pointer + length returned? Why not a normal copied string?

That’s what I don’t get. Not sure LLVM’s IR matters here? :thinking:

I want to use wasm written by rust to handle some data, the data is not necessarily integer and floating point but there is no string type supported in the documentation of wasm, if it is a boolean value I can use 0/1 instead, so I can’t return the string properly. wasmex is the hex library used by elixir to call wasm.

We can’t know the length like “hello world”, so we need to return both the start address and the length of the memory. That’s why I mentioned above that rust can’t return multiple values, so I had to figure out how to get his length and address when calling elixir. My idea was to combine memory and length into one integer return, and then split it up into length and memory address when elixir is called。

Thank you very much, I don’t know if this answer is the answer you want. Because my English is very poor, I may not be able to understand you correctly. I’m sorry, if you have questions you can keep asking, I want to learn more. :grinning: