extern crate bindgen;
use autotools::Config;
use std::collections::HashSet;
use std::env;
use std::path::PathBuf;
use std::process::Command;
static WOLFSSL_VERSION: &str = "wolfssl-5.5.3-stable";
#[derive(Debug)]
struct IgnoreMacros(HashSet<String>);
impl bindgen::callbacks::ParseCallbacks for IgnoreMacros {
fn will_parse_macro(&self, name: &str) -> bindgen::callbacks::MacroParsingBehavior {
if self.0.contains(name) {
bindgen::callbacks::MacroParsingBehavior::Ignore
} else {
bindgen::callbacks::MacroParsingBehavior::Default
}
}
}
fn extract_wolfssl(dest: &str) -> std::io::Result<()> {
Command::new("tar")
.arg("-zxvf")
.arg(format!("vendor/{}.tar.gz", WOLFSSL_VERSION))
.arg("-C")
.arg(dest)
.status()
.unwrap();
Ok(())
}
fn build_wolfssl(dest: &str) -> PathBuf {
let mut conf = Config::new(format!("{}/{}", dest, WOLFSSL_VERSION));
conf.reconf("-ivf")
.enable_static()
.disable_shared()
.enable("tls13", None)
.disable("oldtls", None)
.enable("singlethreaded", None)
.enable("dtls", None)
.enable("sp", None)
.enable("sp-asm", None)
.enable("dtls-mtu", None)
.disable("sha3", None)
.disable("dh", None)
.enable("supportedcurves", None)
.enable("curve25519", None)
.enable("secure-renegotiation", None)
.enable("sni", None)
.cflag("-g")
.cflag("-fPIC")
.cflag("-DWOLFSSL_DTLS_ALLOW_FUTURE")
.cflag("-DWOLFSSL_MIN_RSA_BITS=2048")
.cflag("-DWOLFSSL_MIN_ECC_BITS=256");
if cfg!(feature = "postquantum") {
if let Some(include) = std::env::var_os("DEP_OQS_ROOT") {
let oqs_path = &include.into_string().unwrap();
conf.cflag(format!("-I{}/build/include/", oqs_path));
conf.ldflag(format!("-L{}/build/lib/", oqs_path));
conf.with("liboqs", None);
} else {
panic!("Post Quantum requested but liboqs appears to be missing?");
}
}
if build_target::target_arch().unwrap() == build_target::Arch::X86_64 {
conf.enable("intelasm", None);
conf.enable("aesni", None);
}
if build_target::target_arch().unwrap() == build_target::Arch::AARCH64 {
conf.enable("armasm", None);
}
if build_target::target_arch().unwrap() == build_target::Arch::ARM {
conf.enable("armasm", None);
}
conf.build()
}
fn main() -> std::io::Result<()> {
let dst_string = env::var("OUT_DIR").unwrap();
extract_wolfssl(&dst_string)?;
let dst = build_wolfssl(&dst_string);
let mut hash_ignored_macros = HashSet::new();
for i in &[
"IPPORT_RESERVED",
"EVP_PKEY_DH",
"BIO_CLOSE",
"BIO_NOCLOSE",
"CRYPTO_LOCK",
"ASN1_STRFLGS_ESC_MSB",
"SSL_MODE_RELEASE_BUFFERS",
"GEN_IPADD",
"EVP_PKEY_RSA",
] {
hash_ignored_macros.insert(i.to_string());
}
let ignored_macros = IgnoreMacros(hash_ignored_macros);
let bindings = bindgen::Builder::default()
.header("wrapper.h")
.clang_arg(format!("-I{}/include/", dst_string))
.parse_callbacks(Box::new(ignored_macros))
.rustfmt_bindings(true)
.blocklist_file("/usr/include/stdlib.h")
.generate()
.expect("Unable to generate bindings");
bindings
.write_to_file(dst.join("bindings.rs"))
.expect("Couldn't write bindings!");
println!("cargo:rustc-link-lib=static=wolfssl");
if cfg!(feature = "postquantum") {
println!("cargo:rustc-link-lib=static=oqs");
}
println!("cargo:rustc-link-search=native={}/lib/", dst_string);
println!("cargo:include={}", dst_string);
println!("cargo:rerun-if-changed=wrapper.h");
Ok(())
}