hugr-llvm 0.25.0

A general and extensible crate for lowering HUGRs into LLVM IR
Documentation
// This module provides a *very* basic panic runtime for LLVM execution
// tests involving panics.
//
// Note that this is strictly for *execution testing* and is not used in
// production! See `crate::emit::test::PanicTestPreludeCodegen` and
// `crate::emit::test::Emission::exec_panicking` for the test harness.
//
// The progam entry point should be invoked via the `trampoline` function
// which launches the program. Calls to `panic_exit` will exit the program
// and make `trampoline` return prematurely. Note that the runtime will
// perform a direct jump to the exit and no unrolling is performed. This
// is fine for our use case since stack frames from hugr-llvm generated
// bytecode don't require cleanup. However, using `panic_exit` to jump over
// Rust frames is undefined behaviour.
//
// Note that the implementation relies on setjmp-longjmp (SJLJ) jumping to
// exit the program. As SJLJ is not available in Rust, this had to be
// implemented in C in linked into the Rust binary.

#include <setjmp.h>
#include <string.h>

// Runs the `entry` function inside the panic runtime.
//
// # Safety
//
// `jmp_buf` must have size of at least `jmp_buf_size()`.
void trampoline(void* jmp_buf, void entry()) {
    if (!setjmp(jmp_buf)) {
        entry();
    }
}

// Exits a program launched by `trampoline`.
//
// This function will never return. Also copies the first `msg_limit`
// characters of the string `msg` into `msg_buf`.
//
// # Safety
//
// Calls to this function are only allowed within stack frames that were
// launched by `trampoline`. Furthermore, `panic_exit` may only be used
// to jump over C or LLVM execution engine frames. Using `panic_exit` to
// jump over Rust frames is undefined behaviour.
//
// `jmp_buf` must be the same buffer given to `trampoline` and `msg_buf`
// must have size of at least `msg_limit`.
void panic_exit(void* jmp_buf, char* msg_buf, char* msg, size_t msg_limit) {
    strncpy(msg_buf, msg, msg_limit);
    longjmp(jmp_buf, 1);
}

// Minimum size that needs to be alloced for jump buffers passed to
// `trampoline` and `panic_exit`.
size_t jmp_buf_size() {
    return sizeof(jmp_buf);
}