#![allow(clippy::upper_case_acronyms)]
use std::{
env,
ffi::{OsStr, OsString},
ops::Deref,
path::PathBuf,
process::Command,
sync::Arc,
};
use crate::Tool;
use crate::ToolFamily;
const MSVC_FAMILY: ToolFamily = ToolFamily::Msvc { clang_cl: false };
#[derive(Copy, Clone, PartialEq, Eq)]
enum TargetArch {
X86,
X64,
Arm,
Arm64,
Arm64ec,
}
impl TargetArch {
fn new(arch: &str) -> Option<Self> {
match arch {
"x64" | "x86_64" => Some(Self::X64),
"arm64" | "aarch64" => Some(Self::Arm64),
"arm64ec" => Some(Self::Arm64ec),
"x86" | "i686" | "i586" => Some(Self::X86),
"arm" | "thumbv7a" => Some(Self::Arm),
_ => None,
}
}
#[cfg(windows)]
fn as_vs_arch(&self) -> &'static str {
match self {
Self::X64 => "x64",
Self::Arm64 | Self::Arm64ec => "arm64",
Self::X86 => "x86",
Self::Arm => "arm",
}
}
}
pub(crate) enum Env {
Owned(OsString),
Arced(Arc<OsStr>),
}
impl AsRef<OsStr> for Env {
fn as_ref(&self) -> &OsStr {
self.deref()
}
}
impl Deref for Env {
type Target = OsStr;
fn deref(&self) -> &Self::Target {
match self {
Env::Owned(os_str) => os_str,
Env::Arced(os_str) => os_str,
}
}
}
impl From<Env> for PathBuf {
fn from(env: Env) -> Self {
match env {
Env::Owned(os_str) => PathBuf::from(os_str),
Env::Arced(os_str) => PathBuf::from(os_str.deref()),
}
}
}
pub(crate) trait EnvGetter {
fn get_env(&self, name: &'static str) -> Option<Env>;
}
struct StdEnvGetter;
impl EnvGetter for StdEnvGetter {
#[allow(clippy::disallowed_methods)]
fn get_env(&self, name: &'static str) -> Option<Env> {
env::var_os(name).map(Env::Owned)
}
}
pub fn find(arch_or_target: &str, tool: &str) -> Option<Command> {
find_tool(arch_or_target, tool).map(|c| c.to_command())
}
pub fn find_tool(arch_or_target: &str, tool: &str) -> Option<Tool> {
let full_arch = if let Some((full_arch, rest)) = arch_or_target.split_once("-") {
if !rest.contains("msvc") {
return None;
}
full_arch
} else {
arch_or_target
};
find_tool_inner(full_arch, tool, &StdEnvGetter)
}
pub(crate) fn find_tool_inner(
full_arch: &str,
tool: &str,
env_getter: &dyn EnvGetter,
) -> Option<Tool> {
let target = TargetArch::new(full_arch)?;
if tool.contains("msbuild") {
return impl_::find_msbuild(target, env_getter);
}
if tool.contains("devenv") {
return impl_::find_devenv(target, env_getter);
}
impl_::find_msvc_environment(tool, target, env_getter)
.or_else(|| impl_::find_msvc_15plus(tool, target, env_getter))
.or_else(|| impl_::find_msvc_14(tool, target, env_getter))
}
#[derive(Debug, PartialEq, Eq, Copy, Clone)]
#[non_exhaustive]
pub enum VsVers {
#[deprecated(
note = "Visual Studio 12 is no longer supported. cc will never return this value."
)]
Vs12,
Vs14,
Vs15,
Vs16,
Vs17,
}
#[allow(clippy::disallowed_methods)]
pub fn find_vs_version() -> Result<VsVers, String> {
fn has_msbuild_version(version: &str) -> bool {
impl_::has_msbuild_version(version, &StdEnvGetter)
}
match std::env::var("VisualStudioVersion") {
Ok(version) => match &version[..] {
"17.0" => Ok(VsVers::Vs17),
"16.0" => Ok(VsVers::Vs16),
"15.0" => Ok(VsVers::Vs15),
"14.0" => Ok(VsVers::Vs14),
vers => Err(format!(
"\n\n\
unsupported or unknown VisualStudio version: {}\n\
if another version is installed consider running \
the appropriate vcvars script before building this \
crate\n\
",
vers
)),
},
_ => {
if has_msbuild_version("17.0") {
Ok(VsVers::Vs17)
} else if has_msbuild_version("16.0") {
Ok(VsVers::Vs16)
} else if has_msbuild_version("15.0") {
Ok(VsVers::Vs15)
} else if has_msbuild_version("14.0") {
Ok(VsVers::Vs14)
} else {
Err("\n\n\
couldn't determine visual studio generator\n\
if VisualStudio is installed, however, consider \
running the appropriate vcvars script before building \
this crate\n\
"
.to_string())
}
}
}
}
#[cfg(windows)]
mod impl_ {
use crate::windows::com;
use crate::windows::registry::{RegistryKey, LOCAL_MACHINE};
use crate::windows::setup_config::SetupConfiguration;
use crate::windows::vs_instances::{VsInstances, VswhereInstance};
use crate::windows::windows_sys::{
GetMachineTypeAttributes, GetProcAddress, LoadLibraryA, UserEnabled, HMODULE,
IMAGE_FILE_MACHINE_AMD64, MACHINE_ATTRIBUTES, S_OK,
};
use std::convert::TryFrom;
use std::env;
use std::ffi::OsString;
use std::fs::File;
use std::io::Read;
use std::iter;
use std::mem;
use std::path::{Path, PathBuf};
use std::process::Command;
use std::str::FromStr;
use std::sync::atomic::{AtomicBool, Ordering};
use std::sync::Once;
use super::{EnvGetter, TargetArch, MSVC_FAMILY};
use crate::Tool;
struct MsvcTool {
tool: PathBuf,
libs: Vec<PathBuf>,
path: Vec<PathBuf>,
include: Vec<PathBuf>,
}
struct LibraryHandle(HMODULE);
impl LibraryHandle {
fn new(name: &[u8]) -> Option<Self> {
let handle = unsafe { LoadLibraryA(name.as_ptr() as _) };
(!handle.is_null()).then_some(Self(handle))
}
unsafe fn get_proc_address<F>(&self, name: &[u8]) -> Option<F> {
let symbol = GetProcAddress(self.0, name.as_ptr() as _);
symbol.map(|symbol| mem::transmute_copy(&symbol))
}
}
type GetMachineTypeAttributesFuncType =
unsafe extern "system" fn(u16, *mut MACHINE_ATTRIBUTES) -> i32;
const _: () = {
let _: GetMachineTypeAttributesFuncType = GetMachineTypeAttributes;
};
fn is_amd64_emulation_supported_inner() -> Option<bool> {
let kernel32 = LibraryHandle::new(b"kernel32.dll\0")?;
let get_machine_type_attributes = unsafe {
kernel32
.get_proc_address::<GetMachineTypeAttributesFuncType>(b"GetMachineTypeAttributes\0")
}?;
let mut attributes = Default::default();
if unsafe { get_machine_type_attributes(IMAGE_FILE_MACHINE_AMD64, &mut attributes) } == S_OK
{
Some((attributes & UserEnabled) != 0)
} else {
Some(false)
}
}
fn is_amd64_emulation_supported() -> bool {
static LOAD_VALUE: Once = Once::new();
static IS_SUPPORTED: AtomicBool = AtomicBool::new(false);
LOAD_VALUE.call_once(|| {
IS_SUPPORTED.store(
is_amd64_emulation_supported_inner().unwrap_or(false),
Ordering::Relaxed,
);
});
IS_SUPPORTED.load(Ordering::Relaxed)
}
impl MsvcTool {
fn new(tool: PathBuf) -> MsvcTool {
MsvcTool {
tool,
libs: Vec::new(),
path: Vec::new(),
include: Vec::new(),
}
}
fn into_tool(self, env_getter: &dyn EnvGetter) -> Tool {
let MsvcTool {
tool,
libs,
path,
include,
} = self;
let mut tool = Tool::with_family(tool, MSVC_FAMILY);
add_env(&mut tool, "LIB", libs, env_getter);
add_env(&mut tool, "PATH", path, env_getter);
add_env(&mut tool, "INCLUDE", include, env_getter);
tool
}
}
fn is_vscmd_target(target: TargetArch, env_getter: &dyn EnvGetter) -> Option<bool> {
is_vscmd_target_env(target, env_getter).or_else(|| is_vscmd_target_cl(target, env_getter))
}
fn is_vscmd_target_env(target: TargetArch, env_getter: &dyn EnvGetter) -> Option<bool> {
let vscmd_arch = env_getter.get_env("VSCMD_ARG_TGT_ARCH")?;
Some(target.as_vs_arch() == vscmd_arch.as_ref())
}
fn is_vscmd_target_cl(target: TargetArch, env_getter: &dyn EnvGetter) -> Option<bool> {
let cmd_target = vscmd_target_cl(env_getter)?;
Some(target.as_vs_arch() == cmd_target)
}
fn vscmd_target_cl(env_getter: &dyn EnvGetter) -> Option<&'static str> {
let cl_exe = env_getter.get_env("PATH").and_then(|path| {
env::split_paths(&path)
.map(|p| p.join("cl.exe"))
.find(|p| p.exists())
})?;
let mut cl = Command::new(cl_exe);
cl.stderr(std::process::Stdio::piped())
.stdout(std::process::Stdio::null());
let out = cl.output().ok()?;
let cl_arch = out
.stderr
.split(|&b| b == b'\n' || b == b'\r')
.next()?
.rsplit(|&b| b == b' ')
.next()?;
match cl_arch {
b"x64" => Some("x64"),
b"x86" => Some("x86"),
b"ARM64" => Some("arm64"),
b"ARM" => Some("arm"),
_ => None,
}
}
pub(super) fn find_msvc_environment(
tool: &str,
target: TargetArch,
env_getter: &dyn EnvGetter,
) -> Option<Tool> {
if env_getter.get_env("VCINSTALLDIR").is_none()
&& env_getter.get_env("VSTEL_MSBuildProjectFullPath").is_none()
{
return None;
}
if is_vscmd_target(target, env_getter) == Some(false) {
let vs_install_dir: PathBuf = env_getter.get_env("VSINSTALLDIR")?.into();
tool_from_vs15plus_instance(tool, target, &vs_install_dir, env_getter)
} else {
env_getter
.get_env("PATH")
.and_then(|path| {
env::split_paths(&path)
.map(|p| p.join(tool))
.find(|p| p.exists())
})
.map(|path| Tool::with_family(path, MSVC_FAMILY))
}
}
fn find_msbuild_vs17(target: TargetArch, env_getter: &dyn EnvGetter) -> Option<Tool> {
find_tool_in_vs16plus_path(r"MSBuild\Current\Bin\MSBuild.exe", target, "17", env_getter)
}
#[allow(bare_trait_objects)]
fn vs16plus_instances(
target: TargetArch,
version: &'static str,
env_getter: &dyn EnvGetter,
) -> Box<Iterator<Item = PathBuf>> {
let instances = if let Some(instances) = vs15plus_instances(target, env_getter) {
instances
} else {
return Box::new(iter::empty());
};
Box::new(instances.into_iter().filter_map(move |instance| {
let installation_name = instance.installation_name()?;
if installation_name.starts_with(&format!("VisualStudio/{}.", version))
|| installation_name.starts_with(&format!("VisualStudioPreview/{}.", version))
{
Some(instance.installation_path()?)
} else {
None
}
}))
}
fn find_tool_in_vs16plus_path(
tool: &str,
target: TargetArch,
version: &'static str,
env_getter: &dyn EnvGetter,
) -> Option<Tool> {
vs16plus_instances(target, version, env_getter)
.filter_map(|path| {
let path = path.join(tool);
if !path.is_file() {
return None;
}
let mut tool = Tool::with_family(path, MSVC_FAMILY);
if target == TargetArch::X64 {
tool.env.push(("Platform".into(), "X64".into()));
}
if matches!(target, TargetArch::Arm64 | TargetArch::Arm64ec) {
tool.env.push(("Platform".into(), "ARM64".into()));
}
Some(tool)
})
.next()
}
fn find_msbuild_vs16(target: TargetArch, env_getter: &dyn EnvGetter) -> Option<Tool> {
find_tool_in_vs16plus_path(r"MSBuild\Current\Bin\MSBuild.exe", target, "16", env_getter)
}
fn vs15plus_instances(target: TargetArch, env_getter: &dyn EnvGetter) -> Option<VsInstances> {
vs15plus_instances_using_com()
.or_else(|| vs15plus_instances_using_vswhere(target, env_getter))
}
fn vs15plus_instances_using_com() -> Option<VsInstances> {
com::initialize().ok()?;
let config = SetupConfiguration::new().ok()?;
let enum_setup_instances = config.enum_all_instances().ok()?;
Some(VsInstances::ComBased(enum_setup_instances))
}
fn vs15plus_instances_using_vswhere(
target: TargetArch,
env_getter: &dyn EnvGetter,
) -> Option<VsInstances> {
let program_files_path = env_getter
.get_env("ProgramFiles(x86)")
.or_else(|| env_getter.get_env("ProgramFiles"))?;
let program_files_path = Path::new(program_files_path.as_ref());
let vswhere_path =
program_files_path.join(r"Microsoft Visual Studio\Installer\vswhere.exe");
if !vswhere_path.exists() {
return None;
}
let tools_arch = match target {
TargetArch::X86 | TargetArch::X64 => Some("x86.x64"),
TargetArch::Arm => Some("ARM"),
TargetArch::Arm64 | TargetArch::Arm64ec => Some("ARM64"),
};
let vswhere_output = Command::new(vswhere_path)
.args([
"-latest",
"-products",
"*",
"-requires",
&format!("Microsoft.VisualStudio.Component.VC.Tools.{}", tools_arch?),
"-format",
"text",
"-nologo",
])
.stderr(std::process::Stdio::inherit())
.output()
.ok()?;
let vs_instances =
VsInstances::VswhereBased(VswhereInstance::try_from(&vswhere_output.stdout).ok()?);
Some(vs_instances)
}
fn parse_version(version: &str) -> Option<Vec<u16>> {
version
.split('.')
.map(|chunk| u16::from_str(chunk).ok())
.collect()
}
pub(super) fn find_msvc_15plus(
tool: &str,
target: TargetArch,
env_getter: &dyn EnvGetter,
) -> Option<Tool> {
let iter = vs15plus_instances(target, env_getter)?;
iter.into_iter()
.filter_map(|instance| {
let version = parse_version(&instance.installation_version()?)?;
let instance_path = instance.installation_path()?;
let tool = tool_from_vs15plus_instance(tool, target, &instance_path, env_getter)?;
Some((version, tool))
})
.max_by(|(a_version, _), (b_version, _)| a_version.cmp(b_version))
.map(|(_version, tool)| tool)
}
fn find_tool_in_vs15_path(
tool: &str,
target: TargetArch,
env_getter: &dyn EnvGetter,
) -> Option<Tool> {
let mut path = match vs15plus_instances(target, env_getter) {
Some(instances) => instances
.into_iter()
.filter_map(|instance| instance.installation_path())
.map(|path| path.join(tool))
.find(|path| path.is_file()),
None => None,
};
if path.is_none() {
let key = r"SOFTWARE\WOW6432Node\Microsoft\VisualStudio\SxS\VS7";
path = LOCAL_MACHINE
.open(key.as_ref())
.ok()
.and_then(|key| key.query_str("15.0").ok())
.map(|path| PathBuf::from(path).join(tool))
.and_then(|path| if path.is_file() { Some(path) } else { None });
}
path.map(|path| {
let mut tool = Tool::with_family(path, MSVC_FAMILY);
if target == TargetArch::X64 {
tool.env.push(("Platform".into(), "X64".into()));
} else if matches!(target, TargetArch::Arm64 | TargetArch::Arm64ec) {
tool.env.push(("Platform".into(), "ARM64".into()));
}
tool
})
}
fn tool_from_vs15plus_instance(
tool: &str,
target: TargetArch,
instance_path: &Path,
env_getter: &dyn EnvGetter,
) -> Option<Tool> {
let (root_path, bin_path, host_dylib_path, lib_path, alt_lib_path, include_path) =
vs15plus_vc_paths(target, instance_path, env_getter)?;
let tool_path = bin_path.join(tool);
if !tool_path.exists() {
return None;
};
let mut tool = MsvcTool::new(tool_path);
tool.path.push(bin_path.clone());
tool.path.push(host_dylib_path);
if let Some(alt_lib_path) = alt_lib_path {
tool.libs.push(alt_lib_path);
}
tool.libs.push(lib_path);
tool.include.push(include_path);
if let Some((atl_lib_path, atl_include_path)) = atl_paths(target, &root_path) {
tool.libs.push(atl_lib_path);
tool.include.push(atl_include_path);
}
add_sdks(&mut tool, target, env_getter)?;
Some(tool.into_tool(env_getter))
}
fn vs15plus_vc_paths(
target_arch: TargetArch,
instance_path: &Path,
env_getter: &dyn EnvGetter,
) -> Option<(PathBuf, PathBuf, PathBuf, PathBuf, Option<PathBuf>, PathBuf)> {
let version = vs15plus_vc_read_version(instance_path)?;
let hosts = match host_arch() {
X86 => &["X86"],
X86_64 => &["X64"],
AARCH64 => {
if is_amd64_emulation_supported() {
&["ARM64", "X64", "X86"][..]
} else {
&["ARM64", "X86"]
}
}
_ => return None,
};
let target_dir = target_arch.as_vs_arch();
let path = instance_path.join(r"VC\Tools\MSVC").join(version);
let (host_path, host) = hosts.iter().find_map(|&x| {
let candidate = path.join("bin").join(format!("Host{}", x));
if candidate.join(target_dir).exists() {
Some((candidate, x))
} else {
None
}
})?;
let bin_path = host_path.join(target_dir);
let host_dylib_path = host_path.join(host.to_lowercase());
let lib_fragment = if use_spectre_mitigated_libs(env_getter) {
r"lib\spectre"
} else {
"lib"
};
let lib_path = path.join(lib_fragment).join(target_dir);
let alt_lib_path =
(target_arch == TargetArch::Arm64ec).then(|| path.join(lib_fragment).join("arm64ec"));
let include_path = path.join("include");
Some((
path,
bin_path,
host_dylib_path,
lib_path,
alt_lib_path,
include_path,
))
}
fn vs15plus_vc_read_version(dir: &Path) -> Option<String> {
let mut version_path: PathBuf =
dir.join(r"VC\Auxiliary\Build\Microsoft.VCToolsVersion.default.txt");
let mut version_file = if let Ok(f) = File::open(&version_path) {
f
} else {
let mut version_file = String::new();
version_path.pop();
for file in version_path.read_dir().ok()? {
let name = file.ok()?.file_name();
let name = name.to_str()?;
if name.starts_with("Microsoft.VCToolsVersion.v")
&& name.ends_with(".default.txt")
&& name > &version_file
{
version_file.replace_range(.., name);
}
}
if version_file.is_empty() {
return None;
}
version_path.push(version_file);
File::open(version_path).ok()?
};
let mut version = String::new();
version_file.read_to_string(&mut version).ok()?;
version.truncate(version.trim_end().len());
Some(version)
}
fn use_spectre_mitigated_libs(env_getter: &dyn EnvGetter) -> bool {
env_getter
.get_env("VSCMD_ARG_VCVARS_SPECTRE")
.map(|env| env.as_ref() == "spectre")
.unwrap_or_default()
}
fn atl_paths(target: TargetArch, path: &Path) -> Option<(PathBuf, PathBuf)> {
let atl_path = path.join("atlmfc");
let sub = target.as_vs_arch();
if atl_path.exists() {
Some((atl_path.join("lib").join(sub), atl_path.join("include")))
} else {
None
}
}
pub(super) fn find_msvc_14(
tool: &str,
target: TargetArch,
env_getter: &dyn EnvGetter,
) -> Option<Tool> {
let vcdir = get_vc_dir("14.0")?;
let mut tool = get_tool(tool, &vcdir, target)?;
add_sdks(&mut tool, target, env_getter)?;
Some(tool.into_tool(env_getter))
}
fn add_sdks(tool: &mut MsvcTool, target: TargetArch, env_getter: &dyn EnvGetter) -> Option<()> {
let sub = target.as_vs_arch();
let (ucrt, ucrt_version) = get_ucrt_dir()?;
let host = match host_arch() {
X86 => "x86",
X86_64 => "x64",
AARCH64 => "arm64",
_ => return None,
};
tool.path
.push(ucrt.join("bin").join(&ucrt_version).join(host));
let ucrt_include = ucrt.join("include").join(&ucrt_version);
tool.include.push(ucrt_include.join("ucrt"));
let ucrt_lib = ucrt.join("lib").join(&ucrt_version);
tool.libs.push(ucrt_lib.join("ucrt").join(sub));
if let Some((sdk, version)) = get_sdk10_dir(env_getter) {
tool.path.push(sdk.join("bin").join(host));
let sdk_lib = sdk.join("lib").join(&version);
tool.libs.push(sdk_lib.join("um").join(sub));
let sdk_include = sdk.join("include").join(&version);
tool.include.push(sdk_include.join("um"));
tool.include.push(sdk_include.join("cppwinrt"));
tool.include.push(sdk_include.join("winrt"));
tool.include.push(sdk_include.join("shared"));
} else if let Some(sdk) = get_sdk81_dir() {
tool.path.push(sdk.join("bin").join(host));
let sdk_lib = sdk.join("lib").join("winv6.3");
tool.libs.push(sdk_lib.join("um").join(sub));
let sdk_include = sdk.join("include");
tool.include.push(sdk_include.join("um"));
tool.include.push(sdk_include.join("winrt"));
tool.include.push(sdk_include.join("shared"));
}
Some(())
}
fn add_env(
tool: &mut Tool,
env: &'static str,
paths: Vec<PathBuf>,
env_getter: &dyn EnvGetter,
) {
let prev = env_getter.get_env(env);
let prev = prev.as_ref().map(AsRef::as_ref).unwrap_or_default();
let prev = env::split_paths(&prev);
let new = paths.into_iter().chain(prev);
tool.env
.push((env.to_string().into(), env::join_paths(new).unwrap()));
}
fn get_tool(tool: &str, path: &Path, target: TargetArch) -> Option<MsvcTool> {
bin_subdir(target)
.into_iter()
.map(|(sub, host)| {
(
path.join("bin").join(sub).join(tool),
path.join("bin").join(host),
)
})
.filter(|(path, _)| path.is_file())
.map(|(path, host)| {
let mut tool = MsvcTool::new(path);
tool.path.push(host);
tool
})
.filter_map(|mut tool| {
let sub = vc_lib_subdir(target);
tool.libs.push(path.join("lib").join(sub));
tool.include.push(path.join("include"));
let atlmfc_path = path.join("atlmfc");
if atlmfc_path.exists() {
tool.libs.push(atlmfc_path.join("lib").join(sub));
tool.include.push(atlmfc_path.join("include"));
}
Some(tool)
})
.next()
}
fn get_vc_dir(ver: &str) -> Option<PathBuf> {
let key = r"SOFTWARE\Microsoft\VisualStudio\SxS\VC7";
let key = LOCAL_MACHINE.open(key.as_ref()).ok()?;
let path = key.query_str(ver).ok()?;
Some(path.into())
}
fn get_ucrt_dir() -> Option<(PathBuf, String)> {
let key = r"SOFTWARE\Microsoft\Windows Kits\Installed Roots";
let key = LOCAL_MACHINE.open(key.as_ref()).ok()?;
let root = key.query_str("KitsRoot10").ok()?;
let readdir = Path::new(&root).join("lib").read_dir().ok()?;
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("ucrt").is_dir())
.unwrap_or(false)
})
.max()?;
let version = max_libdir.components().last().unwrap();
let version = version.as_os_str().to_str().unwrap().to_string();
Some((root.into(), version))
}
fn get_sdk10_dir(env_getter: &dyn EnvGetter) -> Option<(PathBuf, String)> {
if let (Some(root), Some(version)) = (
env_getter.get_env("WindowsSdkDir"),
env_getter
.get_env("WindowsSDKVersion")
.as_ref()
.and_then(|version| version.as_ref().to_str()),
) {
return Some((
PathBuf::from(root),
version.trim_end_matches('\\').to_string(),
));
}
let key = r"SOFTWARE\Microsoft\Microsoft SDKs\Windows\v10.0";
let key = LOCAL_MACHINE.open(key.as_ref()).ok()?;
let root = key.query_str("InstallationFolder").ok()?;
let readdir = Path::new(&root).join("lib").read_dir().ok()?;
let mut dirs = readdir
.filter_map(|dir| dir.ok())
.map(|dir| dir.path())
.collect::<Vec<_>>();
dirs.sort();
let dir = dirs
.into_iter()
.rev()
.find(|dir| dir.join("um").join("x64").join("kernel32.lib").is_file())?;
let version = dir.components().last().unwrap();
let version = version.as_os_str().to_str().unwrap().to_string();
Some((root.into(), version))
}
fn get_sdk81_dir() -> Option<PathBuf> {
let key = r"SOFTWARE\Microsoft\Microsoft SDKs\Windows\v8.1";
let key = LOCAL_MACHINE.open(key.as_ref()).ok()?;
let root = key.query_str("InstallationFolder").ok()?;
Some(root.into())
}
const PROCESSOR_ARCHITECTURE_INTEL: u16 = 0;
const PROCESSOR_ARCHITECTURE_AMD64: u16 = 9;
const PROCESSOR_ARCHITECTURE_ARM64: u16 = 12;
const X86: u16 = PROCESSOR_ARCHITECTURE_INTEL;
const X86_64: u16 = PROCESSOR_ARCHITECTURE_AMD64;
const AARCH64: u16 = PROCESSOR_ARCHITECTURE_ARM64;
fn bin_subdir(target: TargetArch) -> Vec<(&'static str, &'static str)> {
match (target, host_arch()) {
(TargetArch::X86, X86) => vec![("", "")],
(TargetArch::X86, X86_64) => vec![("amd64_x86", "amd64"), ("", "")],
(TargetArch::X64, X86) => vec![("x86_amd64", "")],
(TargetArch::X64, X86_64) => vec![("amd64", "amd64"), ("x86_amd64", "")],
(TargetArch::Arm, X86) => vec![("x86_arm", "")],
(TargetArch::Arm, X86_64) => vec![("amd64_arm", "amd64"), ("x86_arm", "")],
_ => vec![],
}
}
fn vc_lib_subdir(target: TargetArch) -> &'static str {
match target {
TargetArch::X86 => "",
TargetArch::X64 => "amd64",
TargetArch::Arm => "arm",
TargetArch::Arm64 | TargetArch::Arm64ec => "arm64",
}
}
#[allow(bad_style)]
fn host_arch() -> u16 {
type DWORD = u32;
type WORD = u16;
type LPVOID = *mut u8;
type DWORD_PTR = usize;
#[repr(C)]
struct SYSTEM_INFO {
wProcessorArchitecture: WORD,
_wReserved: WORD,
_dwPageSize: DWORD,
_lpMinimumApplicationAddress: LPVOID,
_lpMaximumApplicationAddress: LPVOID,
_dwActiveProcessorMask: DWORD_PTR,
_dwNumberOfProcessors: DWORD,
_dwProcessorType: DWORD,
_dwAllocationGranularity: DWORD,
_wProcessorLevel: WORD,
_wProcessorRevision: WORD,
}
extern "system" {
fn GetNativeSystemInfo(lpSystemInfo: *mut SYSTEM_INFO);
}
unsafe {
let mut info = mem::zeroed();
GetNativeSystemInfo(&mut info);
info.wProcessorArchitecture
}
}
fn max_version(key: &RegistryKey) -> Option<(OsString, RegistryKey)> {
let mut max_vers = 0;
let mut max_key = None;
for subkey in key.iter().filter_map(|k| k.ok()) {
let val = subkey
.to_str()
.and_then(|s| s.trim_start_matches('v').replace('.', "").parse().ok());
let val = match val {
Some(s) => s,
None => continue,
};
if val > max_vers {
if let Ok(k) = key.open(&subkey) {
max_vers = val;
max_key = Some((subkey, k));
}
}
}
max_key
}
#[inline(always)]
pub(super) fn has_msbuild_version(version: &str, env_getter: &dyn EnvGetter) -> bool {
match version {
"17.0" => {
find_msbuild_vs17(TargetArch::X64, env_getter).is_some()
|| find_msbuild_vs17(TargetArch::X86, env_getter).is_some()
|| find_msbuild_vs17(TargetArch::Arm64, env_getter).is_some()
}
"16.0" => {
find_msbuild_vs16(TargetArch::X64, env_getter).is_some()
|| find_msbuild_vs16(TargetArch::X86, env_getter).is_some()
|| find_msbuild_vs16(TargetArch::Arm64, env_getter).is_some()
}
"15.0" => {
find_msbuild_vs15(TargetArch::X64, env_getter).is_some()
|| find_msbuild_vs15(TargetArch::X86, env_getter).is_some()
|| find_msbuild_vs15(TargetArch::Arm64, env_getter).is_some()
}
"14.0" => LOCAL_MACHINE
.open(&OsString::from(format!(
"SOFTWARE\\Microsoft\\MSBuild\\ToolsVersions\\{}",
version
)))
.is_ok(),
_ => false,
}
}
pub(super) fn find_devenv(target: TargetArch, env_getter: &dyn EnvGetter) -> Option<Tool> {
find_devenv_vs15(target, env_getter)
}
fn find_devenv_vs15(target: TargetArch, env_getter: &dyn EnvGetter) -> Option<Tool> {
find_tool_in_vs15_path(r"Common7\IDE\devenv.exe", target, env_getter)
}
pub(super) fn find_msbuild(target: TargetArch, env_getter: &dyn EnvGetter) -> Option<Tool> {
if let Some(r) = find_msbuild_vs17(target, env_getter) {
Some(r)
} else if let Some(r) = find_msbuild_vs16(target, env_getter) {
return Some(r);
} else if let Some(r) = find_msbuild_vs15(target, env_getter) {
return Some(r);
} else {
find_old_msbuild(target)
}
}
fn find_msbuild_vs15(target: TargetArch, env_getter: &dyn EnvGetter) -> Option<Tool> {
find_tool_in_vs15_path(r"MSBuild\15.0\Bin\MSBuild.exe", target, env_getter)
}
fn find_old_msbuild(target: TargetArch) -> Option<Tool> {
let key = r"SOFTWARE\Microsoft\MSBuild\ToolsVersions";
LOCAL_MACHINE
.open(key.as_ref())
.ok()
.and_then(|key| {
max_version(&key).and_then(|(_vers, key)| key.query_str("MSBuildToolsPath").ok())
})
.map(|path| {
let mut path = PathBuf::from(path);
path.push("MSBuild.exe");
let mut tool = Tool::with_family(path, MSVC_FAMILY);
if target == TargetArch::X64 {
tool.env.push(("Platform".into(), "X64".into()));
}
tool
})
}
}
#[cfg(not(windows))]
mod impl_ {
use std::{env, ffi::OsStr};
use super::{EnvGetter, TargetArch, MSVC_FAMILY};
use crate::Tool;
#[inline(always)]
pub(super) fn find_msbuild(_target: TargetArch, _: &dyn EnvGetter) -> Option<Tool> {
None
}
#[inline(always)]
pub(super) fn find_devenv(_target: TargetArch, _: &dyn EnvGetter) -> Option<Tool> {
None
}
pub(super) fn find_msvc_environment(
tool: &str,
_target: TargetArch,
env_getter: &dyn EnvGetter,
) -> Option<Tool> {
let vc_install_dir = env_getter.get_env("VCINSTALLDIR")?;
let vs_install_dir = env_getter.get_env("VSINSTALLDIR")?;
let get_tool = |install_dir: &OsStr| {
env::split_paths(install_dir)
.map(|p| p.join(tool))
.find(|p| p.exists())
.map(|path| Tool::with_family(path, MSVC_FAMILY))
};
get_tool(vc_install_dir.as_ref())
.or_else(|| get_tool(vs_install_dir.as_ref()))
.or_else(|| {
env_getter
.get_env("PATH")
.as_ref()
.map(|path| path.as_ref())
.and_then(get_tool)
})
}
#[inline(always)]
pub(super) fn find_msvc_15plus(
_tool: &str,
_target: TargetArch,
_: &dyn EnvGetter,
) -> Option<Tool> {
None
}
#[inline(always)]
pub(super) fn find_msvc_14(
_tool: &str,
_target: TargetArch,
_: &dyn EnvGetter,
) -> Option<Tool> {
None
}
#[inline(always)]
pub(super) fn has_msbuild_version(_version: &str, _: &dyn EnvGetter) -> bool {
false
}
}