summaryrefslogtreecommitdiff
path: root/crates/windows-kernel-build/src
diff options
context:
space:
mode:
authorFuwn <[email protected]>2022-01-03 03:20:12 -0800
committerFuwn <[email protected]>2022-01-03 03:20:12 -0800
commit85db2b507f3f69b32811c54a89d9ac7bbbc46121 (patch)
tree2efd66da452f8a6a2cc6c91584c925f237506ddf /crates/windows-kernel-build/src
downloaddriver-85db2b507f3f69b32811c54a89d9ac7bbbc46121.tar.xz
driver-85db2b507f3f69b32811c54a89d9ac7bbbc46121.zip
feat(driver): commit primer
Diffstat (limited to 'crates/windows-kernel-build/src')
-rw-r--r--crates/windows-kernel-build/src/lib.rs92
1 files changed, 92 insertions, 0 deletions
diff --git a/crates/windows-kernel-build/src/lib.rs b/crates/windows-kernel-build/src/lib.rs
new file mode 100644
index 0000000..d312b82
--- /dev/null
+++ b/crates/windows-kernel-build/src/lib.rs
@@ -0,0 +1,92 @@
+use std::path::PathBuf;
+
+use thiserror::Error;
+use winreg::{enums::HKEY_LOCAL_MACHINE, RegKey};
+
+#[derive(Debug, Error)]
+pub enum Error {
+ #[error(transparent)]
+ IoError(#[from] std::io::Error),
+ #[error("cannot find the directory")]
+ DirectoryNotFound,
+}
+
+pub enum DirectoryType {
+ Include,
+ Library,
+}
+
+/// Retrieves the path to the Windows Kits directory. The default should be
+/// `C:\Program Files (x86)\Windows Kits\10`.
+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())
+}
+
+/// Retrieves the path to the kernel mode libraries. The path may look something
+/// like: `C:\Program Files (x86)\Windows Kits\10\lib\10.0.18362.0\km`.
+pub fn get_km_dir(dir_type: DirectoryType) -> Result<PathBuf, Error> {
+ // We first append lib to the path and read the directory..
+ let dir = get_windows_kits_dir()?
+ .join(match dir_type {
+ DirectoryType::Include => "Include",
+ DirectoryType::Library => "Lib",
+ })
+ .read_dir()?;
+
+ // In the lib directory we may have one or more directories named after the
+ // version of Windows, we will be looking for the highest version number.
+ 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)?;
+
+ // Finally append km to the path to get the path to the kernel mode libraries.
+ Ok(dir.join("km"))
+}
+
+pub fn build() -> Result<(), Error> {
+ // Get the path to the kernel libraries.
+ let dir = get_km_dir(DirectoryType::Library).unwrap();
+
+ // Append the architecture based on our target.
+ 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);
+
+ // Specify the link path.
+ println!("cargo:rustc-link-search=native={}", dir.to_str().unwrap());
+
+ // Ensure the right linker flags are passed for building a driver.
+ 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(())
+}