Dividing the runtime into two distinct parts makes a lot of sense when we take a look at how Rust models asynchronous tasks. If you read the documentation for Future (https://doc.rust-lang.org/std/future/trait.Future.html) and Waker (https://doc.rust-lang.org/std/task/struct.Waker.html), you’ll see that Rust doesn’t only define a Future trait and a Waker type but also comes with important information on how they’re supposed to be used.
One example of this is that Future traits are inert, as we covered in Chapter 6. Another example is that a call to Waker::wake will guarantee at least one call to Future::poll on the corresponding task.
So, already by reading the documentation, you will see that there is at least some thought put into how runtimes should behave.
The reason for learning this pattern is that it’s almost a glove-to-hand fit for Rust’s asynchronous model.
Since many readers, including me, will not have English as a first language, I’ll explain the names here at the start since, well, they seem to be easy to misunderstand.
If the name reactor gives you associations with nuclear reactors, and you start thinking of reactors as something that powers, or drives, a runtime, drop that thought right now. A reactor is simply something that reacts to a whole set of incoming events and dispatches them one by one to a handler. It’s an event loop, and in our case, it dispatches events to an executor. Events that are handled by a reactor could be anything from a timer that expires, an interrupt if you write programs for embedded systems, or an I/O event such as a READABLE event on TcpStream.
You could have several kinds of reactors running in the same runtime.
If the name executor gives you associations to executioners (the medieval times kind) or executables, drop that thought as well. If you look up what an executor is, it’s a person, often a lawyer, who administers a person’s will. Most often, since that person is dead. Which is also the point where whatever mental model the naming suggests to you falls apart since nothing, and no one, needs to come in harm’s way for the executor to have work to do in an asynchronous runtime, but I digress.
The important point is that an executor simply decides who gets time on the CPU to progress and when they get it. The executor must also call Future::poll and advance the state machines to their next state. It’s a type of scheduler.
It can be frustrating to get the wrong idea from the start since the subject matter is already complex enough without thinking about how on earth nuclear reactors and executioners fit in the whole picture.
Since reactors will respond to events, they need some integration with the source of the event. If we continue using TcpStream as an example, something will call read or write on it, and at that point, the reactor needs to know that it should track certain events on that source.
For this reason, non-blocking I/O primitives and reactors need tight integration, and depending on how you look at it, the I/O primitives will either have to bring their own reactor or you’ll have a reactor that provides I/O primitives such as sockets, ports, and streams.
Now that we’ve covered some of the overarching design, we can start writing some code.
Runtimes tend to get complex pretty quickly, so to keep this as simple as possible, we’ll avoid any error handling in our code and use unwrap or expect for everything. We’ll also choose simplicity over cleverness and readability over efficiency to the best of our abilities.
Our first task will be to take the first example we wrote in Chapter 7 and improve it by avoiding having to actively poll it to make progress. Instead, we lean on what we learned about non-blocking I/O and epoll in the earlier chapters.