Stream: git-wasmtime

Topic: wasmtime / Issue #2864 writev only outputs first paramete...


view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 18:12):

expnkx labeled Issue #2864:

Thanks for filing a bug report! Please fill out the TODOs below.

Test Case

#include<fast_io.h>

int main()
{
print(fast_io::out(),"Hello World\n",fast_io::posix_clock_gettime(fast_io::posix_clock_id::realtime));
}

Expected Results

It should output all results.

Actual Results

It only outputs the first one

Versions and Environment

Wasmtime version or commit: main

Operating system: windows

Architecture: x86_64

Extra Info

-stdlib=libstdc++

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 18:12):

expnkx opened Issue #2864:

Thanks for filing a bug report! Please fill out the TODOs below.

Test Case

#include<fast_io.h>

int main()
{
print(fast_io::out(),"Hello World\n",fast_io::posix_clock_gettime(fast_io::posix_clock_id::realtime));
}

Expected Results

It should output all results.

Actual Results

It only outputs the first one

Versions and Environment

Wasmtime version or commit: main

Operating system: windows

Architecture: x86_64

Extra Info

-stdlib=libstdc++

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 18:13):

expnkx edited Issue #2864:

Thanks for filing a bug report! Please fill out the TODOs below.

Test Case

https://github.com/expnkx/fast_io/blob/02cc01ceea5cba4a40e35dc4ae1fd64e68148e2d/include/fast_io_hosted/platforms/posix.h#L1512

#include<fast_io.h>

int main()
{
print(fast_io::out(),"Hello World\n",fast_io::posix_clock_gettime(fast_io::posix_clock_id::realtime));
}

Expected Results

It should output all results.

Actual Results

It only outputs the first one

Versions and Environment

Wasmtime version or commit: main

Operating system: windows

Architecture: x86_64

Extra Info

-stdlib=libstdc++

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 18:13):

expnkx edited Issue #2864:

Thanks for filing a bug report! Please fill out the TODOs below.

Test Case

https://github.com/expnkx/fast_io/blob/02cc01ceea5cba4a40e35dc4ae1fd64e68148e2d/include/fast_io_hosted/platforms/posix.h#L1512

#include<fast_io.h>

int main()
{
    print(fast_io::out(),"Hello World\n",fast_io::posix_clock_gettime(fast_io::posix_clock_id::realtime));
}

Expected Results

It should output all results.

Actual Results

It only outputs the first one

Versions and Environment

Wasmtime version or commit: main

Operating system: windows

Architecture: x86_64

Extra Info

-stdlib=libstdc++

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 18:21):

sunfishcode commented on Issue #2864:

As with POSIX writev, __wasi_fd_write may write fewer bytes than requested. It should return the number of bytes read, so that you can call it again with the unwritten portion of the buffer(s).

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 20:44):

expnkx commented on Issue #2864:

As with POSIX writev, __wasi_fd_write may write fewer bytes than requested. It should return the number of bytes read, so that you can call it again with the unwritten portion of the buffer(s).

The problem is that these approaches break the atomicity of syscalls. And it might in a situation it only writes zero-bytes example. Also it leads to binary bloat.

It requires a fix in wasmtime tbh.

https://github.com/expnkx/fast_io/blob/725695664091bc4e8fbb05df446aaad93642bbf5/include/fast_io_hosted/platforms/win32.h#L613
Here is my implementation on windows. Use LockFile to lock the file

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 20:44):

expnkx edited a comment on Issue #2864:

As with POSIX writev, __wasi_fd_write may write fewer bytes than requested. It should return the number of bytes read, so that you can call it again with the unwritten portion of the buffer(s).

The problem is that these approaches break the atomicity of syscalls. And it might in a situation it only writes zero-bytes example, the users have no idea how to deal with it. Also, it leads to binary bloat.

It requires a fix in wasmtime tbh.

https://github.com/expnkx/fast_io/blob/725695664091bc4e8fbb05df446aaad93642bbf5/include/fast_io_hosted/platforms/win32.h#L613
Here is my implementation on windows. Use LockFile to lock the file

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 20:53):

bjorn3 commented on Issue #2864:

What if part of the data could be written and then an error is returned? It would be incorrect to suggest that no data was written by returning an error. Returning only how many bytes werw actually written would still require the loop. The error could be EWOULDBLOCK for example which in case of async usage indicates that it will be possible in the future to write more data. In that case the application needs to know how many bytes were written and try again later. Moving thr write loop into the write/writev call also doesn't match the expectations of pretty much every unix application.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 20:56):

bjorn3 commented on Issue #2864:

And it might in a situation it only writes zero-bytes example, the users have no idea how to deal with it.

I don't think that is possible. For read itbis possible though and indicates EOF.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 20:59):

expnkx commented on Issue #2864:

And it might in a situation it only writes zero-bytes example, the users have no idea how to deal with it.

I don't think that is possible. For read itbis possible though and indicates EOF.

This is not readv or read. This is writev. It does not mean it is EOF.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:00):

expnkx edited a comment on Issue #2864:

