aya-rustc-llvm-proxy 0.10.0

Dynamically proxy LLVM calls into Rust own shared library
Documentation
#![deny(warnings)]
#![allow(
    non_snake_case,
    unused_imports,
    unused_macros,
    deprecated,
    clippy::missing_safety_doc
)]

//! This is a **fork** of the [rustc-llvm-proxy](https://github.com/denzp/rustc-llvm-proxy) crate.
//!
//! Dynamically proxy LLVM calls into Rust own shared library! 🎉
//!
//! ## Use cases
//! Normally there is no much need for the crate, except a couple of exotic cases:
//!
//! * Your crate is some kind build process helper that leverages LLVM (e.g. [bpf-linker](https://github.com/aya-rs/bpf-linker)),
//! * Your crate needs to stay up to date with Rust LLVM version (again [bpf-linker](https://github.com/aya-rs/bpf-linker)),
//! * You would prefer not to have dependencies on host LLVM libs (as always [bpf-linker](https://github.com/aya-rs/bpf-linker)).
//!
//! ## Usage
//! First, you need to make sure no other crate links your binary against system LLVM library.
//! In case you are using `llvm-sys`, this can be achieved with a special feature:
//!
//! ``` toml
//! [dependencies.llvm-sys]
//! version = "70"
//! features = ["no-llvm-linking"]
//! ```
//!
//! Then all you need to do is to include the crate into your project:
//!
//! ``` toml
//! [dependencies]
//! rustc-llvm-proxy = "0.4"
//! ```
//!
//! ``` rust
//! extern crate aya_rustc_llvm_proxy;
//! ```

use libloading::Library;

pub mod init;

static SHARED_LIB: std::sync::LazyLock<Library> = std::sync::LazyLock::new(|| {
    for (var, is_bin) in [
        ("LD_LIBRARY_PATH", false),
        ("DYLD_FALLBACK_LIBRARY_PATH", false),
        ("PATH", true),
    ] {
        let Some(unparsed) = std::env::var_os(var) else {
            continue;
        };
        let paths = std::env::split_paths(&unparsed);
        for mut path in paths {
            if is_bin {
                path.pop();
                path.push("lib");
            }
            let files = match path.read_dir() {
                Ok(files) => files,
                Err(err) => {
                    eprintln!("unable to read dir {}: {}", path.display(), err);
                    continue;
                }
            };
            for (i, file) in files.enumerate() {
                let file = match file {
                    Ok(file) => file,
                    Err(err) => {
                        eprintln!(
                            "unable to read dir entry {} in {}: {}",
                            i,
                            path.display(),
                            err
                        );
                        continue;
                    }
                };
                let path = file.path();
                let Some(stem) = path.file_stem() else {
                    continue;
                };
                let Some(stem) = stem.to_str() else { continue };
                if stem.starts_with("libLLVM") {
                    match unsafe { Library::new(&path) } {
                        Ok(library) => return library,
                        Err(error) => {
                            eprintln!(
                                "unable to open LLVM shared lib {}: {}",
                                path.display(),
                                error
                            );
                            continue;
                        }
                    }
                }
            }
        }
    }
    panic!("unable to find LLVM shared lib")
});

/// LLVM C-API symbols with dynamic resolving.
pub mod proxy {
    use super::SHARED_LIB;

    include!(concat!(env!("OUT_DIR"), "/llvm_gen.rs"));
}