I am building an embedded SQL database and experimenting with cranelift to generate efficient query operators. One of the main issues with jitted query executors is that they quickly become rather complex. You often want to inline calls made in loops, meaning that you have to emit code for many primitives and data structures. The task quickly becomes daunting if you add to that specialization for all data types. Now, I was wondering if it would be possible to write some library in rust, and store that in the program static assets. When it comes time to build the query, I could assemble fragments from that library, inlining where necessary. Now I have a couple of questions:
Could something like that be hacked together with rustc_codegen_cranelift
Let's say I can statically generate a set of functions. Could I simply declare/define them in a module at program startup and use them within the query I build at runtime?
Is there any way those functions could be inlined? Does Cranelift even support that?
If inlining functions are not an option, would there be a way to compile and store the basic blocks of the function bodies and stitch them together at runtime?
does this approach even make any sense?
Sorry if any of those questions seem obvious, I'm not very experienced working with JITs :)
one more question:
database queries are definitely outside of my expertise, but I'll try to take a crack at answering your questions
Could something like that be hacked together with rustc_codegen_cranelift
Is your input language Rust? If not, then I don't think it makes sense to use rustc_codegen_cranelift
. you would want to write your own input-language-to-CLIF frontend instead.
Is there any way those functions could be inlined? Does Cranelift even support that?
Cranelift does not have an inlining pass today. It is something we want to add as an option eventually, but probably not something we would run automatically. In the meantime, it is expected that the CLIF producer (eg wasmtime's wasm-to-clif frontend or rustc_codegen_cranelift
) has already done inlining where it was beneficial or desired.
does this approach even make any sense?
without knowing more, I suspect that you probably want to write your own query-language-to-clif frontend. other than that, I think it does make sense.
I remember seeing a paper not too too long ago that benchmarked a bunch of different JIT compilers/frameworks in the context of JITing database queries. It may make sense for you to track that paper down and read it for more context and inspiration. If I can dredge it up, I'll link it here.
Would it be possible to attach debug information to those fragments to pass compiled queries through a debugger?
yes, primarily through cranelift_frontend::FunctionBuilder::set_val_label
. although you may find gaps around certain corners of DWARF or incompleteness. certainly we don't do the best job of preserving debug info across all our optimizations; would certainly appreciate issues filed for any specific cases you run into so we can track, diagnose, and fix them. for the best debugging fidelity, disable optimizations.
fitzgen (he/him) said:
Could something like that be hacked together with rustc_codegen_cranelift
Is your input language Rust? If not, then I don't think it makes sense to use
rustc_codegen_cranelift
. you would want to write your own input-language-to-CLIF frontend instead.
if I understood the proposal above correctly, the idea was to write a runtime library of sorts in Rust, and then take its functions and inline them directly into Cranelift-compiled JIT'd query code. Is that right, @marin ?
That seems plausible to me -- the technical bit that makes it work is that rustc_codegen_cranelift
is ABI-compatible with LLVM Rust (the default/main toolchain), so if you access internal data structures, etc., everything should work just as if you had called a separately compiled function. If it doesn't, we would consider that a bug (either in cg_clif
or Cranelift, depending on the issue).
I also suspect you'd want to build a purpose-built inliner for this, at least at first, rather than try to build general framework in Cranelift: you have more information about intent and performance tradeoffs than a generic inliner's heuristics would, so you could know that, for example, you always want to inline simple field accessor functions in your runtime, and that they don't call back into the query code. The "code transcription" part of an inliner is the relatively simple part; the heuristics and the plumbing to operate over connected components of the callgraph is much harder, and that's the part you wouldn't need if you do "manual inlining" at runtime-library calls.
@Chris Fallin yes that's correct! I wasn't even expecting to be able to work with rust datatype, but rather, just being able to write the operator in rust leverage monomorphisation to generate a specialized implementation for all my database types and reduce the maintenance burden. At runtime, I would assemble those building blocks into pipelines
@fitzgen (he/him) are you thinking about this paper? https://www.vldb.org/pvldb/vol11/p2209-kersten.pdf
don't think it was that paper; the paper I'm thinking of included cranelift in its benchmark evaluation. I remember having some nitpicks about the way they used or presented cranelift but otherwise thinking it was good. it compared building the "same" query JIT with multiple approaches/frameworks, iirc
ah, yeah, that paper is right on the tip of my neurons too, unfortunately not recalling the author or title... I'll dig a bit
As a first step, I will probably stick to compiling expressions in where clauses and maybe an aggregation operator. But even then, take the group by
aggregation clause, for example. You could implement that as a hashmap, with a specialization for every possible key type. Being able to leverage the Rust ecosystem would be really neat.
I'll run a search on google scholars, if it mentions cranelift it should pop up :)
Last updated: Jan 24 2025 at 00:11 UTC