In this chapter, we take a deep dive into a very popular way of handling concurrency. There is no better way of getting a fundamental understanding of the subject than doing it yourself. Fortunately, even though the topic is a little complex, we only need around 200 lines of code to get a fully working example in the end.
What makes the topic complex is that it requires quite a bit of fundamental understanding of how CPUs, operating systems, and assembly work. This complexity is also what makes this topic so interesting. If you explore and work through this example in detail, you will be rewarded with an eye-opening understanding of topics you might only have heard about or only have a rudimentary understanding of. You will also get the chance to get to know a few aspects of the Rust language that you haven’t seen before, expanding your knowledge of both Rust and programming in general.
We start off by introducing a little background knowledge that we need before we start writing code. Once we have that in place, we’ll start with some small examples that will allow us to show and discuss the most technical and difficult parts of our example in detail so we can introduce the topics gradually. Lastly, we’ll build on the knowledge we’ve gained and create our main example, which is a working example of fibers implemented in Rust.
As a bonus, you’ll get two expanded versions of the example in the repository to inspire you to go on and change, adapt, and build upon what we’ve created to make it your own.
I’ll list the main topics here so you can refer to them later on:
- How to use the repository alongside the book
- Background information
- An example we can build upon
- The stack
- Implementing our own fibers
- Final thoughts
Note
In this chapter, we’ll use the terms “fibers” and “green threads” to refer to this exact implementation of stackful coroutines. The term “threads” in this chapter, which is used in the code we write, will refer to the green threads/fibers we implement in our example and not OS threads.
Technical requirements
To run the examples, you will need a computer running on a CPU using the x86-64 instruction set. Most popular desktop, server, and laptop CPUs out there today use this instruction set, as do most modern CPUs from Intel and AMD (which are most CPU models from these manufacturers produced in the last 10–15 years).
One caveat is that the modern M-series Macs use the ARM ISA (instruction set), which won’t be compatible with the examples we write here. However, older Intel-based Macs do, so you should be able to use a Mac to follow along if you don’t have the latest version.
If you don’t have a computer using this instruction set available, you have a few options to install Rust and run the examples:
- Mac users on M-series chips can use Rosetta (which ships with newer MacOS versions) and get the examples working with just four simple steps. You’ll find the instructions in the repository under ch05/How-to-MacOS-M.md.
- https://mac.getutm.app/ (some even have a free layer) a remote server running Linux on x86-64. I have experience with Linode’s offering (/), but there are many more options out there.
To follow along with the examples in the book, you also need a Unix-based operating system. The example code will work natively on any Linux and BSD operating system (such as Ubuntu or macOS) as long as it’s running on an x86-64 CPU.
If you’re on Windows, there is a version of the example in the repository that works natively with Windows too, but to follow along with the book, my clear recommendation is to set up Windows Subsystem for Linux (WSL) (https://learn.microsoft.com/en-us/windows/wsl/install), install Rust, and follow along using Rust on WSL.
I personally use VS Code as my editor, as it makes it very easy to switch between using a Linux version on WSL and Windows—simply press Ctrl + Shift + P and search for the Reopen folder in WSL.
Leave a Reply