Skip to content
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

add Interest::PRIORITY and ready and ready_mut functions to AsyncFd #5566

Merged
merged 25 commits into from
Jun 6, 2023
Merged
Show file tree
Hide file tree
Changes from 24 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
b8f1d0d
stream: add `StreamNotifyClose` (#4851)
aviramha Apr 16, 2023
f0ddd93
add a test awaiting priority
folkertdev Mar 21, 2023
3fcc4ef
add target cfg for docs
folkertdev Mar 21, 2023
9780946
on CI, AsRawFd is not yet in std::os::fs
folkertdev Mar 21, 2023
8d33a34
more conditional compilation
folkertdev Mar 21, 2023
5fdb52f
actually read out-of-band data in test
folkertdev Mar 22, 2023
024f16e
rename priority -> priority_ready
folkertdev Apr 5, 2023
fd24209
keep original interest for PollEvented::new
folkertdev Apr 5, 2023
735dd9f
add READ_CLOSED to the priority readiness
folkertdev Apr 16, 2023
3ea6402
add special waker for priority events
folkertdev Apr 16, 2023
69c01cf
keep old behavior for AsyncFd::new
folkertdev May 19, 2023
07fea45
add `AsyncFd::ready` instead of `AsyncFd::priority_ready`
folkertdev May 19, 2023
4a254e9
remove a test that does not work
folkertdev May 19, 2023
720d8fc
fix doc comment
folkertdev May 19, 2023
b0d1131
remove now-unused priority waker
folkertdev May 19, 2023
2700d7f
allow fine-grained clearing of readiness on AsyncFd
folkertdev May 23, 2023
462c926
document clearing of readiness on the ready functions
folkertdev May 24, 2023
920e1fb
rename is_priority_ready -> is_priority
folkertdev May 24, 2023
c3de66e
de-duplicate code in the Debug instance
folkertdev May 24, 2023
034bd78
clear the event if all readiness flags are cleared
folkertdev May 24, 2023
4633bca
add test that readiness is actually cleared
folkertdev May 24, 2023
395fe37
rename clear_ready_exact -> clear_ready_matching
folkertdev May 24, 2023
81cb2bb
replicate the tcp combined interest example
folkertdev May 29, 2023
6837709
Merge branch 'master' into priority-interest
carllerche Jun 6, 2023
ab05c9f
Update tokio/Cargo.toml
carllerche Jun 6, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion tokio/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -103,7 +103,7 @@ pin-project-lite = "0.2.0"

# Everything else is optional...
bytes = { version = "1.0.0", optional = true }
mio = { version = "0.8.4", optional = true, default-features = false }
mio = { version = "0.8.6", optional = true, default-features = false }
carllerche marked this conversation as resolved.
Show resolved Hide resolved
num_cpus = { version = "1.8.0", optional = true }
parking_lot = { version = "0.12.0", optional = true }

Expand Down
436 changes: 416 additions & 20 deletions tokio/src/io/async_fd.rs

Large diffs are not rendered by default.

26 changes: 26 additions & 0 deletions tokio/src/io/interest.rs
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,11 @@ impl Interest {
/// Writable interest includes write-closed events.
pub const WRITABLE: Interest = Interest(mio::Interest::WRITABLE);

/// Returns a `Interest` set representing priority completion interests.
#[cfg(any(target_os = "linux", target_os = "android"))]
#[cfg_attr(docsrs, doc(cfg(any(target_os = "linux", target_os = "android"))))]
pub const PRIORITY: Interest = Interest(mio::Interest::PRIORITY);
satakuma marked this conversation as resolved.
Show resolved Hide resolved

/// Returns true if the value includes readable interest.
///
/// # Examples
Expand Down Expand Up @@ -78,6 +83,25 @@ impl Interest {
self.0.is_writable()
}

/// Returns true if the value includes priority interest.
///
/// # Examples
///
/// ```
/// use tokio::io::Interest;
///
/// assert!(!Interest::READABLE.is_priority());
/// assert!(Interest::PRIORITY.is_priority());
///
/// let both = Interest::READABLE | Interest::PRIORITY;
/// assert!(both.is_priority());
/// ```
#[cfg(any(target_os = "linux", target_os = "android"))]
#[cfg_attr(docsrs, doc(cfg(any(target_os = "linux", target_os = "android"))))]
pub const fn is_priority(self) -> bool {
self.0.is_priority()
}

/// Add together two `Interest` values.
///
/// This function works from a `const` context.
Expand All @@ -104,6 +128,8 @@ impl Interest {
match self {
Interest::READABLE => Ready::READABLE | Ready::READ_CLOSED,
Interest::WRITABLE => Ready::WRITABLE | Ready::WRITE_CLOSED,
#[cfg(any(target_os = "linux", target_os = "android"))]
Interest::PRIORITY => Ready::PRIORITY | Ready::READ_CLOSED,
_ => Ready::EMPTY,
}
}
Expand Down
3 changes: 3 additions & 0 deletions tokio/src/io/poll_evented.rs
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,9 @@ cfg_io_driver! {
impl<E: Source> PollEvented<E> {
/// Creates a new `PollEvented` associated with the default reactor.
///
/// The returned `PollEvented` has readable and writable interests. For more control, use
/// [`Self::new_with_interest`].
///
/// # Panics
///
/// This function panics if thread-local runtime is not set.
Expand Down
55 changes: 51 additions & 4 deletions tokio/src/io/ready.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ const READABLE: usize = 0b0_01;
const WRITABLE: usize = 0b0_10;
const READ_CLOSED: usize = 0b0_0100;
const WRITE_CLOSED: usize = 0b0_1000;
#[cfg(any(target_os = "linux", target_os = "android"))]
const PRIORITY: usize = 0b1_0000;

/// Describes the readiness state of an I/O resources.
///
Expand All @@ -31,7 +33,17 @@ impl Ready {
/// Returns a `Ready` representing write closed readiness.
pub const WRITE_CLOSED: Ready = Ready(WRITE_CLOSED);

/// Returns a `Ready` representing priority readiness.
#[cfg(any(target_os = "linux", target_os = "android"))]
#[cfg_attr(docsrs, doc(cfg(any(target_os = "linux", target_os = "android"))))]
pub const PRIORITY: Ready = Ready(PRIORITY);

/// Returns a `Ready` representing readiness for all operations.
#[cfg(any(target_os = "linux", target_os = "android"))]
pub const ALL: Ready = Ready(READABLE | WRITABLE | READ_CLOSED | WRITE_CLOSED | PRIORITY);

/// Returns a `Ready` representing readiness for all operations.
#[cfg(not(any(target_os = "linux", target_os = "android")))]
pub const ALL: Ready = Ready(READABLE | WRITABLE | READ_CLOSED | WRITE_CLOSED);

// Must remain crate-private to avoid adding a public dependency on Mio.
Expand Down Expand Up @@ -65,6 +77,13 @@ impl Ready {
ready |= Ready::WRITE_CLOSED;
}

#[cfg(any(target_os = "linux", target_os = "android"))]
{
if event.is_priority() {
ready |= Ready::PRIORITY;
}
}

ready
}

Expand Down Expand Up @@ -144,6 +163,23 @@ impl Ready {
self.contains(Ready::WRITE_CLOSED)
}

/// Returns `true` if the value includes priority `readiness`.
///
/// # Examples
///
/// ```
/// use tokio::io::Ready;
///
/// assert!(!Ready::EMPTY.is_priority());
/// assert!(!Ready::WRITABLE.is_priority());
/// assert!(Ready::PRIORITY.is_priority());
/// ```
#[cfg(any(target_os = "linux", target_os = "android"))]
#[cfg_attr(docsrs, doc(cfg(any(target_os = "linux", target_os = "android"))))]
pub fn is_priority(self) -> bool {
self.contains(Ready::PRIORITY)
}

/// Returns true if `self` is a superset of `other`.
///
/// `other` may represent more than one readiness operations, in which case
Expand Down Expand Up @@ -191,6 +227,12 @@ cfg_io_readiness! {
ready |= Ready::WRITE_CLOSED;
}

#[cfg(any(target_os = "linux", target_os = "android"))]
if interest.is_priority() {
ready |= Ready::PRIORITY;
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

note for the future: if we add the Ready::READ_CLOSED to mask above, we should also add it here

ready |= Ready::READ_CLOSED;
}

ready
}

Expand Down Expand Up @@ -240,11 +282,16 @@ impl ops::Sub<Ready> for Ready {

impl fmt::Debug for Ready {
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt.debug_struct("Ready")
.field("is_readable", &self.is_readable())
let mut fmt = fmt.debug_struct("Ready");

fmt.field("is_readable", &self.is_readable())
.field("is_writable", &self.is_writable())
.field("is_read_closed", &self.is_read_closed())
.field("is_write_closed", &self.is_write_closed())
.finish()
.field("is_write_closed", &self.is_write_closed());

#[cfg(any(target_os = "linux", target_os = "android"))]
folkertdev marked this conversation as resolved.
Show resolved Hide resolved
fmt.field("is_priority", &self.is_priority());

fmt.finish()
}
}
12 changes: 12 additions & 0 deletions tokio/src/runtime/io/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,18 @@ pub(crate) struct ReadyEvent {
is_shutdown: bool,
}

cfg_net_unix!(
impl ReadyEvent {
pub(crate) fn with_ready(&self, ready: Ready) -> Self {
Self {
ready,
tick: self.tick,
is_shutdown: self.is_shutdown,
}
}
}
);

struct IoDispatcher {
allocator: slab::Allocator<ScheduledIo>,
is_shutdown: bool,
Expand Down
86 changes: 86 additions & 0 deletions tokio/tests/io_async_fd.rs
Original file line number Diff line number Diff line change
Expand Up @@ -599,3 +599,89 @@ fn driver_shutdown_wakes_poll_race() {
assert_err!(futures::executor::block_on(poll_writable(&afd_a)));
}
}

#[tokio::test]
#[cfg(any(target_os = "linux", target_os = "android"))]
async fn priority_event_on_oob_data() {
use std::net::SocketAddr;

use tokio::io::Interest;

let addr: SocketAddr = "127.0.0.1:0".parse().unwrap();

let listener = std::net::TcpListener::bind(addr).unwrap();
let addr = listener.local_addr().unwrap();

let client = std::net::TcpStream::connect(addr).unwrap();
let client = AsyncFd::with_interest(client, Interest::PRIORITY).unwrap();

let (stream, _) = listener.accept().unwrap();

// Sending out of band data should trigger priority event.
send_oob_data(&stream, b"hello").unwrap();

let _ = client.ready(Interest::PRIORITY).await.unwrap();
}

#[cfg(any(target_os = "linux", target_os = "android"))]
fn send_oob_data<S: AsRawFd>(stream: &S, data: &[u8]) -> io::Result<usize> {
unsafe {
let res = libc::send(
stream.as_raw_fd(),
data.as_ptr().cast(),
data.len(),
libc::MSG_OOB,
);
if res == -1 {
Err(io::Error::last_os_error())
} else {
Ok(res as usize)
}
}
}

#[tokio::test]
async fn clear_ready_matching_clears_ready() {
use tokio::io::{Interest, Ready};

let (a, mut b) = socketpair();

let afd_a = AsyncFd::new(a).unwrap();
b.write_all(b"0").unwrap();

let mut guard = afd_a
.ready(Interest::READABLE | Interest::WRITABLE)
.await
.unwrap();

assert_eq!(guard.ready(), Ready::READABLE | Ready::WRITABLE);

guard.clear_ready_matching(Ready::READABLE);
assert_eq!(guard.ready(), Ready::WRITABLE);

guard.clear_ready_matching(Ready::WRITABLE);
assert_eq!(guard.ready(), Ready::EMPTY);
}

#[tokio::test]
async fn clear_ready_matching_clears_ready_mut() {
use tokio::io::{Interest, Ready};

let (a, mut b) = socketpair();

let mut afd_a = AsyncFd::new(a).unwrap();
b.write_all(b"0").unwrap();

let mut guard = afd_a
.ready_mut(Interest::READABLE | Interest::WRITABLE)
.await
.unwrap();

assert_eq!(guard.ready(), Ready::READABLE | Ready::WRITABLE);

guard.clear_ready_matching(Ready::READABLE);
assert_eq!(guard.ready(), Ready::WRITABLE);

guard.clear_ready_matching(Ready::WRITABLE);
assert_eq!(guard.ready(), Ready::EMPTY);
}