use std::collections::HashMap;
use std::env;
use std::fs;
use std::io;
use std::io::prelude::*;
use std::path::{Path, PathBuf};
use std::process;
#[cfg(feature = "toml")]
extern crate toml;
#[derive(PartialEq, Eq, Hash, Debug)]
pub enum VersionInfo {
FILEVERSION,
PRODUCTVERSION,
FILEOS,
FILETYPE,
FILESUBTYPE,
FILEFLAGSMASK,
FILEFLAGS,
}
#[derive(Debug)]
struct Icon {
path: String,
name_id: String,
}
#[derive(Debug)]
pub struct WindowsResource {
toolkit_path: PathBuf,
properties: HashMap<String, String>,
version_info: HashMap<VersionInfo, u64>,
rc_file: Option<String>,
icons: Vec<Icon>,
language: u16,
manifest: Option<String>,
manifest_file: Option<String>,
output_directory: String,
windres_path: String,
ar_path: String,
add_toolkit_include: bool,
append_rc_content: String,
}
#[allow(clippy::new_without_default)]
impl WindowsResource {
pub fn new() -> Self {
let mut props: HashMap<String, String> = HashMap::new();
let mut ver: HashMap<VersionInfo, u64> = HashMap::new();
props.insert(
"FileVersion".to_string(),
env::var("CARGO_PKG_VERSION").unwrap(),
);
props.insert(
"ProductVersion".to_string(),
env::var("CARGO_PKG_VERSION").unwrap(),
);
props.insert(
"ProductName".to_string(),
env::var("CARGO_PKG_NAME").unwrap(),
);
props.insert(
"FileDescription".to_string(),
env::var("CARGO_PKG_NAME").unwrap(),
);
#[cfg(feature = "toml")]
parse_cargo_toml(&mut props).unwrap();
let mut version = 0_u64;
version |= env::var("CARGO_PKG_VERSION_MAJOR")
.unwrap()
.parse()
.unwrap_or(0)
<< 48;
version |= env::var("CARGO_PKG_VERSION_MINOR")
.unwrap()
.parse()
.unwrap_or(0)
<< 32;
version |= env::var("CARGO_PKG_VERSION_PATCH")
.unwrap()
.parse()
.unwrap_or(0)
<< 16;
ver.insert(VersionInfo::FILEVERSION, version);
ver.insert(VersionInfo::PRODUCTVERSION, version);
ver.insert(VersionInfo::FILEOS, 0x00040004);
ver.insert(VersionInfo::FILETYPE, 1);
ver.insert(VersionInfo::FILESUBTYPE, 0);
ver.insert(VersionInfo::FILEFLAGSMASK, 0x3F);
ver.insert(VersionInfo::FILEFLAGS, 0);
let sdk = if cfg!(target_env = "msvc") {
match get_sdk() {
Ok(mut v) => v.pop().unwrap(),
Err(_) => PathBuf::new(),
}
} else if cfg!(windows) {
PathBuf::from("\\")
} else {
PathBuf::from("/")
};
let prefix = if let Ok(cross) = env::var("CROSS_COMPILE") {
cross
} else if env::var_os("HOST").unwrap() != env::var_os("TARGET").unwrap()
&& cfg!(not(target_env = "msvc"))
{
match env::var("TARGET").unwrap().as_str() {
"x86_64-pc-windows-gnu" => "x86_64-w64-mingw32-",
"i686-pc-windows-gnu" => "i686-w64-mingw32-",
"i586-pc-windows-gnu" => "i586-w64-mingw32-",
"aarch64-pc-windows-gnu" => "llvm-",
"x86_64-pc-windows-gnullvm"
| "i686-pc-windows-gnullvm"
| "aarch64-pc-windows-gnullvm" => "llvm-",
_ => {
println!(
"cargo:warning=unknown Windows target used for cross-compilation; \
invoking unprefixed windres"
);
""
}
}
.into()
} else {
"".into()
};
let windres_path = if let Ok(windres) = env::var("WINDRES") {
windres
} else {
format!("{}windres", prefix)
};
let ar_path = if let Ok(ar) = env::var("AR") {
ar
} else {
format!("{}ar", prefix)
};
WindowsResource {
toolkit_path: sdk,
properties: props,
version_info: ver,
rc_file: None,
icons: Vec::new(),
language: 0,
manifest: None,
manifest_file: None,
output_directory: env::var("OUT_DIR").unwrap_or_else(|_| ".".to_string()),
windres_path,
ar_path,
add_toolkit_include: false,
append_rc_content: String::new(),
}
}
pub fn set<'a>(&mut self, name: &'a str, value: &'a str) -> &mut Self {
self.properties.insert(name.to_string(), value.to_string());
self
}
pub fn set_toolkit_path<'a>(&mut self, path: &'a str) -> &mut Self {
self.toolkit_path = PathBuf::from(path);
self
}
pub fn set_language(&mut self, language: u16) -> &mut Self {
self.language = language;
self
}
pub fn set_icon<'a>(&mut self, path: &'a str) -> &mut Self {
const DEFAULT_APPLICATION_ICON_ID: &str = "1";
self.set_icon_with_id(path, DEFAULT_APPLICATION_ICON_ID)
}
pub fn set_icon_with_id<'a>(&mut self, path: &'a str, name_id: &'a str) -> &mut Self {
self.icons.push(Icon {
path: path.into(),
name_id: name_id.into(),
});
self
}
pub fn set_version_info(&mut self, field: VersionInfo, value: u64) -> &mut Self {
self.version_info.insert(field, value);
self
}
pub fn set_manifest<'a>(&mut self, manifest: &'a str) -> &mut Self {
self.manifest_file = None;
self.manifest = Some(manifest.to_string());
self
}
pub fn set_manifest_file<'a>(&mut self, file: &'a str) -> &mut Self {
self.manifest_file = Some(file.to_string());
self.manifest = None;
self
}
pub fn set_windres_path(&mut self, path: &str) -> &mut Self {
self.windres_path = path.to_string();
self
}
pub fn set_ar_path(&mut self, path: &str) -> &mut Self {
self.ar_path = path.to_string();
self
}
pub fn add_toolkit_include(&mut self, add: bool) -> &mut Self {
self.add_toolkit_include = add;
self
}
pub fn write_resource_file<P: AsRef<Path>>(&self, path: P) -> io::Result<()> {
let mut f = fs::File::create(path)?;
writeln!(f, "#pragma code_page(65001)")?;
writeln!(f, "1 VERSIONINFO")?;
for (k, v) in self.version_info.iter() {
match *k {
VersionInfo::FILEVERSION | VersionInfo::PRODUCTVERSION => writeln!(
f,
"{:?} {}, {}, {}, {}",
k,
(*v >> 48) as u16,
(*v >> 32) as u16,
(*v >> 16) as u16,
*v as u16
)?,
_ => writeln!(f, "{:?} {:#x}", k, v)?,
};
}
writeln!(f, "{{\nBLOCK \"StringFileInfo\"")?;
writeln!(f, "{{\nBLOCK \"{:04x}04b0\"\n{{", self.language)?;
for (k, v) in self.properties.iter() {
if !v.is_empty() {
writeln!(
f,
"VALUE \"{}\", \"{}\"",
escape_string(k),
escape_string(v)
)?;
}
}
writeln!(f, "}}\n}}")?;
writeln!(f, "BLOCK \"VarFileInfo\" {{")?;
writeln!(f, "VALUE \"Translation\", {:#x}, 0x04b0", self.language)?;
writeln!(f, "}}\n}}")?;
for icon in &self.icons {
writeln!(
f,
"{} ICON \"{}\"",
escape_string(&icon.name_id),
escape_string(&icon.path)
)?;
}
if let Some(e) = self.version_info.get(&VersionInfo::FILETYPE) {
if let Some(manf) = self.manifest.as_ref() {
writeln!(f, "{} 24", e)?;
writeln!(f, "{{")?;
for line in manf.lines() {
writeln!(f, "\" {} \"", escape_string(line.trim()))?;
}
writeln!(f, "}}")?;
} else if let Some(manf) = self.manifest_file.as_ref() {
writeln!(f, "{} 24 \"{}\"", e, escape_string(manf))?;
}
}
writeln!(f, "{}", self.append_rc_content)?;
Ok(())
}
pub fn set_resource_file<'a>(&mut self, path: &'a str) -> &mut Self {
self.rc_file = Some(path.to_string());
self
}
pub fn append_rc_content<'a>(&mut self, content: &'a str) -> &mut Self {
if !(self.append_rc_content.ends_with('\n') || self.append_rc_content.is_empty()) {
self.append_rc_content.push('\n');
}
self.append_rc_content.push_str(content);
self
}
pub fn set_output_directory<'a>(&mut self, path: &'a str) -> &mut Self {
self.output_directory = path.to_string();
self
}
fn compile_with_toolkit_gnu<'a>(&self, input: &'a str, output_dir: &'a str) -> io::Result<()> {
let output = PathBuf::from(output_dir).join("resource.o");
let input = PathBuf::from(input);
let status = process::Command::new(&self.windres_path)
.current_dir(&self.toolkit_path)
.arg(format!("-I{}", env::var("CARGO_MANIFEST_DIR").unwrap()))
.arg(format!("{}", input.display()))
.arg(format!("{}", output.display()))
.status()?;
if !status.success() {
return Err(io::Error::new(
io::ErrorKind::Other,
"Could not compile resource file",
));
}
let libname = PathBuf::from(output_dir).join("libresource.a");
let status = process::Command::new(&self.ar_path)
.current_dir(&self.toolkit_path)
.arg("rsc")
.arg(format!("{}", libname.display()))
.arg(format!("{}", output.display()))
.status()?;
if !status.success() {
return Err(io::Error::new(
io::ErrorKind::Other,
"Could not create static library for resource file",
));
}
println!("cargo:rustc-link-search=native={}", output_dir);
if version_check::is_min_version("1.61.0").unwrap_or(true) {
println!("cargo:rustc-link-lib=static:+whole-archive=resource");
} else {
println!("cargo:rustc-link-lib=static=resource");
}
Ok(())
}
pub fn compile(&self) -> io::Result<()> {
let output = PathBuf::from(&self.output_directory);
let rc = output.join("resource.rc");
if self.rc_file.is_none() {
self.write_resource_file(&rc)?;
}
let rc = if let Some(s) = self.rc_file.as_ref() {
s.clone()
} else {
rc.to_str().unwrap().to_string()
};
let target_env = std::env::var("CARGO_CFG_TARGET_ENV").unwrap();
match target_env.as_str() {
"gnu" => self.compile_with_toolkit_gnu(rc.as_str(), &self.output_directory),
"msvc" => self.compile_with_toolkit_msvc(rc.as_str(), &self.output_directory),
_ => Err(io::Error::new(
io::ErrorKind::Other,
"Can only compile resource file when target_env is \"gnu\" or \"msvc\"",
)),
}
}
fn compile_with_toolkit_msvc<'a>(&self, input: &'a str, output_dir: &'a str) -> io::Result<()> {
let rc_exe = if let Some(rc_path) = std::env::var_os("RC_PATH") {
PathBuf::from(rc_path)
} else if cfg!(unix) {
PathBuf::from("llvm-rc")
} else {
let rc_exe = PathBuf::from(&self.toolkit_path).join("rc.exe");
if !rc_exe.exists() {
if cfg!(target_arch = "x86_64") {
PathBuf::from(&self.toolkit_path).join(r"bin\x64\rc.exe")
} else {
PathBuf::from(&self.toolkit_path).join(r"bin\x86\rc.exe")
}
} else {
rc_exe
}
};
println!("Selected RC path: '{}'", rc_exe.display());
let output = PathBuf::from(output_dir).join("resource.lib");
let input = PathBuf::from(input);
let mut command = process::Command::new(&rc_exe);
let command = command.arg(format!("/I{}", env::var("CARGO_MANIFEST_DIR").unwrap()));
if self.add_toolkit_include {
let root = win_sdk_include_root(&rc_exe);
println!("Adding toolkit include: {}", root.display());
command.arg(format!("/I{}", root.join("um").display()));
command.arg(format!("/I{}", root.join("shared").display()));
}
let status = command
.arg(format!("/fo{}", output.display()))
.arg(format!("{}", input.display()))
.output()?;
println!(
"RC Output:\n{}\n------",
String::from_utf8_lossy(&status.stdout)
);
println!(
"RC Error:\n{}\n------",
String::from_utf8_lossy(&status.stderr)
);
if !status.status.success() {
return Err(io::Error::new(
io::ErrorKind::Other,
"Could not compile resource file",
));
}
println!("cargo:rustc-link-search=native={}", output_dir);
println!("cargo:rustc-link-lib=dylib=resource");
Ok(())
}
}
fn get_sdk() -> io::Result<Vec<PathBuf>> {
let output = process::Command::new("reg")
.arg("query")
.arg(r"HKLM\SOFTWARE\Microsoft\Windows Kits\Installed Roots")
.arg("/reg:32")
.output()?;
if !output.status.success() {
return Err(io::Error::new(
io::ErrorKind::Other,
format!(
"Querying the registry failed with error message:\n{}",
String::from_utf8(output.stderr)
.map_err(|e| io::Error::new(io::ErrorKind::Other, e.to_string()))?
),
));
}
let lines = String::from_utf8(output.stdout)
.map_err(|e| io::Error::new(io::ErrorKind::Other, e.to_string()))?;
let mut kits: Vec<PathBuf> = Vec::new();
let mut lines: Vec<&str> = lines.lines().collect();
lines.reverse();
for line in lines {
if line.trim().starts_with("KitsRoot") {
let kit: String = line
.chars()
.skip(line.find("REG_SZ").unwrap() + 6)
.skip_while(|c| c.is_whitespace())
.collect();
let p = PathBuf::from(&kit);
let rc = if cfg!(target_arch = "x86_64") {
p.join(r"bin\x64\rc.exe")
} else {
p.join(r"bin\x86\rc.exe")
};
if rc.exists() {
println!("{:?}", rc);
kits.push(rc.parent().unwrap().to_owned());
}
if let Ok(bin) = p.join("bin").read_dir() {
for e in bin.filter_map(|e| e.ok()) {
let p = if cfg!(target_arch = "x86_64") {
e.path().join(r"x64\rc.exe")
} else {
e.path().join(r"x86\rc.exe")
};
if p.exists() {
println!("{:?}", p);
kits.push(p.parent().unwrap().to_owned());
}
}
}
}
}
if kits.is_empty() {
return Err(io::Error::new(
io::ErrorKind::Other,
"Can not find Windows SDK",
));
}
Ok(kits)
}
#[cfg(feature = "toml")]
fn parse_cargo_toml(props: &mut HashMap<String, String>) -> io::Result<()> {
let cargo = Path::new(&env::var("CARGO_MANIFEST_DIR").unwrap()).join("Cargo.toml");
let mut f = fs::File::open(cargo)?;
let mut cargo_toml = String::new();
f.read_to_string(&mut cargo_toml)?;
if let Ok(ml) = cargo_toml.parse::<toml::Value>() {
if let Some(pkg) = ml.get("package") {
if let Some(pkg) = pkg.get("metadata") {
if let Some(pkg) = pkg.get("winresource") {
if let Some(pkg) = pkg.as_table() {
for (k, v) in pkg {
if let Some(v) = v.as_str() {
props.insert(k.clone(), v.to_string());
} else {
println!("package.metadata.winresource.{} is not a string", k);
}
}
} else {
println!("package.metadata.winresource is not a table");
}
} else {
println!("package.metadata.winresource does not exist");
}
} else {
println!("package.metadata does not exist");
}
} else {
println!("package does not exist");
}
} else {
println!("TOML parsing error")
}
Ok(())
}
fn escape_string(string: &str) -> String {
let mut escaped = String::new();
for chr in string.chars() {
match chr {
'"' => escaped.push_str("\"\""),
'\'' => escaped.push_str("\\'"),
'\\' => escaped.push_str("\\\\"),
'\n' => escaped.push_str("\\n"),
'\t' => escaped.push_str("\\t"),
'\r' => escaped.push_str("\\r"),
_ => escaped.push(chr),
};
}
escaped
}
fn win_sdk_include_root(path: &Path) -> PathBuf {
let mut tools_path = PathBuf::new();
let mut iter = path.iter();
while let Some(p) = iter.next() {
if p == "bin" {
let version = iter.next().unwrap();
tools_path.push("Include");
if version.to_string_lossy().starts_with("10.") {
tools_path.push(version);
}
break;
} else {
tools_path.push(p);
}
}
tools_path
}
#[cfg(test)]
mod tests {
use super::escape_string;
use super::win_sdk_include_root;
#[test]
fn string_escaping() {
assert_eq!(&escape_string(""), "");
assert_eq!(&escape_string("foo"), "foo");
assert_eq!(&escape_string(r#""Hello""#), r#"""Hello"""#);
assert_eq!(
&escape_string(r"C:\Program Files\Foobar"),
r"C:\\Program Files\\Foobar"
);
}
#[test]
fn toolkit_include_win10() {
use std::path::Path;
let res = win_sdk_include_root(Path::new(
r"C:\Program Files (x86)\Windows Kits\10\bin\10.0.17763.0\x64\rc.exe",
));
assert_eq!(
res.as_os_str(),
r"C:\Program Files (x86)\Windows Kits\10\Include\10.0.17763.0"
);
}
#[test]
fn toolkit_include_win8() {
use std::path::Path;
let res = win_sdk_include_root(Path::new(
r"C:\Program Files (x86)\Windows Kits\8.1\bin\x86\rc.exe",
));
assert_eq!(
res.as_os_str(),
r"C:\Program Files (x86)\Windows Kits\8.1\Include"
);
}
}