mirror of
https://github.com/fish-shell/fish-shell.git
synced 2025-01-21 18:40:44 +08:00
Add native read_retry() and write_retry() methods
These are equivalent to read_loop() and write_loop() but operate on native Rust types without libc ffi.
This commit is contained in:
parent
bdde2b2b35
commit
4e50ae34da
|
@ -21,6 +21,7 @@ use libc::{EIO, O_WRONLY, SIGTTOU, SIG_IGN, STDERR_FILENO, STDIN_FILENO, STDOUT_
|
|||
use once_cell::sync::OnceCell;
|
||||
use std::env;
|
||||
use std::ffi::{CStr, CString, OsStr, OsString};
|
||||
use std::io::{Read, Write};
|
||||
use std::mem;
|
||||
use std::ops::{Deref, DerefMut};
|
||||
use std::os::unix::prelude::*;
|
||||
|
@ -1515,6 +1516,49 @@ pub fn read_loop<Fd: AsRawFd>(fd: &Fd, buf: &mut [u8]) -> std::io::Result<usize>
|
|||
}
|
||||
}
|
||||
|
||||
/// Provides write methods for types implementing [`Write`] that continue on
|
||||
/// EINTR or EAGAIN. Like [`Write::write_all`] but also handles EAGAIN.
|
||||
trait LoopedWrite {
|
||||
#[inline(always)]
|
||||
fn write_retry(&mut self, buf: &[u8]) -> std::io::Result<usize>
|
||||
where
|
||||
Self: Write,
|
||||
{
|
||||
let mut written = 0;
|
||||
while written != buf.len() {
|
||||
match self.write(&buf[written..]) {
|
||||
Ok(bytes) => written += bytes,
|
||||
Err(e) if e.kind() == std::io::ErrorKind::Interrupted => continue,
|
||||
Err(e) if e.kind() == std::io::ErrorKind::WouldBlock => continue,
|
||||
Err(e) => return Err(e),
|
||||
}
|
||||
}
|
||||
Ok(written)
|
||||
}
|
||||
}
|
||||
|
||||
/// Provides read methods for types implementing [`Read`] that continue on
|
||||
/// EINTR or EAGAIN.
|
||||
trait LoopedRead {
|
||||
#[inline(always)]
|
||||
fn read_retry(&mut self, buf: &mut [u8]) -> std::io::Result<usize>
|
||||
where
|
||||
Self: Read,
|
||||
{
|
||||
loop {
|
||||
match self.read(buf) {
|
||||
Ok(read) => return Ok(read),
|
||||
Err(e) if e.kind() == std::io::ErrorKind::Interrupted => continue,
|
||||
Err(e) if e.kind() == std::io::ErrorKind::WouldBlock => continue,
|
||||
Err(e) => return Err(e),
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<W: Write> LoopedWrite for W {}
|
||||
impl<R: Read> LoopedRead for R {}
|
||||
|
||||
/// Write the given paragraph of output, redoing linebreaks to fit \p termsize.
|
||||
pub fn reformat_for_screen(msg: &wstr, termsize: &Termsize) -> WString {
|
||||
let mut buff = WString::new();
|
||||
|
|
Loading…
Reference in New Issue
Block a user