Before we round off this chapter, I want to point out that it should now be clear to us why coroutines aren’t really pre-emptable. If you remember back in Chapter 2, we said that a stackful coroutine (such as our fibers/green threads example) could be pre-empted and its execution could be paused at any point. That’s because they have a stack, and pausing a task is as simple as storing the current execution state to the stack and jumping to another task.
That’s not possible here. The only places we can stop and resume execution are at the pre-defined suspension points that we manually tagged with wait.
In theory, if you have a tightly integrated system where you control the compiler, the coroutine definition, the scheduler, and the I/O primitives, you could add additional states to the state machine and create additional points where the task could be suspended/resumed. These suspension points could be opaque to the user and treated differently than normal wait/suspension points.
For example, every time you encounter a normal function call, you could add a suspension point (a new state to our state machine) where you check in with the scheduler if the current task has used up its time budget or something like that. If it has, you could schedule another task to run and resume the task at a later point even though this didn’t happen in a cooperative manner.
However, even though this would be invisible to the user, it’s not the same as being able to stop/resume execution from any point in your code. It would also go against the usually implied cooperative nature of coroutines.
Summary
Good job! In this chapter, we introduced quite a bit of code and set up an example that we’ll continue using in the following chapters.
So far, we’ve focused on futures and async/await to model and create tasks that can be paused and resumed at specific points. We know this is a prerequisite to having tasks that are in progress at the same time. We did this by introducing our own simplified Future trait and our own coroutine/wait syntax that’s way more limited than Rust’s futures and async/await syntax, but it’s easier to understand and get a mental idea of how this works in contrast to fibers/green threads (at least I hope so).
We have also discussed the difference between eager and lazy coroutines and how they impact how you achieve concurrency. We took inspiration from Tokio’s join_all function and implemented our own version of it.
In this chapter, we simply created tasks that could be paused and resumed. There are no event loops, scheduling, or anything like that yet, but don’t worry. They’re exactly what we’ll go through in the next chapter. The good news is that getting a clear idea of coroutines, like we did in this chapter, is one of the most difficult things to do.