Skip to content

Commit

Permalink
Use nix::select directly to fix MacOS support
Browse files Browse the repository at this point in the history
On MacOS, `poll` and `kqueue` are not supported for tty[1]. So let's ditch
mio completely and directly call `select` ourselves. It's not a lot of
work anyway.

This fixes MacOS support.

[1]: tokio-rs/mio#1377
  • Loading branch information
zeenix committed Mar 9, 2024
1 parent b4ea893 commit 5b019fd
Show file tree
Hide file tree
Showing 2 changed files with 27 additions and 16 deletions.
3 changes: 1 addition & 2 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -14,8 +14,7 @@ readme = "README.md"
thiserror = "1.0"

[target.'cfg(unix)'.dependencies]
mio = { version = "0.8", features = ["os-ext"] }
nix = "0.22"
nix = { version = "0.28", features = ["poll"] }

[dev-dependencies]
crossterm = "0.21"
40 changes: 26 additions & 14 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,32 +24,44 @@ pub fn query_buffer<MS: Into<u64>>(
buffer: &mut [u8],
timeout_ms: MS,
) -> Result<usize, XQError> {
use mio::{unix::SourceFd, Events, Interest, Poll, Token};
use nix::sys::{
select::{select, FdSet},
time::TimeVal,
};
use std::{
fs::File,
io::{self, Read, Write},
os::fd::AsRawFd,
os::fd::{AsFd, AsRawFd},
time::Duration,
};
let stdout = io::stdout();
let mut stdout = stdout.lock();
write!(stdout, "{}", query)?;
stdout.flush()?;
let mut stdin = File::open("/dev/tty")?;
let mut poll = Poll::new()?;
let mut events = Events::with_capacity(1024);
let stdin_raw_fd = stdin.as_raw_fd();
let mut stdin_fd = SourceFd(&stdin_raw_fd); // fancy way to pass the 0 const
poll.registry()
.register(&mut stdin_fd, Token(0), Interest::READABLE)?;
let timeout = std::time::Duration::from_millis(timeout_ms.into());
poll.poll(&mut events, Some(timeout))?;
for event in &events {
if event.token() == Token(0) {
let stdin_fd = stdin.as_fd();
let mut fd_set = FdSet::new();
fd_set.insert(stdin_fd);

let timeout_us = Duration::from_millis(timeout_ms.into())
.as_micros()
.try_into()
.unwrap();
let mut tv = TimeVal::new(0, timeout_us);
match select(
stdin_fd.as_raw_fd() + 1,
Some(&mut fd_set),
None,
None,
Some(&mut tv),
) {
Ok(n) if n == 0 => Err(XQError::Timeout),
Ok(_) => {
let bytes_written = stdin.read(buffer)?;
return Ok(bytes_written);
Ok(bytes_written)
}
Err(e) => Err(XQError::IO(e.into())),
}
Err(XQError::Timeout) // no file descriptor was ready in time
}

#[cfg(not(unix))]
Expand Down

0 comments on commit 5b019fd

Please sign in to comment.