extern crate bindgen;
extern crate pkg_config;
use std::env;
use std::path::{Path, PathBuf};
use std::process::Command;
fn main() {
let vendored = env::var("CARGO_FEATURE_VENDORED").is_ok();
let zlib_ng_compat = env::var("CARGO_FEATURE_ZLIB_NG_COMPAT").is_ok();
println!("cargo:rerun-if-env-changed=PARASAIL_NO_VENDOR");
let force_no_vendor = env::var("PARASAIL_NO_VENDOR").map_or(false, |v| v != "0");
if force_no_vendor {
if try_system_parasail().is_err() {
panic!(
"The environment variable `PARASAIL_NO_VENDOR` is set, but no system parasail was found.
Please install parasail or unset `PARASAIL_NO_VENDOR` or use `PARASAIL_NO_VENDOR=0`."
);
}
return;
}
let use_system_parasail = !vendored && !zlib_ng_compat;
if use_system_parasail && try_system_parasail().is_ok() {
return;
}
build_parasail();
}
fn bindgen_build(header: &str) {
let out_path = PathBuf::from(env::var("OUT_DIR").unwrap());
let bindings = bindgen::Builder::default()
.header(header)
.allowlist_item("parasail_.*")
.parse_callbacks(Box::new(bindgen::CargoCallbacks::new()))
.generate()
.expect("Unable to generate bindings");
bindings
.write_to_file(out_path.join("parasail_bindings.rs"))
.expect("Couldn't write bindings!");
}
fn try_system_parasail() -> Result<pkg_config::Library, pkg_config::Error> {
let mut cfg = pkg_config::Config::new();
match cfg.atleast_version("2.4.2").probe("parasail") {
Ok(lib) => {
println!(
"cargo:rustc-link-search=native={}",
lib.link_paths[0].display()
);
println!("cargo:rustc-link-lib=parasail");
bindgen_build("wrapper.h");
Ok(lib)
}
Err(e) => {
println!("cargo:warning=Could not find system parasail: {e}",);
Err(e)
}
}
}
fn build_parasail() {
println!("cargo:rustc-cfg=parasail_vendored");
let project_dir = env::var("CARGO_MANIFEST_DIR").unwrap();
if !Path::new("parasail/src").exists() {
let _ = Command::new("git")
.args(["submodule", "update", "--init", "parasail"])
.status();
}
let parasail_dir = PathBuf::from(&project_dir)
.join("parasail")
.canonicalize()
.expect("Failed to find parasail directory");
let out_dir = env::var("OUT_DIR").unwrap();
let parasail_build = PathBuf::from(&out_dir).join("parasail_build");
std::fs::create_dir_all(¶sail_build).unwrap();
let headers_path = parasail_dir.join("parasail.h");
let headers_path_str = headers_path.to_str().unwrap();
println!("cargo:warning=Checking for CMake...");
if Command::new("cmake").arg("--version").output().is_err() {
println!("cargo:warning===========================================");
println!("cargo:warning=Failed to find CMake installation!");
println!("cargo:warning=This crate requires CMake to build.");
println!("cargo:warning=Please install CMake");
panic!("CMake not found and required to build libparasail-sys");
}
assert!(
Command::new("cmake")
.args([
"-DBUILD_SHARED_LIBS=OFF",
"-DCMAKE_BUILD_TYPE=Release",
"-DCMAKE_POSITION_INDEPENDENT_CODE=ON",
parasail_dir.to_str().unwrap()
])
.current_dir(¶sail_build)
.status()
.expect("Error: cmake command failed. Is cmake installed?")
.success(),
"Failed to cmake"
);
assert!(
Command::new("cmake")
.args([
"--build",
¶sail_build.to_str().unwrap(),
"--target parasail"
])
.current_dir(¶sail_build)
.status()
.unwrap()
.success(),
"Failed to make"
);
println!(
"cargo:rustc-link-search=native={}",
parasail_build.display()
);
println!("cargo:rustc-link-lib=static=parasail");
bindgen_build(headers_path_str);
}