use std::{ env, fs, path::{Path, PathBuf}, }; fn find_package(name: &str) -> Vec { let vcpkg_root = std::env::var("VCPKG_ROOT").unwrap(); let mut path: PathBuf = vcpkg_root.into(); let target_os = std::env::var("CARGO_CFG_TARGET_OS").unwrap(); let mut target_arch = std::env::var("CARGO_CFG_TARGET_ARCH").unwrap(); if target_arch == "x86_64" { target_arch = "x64".to_owned(); } else if target_arch == "aarch64" { target_arch = "arm64".to_owned(); } else { target_arch = "arm".to_owned(); } let target = if target_os == "macos" { "x64-osx".to_owned() } else if target_os == "windows" { "x64-windows-static".to_owned() } else if target_os == "android" { format!("{}-android-static", target_arch) } else { "x64-linux".to_owned() }; println!("cargo:info={}", target); path.push("installed"); path.push(target); let mut lib = name.trim_start_matches("lib").to_string(); if lib == "vpx" && target_os == "windows" { lib = format!("{}mt", lib); } println!("{}", format!("cargo:rustc-link-lib=static={}", lib)); println!( "{}", format!( "cargo:rustc-link-search={}", path.join("lib").to_str().unwrap() ) ); let include = path.join("include"); println!("{}", format!("cargo:include={}", include.to_str().unwrap())); vec![include] } fn generate_bindings( ffi_header: &Path, include_paths: &[PathBuf], ffi_rs: &Path, exact_file: &Path, ) { let mut b = bindgen::builder() .header(ffi_header.to_str().unwrap()) .whitelist_type("^[vV].*") .whitelist_var("^[vV].*") .whitelist_function("^[vV].*") .rustified_enum("^v.*") .trust_clang_mangling(false) .layout_tests(false) // breaks 32/64-bit compat .generate_comments(false); // vpx comments have prefix /*!\ for dir in include_paths { b = b.clang_arg(format!("-I{}", dir.display())); } b.generate().unwrap().write_to_file(ffi_rs).unwrap(); fs::copy(ffi_rs, exact_file).ok(); // ignore failure } fn gen_vpx() { let includes = find_package("libvpx"); let src_dir = env::var_os("CARGO_MANIFEST_DIR").unwrap(); let src_dir = Path::new(&src_dir); let out_dir = env::var_os("OUT_DIR").unwrap(); let out_dir = Path::new(&out_dir); let ffi_header = src_dir.join("vpx_ffi.h"); println!("rerun-if-changed={}", ffi_header.display()); for dir in &includes { println!("rerun-if-changed={}", dir.display()); } let ffi_rs = out_dir.join("vpx_ffi.rs"); let exact_file = src_dir.join("generated").join("vpx_ffi.rs"); generate_bindings(&ffi_header, &includes, &ffi_rs, &exact_file); } fn main() { // note: all link symbol names in x86 (32-bit) are prefixed wth "_". // run "rustup show" to show current default toolchain, if it is stable-x86-pc-windows-msvc, // please install x64 toolchain by "rustup toolchain install stable-x86_64-pc-windows-msvc", // then set x64 to default by "rustup default stable-x86_64-pc-windows-msvc" let target = target_build_utils::TargetInfo::new(); if target.unwrap().target_pointer_width() != "64" { panic!("Only support 64bit system"); } env::remove_var("CARGO_CFG_TARGET_FEATURE"); env::set_var("CARGO_CFG_TARGET_FEATURE", "crt-static"); find_package("libyuv"); gen_vpx(); // there is problem with cfg(target_os) in build.rs, so use our workaround let target_os = std::env::var("CARGO_CFG_TARGET_OS").unwrap(); if target_os == "android" || target_os == "ios" { // nothing } else if cfg!(windows) { // The first choice is Windows because DXGI is amazing. println!("cargo:rustc-cfg=dxgi"); } else if cfg!(target_os = "macos") { // Quartz is second because macOS is the (annoying) exception. println!("cargo:rustc-cfg=quartz"); } else if cfg!(unix) { // On UNIX we pray that X11 (with XCB) is available. println!("cargo:rustc-cfg=x11"); } }