yield is a reserved word in Rust, so we can’t name our function that. If that was not the case, it would be my preferred name for it over the slightly more cryptic t_yield.
This is the return function that we call when a thread is finished. return is another reserved keyword in Rust, so we name this t_return(). Make a note that the user of our threads does not call this; we set up our stack so this is called when the task is done:
fn t_return(&mut self) {
if self.current != 0 {
self.threads[self.current].state = State::Available;
self.t_yield();
}
}
If the calling thread is the base_thread, we won’t do anything. Our runtime will call t_yield for us on the base thread. If it’s called from a spawned thread, we know it’s finished since all threads will have a guard function on top of their stack (which we’ll show further down), and the only place where this function is called is on our guard function.
We set its state to Available, letting the runtime know it’s ready to be assigned a new task, and then immediately call t_yield, which will schedule a new thread to be run.
So, finally, we get to the heart of our runtime: the t_yield function.
The first part of this function is our scheduler. We simply go through all the threads and see if any are in the Ready state, which indicates that it has a task it is ready to make progress. This could be a database call that has returned in a real-world application.
If no thread is Ready, we’re all done. This is an extremely simple scheduler using only a round-robin algorithm. A real scheduler might have a much more sophisticated way of deciding what task to run next.
If we find a thread that’s ready to be run, we change the state of the current thread from Running to Ready.
Let’s present the function before we go on to explain the last part of it:
#[inline(never)]
fn t_yield(&mut self) -> bool {
let mut pos = self.current;
while self.threads[pos].state != State::Ready {
pos += 1;
if pos == self.threads.len() {
pos = 0;
}
if pos == self.current {
return false;
}
}
if self.threads[self.current].state != State::Available {
self.threads[self.current].state = State::Ready;
}
self.threads[pos].state = State::Running;
let old_pos = self.current;
self.current = pos;
unsafe {
let old: *mut ThreadContext = &mut self.threads[old_pos].ctx;
let new: *const ThreadContext = &self.threads[pos].ctx;
asm!(“call switch”, in(“rdi”) old, in(“rsi”) new, clobber_abi(“C”));
}
self.threads.len() > 0
}
The next thing we do is to call the function switch, which will save the current context (the old context) and load the new context into the CPU. The new context is either a new task or all the information the CPU needs to resume work on an existing task.
Our switch function, which we will cover a little further down, takes two arguments and is marked as #[naked]. Naked functions are not like normal functions. They don’t accept formal arguments, for example, so we can’t simply call it in Rust as a normal function like switch(old, new).
You see, usually, when we call a function with two arguments, the compiler will place each argument in a register described by the calling convention for the platform. However, when we call a #[naked] function, we need to take care of this ourselves. Therefore, we pass in the address to our old and new ThreadContext using assembly. rdi is the register for the first argument in the System V ABI calling convention and rsi is the register used for the second argument.
The #[inline(never)] attribute prevents the compiler from simply substituting a call to our function with a copy of the function content wherever it’s called (this is what inlining means). This is almost never a problem on debug builds, but in this case, our program will fail if the compiler inlines this function in a release build. The issue manifests itself by the runtime exiting before all the tasks are finished. Since we store Runtime as a static usize that we then cast as a *mut pointer (which is almost guaranteed to cause UB), it’s most likely caused by the compiler making the wrong assumptions when this function is inlined and called by casting and dereferencing RUNTIME in one of the helper methods that will be outlined. Just make a note that this is probably avoidable if we change our design; it’s not something worth dwelling on for too long in this specific case.