Skip to content

Commit 50196d3

Browse files
committed
Fix io::default_read_to_end uses of read_buf
1 parent 976ef15 commit 50196d3

File tree

1 file changed

+11
-7
lines changed

1 file changed

+11
-7
lines changed

library/std/src/io/mod.rs

+11-7
Original file line numberDiff line numberDiff line change
@@ -477,13 +477,14 @@ pub(crate) fn default_read_to_end<R: Read + ?Sized>(
477477
}
478478

479479
let mut cursor = read_buf.unfilled();
480-
loop {
480+
let result = loop {
481481
match r.read_buf(cursor.reborrow()) {
482-
Ok(()) => break,
483482
Err(e) if e.is_interrupted() => continue,
484-
Err(e) => return Err(e),
483+
// Do not stop now in case of error: we might have received both data
484+
// and an error
485+
res => break res,
485486
}
486-
}
487+
};
487488

488489
let unfilled_but_initialized = cursor.init_ref().len();
489490
let bytes_read = cursor.written();
@@ -493,15 +494,18 @@ pub(crate) fn default_read_to_end<R: Read + ?Sized>(
493494
return Ok(buf.len() - start_len);
494495
}
495496

496-
// store how much was initialized but not filled
497-
initialized = unfilled_but_initialized;
498-
499497
// SAFETY: BorrowedBuf's invariants mean this much memory is initialized.
500498
unsafe {
501499
let new_len = bytes_read + buf.len();
502500
buf.set_len(new_len);
503501
}
504502

503+
// Now that all data is pushed to the vector, we can fail without data loss
504+
result?;
505+
506+
// store how much was initialized but not filled
507+
initialized = unfilled_but_initialized;
508+
505509
// Use heuristics to determine the max read size if no initial size hint was provided
506510
if size_hint.is_none() {
507511
// The reader is returning short reads but it doesn't call ensure_init().

0 commit comments

Comments
 (0)