We would get a value of 0 if a timeout elapses before an event has happened.
The last thing we do is to make an unsafe call to events.set_len(res as usize). This function is unsafe since we could potentially set the length so that we would access memory that’s not been initialized yet in safe Rust. We know from the guarantee the operating system gives us that the number of events it returns is pointing to valid data in our Vec, so this is safe in our case.
Next up is our Registry struct. We will only implement one method, called register, and lastly, we’ll implement the Drop trait for it, closing the epoll instance:
ch04/a-epoll/src/poll.rs
impl Registry {
pub fn register(&self, source: &TcpStream, token: usize, interests: i32) -> Result<()> {
let mut event = ffi::Event {
events: interests as u32,
epoll_data: token,
};
let op = ffi::EPOLL_CTL_ADD;
let res = unsafe {
ffi::epoll_ctl(self.raw_fd, op, source.as_raw_fd(), &mut event)
};
if res < 0 {
return Err(io::Error::last_os_error());
}
Ok(())
}
}
The register function takes a &TcpStream as a source, a token of type usize, and a bitmask named interests, which is of type i32.
Note
This is where mio does things differently. The source argument is specific to each platform. Instead of having the implementation of register on Registry, it’s handled in a platform-specific way in the source argument it receives.
The first thing we do is to create an ffi::Event object. The events field is simply set to the bitmask we received and named interests, and epoll_data is set to the value we passed in the token argument.
The operation we want to perform on the epoll queue is adding interest in events on a new file descriptor. Therefore, we set the op argument to the ffi::EPOLL_CTL_ADD constant value.
Next up is the call to ffi::epoll_ctl. We pass in the file descriptor to the epoll instance first, then we pass in the op argument to indicate what kind of operation we want to perform. The last two arguments are the file descriptor we want the queue to track and the Event object we created to indicate what kind of events we’re interested in getting notifications for.
The last part of the function body is simply the error handling, which should be familiar by now.
The last part of poll.rs is the Drop implementation for Registry:
ch04/a-epoll/src/poll.rs
impl Drop for Registry {
fn drop(&mut self) {
let res = unsafe { ffi::close(self.raw_fd) };
if res < 0 {
let err = io::Error::last_os_error();
eprintln!(“ERROR: {err:?}”);
}
}
}
The Drop implementation simply calls ffi::close on the epoll file descriptor. Adding a panic to drop is rarely a good idea since drop can be called within a panic already, which will cause the process to simply abort. mio logs errors if they occur in its Drop implementation but doesn’t handle them in any other way. For our simple example, we’ll just print the error so we can see if anything goes wrong since we don’t implement any kind of logging here.
The last part is the code for running our example, and that leads us to main.rs.
Leave a Reply