mio describes itself as a “fast, low-level I/O library for Rust focusing on non-blocking APIs and event notification for building performance I/O apps with as little overhead as possible over the OS abstractions.”
mio drives the event queue in Tokio, which is one of the most popular and widely used asynchronous runtimes in Rust. This means that mio is driving I/O for popular frameworks such as Actix Web (https://actix.rs/), Warp (https://github.com/seanmonstar/warp), and Rocket (https://rocket.rs/).
The version of mio we’ll use as design inspiration in this example is version 0.8.8. The API has changed in the past and may change in the future, but the parts of the API we cover here have been stable since 2019, so it’s a good bet that there will not be significant changes to it in the near future.
As is the case with all cross-platform abstractions, it’s often necessary to go the route of choosing the least common denominator. Some choices will limit flexibility and efficiency on one or more platforms in the pursuit of having a unified API that works with all of them. We’ll discuss some of those choices in this chapter.
Before we go further, let’s create a blank project and give it a name. We’ll refer to it as a-epoll going forward, but you will of course need to replace that with the name you choose.
Enter the folder and type the cargo init command.
In this example, we’ll divide the project into a few modules, and we’ll split the code up into the following files:
src
|– ffi.rs
|– main.rs
|– poll.rs
Their descriptions are as follows:
- ffi.rs: This module will contain the code related to the syscalls we need to communicate with the host operating system
- main.rs: This is the example program itself
- poll.rs: This module contains the main abstraction, which is a thin layer over epoll
Next, create the four files, mentioned in the preceding list, in the src folder.
In main.rs, we need to declare the modules as well:
a-epoll/src/main.rs
mod ffi;
mod poll;
Now that we have our project set up, we can start by going through how we’ll design the API we’ll use. The main abstraction is in poll.rs, so go ahead and open that file.
Let’s start by stubbing out the structures and functions we need. It’s easier to discuss them when we have them in front of us:
a-epoll/src/poll.rs
use std::{io::{self, Result}, net::TcpStream, os::fd::AsRawFd};
use crate::ffi;
type Events = Vec<ffi::Event>;
pub struct Poll {
registry: Registry,
}
impl Poll {
pub fn new() -> Result<Self> {
todo!()
}
pub fn registry(&self) -> &Registry {
&self.registry
}
pub fn poll(&mut self, events: &mut Events, timeout: Option<i32>) -> Result<()> {
todo!()
}
}
pub struct Registry {
raw_fd: i32,
}
impl Registry {
pub fn register(&self, source: &TcpStream, token: usize, interests: i32) -> Result<()>
{
todo!()
}
}
impl Drop for Registry {
fn drop(&mut self) {
todo!()
}
}
We’ve replaced all the implementations with todo!() for now. This macro will let us compile the program even though we’ve yet to implement the function body. If our execution ever reaches todo!(), it will panic.
The first thing you’ll notice is that we’ll pull the ffi module in scope in addition to some types from the standard library.
Leave a Reply