And it might in a situation it only writes zero-bytes example, the users have no idea how to deal with it.

I don't think that is possible. For read itbis possible though and indicates EOF.

This is not readv or read. This is writev. It does not mean it is EOF.
Checking size is very bad idea imo for this since it can lead to block of the process.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:00):

bjorn3 edited a comment on Issue #2864:

And it might in a situation it only writes zero-bytes example, the users have no idea how to deal with it.

I don't think that is possible. For read it is possible though and indicates EOF.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:08):

sunfishcode commented on Issue #2864:

The code bloat is a consequence of POSIX here; no implementation on a POSIX host can guarantee that it'll write all of all the buffers atomically without being interrupted. There has to be a loop somewhere, and if we follow POSIX, we can't put the loop in the WASI implementation because that would make it non-atomic.

WASI is working on some new I/O facilities which don't follow POSIX as closely, and when those are available we may have other options here. But right now, we have __wasi_fd_write and it does follow POSIX, and POSIX requires you to have a loop if you want all the buffers printed.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:14):

expnkx commented on Issue #2864:

The code bloat is a consequence of POSIX here; no implementation on a POSIX host can guarantee that it'll write all of all the buffers atomically without being interrupted. There has to be a loop somewhere, and if we follow POSIX, we can't put the loop in the WASI implementation because that would make it non-atomic.

WASI is working on some new I/O facilities which don't follow POSIX as closely, and when those are available we may have other options here. But right now, we have __wasi_fd_write and it does follow POSIX, and POSIX requires you to have a loop if you want all the buffers printed.

this has nothing to do with the code works on Linux but not on windows. Windows can work as long as you lock the file.
It is absolutely the project's fault, not POSIX. POSIX never said you can provide a buggy implementation.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:14):

expnkx edited a comment on Issue #2864:

The code bloat is a consequence of POSIX here; no implementation on a POSIX host can guarantee that it'll write all of all the buffers atomically without being interrupted. There has to be a loop somewhere, and if we follow POSIX, we can't put the loop in the WASI implementation because that would make it non-atomic.

WASI is working on some new I/O facilities which don't follow POSIX as closely, and when those are available we may have other options here. But right now, we have __wasi_fd_write and it does follow POSIX, and POSIX requires you to have a loop if you want all the buffers printed.

this has nothing to do with the code works on Linux but not on windows. Windows can work as long as you lock the file. I write the same code too and I never have any issues on windows.
It is absolutely the project's fault, not POSIX. POSIX never said you can provide a buggy implementation.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:18):

expnkx edited a comment on Issue #2864:

The code bloat is a consequence of POSIX here; no implementation on a POSIX host can guarantee that it'll write all of all the buffers atomically without being interrupted. There has to be a loop somewhere, and if we follow POSIX, we can't put the loop in the WASI implementation because that would make it non-atomic.

WASI is working on some new I/O facilities which don't follow POSIX as closely, and when those are available we may have other options here. But right now, we have __wasi_fd_write and it does follow POSIX, and POSIX requires you to have a loop if you want all the buffers printed.

this has nothing to do with the code works on Linux but not on windows. Windows can work as long as you lock the file. I write the same code too and I never have any issues on windows.
It is absolutely the project's fault, not POSIX. POSIX never said you can provide a buggy implementation.

Atomicity of syscalls cannot be guaranteed with writev either. an example is a remote console or pipe on Linux. writev is not atomic either. However, file operations beside NFS are always atomic.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:19):

expnkx edited a comment on Issue #2864:

The code bloat is a consequence of POSIX here; no implementation on a POSIX host can guarantee that it'll write all of all the buffers atomically without being interrupted. There has to be a loop somewhere, and if we follow POSIX, we can't put the loop in the WASI implementation because that would make it non-atomic.

WASI is working on some new I/O facilities which don't follow POSIX as closely, and when those are available we may have other options here. But right now, we have __wasi_fd_write and it does follow POSIX, and POSIX requires you to have a loop if you want all the buffers printed.

this has nothing to do with the code works on Linux but not on windows. Windows can work as long as you lock the file. I write the same code too and I never have any issues on windows.
It is absolutely the project's fault, not POSIX. POSIX never said you can provide a buggy implementation.

Atomicity of syscalls cannot be guaranteed with writev either. an example is a remote console or pipe on Linux. writev is not atomic either. However, file operations besides NFS are always atomic.

The code works just fine on windows and it ensures atomicity for file operations.
https://github.com/expnkx/fast_io/blob/725695664091bc4e8fbb05df446aaad93642bbf5/include/fast_io_hosted/platforms/win32.h#L613

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:21):

bjorn3 commented on Issue #2864:

The implementation is not buggy on windows. It can behave in exactly the same way on linux too. You are just lucky that your the amount of text you are printing is smaller than the pipe buffer size (64k on linux) and that your terminal immediately empties the pipe as soon as data gets received. If you were to write more than 64k at once, you are guaranteed to only get a partial write and when your terminal is too slow, you may get a partial write. This is just the semantics of the write function as defined by POSIX.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:34):

iximeow commented on Issue #2864:

