无法使用 bindgen 进行 llvm 绑定

Unable to make llvm bindings using bindgen

我正在努力完成 Kaleidoscope Rust 教程。

但我似乎对 codegen 很感兴趣。

我的 build.rs :

extern crate bindgen;

use std::env;
use std::path::PathBuf;

fn main(){
    println!("cargo:rustc-link-lib=llvm");
    println!("cargo:rerun-if-changed=wrapper.h");

    let bindings = bindgen::Builder::default()
        .header("wrapper.h")
        .clang_arg("llvm-config --cxxflags --ldflags --system-libs --libs core")
        .parse_callbacks(Box::new(bindgen::CargoCallbacks))
        .generate()
        .expect("unable to generate bindings");

    let out_path = PathBuf::from(env::var("OUT_DIR").unwrap());
    bindings
        .write_to_file(out_path.join("bindings.rs"))
        .expect("couldn't write bindings!");
}

我的 wrapper.h :

#include "llvm/ADT/APFloat.h"
#include "llvm/ADT/STLExtras.h"
#include "llvm/IR/BasicBlock.h"
#include "llvm/IR/Constants.h"
#include "llvm/IR/DerivedTypes.h"
#include "llvm/IR/Function.h"
#include "llvm/IR/IRBuilder.h"
#include "llvm/IR/LLVMContext.h"
#include "llvm/IR/Module.h"
#include "llvm/IR/Type.h"
#include "llvm/IR/Verifier.h"

我希望能够传递教程中的以下标志 clang++ `llvm-config --cxxflags --ldflags --system-libs --libs core`

我得到的当前错误:

build-script-build` (exit code: 101)
--- stdout
cargo:rustc-link-lib=llvm
cargo:rerun-if-changed=wrapper.h

--- stderr
warning: llvm-config --cxxflags --ldflags --system-libs --libs core: 'linker' input unused [-Wunused-command-line-argument]
/usr/include/llvm/Support/Compiler.h:19:10: fatal error: 'new' file not found
warning: llvm-config --cxxflags --ldflags --system-libs --libs core: 'linker' input unused [-Wunused-command-line-argument], err: false
/usr/include/llvm/Support/Compiler.h:19:10: fatal error: 'new' file not found, err: true
thread 'main' panicked at 'unable to generate bindings: ()', src/libcore/result.rs:1188:5
note: run with `RUST_BACKTRACE=1` environment variable to display a backtrace.

你按字面意义将 llvm-config --cxxflags --ldflags --system-libs --libs core 传递给 clang,但你需要做的是执行 llvm-config --cxxflags --ldflags --system-libs --libs core 并将结果参数传递给 clang(这就是反引号在 shell).

所以在 bindgen 调用之前,做这样的事情(未测试):

use std::{process::Command, str::from_utf8};

let llvm_config_out = Command::new("llvm-config")
    .args(&["--cxxflags", "--ldflags", "--system-libs", "--libs", "core"])
    .output()
    .expect("failed to execute llvm-config");

let llvm_clang_args = llvm_config_out
    .stdout
    .split(|byte| byte.is_ascii_whitespace())
    .map(|arg| str::from_utf8(arg).unwrap());

然后您可以将 llvm_clang_args 作为 .clang_args(llvm_clang_args) 传递给 bindgen 构建器(不是 .clang_arg,注意额外的 s)。

感谢jplatte指出执行llvm-config --cxxflags --ldflags --system-libs --libs core并将其提供给clang_args

错误仍然存​​在是因为尝试绑定 c 而不是 c++,最终 build.rs 在将 wrapper.h 重命名为 wrapper.hpp 并传递 -x -- c++

extern crate bindgen;

use std::env;
use std::path::PathBuf;
use std::{process::Command, str};

fn main(){
    println!("cargo:rustc-link-lib=llvm");
    println!("cargo:rerun-if-changed=wrapper.hpp");

    let llvm_config_out = Command::new("llvm-config")
        .args(&["--cxxflags", "--ldflags", "--system-libs", "--libs", "core"])
        .output()
        .expect("failed to execute llvm-config");

    let llvm_clang_args = llvm_config_out
        .stdout
        .split(|byte| byte.is_ascii_whitespace())
        .map(|arg| str::from_utf8(arg).unwrap());

    let bindings = bindgen::Builder::default()
        .header("wrapper.hpp")
        .clang_arg("-x").clang_arg("c++") // c++ flag
        .clang_args(llvm_clang_args)
        .parse_callbacks(Box::new(bindgen::CargoCallbacks))
        .generate()
        .expect("unable to generate bindings");

    let out_path = PathBuf::from(env::var("OUT_DIR").unwrap());
    bindings
        .write_to_file(out_path.join("bindings.rs"))
        .expect("couldn't write bindings!");
}