Hi! I am trying to implement a custom compiler based on Cranelift for a custom language, which will support both AOT and JIT modes, and I was wondering if I could get some help. Here's some of the code that's giving me some trouble:
//! src/codegen/backend/unify.rs
pub trait BackendInternal<'a> {
// ...
fn builder(&mut self) -> Rc<RefCell<FunctionBuilder<'a>>>;
fn new_builder(&mut self) -> Rc<RefCell<FunctionBuilder<'a>>>;
}
impl<'a> BackendInternal<'a> for AotGenerator<'a> {
// ...
fn builder(&mut self) -> Rc<RefCell<FunctionBuilder<'a>>> {
self.builder.as_ref().unwrap().clone()
}
fn new_builder(&mut self) -> Rc<RefCell<FunctionBuilder<'a>>> {
self.internal_new_builder();
self.builder.as_ref().unwrap().clone()
}
}
impl<'a> BackendInternal<'a> for JitGenerator<'a> {
// ...
fn builder(&mut self) -> Rc<RefCell<FunctionBuilder<'a>>> {
self.builder.as_ref().unwrap().clone()
}
fn new_builder(&mut self) -> Rc<RefCell<FunctionBuilder<'a>>> {
self.internal_new_builder();
self.builder.as_ref().unwrap().clone()
}
}
//! src/codegen/aot.rs
impl<'a> AotGenerator<'a> {
// ...
pub(crate) fn internal_new_builder(&mut self) {
let b = FunctionBuilder::new(&mut self.ctx.func, &mut self.builder_ctx);
let c = RefCell::new(b);
self.builder = Some(Rc::new(c));
}
}
Here's the full code: https://github.com/RedstoneWizard08/QuickScript
The issue is that in the internal_new_builder
function, constructing the RefCell
throws the error "lifetime may not live long enough"
. Is there a better way to do this? Please help me. Thanks!
I would like to note that I have tried many different methods, including using &mut
references, no Rc
s, and even without the RefCell
s entirely, and even using Arc
. Nothing seems to work.
This is more of a Rust-memory-idioms issue than a Cranelift issue specifically, but the main question I have is: what is the intended lifetime relationship?
FunctionBuilder
has a lifetime 'a
that denotes the underlying Function
's lifetime; it holds a borrow of this function body, and doesn't own it. That means there has to be some scope on the stack somewhere that owns a Function
(or owns something that owns it, transitively) and the FunctionBuilder
must have a lifetime within that.
The issue here is that AotGenerator
is trying to own the Function
(inside a FunctionBuilderContext
) and the thing that borrows it. That pattern is called a "self-referential struct" and is possible in Rust, but not without tricks (there are crates for this that use unsafe code internally).
I'd recommend unwinding the design a bit, avoiding use of Rc
/RefCell
(these usually indicate that one is trying to work around lifetimes and make them all dynamic, which doesn't mix well with APIs, like Cranelift's, that use explicit lifetimes and borrows). In general, patterns from GC'd languages like Java/Python/..., or from languages with free-and-dangerous pointers like C/C++, are often a little hard to adopt because they don't fit the strict lifetime nesting regime of safe Rust. A simple workaround would be for you to create the FunctionBuilder
in a scope that starts code generation and then pass it as a borrow through params wherever you need it. As a side-benefit, this tends to push toward more functional-style code: it avoids the use of implicit passing through struct members (which at least in my experience, makes more of the plumbing explicit and less error-prone).
Okay, I have tried to do this in the past (coming from a Java background as my first language xD), but unfortunately the structure of my code throughout leans towards using structs and traits, and I don't even want the lifetimes. Unfortunately, not being able to ".clone() and ignore" doesn't work here as the shared state is important, and I don't know what to do.
One of the main issues I'm facing is that the way I have organized this to reduce code duplication and allow for expandability in the future is that I use different functions in trait impls to do everything, and they rely on this .builder()
method of BackendInternal
to use the FunctionBuilder
context.
I'd also like to mention that I am trying to avoid passing the FunctionBuilder
object between the different impls as much as possible. I like having it kept in one place and being accessed as needed.
OK, it's hard to say much more without seeing your whole codebase; but again I'd recommend reconsidering the overall object-ownership design. I've found that Rust pushes me strongly away from certain designs, but often when something meets unexpected resistance, it's because there is some hidden tension -- here for example, you have a backend-thing that owns the function, but a function-builder that also owns the function, simultaneously, and that can result all sorts of unexpected bugs where state changes under your feet. (Rust strongly discourages "shared ownership" where there are multiple mutable paths to one thing.) So my only real recommendation (or at least the most honest one!) to an issue of "I prefer designing my software differently" is "learn the idiomatic Rust way first". I know it sounds cliche, sorry; but there's no magic trick to reveal here to solve the problem more easily
The whole codebase is here, I'm typing another response right now so I'm gonna get this out quickly first: https://github.com/RedstoneWizard08/QuickScript
I do know that the object-ownership design is probably a bad move. I just don't know how else to do this in a clean way. I can try to move it into a more function-based system, but to me I think that's less intuitive as I am trying to both build an API that others may be able to use or base code off of, and an API that I can come back to in a few years and quickly figure out what's going on. I guess that maybe I'm just not used to functional programming yet.
that's no problem, I just meant it as sort of a reference point if it helped to clarify; the important bit is the "explicit plumbing", i.e., passing the borrows around
One pattern I have found to help is to gather up a "context struct" of arguments that exist (as borrows or owned) during some phase; for example, while compiling a single function
So you have some toplevel generate_function
that creates the Function
, and then you create your JitContext
that has a FunctionBuilder
inside of it, and the JitContext
is passed down the stack
the thing that this does that storing an Option<...>
borrow in a field on your self
struct does not is that it makes the lifetimes explicitly nested: Rust can more easily reason about parameters because, if a borrow is passed into a function, unless explicitly annotated it lasts only for that function
Interesting. I think that might be the way to go. I am loosely basing this on the Cranelift JIT example, and that is how they did it, I just didn't think it was as organized.
whereas a field on self
lasts as long as self
, so it's harder to prove to the type system that the borrow goes away before the borrowed thing does
(that's the heart of the "self-referential struct" problem)
Okay, I think I'll give that a shot then.
cool, best of luck! and do feel free to ask more here if still stuck :-)
Okay! Thanks though!
BTW, just to clarify, you meant something like this, right?
use std::collections::HashMap;
use cranelift_frontend::{FunctionBuilder, Variable};
use cranelift_module::DataId;
pub struct CodegenContext<'a> {
pub locals: HashMap<String, DataId>,
pub vars: HashMap<String, Variable>,
pub builder: FunctionBuilder<'a>,
}
Okay, now the only issue is this:
let mut builder = FunctionBuilder::new(&mut self.ctx().func, self.builder_ctx());
It says, "cannot borrow '*self' as mutable more than once at a time"
.
omg... no errors... wow! thank you so much!!!
Jacob Sapoznikow has marked this topic as resolved.
cool, glad that worked out!
Lee Wei has marked this topic as unresolved.
Lee Wei has marked this topic as resolved.
Sorry, misclicked the button. :sweat_smile:
Last updated: Jan 24 2025 at 00:11 UTC