Frenetic is an implementation of stackful coroutines. It is written in Rust and LLVM. Notably, this approach does not require any system calls or hand-crafted assembly at all.
use frenetic::{Coroutine, Generator, GeneratorState};
use core::pin::Pin;
// You'll need to create a stack before using Frenetic coroutines.
let mut stack = [0u8; 4096 * 8];
// Then, you can initialize with `Coroutine::new`.
let mut coro = Coroutine::new(&mut stack, |c| {
let c = c.r#yield(1)?; // Yield an integer value.
c.done("foo") // Return a string value.
});
// You can also interact with the yielded and returned values.
match Pin::new(&mut coro).resume() {
GeneratorState::Yielded(1) => {}
_ => panic!("unexpected return from resume"),
}
match Pin::new(&mut coro).resume() {
GeneratorState::Complete("foo") => {}
_ => panic!("unexpected return from resume"),
}
That's it!
The current API consists of a few basic primitives:
Coroutine::new
Spawns a new coroutine. Requires a stack and a function to be executed.
NOTE: The caller is responsible for properly allocating this stack. We recommend the stack includes a guard page.
Control::r#yield
Halts the current coroutine's execution and passes control back to the parent.
Control::done
Marks the current coroutine as done, and finishes.
Generator::resume
Resumes a halted coroutine.