19df49d7eSJed Brown extern crate bindgen; 29df49d7eSJed Brown extern crate pkg_config; 39df49d7eSJed Brown 4e7f21013SJed Brown use std::path::{Path, PathBuf}; 59df49d7eSJed Brown use std::process::Command; 69df49d7eSJed Brown 79df49d7eSJed Brown fn main() { 89df49d7eSJed Brown let out_dir = PathBuf::from(env("OUT_DIR").unwrap()); 99df49d7eSJed Brown let statik = env("CARGO_FEATURE_STATIC").is_some(); 109df49d7eSJed Brown let system = env("CARGO_FEATURE_SYSTEM").is_some(); 119df49d7eSJed Brown 129df49d7eSJed Brown let ceed_pc = if system { 139df49d7eSJed Brown "ceed".to_string() 149df49d7eSJed Brown } else { 159df49d7eSJed Brown // Install libceed.a or libceed.so to $OUT_DIR/lib 169df49d7eSJed Brown let makeflags = env("CARGO_MAKEFLAGS").unwrap(); 179df49d7eSJed Brown let mut make = Command::new("make"); 189df49d7eSJed Brown make.arg("install") 199df49d7eSJed Brown .arg(format!("prefix={}", out_dir.to_string_lossy())) 20dfd5c3f2SJed Brown .arg(format!( 21dfd5c3f2SJed Brown "OBJDIR={}", 22dfd5c3f2SJed Brown out_dir.join("build").to_string_lossy() 23dfd5c3f2SJed Brown )) 24dfd5c3f2SJed Brown .arg(format!( 25dfd5c3f2SJed Brown "LIBDIR={}", 26dfd5c3f2SJed Brown out_dir.join("build").join("lib").to_string_lossy() 27dfd5c3f2SJed Brown )) 289df49d7eSJed Brown .env("MAKEFLAGS", makeflags) 299df49d7eSJed Brown .current_dir("c-src"); 309df49d7eSJed Brown if statik { 319df49d7eSJed Brown make.arg("STATIC=1"); 329df49d7eSJed Brown } 339df49d7eSJed Brown run(&mut make); 349df49d7eSJed Brown out_dir 359df49d7eSJed Brown .join("lib") 369df49d7eSJed Brown .join("pkgconfig") 379df49d7eSJed Brown .join("ceed.pc") 389df49d7eSJed Brown .to_string_lossy() 399df49d7eSJed Brown .into_owned() 409df49d7eSJed Brown }; 419df49d7eSJed Brown pkg_config::Config::new() 429df49d7eSJed Brown .statik(statik) 43*d66340f5SJed Brown .atleast_version("0.9") 449df49d7eSJed Brown .probe(&ceed_pc) 459df49d7eSJed Brown .unwrap(); 469df49d7eSJed Brown 479df49d7eSJed Brown // Tell cargo to invalidate the built crate whenever the wrapper changes 489df49d7eSJed Brown println!("cargo:rerun-if-changed=c-src/include/ceed/ceed.h"); 499df49d7eSJed Brown println!("cargo:rerun-if-changed=c-src/Makefile"); 50e7f21013SJed Brown if Path::new("c-src/config.mk").is_file() { 519df49d7eSJed Brown println!("cargo:rerun-if-changed=c-src/config.mk"); 52e7f21013SJed Brown } 539df49d7eSJed Brown 549df49d7eSJed Brown // The bindgen::Builder is the main entry point 559df49d7eSJed Brown // to bindgen, and lets you build up options for 569df49d7eSJed Brown // the resulting bindings. 579df49d7eSJed Brown let bindings = bindgen::Builder::default() 589df49d7eSJed Brown // The input header we would like to generate 599df49d7eSJed Brown // bindings for. 609df49d7eSJed Brown .header("c-src/include/ceed/ceed.h") 619df49d7eSJed Brown .allowlist_function("Ceed.*") 629df49d7eSJed Brown .allowlist_type("Ceed.*") 639df49d7eSJed Brown .allowlist_var("Ceed.*") 649df49d7eSJed Brown .allowlist_var("CEED_.*") 659df49d7eSJed Brown // Don't chase recursive definitions of these system types 669df49d7eSJed Brown .opaque_type("FILE") 679df49d7eSJed Brown .opaque_type("va_list") 689df49d7eSJed Brown // Accessing directly here, but perhaps should use libc crate 699df49d7eSJed Brown .allowlist_function("fclose") 709df49d7eSJed Brown .allowlist_function("open_memstream") 719df49d7eSJed Brown // Tell cargo to not mangle the function names 729df49d7eSJed Brown .trust_clang_mangling(false) 739df49d7eSJed Brown // Tell cargo to invalidate the built crate whenever any of the 749df49d7eSJed Brown // included header files changed. 759df49d7eSJed Brown .parse_callbacks(Box::new(bindgen::CargoCallbacks)) 769df49d7eSJed Brown // Finish the builder and generate the bindings. 779df49d7eSJed Brown .generate() 789df49d7eSJed Brown // Unwrap the Result and panic on failure. 799df49d7eSJed Brown .expect("Unable to generate bindings"); 809df49d7eSJed Brown 819df49d7eSJed Brown // Write the bindings to the $OUT_DIR/bindings.rs file. 829df49d7eSJed Brown bindings 839df49d7eSJed Brown .write_to_file(out_dir.join("bindings.rs")) 849df49d7eSJed Brown .expect("Couldn't write bindings!"); 859df49d7eSJed Brown } 869df49d7eSJed Brown 879df49d7eSJed Brown fn env(k: &str) -> Option<String> { 889df49d7eSJed Brown std::env::var(k).ok() 899df49d7eSJed Brown } 909df49d7eSJed Brown 919df49d7eSJed Brown fn run(command: &mut Command) { 929df49d7eSJed Brown println!("Running: `{:?}`", command); 939df49d7eSJed Brown assert!(command.status().unwrap().success()); 949df49d7eSJed Brown } 95