use std::env::var;
use std::path::Path;
use std::path::PathBuf;
use thiserror::Error;
use winreg::RegKey;
use winreg::enums::*;
use failure::format_err;
#[derive(Debug, Error)]
pub enum Error {
#[error(transparent)]
IoError(#[from] std::io::Error),
#[error("cannot find the directory")]
DirectoryNotFound,
}
pub enum DirectoryType {
Include,
Library,
}
pub fn get_windows_kits_dir() -> Result<PathBuf, Error> {
let hklm = RegKey::predef(HKEY_LOCAL_MACHINE);
let key = r"SOFTWARE\Microsoft\Windows Kits\Installed Roots";
let dir: String = hklm.open_subkey(key)?.get_value("KitsRoot10")?;
Ok(dir.into())
}
pub fn get_km_dir(dir_type: DirectoryType) -> Result<PathBuf, Error> {
let dir = get_windows_kits_dir()?
.join(match dir_type {
DirectoryType::Include => "Include",
DirectoryType::Library => "Lib",
})
.read_dir()?;
let dir = dir
.filter_map(|dir| dir.ok())
.map(|dir| dir.path())
.filter(|dir| {
dir.components()
.last()
.and_then(|c| c.as_os_str().to_str())
.map(|c| c.starts_with("10.") && dir.join("km").is_dir())
.unwrap_or(false)
})
.max()
.ok_or_else(|| Error::DirectoryNotFound)?;
Ok(dir.join("km"))
}
pub fn build() -> Result<(), Error> {
let dir = get_km_dir(DirectoryType::Library).unwrap();
let target = std::env::var("TARGET").unwrap();
let arch = if target.contains("x86_64") {
"x64"
} else if target.contains("i686") {
"x86"
} else {
panic!("The target {} is currently not supported.", target);
};
let dir = dir.join(arch);
println!("cargo:rustc-link-search=native={}", dir.to_str().unwrap());
println!("cargo:rustc-link-arg=/NODEFAULTLIB");
println!("cargo:rustc-link-arg=/SUBSYSTEM:NATIVE");
println!("cargo:rustc-link-arg=/DRIVER");
println!("cargo:rustc-link-arg=/DYNAMICBASE");
println!("cargo:rustc-link-arg=/MANIFEST:NO");
println!("cargo:rustc-link-arg=/ENTRY:driver_entry");
println!("cargo:rustc-link-arg=/MERGE:.edata=.rdata");
println!("cargo:rustc-link-arg=/MERGE:.rustc=.data");
println!("cargo:rustc-link-arg=/INTEGRITYCHECK");
Ok(())
}
pub fn get_km_dir_path(windows_kits_dir: &PathBuf) -> Result<PathBuf, failure::Error> {
let readdir = Path::new(windows_kits_dir).join("lib").read_dir()?;
let max_libdir = readdir
.filter_map(|dir| dir.ok())
.map(|dir| dir.path())
.filter(|dir| {
dir.components()
.last()
.and_then(|c| c.as_os_str().to_str())
.map(|c| c.starts_with("10.") && dir.join("km").is_dir())
.unwrap_or(false)
})
.max()
.ok_or_else(|| format_err!("Can not find a valid km dir in `{:?}`", windows_kits_dir))?;
Ok(max_libdir.join("km"))
}
pub fn internal_link_search() {
let windows_kits_dir = get_windows_kits_dir().unwrap();
let km_dir = get_km_dir_path(&windows_kits_dir).unwrap();
let target = var("TARGET").unwrap();
let arch = if target.contains("x86_64") {
"x64"
} else if target.contains("i686") {
"x86"
} else {
panic!("Only support x86_64 and i686!");
};
let lib_dir = km_dir.join(arch);
println!(
"cargo:rustc-link-search=native={}",
lib_dir.to_str().unwrap()
);
println!("cargo:rustc-link-lib=msvcrt");
}
pub fn extra_link_search() {}
pub fn build_driver() {
if var(format!(
"CARGO_FEATURE_{}",
"extra_link_search".to_uppercase()
))
.is_ok()
{
extra_link_search()
} else {
internal_link_search()
}
}