-
Notifications
You must be signed in to change notification settings - Fork 373
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Fix libc::read shim: make it write to a buffer correct amount of bytes. Add tests for new behavior #3720
Conversation
Thanks! It would be nice to have some tests that check the libc APIs (open/read/write/close) directly, basically a libc version of |
@RalfJung , okay. I will remove draft status, when I'm done |
I am not sure if that generates a notification... please write @rustbot author |
@RalfJung , okay |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah, that's what this is about. :) Good catch, and thanks for the PR!
src/shims/unix/fd.rs
Outdated
@@ -394,7 +394,7 @@ pub trait EvalContextExt<'tcx>: crate::MiriInterpCxExt<'tcx> { | |||
match result { | |||
Ok(read_bytes) => { | |||
// If reading to `bytes` did not fail, we write those bytes to the buffer. | |||
this.write_bytes_ptr(buf, bytes)?; | |||
this.write_bytes_ptr(buf, bytes[..usize::try_from(read_bytes).unwrap()].to_vec())?; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this.write_bytes_ptr(buf, bytes[..usize::try_from(read_bytes).unwrap()].to_vec())?; | |
// Crucially, if fewer than `bytes.len()` bytes were read, only write | |
// that much into the output buffer! | |
this.write_bytes_ptr(buf, bytes[..usize::try_from(read_bytes).unwrap()].to_vec())?; |
Also, to_vec
should not be needed. .iter().copied()
should work.
@@ -0,0 +1,25 @@ | |||
// We test that if we requested to read 4 bytes, but actually read 3 bytes, then 3 bytes (not 4) will be initialized | |||
//@ignore-target-windows: we have no file deletion on Windows |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
//@ignore-target-windows: we have no file deletion on Windows | |
//@ignore-target-windows: no file system support on Windows |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What is the point of this test? I don't think this is a likely enough error to justify spending CI time on.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please move these into libc-fs.rs
@rustbot author |
tests/pass-dep/libc/libc-fs.rs
Outdated
remove_file(&path).unwrap(); | ||
} | ||
{ | ||
// We test that if we requested to read 4 bytes, but actually read 3 bytes, then 3 bytes (not 4) will be overwritten, and remaining byte will be left as is |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
// We test that if we requested to read 4 bytes, but actually read 3 bytes, then 3 bytes (not 4) will be overwritten, and remaining byte will be left as is | |
// We test that if we requested to read 4 bytes, but actually read 3 bytes, then | |
// 3 bytes (not 4) will be overwritten, and the remaining byte will be left as-is. |
tests/pass-dep/libc/libc-fs.rs
Outdated
fn test_read_and_uninit() { | ||
use std::mem::MaybeUninit; | ||
{ | ||
// We test that libc::read initializes its buffer |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
// We test that libc::read initializes its buffer | |
// We test that libc::read initializes its buffer. |
@@ -0,0 +1,26 @@ | |||
// We test that if we requested to read 4 bytes, but actually read 3 bytes, then 3 bytes (not 4) will be initialized |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
// We test that if we requested to read 4 bytes, but actually read 3 bytes, then 3 bytes (not 4) will be initialized | |
//! We test that if we requested to read 4 bytes, but actually read 3 bytes, | |
//! then 3 bytes (not 4) will be initialized. |
Just some final nits, please make these comments proper sentences and line-wrap them. Usually I'd just apply these patches myself, but you seem to have disabled the option in the PR to let maintainers push to your branch. I can only recommend to enable that option, it saves you some work that maintainers can do for you. :) @rustbot author |
…s. Add tests for the new behavior. libc::read shim had a bug: if underlying real call libc::read(fd, buf, N) returns M, then libc::read shim writes N bytes to buf instead of M. Remaining N - M bytes are filled with zeros. This commit fixes this bug and adds tests for new behavior
Thanks, looks good! |
☀️ Test successful - checks-actions |
libc::read shim had a bug: if underlying real call libc::read(fd, buf, N) returns M, then
libc::read shim writes N bytes to buf instead of M. Remaining N - M bytes are filled with zeros.
This commit fixes this bug and adds tests for new behavior