And it might in a situation it only writes zero-bytes example

i also don't think this is possible - wasi-libc, wasmtime, and cap-std just pass the iovecs to Rust's stdlib. the default_write_vectored impl there selects the first _nonempty_ iovec for writing. if write_vectored returns zero it's because the system returned zero; either the iovec selected for writing was zero-length (and all iovs were zero-length), or a non-empty vector was selected for writing and the OS itself wrote zero bytes.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:35):

iximeow edited a comment on Issue #2864:

And it might in a situation it only writes zero-bytes example

i also don't think this is possible - wasi-libc, wasmtime, and cap-std currently just pass the iovecs to Rust's stdlib. the default_write_vectored impl there selects the first _nonempty_ iovec for writing. if write_vectored returns zero it's because the system returned zero; either the iovec selected for writing was zero-length (and all iovs were zero-length), or a non-empty vector was selected for writing and the OS itself wrote zero bytes.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:38):

expnkx commented on Issue #2864:

And it might in a situation it only writes zero-bytes example

i also don't think this is possible - wasi-libc, wasmtime, and cap-std currently just pass the iovecs to Rust's stdlib. the default_write_vectored impl there selects the first _nonempty_ iovec for writing. if write_vectored returns zero it's because the system returned zero; either the iovec selected for writing was zero-length (and all iovs were zero-length), or a non-empty vector was selected for writing and the OS itself wrote zero bytes.

wraps around rust's stdlib are not something that is supposed to work with wasm imo. WASI is to abstract OS interface, not to abstract language interface.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:39):

expnkx edited a comment on Issue #2864:

And it might in a situation it only writes zero-bytes example

i also don't think this is possible - wasi-libc, wasmtime, and cap-std currently just pass the iovecs to Rust's stdlib. the default_write_vectored impl there selects the first _nonempty_ iovec for writing. if write_vectored returns zero it's because the system returned zero; either the iovec selected for writing was zero-length (and all iovs were zero-length), or a non-empty vector was selected for writing and the OS itself wrote zero bytes.

wraps around rust's stdlib are not something that is supposed to work with wasm imo. WASI is to abstract OS interface, not to abstract language interface.
BTW I do see IO performance bottleneck with using Rust stdlib.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 21:54):

sunfishcode commented on Issue #2864:

POSIX says "If write() is interrupted by a signal after it successfully writes some data, it shall return the number of bytes written". So even on Linux, where things may appear to work, they're not guaranteed to work in all cases. The system call could still be interrupted in some circumstances, and could perform an incomplete write in some circumstances. Consequently, Wasm code needs a loop if it wants all the buffers printed.

In theory, a __wasi_fd_write call could return zero. But this shouldn't be any different from it returning any other number less than the requested number. If the wasm code has a loop, the zero case isn't special.

WASI today is a synchronous API. It blocks. Many people are obviously interested in async I/O; various approaches are in development, but not available yet. So for now, it's a blocking API, and adding a loop around it doesn't introduce any new blocking.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 22:00):

expnkx commented on Issue #2864:

POSIX says "If write() is interrupted by a signal after it successfully writes some data, it shall return the number of bytes written". So even on Linux, where things may appear to work, they're not guaranteed to work in all cases. The system call could still be interrupted in some circumstances, and could perform an incomplete write in some circumstances. Consequently, Wasm code needs a loop if it wants all the buffers printed.

In theory, a __wasi_fd_write call could return zero. But this shouldn't be any different from it returning any other number less than the requested number. If the wasm code has a loop, the zero case isn't special.

WASI today is a synchronous API. It blocks. Many people are obviously interested in async I/O; various approaches are in development, but not available yet. So for now, it's a blocking API, and adding a loop around it doesn't introduce any new blocking.

if the API keeps returning 0, you get a block.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 28 2021 at 22:18):

sunfishcode commented on Issue #2864:

if the API keeps returning 0, you get a block.

That is correct. The API itself can also block.

view this post on Zulip Wasmtime GitHub notifications bot (May 07 2021 at 18:55):

sunfishcode commented on Issue #2864:

The implementation is working as intended. To write a complete string reliably with the current WASI APIs, it's necessary to wrap the __wasi_fd_write call in a loop, as libc does.

view this post on Zulip Wasmtime GitHub notifications bot (May 07 2021 at 18:55):

sunfishcode closed Issue #2864:

Thanks for filing a bug report! Please fill out the TODOs below.

Test Case

https://github.com/expnkx/fast_io/blob/02cc01ceea5cba4a40e35dc4ae1fd64e68148e2d/include/fast_io_hosted/platforms/posix.h#L1512

#include<fast_io.h>

int main()
{
    print(fast_io::out(),"Hello World\n",fast_io::posix_clock_gettime(fast_io::posix_clock_id::realtime));
}

Expected Results

It should output all results.

Actual Results

It only outputs the first one

Versions and Environment

Wasmtime version or commit: main

Operating system: windows

Architecture: x86_64

Extra Info

-stdlib=libstdc++


Last updated: Jan 24 2025 at 00:11 UTC