Skip to content

Commit 15d69c9

Browse files
committed
Pre-allocate buffers in File::open_buffered and create_buffered
1 parent eb07a61 commit 15d69c9

File tree

5 files changed

+38
-3
lines changed

5 files changed

+38
-3
lines changed

std/src/fs.rs

+8-2
Original file line numberDiff line numberDiff line change
@@ -407,7 +407,10 @@ impl File {
407407
/// ```
408408
#[unstable(feature = "file_buffered", issue = "none")]
409409
pub fn open_buffered<P: AsRef<Path>>(path: P) -> io::Result<io::BufReader<File>> {
410-
File::open(path).map(io::BufReader::new)
410+
// Allocate the buffer *first* so we don't affect the filesystem otherwise.
411+
let buffer = io::BufReader::<Self>::try_new_buffer()?;
412+
let file = File::open(path)?;
413+
Ok(io::BufReader::with_buffer(file, buffer))
411414
}
412415

413416
/// Opens a file in write-only mode.
@@ -472,7 +475,10 @@ impl File {
472475
/// ```
473476
#[unstable(feature = "file_buffered", issue = "none")]
474477
pub fn create_buffered<P: AsRef<Path>>(path: P) -> io::Result<io::BufWriter<File>> {
475-
File::create(path).map(io::BufWriter::new)
478+
// Allocate the buffer *first* so we don't affect the filesystem otherwise.
479+
let buffer = io::BufWriter::<Self>::try_new_buffer()?;
480+
let file = File::create(path)?;
481+
Ok(io::BufWriter::with_buffer(file, buffer))
476482
}
477483

478484
/// Creates a new file in read-write mode; error if the file exists.

std/src/io/buffered/bufreader.rs

+8
Original file line numberDiff line numberDiff line change
@@ -74,6 +74,14 @@ impl<R: Read> BufReader<R> {
7474
BufReader::with_capacity(DEFAULT_BUF_SIZE, inner)
7575
}
7676

77+
pub(crate) fn try_new_buffer() -> io::Result<Buffer> {
78+
Buffer::try_with_capacity(DEFAULT_BUF_SIZE)
79+
}
80+
81+
pub(crate) fn with_buffer(inner: R, buf: Buffer) -> Self {
82+
Self { inner, buf }
83+
}
84+
7785
/// Creates a new `BufReader<R>` with the specified buffer capacity.
7886
///
7987
/// # Examples

std/src/io/buffered/bufreader/buffer.rs

+11-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@
1010
//! without encountering any runtime bounds checks.
1111
1212
use crate::cmp;
13-
use crate::io::{self, BorrowedBuf, Read};
13+
use crate::io::{self, BorrowedBuf, ErrorKind, Read};
1414
use crate::mem::MaybeUninit;
1515

1616
pub struct Buffer {
@@ -36,6 +36,16 @@ impl Buffer {
3636
Self { buf, pos: 0, filled: 0, initialized: 0 }
3737
}
3838

39+
#[inline]
40+
pub fn try_with_capacity(capacity: usize) -> io::Result<Self> {
41+
match Box::try_new_uninit_slice(capacity) {
42+
Ok(buf) => Ok(Self { buf, pos: 0, filled: 0, initialized: 0 }),
43+
Err(_) => {
44+
Err(io::const_io_error!(ErrorKind::OutOfMemory, "failed to allocate read buffer"))
45+
}
46+
}
47+
}
48+
3949
#[inline]
4050
pub fn buffer(&self) -> &[u8] {
4151
// SAFETY: self.pos and self.cap are valid, and self.cap => self.pos, and

std/src/io/buffered/bufwriter.rs

+10
Original file line numberDiff line numberDiff line change
@@ -94,6 +94,16 @@ impl<W: Write> BufWriter<W> {
9494
BufWriter::with_capacity(DEFAULT_BUF_SIZE, inner)
9595
}
9696

97+
pub(crate) fn try_new_buffer() -> io::Result<Vec<u8>> {
98+
Vec::try_with_capacity(DEFAULT_BUF_SIZE).map_err(|_| {
99+
io::const_io_error!(ErrorKind::OutOfMemory, "failed to allocate write buffer")
100+
})
101+
}
102+
103+
pub(crate) fn with_buffer(inner: W, buf: Vec<u8>) -> Self {
104+
Self { inner, buf, panicked: false }
105+
}
106+
97107
/// Creates a new `BufWriter<W>` with at least the specified buffer capacity.
98108
///
99109
/// # Examples

std/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -374,6 +374,7 @@
374374
#![feature(slice_concat_trait)]
375375
#![feature(thin_box)]
376376
#![feature(try_reserve_kind)]
377+
#![feature(try_with_capacity)]
377378
#![feature(vec_into_raw_parts)]
378379
// tidy-alphabetical-end
379380
//

0 commit comments

Comments
 (0)