2017-01-28 04:49:09 +08:00
|
|
|
#include "pch.h"
|
2017-04-28 09:08:52 +08:00
|
|
|
|
|
|
|
#include "PackageSpec.h"
|
2017-04-04 07:29:11 +08:00
|
|
|
#include "VcpkgPaths.h"
|
2016-09-19 11:50:08 +08:00
|
|
|
#include "metrics.h"
|
2017-04-01 08:40:08 +08:00
|
|
|
#include "vcpkg_Files.h"
|
2017-04-28 09:08:52 +08:00
|
|
|
#include "vcpkg_System.h"
|
2017-04-01 18:30:52 +08:00
|
|
|
#include "vcpkg_Util.h"
|
2017-04-28 09:08:52 +08:00
|
|
|
#include "vcpkg_expected.h"
|
2016-09-19 11:50:08 +08:00
|
|
|
|
|
|
|
namespace vcpkg
|
|
|
|
{
|
2017-04-28 09:08:52 +08:00
|
|
|
static bool exists_and_has_equal_or_greater_version(const std::wstring& version_cmd,
|
|
|
|
const std::array<int, 3>& expected_version)
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
|
|
|
static const std::regex re(R"###((\d+)\.(\d+)\.(\d+))###");
|
|
|
|
|
2017-03-14 07:17:47 +08:00
|
|
|
auto rc = System::cmd_execute_and_capture_output(Strings::wformat(LR"(%s)", version_cmd));
|
2017-03-11 08:30:44 +08:00
|
|
|
if (rc.exit_code != 0)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::match_results<std::string::const_iterator> match;
|
|
|
|
auto found = std::regex_search(rc.output, match, re);
|
|
|
|
if (!found)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int d1 = atoi(match[1].str().c_str());
|
|
|
|
int d2 = atoi(match[2].str().c_str());
|
|
|
|
int d3 = atoi(match[3].str().c_str());
|
2017-04-28 09:08:52 +08:00
|
|
|
if (d1 > expected_version[0] || (d1 == expected_version[0] && d2 > expected_version[1]) ||
|
|
|
|
(d1 == expected_version[0] && d2 == expected_version[1] && d3 >= expected_version[2]))
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
|
|
|
// satisfactory version found
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-04-28 09:08:52 +08:00
|
|
|
static Optional<fs::path> find_if_has_equal_or_greater_version(const std::vector<fs::path>& candidate_paths,
|
|
|
|
const std::wstring& version_check_arguments,
|
|
|
|
const std::array<int, 3>& expected_version)
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
2017-04-28 09:08:52 +08:00
|
|
|
auto it = Util::find_if(candidate_paths, [&](const fs::path& p) {
|
|
|
|
const std::wstring cmd = Strings::wformat(LR"("%s" %s)", p.native(), version_check_arguments);
|
|
|
|
return exists_and_has_equal_or_greater_version(cmd, expected_version);
|
|
|
|
});
|
2017-03-11 08:30:44 +08:00
|
|
|
|
|
|
|
if (it != candidate_paths.cend())
|
|
|
|
{
|
2017-03-29 06:05:55 +08:00
|
|
|
return std::move(*it);
|
2017-03-11 08:30:44 +08:00
|
|
|
}
|
|
|
|
|
2017-03-29 06:05:55 +08:00
|
|
|
return nullopt;
|
2017-03-11 08:30:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static std::vector<fs::path> find_from_PATH(const std::wstring& name)
|
|
|
|
{
|
2017-03-14 07:17:47 +08:00
|
|
|
const std::wstring cmd = Strings::wformat(L"where.exe %s", name);
|
2017-03-11 08:30:44 +08:00
|
|
|
auto out = System::cmd_execute_and_capture_output(cmd);
|
|
|
|
if (out.exit_code != 0)
|
|
|
|
{
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2017-04-01 18:30:52 +08:00
|
|
|
return Util::fmap(Strings::split(out.output, "\n"), [](auto&& s) { return fs::path(s); });
|
2017-03-11 08:30:44 +08:00
|
|
|
}
|
|
|
|
|
2017-08-12 07:11:00 +08:00
|
|
|
static fs::path fetch_dependency(const fs::path& scripts_folder,
|
2017-04-28 09:08:52 +08:00
|
|
|
const std::wstring& tool_name,
|
2017-08-12 07:09:45 +08:00
|
|
|
const fs::path& expected_downloaded_path,
|
|
|
|
const std::array<int, 3>& version)
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
|
|
|
const fs::path script = scripts_folder / "fetchDependency.ps1";
|
|
|
|
auto install_cmd = System::create_powershell_script_cmd(script, Strings::wformat(L"-Dependency %s", tool_name));
|
2017-04-04 07:30:11 +08:00
|
|
|
System::ExitCodeAndOutput rc = System::cmd_execute_and_capture_output(install_cmd);
|
2017-03-11 08:30:44 +08:00
|
|
|
if (rc.exit_code)
|
|
|
|
{
|
2017-08-12 07:09:45 +08:00
|
|
|
const std::string version_as_string = Strings::format("%d.%d.%d", version[0], version[1], version[2]);
|
|
|
|
|
|
|
|
System::println(System::Color::error,
|
|
|
|
"Launching powershell failed or was denied when trying to fetch %s version %s.\n"
|
|
|
|
"(No sufficient installed version was found)",
|
|
|
|
Strings::to_utf8(tool_name),
|
|
|
|
version_as_string);
|
2017-08-26 07:03:57 +08:00
|
|
|
{
|
|
|
|
auto locked_metrics = Metrics::g_metrics.lock();
|
|
|
|
locked_metrics->track_property("error", "powershell install failed");
|
2017-08-30 08:02:33 +08:00
|
|
|
locked_metrics->track_property("dependency", tool_name);
|
2017-08-26 07:03:57 +08:00
|
|
|
}
|
2017-03-23 08:46:05 +08:00
|
|
|
Checks::exit_with_code(VCPKG_LINE_INFO, rc.exit_code);
|
2017-03-11 08:30:44 +08:00
|
|
|
}
|
|
|
|
|
2017-03-14 08:54:34 +08:00
|
|
|
const fs::path actual_downloaded_path = Strings::trimmed(rc.output);
|
2017-05-09 10:45:39 +08:00
|
|
|
std::error_code ec;
|
|
|
|
auto eq = fs::stdfs::equivalent(expected_downloaded_path, actual_downloaded_path, ec);
|
2017-04-28 09:08:52 +08:00
|
|
|
Checks::check_exit(VCPKG_LINE_INFO,
|
2017-05-09 10:45:39 +08:00
|
|
|
eq && !ec,
|
2017-04-28 09:08:52 +08:00
|
|
|
"Expected dependency downloaded path to be %s, but was %s",
|
2017-05-09 10:45:39 +08:00
|
|
|
expected_downloaded_path.u8string(),
|
|
|
|
actual_downloaded_path.u8string());
|
2017-03-11 08:30:44 +08:00
|
|
|
return actual_downloaded_path;
|
|
|
|
}
|
|
|
|
|
2017-08-12 07:11:00 +08:00
|
|
|
static fs::path get_cmake_path(const fs::path& downloads_folder, const fs::path& scripts_folder)
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
2017-08-12 00:10:37 +08:00
|
|
|
static constexpr std::array<int, 3> expected_version = {3, 9, 1};
|
2017-03-11 08:30:44 +08:00
|
|
|
static const std::wstring version_check_arguments = L"--version";
|
|
|
|
|
2017-08-12 00:10:37 +08:00
|
|
|
const fs::path downloaded_copy = downloads_folder / "cmake-3.9.1-win32-x86" / "bin" / "cmake.exe";
|
2017-03-11 08:30:44 +08:00
|
|
|
const std::vector<fs::path> from_path = find_from_PATH(L"cmake");
|
|
|
|
|
|
|
|
std::vector<fs::path> candidate_paths;
|
|
|
|
candidate_paths.push_back(downloaded_copy);
|
|
|
|
candidate_paths.insert(candidate_paths.end(), from_path.cbegin(), from_path.cend());
|
2017-04-01 08:24:45 +08:00
|
|
|
candidate_paths.push_back(System::get_ProgramFiles_platform_bitness() / "CMake" / "bin" / "cmake.exe");
|
|
|
|
candidate_paths.push_back(System::get_ProgramFiles_32_bit() / "CMake" / "bin");
|
2017-03-11 08:30:44 +08:00
|
|
|
|
2017-04-28 09:08:52 +08:00
|
|
|
const Optional<fs::path> path =
|
|
|
|
find_if_has_equal_or_greater_version(candidate_paths, version_check_arguments, expected_version);
|
2017-03-29 06:05:55 +08:00
|
|
|
if (auto p = path.get())
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
2017-03-29 06:05:55 +08:00
|
|
|
return *p;
|
2017-03-11 08:30:44 +08:00
|
|
|
}
|
|
|
|
|
2017-08-12 07:09:45 +08:00
|
|
|
return fetch_dependency(scripts_folder, L"cmake", downloaded_copy, expected_version);
|
2017-03-11 08:30:44 +08:00
|
|
|
}
|
|
|
|
|
2017-08-12 07:11:00 +08:00
|
|
|
fs::path get_nuget_path(const fs::path& downloads_folder, const fs::path& scripts_folder)
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
2017-08-08 03:26:30 +08:00
|
|
|
static constexpr std::array<int, 3> expected_version = {4, 1, 0};
|
2017-08-29 10:39:33 +08:00
|
|
|
static const std::wstring version_check_arguments = Strings::WEMPTY;
|
2017-03-11 08:30:44 +08:00
|
|
|
|
2017-08-08 03:26:30 +08:00
|
|
|
const fs::path downloaded_copy = downloads_folder / "nuget-4.1.0" / "nuget.exe";
|
2017-03-11 08:30:44 +08:00
|
|
|
const std::vector<fs::path> from_path = find_from_PATH(L"nuget");
|
|
|
|
|
|
|
|
std::vector<fs::path> candidate_paths;
|
|
|
|
candidate_paths.push_back(downloaded_copy);
|
|
|
|
candidate_paths.insert(candidate_paths.end(), from_path.cbegin(), from_path.cend());
|
|
|
|
|
2017-03-29 06:05:55 +08:00
|
|
|
auto path = find_if_has_equal_or_greater_version(candidate_paths, version_check_arguments, expected_version);
|
|
|
|
if (auto p = path.get())
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
2017-03-29 06:05:55 +08:00
|
|
|
return *p;
|
2017-03-11 08:30:44 +08:00
|
|
|
}
|
|
|
|
|
2017-08-12 07:09:45 +08:00
|
|
|
return fetch_dependency(scripts_folder, L"nuget", downloaded_copy, expected_version);
|
2017-03-11 08:30:44 +08:00
|
|
|
}
|
|
|
|
|
2017-08-12 07:11:00 +08:00
|
|
|
fs::path get_git_path(const fs::path& downloads_folder, const fs::path& scripts_folder)
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
2017-08-12 06:40:06 +08:00
|
|
|
static constexpr std::array<int, 3> expected_version = {2, 14, 1};
|
2017-03-11 08:30:44 +08:00
|
|
|
static const std::wstring version_check_arguments = L"--version";
|
|
|
|
|
2017-08-12 06:40:06 +08:00
|
|
|
const fs::path downloaded_copy = downloads_folder / "MinGit-2.14.1-32-bit" / "cmd" / "git.exe";
|
2017-03-11 08:30:44 +08:00
|
|
|
const std::vector<fs::path> from_path = find_from_PATH(L"git");
|
|
|
|
|
|
|
|
std::vector<fs::path> candidate_paths;
|
|
|
|
candidate_paths.push_back(downloaded_copy);
|
|
|
|
candidate_paths.insert(candidate_paths.end(), from_path.cbegin(), from_path.cend());
|
2017-04-01 08:24:45 +08:00
|
|
|
candidate_paths.push_back(System::get_ProgramFiles_platform_bitness() / "git" / "cmd" / "git.exe");
|
|
|
|
candidate_paths.push_back(System::get_ProgramFiles_32_bit() / "git" / "cmd" / "git.exe");
|
2017-03-11 08:30:44 +08:00
|
|
|
|
2017-04-28 09:08:52 +08:00
|
|
|
const Optional<fs::path> path =
|
|
|
|
find_if_has_equal_or_greater_version(candidate_paths, version_check_arguments, expected_version);
|
2017-03-29 06:05:55 +08:00
|
|
|
if (auto p = path.get())
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
2017-03-29 06:05:55 +08:00
|
|
|
return *p;
|
2017-03-11 08:30:44 +08:00
|
|
|
}
|
|
|
|
|
2017-08-12 07:09:45 +08:00
|
|
|
return fetch_dependency(scripts_folder, L"git", downloaded_copy, expected_version);
|
2017-03-11 08:30:44 +08:00
|
|
|
}
|
|
|
|
|
2017-04-04 07:29:11 +08:00
|
|
|
Expected<VcpkgPaths> VcpkgPaths::create(const fs::path& vcpkg_root_dir)
|
2016-09-19 11:50:08 +08:00
|
|
|
{
|
|
|
|
std::error_code ec;
|
2017-04-12 06:16:39 +08:00
|
|
|
const fs::path canonical_vcpkg_root_dir = fs::stdfs::canonical(vcpkg_root_dir, ec);
|
2016-09-19 11:50:08 +08:00
|
|
|
if (ec)
|
|
|
|
{
|
|
|
|
return ec;
|
|
|
|
}
|
|
|
|
|
2017-04-04 07:29:11 +08:00
|
|
|
VcpkgPaths paths;
|
2016-09-19 11:50:08 +08:00
|
|
|
paths.root = canonical_vcpkg_root_dir;
|
|
|
|
|
|
|
|
if (paths.root.empty())
|
|
|
|
{
|
2017-08-26 07:03:57 +08:00
|
|
|
Metrics::g_metrics.lock()->track_property("error", "Invalid vcpkg root directory");
|
2017-03-23 08:06:09 +08:00
|
|
|
Checks::exit_with_message(VCPKG_LINE_INFO, "Invalid vcpkg root directory: %s", paths.root.string());
|
2016-09-19 11:50:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
paths.packages = paths.root / "packages";
|
|
|
|
paths.buildtrees = paths.root / "buildtrees";
|
|
|
|
paths.downloads = paths.root / "downloads";
|
|
|
|
paths.ports = paths.root / "ports";
|
|
|
|
paths.installed = paths.root / "installed";
|
|
|
|
paths.triplets = paths.root / "triplets";
|
2017-01-24 07:12:43 +08:00
|
|
|
paths.scripts = paths.root / "scripts";
|
2016-09-19 11:50:08 +08:00
|
|
|
|
2017-01-24 07:12:43 +08:00
|
|
|
paths.buildsystems = paths.scripts / "buildsystems";
|
2016-09-19 11:50:08 +08:00
|
|
|
paths.buildsystems_msbuild_targets = paths.buildsystems / "msbuild" / "vcpkg.targets";
|
|
|
|
|
|
|
|
paths.vcpkg_dir = paths.installed / "vcpkg";
|
|
|
|
paths.vcpkg_dir_status_file = paths.vcpkg_dir / "status";
|
|
|
|
paths.vcpkg_dir_info = paths.vcpkg_dir / "info";
|
|
|
|
paths.vcpkg_dir_updates = paths.vcpkg_dir / "updates";
|
|
|
|
|
2017-01-24 07:12:43 +08:00
|
|
|
paths.ports_cmake = paths.scripts / "ports.cmake";
|
2017-03-11 08:30:44 +08:00
|
|
|
|
2016-09-19 11:50:08 +08:00
|
|
|
return paths;
|
|
|
|
}
|
2016-09-24 05:25:17 +08:00
|
|
|
|
2017-04-28 09:08:52 +08:00
|
|
|
fs::path VcpkgPaths::package_dir(const PackageSpec& spec) const { return this->packages / spec.dir(); }
|
2016-09-24 05:25:17 +08:00
|
|
|
|
2017-04-28 09:08:52 +08:00
|
|
|
fs::path VcpkgPaths::port_dir(const PackageSpec& spec) const { return this->ports / spec.name(); }
|
2017-08-22 08:16:14 +08:00
|
|
|
fs::path VcpkgPaths::port_dir(const std::string& name) const { return this->ports / name; }
|
2016-09-29 03:39:05 +08:00
|
|
|
|
2017-04-04 07:29:11 +08:00
|
|
|
fs::path VcpkgPaths::build_info_file_path(const PackageSpec& spec) const
|
2016-11-09 06:12:49 +08:00
|
|
|
{
|
|
|
|
return this->package_dir(spec) / "BUILD_INFO";
|
|
|
|
}
|
|
|
|
|
2017-04-04 07:29:11 +08:00
|
|
|
fs::path VcpkgPaths::listfile_path(const BinaryParagraph& pgh) const
|
2016-11-08 09:54:23 +08:00
|
|
|
{
|
|
|
|
return this->vcpkg_dir_info / (pgh.fullstem() + ".list");
|
|
|
|
}
|
|
|
|
|
2017-04-04 07:29:11 +08:00
|
|
|
bool VcpkgPaths::is_valid_triplet(const Triplet& t) const
|
2016-09-29 03:39:05 +08:00
|
|
|
{
|
2017-04-13 14:15:02 +08:00
|
|
|
for (auto&& path : get_filesystem().get_files_non_recursive(this->triplets))
|
2016-09-29 03:39:05 +08:00
|
|
|
{
|
2017-04-12 06:16:39 +08:00
|
|
|
std::string triplet_file_name = path.stem().generic_u8string();
|
2016-10-04 08:20:52 +08:00
|
|
|
if (t.canonical_name() == triplet_file_name) // TODO: fuzzy compare
|
2016-09-29 03:39:05 +08:00
|
|
|
{
|
2017-04-28 09:08:52 +08:00
|
|
|
// t.value = triplet_file_name; // NOTE: uncomment when implementing fuzzy compare
|
2016-09-29 03:39:05 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2017-03-11 08:30:44 +08:00
|
|
|
|
2017-04-04 07:29:11 +08:00
|
|
|
const fs::path& VcpkgPaths::get_cmake_exe() const
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
|
|
|
return this->cmake_exe.get_lazy([this]() { return get_cmake_path(this->downloads, this->scripts); });
|
|
|
|
}
|
|
|
|
|
2017-04-04 07:29:11 +08:00
|
|
|
const fs::path& VcpkgPaths::get_git_exe() const
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
|
|
|
return this->git_exe.get_lazy([this]() { return get_git_path(this->downloads, this->scripts); });
|
|
|
|
}
|
|
|
|
|
2017-04-04 07:29:11 +08:00
|
|
|
const fs::path& VcpkgPaths::get_nuget_exe() const
|
2017-03-11 08:30:44 +08:00
|
|
|
{
|
|
|
|
return this->nuget_exe.get_lazy([this]() { return get_nuget_path(this->downloads, this->scripts); });
|
|
|
|
}
|
2017-04-01 08:40:08 +08:00
|
|
|
|
2017-04-04 07:29:11 +08:00
|
|
|
static std::vector<std::string> get_VS2017_installation_instances(const VcpkgPaths& paths)
|
2017-04-01 08:40:08 +08:00
|
|
|
{
|
|
|
|
const fs::path script = paths.scripts / "findVisualStudioInstallationInstances.ps1";
|
|
|
|
const std::wstring cmd = System::create_powershell_script_cmd(script);
|
2017-04-04 07:30:11 +08:00
|
|
|
System::ExitCodeAndOutput ec_data = System::cmd_execute_and_capture_output(cmd);
|
2017-04-01 08:40:08 +08:00
|
|
|
Checks::check_exit(VCPKG_LINE_INFO, ec_data.exit_code == 0, "Could not run script to detect VS 2017 instances");
|
|
|
|
return Strings::split(ec_data.output, "\n");
|
|
|
|
}
|
|
|
|
|
2017-04-04 07:27:51 +08:00
|
|
|
static Optional<fs::path> get_VS2015_installation_instance()
|
2017-04-01 08:40:08 +08:00
|
|
|
{
|
2017-05-05 05:58:18 +08:00
|
|
|
const Optional<std::wstring> vs2015_cmntools_optional = System::get_environment_variable(L"VS140COMNTOOLS");
|
2017-04-01 18:30:52 +08:00
|
|
|
if (auto v = vs2015_cmntools_optional.get())
|
|
|
|
{
|
2017-04-28 09:08:52 +08:00
|
|
|
const fs::path vs2015_cmntools = fs::path(*v).parent_path(); // The call to parent_path() is needed because
|
|
|
|
// the env variable has a trailing backslash
|
2017-04-01 18:30:52 +08:00
|
|
|
return vs2015_cmntools.parent_path().parent_path();
|
|
|
|
}
|
2017-04-01 08:40:08 +08:00
|
|
|
|
2017-04-01 18:30:52 +08:00
|
|
|
return nullopt;
|
2017-04-01 08:40:08 +08:00
|
|
|
}
|
|
|
|
|
2017-06-03 00:13:12 +08:00
|
|
|
static std::vector<Toolset> find_toolset_instances(const VcpkgPaths& paths)
|
2017-04-01 08:40:08 +08:00
|
|
|
{
|
2017-08-24 18:26:42 +08:00
|
|
|
using CPU = System::CPUArchitecture;
|
|
|
|
|
2017-04-09 11:19:35 +08:00
|
|
|
const auto& fs = paths.get_filesystem();
|
|
|
|
|
2017-04-01 08:40:08 +08:00
|
|
|
const std::vector<std::string> vs2017_installation_instances = get_VS2017_installation_instances(paths);
|
2017-04-01 17:08:48 +08:00
|
|
|
// Note: this will contain a mix of vcvarsall.bat locations and dumpbin.exe locations.
|
2017-04-01 08:40:08 +08:00
|
|
|
std::vector<fs::path> paths_examined;
|
|
|
|
|
2017-06-03 00:13:12 +08:00
|
|
|
std::vector<Toolset> found_toolsets;
|
|
|
|
|
2017-06-06 13:01:41 +08:00
|
|
|
// VS2015
|
|
|
|
const Optional<fs::path> vs_2015_installation_instance = get_VS2015_installation_instance();
|
|
|
|
if (auto v = vs_2015_installation_instance.get())
|
|
|
|
{
|
|
|
|
const fs::path vs2015_vcvarsall_bat = *v / "VC" / "vcvarsall.bat";
|
|
|
|
|
|
|
|
paths_examined.push_back(vs2015_vcvarsall_bat);
|
|
|
|
if (fs.exists(vs2015_vcvarsall_bat))
|
|
|
|
{
|
|
|
|
const fs::path vs2015_dumpbin_exe = *v / "VC" / "bin" / "dumpbin.exe";
|
|
|
|
paths_examined.push_back(vs2015_dumpbin_exe);
|
2017-08-24 18:26:42 +08:00
|
|
|
|
|
|
|
const fs::path vs2015_bin_dir = vs2015_vcvarsall_bat.parent_path() / "bin";
|
2017-08-26 15:19:51 +08:00
|
|
|
std::vector<ToolsetArchOption> supported_architectures;
|
|
|
|
if (fs.exists(vs2015_bin_dir / "vcvars32.bat"))
|
|
|
|
supported_architectures.push_back({L"x86", CPU::X86, CPU::X86});
|
2017-08-24 18:26:42 +08:00
|
|
|
if (fs.exists(vs2015_bin_dir / "amd64\\vcvars64.bat"))
|
2017-08-26 15:19:51 +08:00
|
|
|
supported_architectures.push_back({L"x64", CPU::X64, CPU::X64});
|
2017-08-24 18:26:42 +08:00
|
|
|
if (fs.exists(vs2015_bin_dir / "x86_amd64\\vcvarsx86_amd64.bat"))
|
2017-08-26 15:19:51 +08:00
|
|
|
supported_architectures.push_back({L"x86_amd64", CPU::X86, CPU::X64});
|
2017-08-24 18:26:42 +08:00
|
|
|
if (fs.exists(vs2015_bin_dir / "x86_arm\\vcvarsx86_arm.bat"))
|
2017-08-26 15:19:51 +08:00
|
|
|
supported_architectures.push_back({L"x86_arm", CPU::X86, CPU::ARM});
|
2017-08-24 18:26:42 +08:00
|
|
|
if (fs.exists(vs2015_bin_dir / "amd64_x86\\vcvarsamd64_x86.bat"))
|
2017-08-26 15:19:51 +08:00
|
|
|
supported_architectures.push_back({L"amd64_x86", CPU::X64, CPU::X86});
|
2017-08-24 18:26:42 +08:00
|
|
|
if (fs.exists(vs2015_bin_dir / "amd64_arm\\vcvarsamd64_arm.bat"))
|
2017-08-26 15:19:51 +08:00
|
|
|
supported_architectures.push_back({L"amd64_arm", CPU::X64, CPU::ARM});
|
2017-08-24 18:26:42 +08:00
|
|
|
|
2017-06-06 13:01:41 +08:00
|
|
|
if (fs.exists(vs2015_dumpbin_exe))
|
|
|
|
{
|
2017-08-26 15:19:51 +08:00
|
|
|
found_toolsets.push_back(
|
|
|
|
{vs2015_dumpbin_exe, vs2015_vcvarsall_bat, L"v140", supported_architectures});
|
2017-06-06 13:01:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-01 08:40:08 +08:00
|
|
|
// VS2017
|
2017-06-03 00:13:12 +08:00
|
|
|
Optional<Toolset> vs2017_toolset;
|
2017-04-01 17:08:48 +08:00
|
|
|
for (const fs::path& instance : vs2017_installation_instances)
|
2017-04-01 08:40:08 +08:00
|
|
|
{
|
2017-04-01 17:08:48 +08:00
|
|
|
const fs::path vc_dir = instance / "VC";
|
|
|
|
|
|
|
|
// Skip any instances that do not have vcvarsall.
|
2017-08-24 18:26:42 +08:00
|
|
|
const fs::path vcvarsall_dir = vc_dir / "Auxiliary" / "Build";
|
|
|
|
const fs::path vcvarsall_bat = vcvarsall_dir / "vcvarsall.bat";
|
2017-04-01 17:08:48 +08:00
|
|
|
paths_examined.push_back(vcvarsall_bat);
|
2017-04-28 09:08:52 +08:00
|
|
|
if (!fs.exists(vcvarsall_bat)) continue;
|
2017-04-01 17:08:48 +08:00
|
|
|
|
2017-08-26 15:19:51 +08:00
|
|
|
// Get all supported architectures
|
2017-08-24 18:26:42 +08:00
|
|
|
std::vector<ToolsetArchOption> supported_architectures;
|
|
|
|
if (fs.exists(vcvarsall_dir / "vcvars32.bat"))
|
2017-08-26 15:19:51 +08:00
|
|
|
supported_architectures.push_back({L"x86", CPU::X86, CPU::X86});
|
2017-08-24 18:26:42 +08:00
|
|
|
if (fs.exists(vcvarsall_dir / "vcvars64.bat"))
|
2017-08-26 15:19:51 +08:00
|
|
|
supported_architectures.push_back({L"amd64", CPU::X64, CPU::X64});
|
2017-08-24 18:26:42 +08:00
|
|
|
if (fs.exists(vcvarsall_dir / "vcvarsx86_amd64.bat"))
|
2017-08-26 15:19:51 +08:00
|
|
|
supported_architectures.push_back({L"x86_amd64", CPU::X86, CPU::X64});
|
2017-08-24 18:26:42 +08:00
|
|
|
if (fs.exists(vcvarsall_dir / "vcvarsx86_arm.bat"))
|
2017-08-26 15:19:51 +08:00
|
|
|
supported_architectures.push_back({L"x86_arm", CPU::X86, CPU::ARM});
|
2017-08-24 18:26:42 +08:00
|
|
|
if (fs.exists(vcvarsall_dir / "vcvarsamd64_x86.bat"))
|
2017-08-26 15:19:51 +08:00
|
|
|
supported_architectures.push_back({L"amd64_x86", CPU::X64, CPU::X86});
|
2017-08-24 18:26:42 +08:00
|
|
|
if (fs.exists(vcvarsall_dir / "vcvarsamd64_arm.bat"))
|
2017-08-26 15:19:51 +08:00
|
|
|
supported_architectures.push_back({L"amd64_arm", CPU::X64, CPU::ARM});
|
2017-08-24 18:26:42 +08:00
|
|
|
|
2017-04-01 17:08:48 +08:00
|
|
|
// Locate the "best" MSVC toolchain version
|
|
|
|
const fs::path msvc_path = vc_dir / "Tools" / "MSVC";
|
2017-04-13 14:15:02 +08:00
|
|
|
std::vector<fs::path> msvc_subdirectories = fs.get_files_non_recursive(msvc_path);
|
2017-04-28 09:08:52 +08:00
|
|
|
Util::unstable_keep_if(msvc_subdirectories, [&fs](const fs::path& path) { return fs.is_directory(path); });
|
2017-04-01 08:40:08 +08:00
|
|
|
|
|
|
|
// Sort them so that latest comes first
|
2017-04-28 09:08:52 +08:00
|
|
|
std::sort(msvc_subdirectories.begin(),
|
|
|
|
msvc_subdirectories.end(),
|
|
|
|
[](const fs::path& left, const fs::path& right) { return left.filename() > right.filename(); });
|
2017-04-01 08:40:08 +08:00
|
|
|
|
|
|
|
for (const fs::path& subdir : msvc_subdirectories)
|
|
|
|
{
|
|
|
|
const fs::path dumpbin_path = subdir / "bin" / "HostX86" / "x86" / "dumpbin.exe";
|
|
|
|
paths_examined.push_back(dumpbin_path);
|
2017-04-09 11:19:35 +08:00
|
|
|
if (fs.exists(dumpbin_path))
|
2017-04-01 08:40:08 +08:00
|
|
|
{
|
2017-08-24 18:26:42 +08:00
|
|
|
vs2017_toolset = Toolset{dumpbin_path, vcvarsall_bat, L"v141", supported_architectures};
|
2017-06-03 00:13:12 +08:00
|
|
|
break;
|
2017-04-01 08:40:08 +08:00
|
|
|
}
|
|
|
|
}
|
2017-06-06 13:01:41 +08:00
|
|
|
if (auto value = vs2017_toolset.get())
|
2017-04-01 08:40:08 +08:00
|
|
|
{
|
2017-06-06 13:01:41 +08:00
|
|
|
found_toolsets.push_back(*value);
|
|
|
|
break;
|
2017-04-01 08:40:08 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-06 13:01:41 +08:00
|
|
|
if (found_toolsets.empty())
|
2017-04-01 08:40:08 +08:00
|
|
|
{
|
2017-06-03 00:13:12 +08:00
|
|
|
System::println(System::Color::error, "Could not locate a complete toolset.");
|
|
|
|
System::println("The following paths were examined:");
|
2017-06-06 13:01:41 +08:00
|
|
|
for (const fs::path& path : paths_examined)
|
2017-06-03 00:13:12 +08:00
|
|
|
{
|
|
|
|
System::println(" %s", path.u8string());
|
|
|
|
}
|
|
|
|
Checks::exit_fail(VCPKG_LINE_INFO);
|
2017-04-01 08:40:08 +08:00
|
|
|
}
|
2017-06-03 00:13:12 +08:00
|
|
|
|
|
|
|
return found_toolsets;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Toolset& VcpkgPaths::get_toolset(const std::string& toolset_version) const
|
|
|
|
{
|
2017-06-06 13:01:41 +08:00
|
|
|
// Invariant: toolsets are non-empty and sorted with newest at back()
|
|
|
|
const auto& vs_toolsets = this->toolsets.get_lazy([this]() { return find_toolset_instances(*this); });
|
|
|
|
|
|
|
|
if (toolset_version.empty())
|
2017-06-03 00:13:12 +08:00
|
|
|
{
|
2017-06-06 13:01:41 +08:00
|
|
|
return vs_toolsets.back();
|
2017-06-03 00:13:12 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-06-06 13:01:41 +08:00
|
|
|
const auto toolset = Util::find_if(vs_toolsets, [&](const Toolset& toolset) {
|
|
|
|
return toolset_version == Strings::to_utf8(toolset.version);
|
|
|
|
});
|
|
|
|
Checks::check_exit(
|
|
|
|
VCPKG_LINE_INFO, toolset != vs_toolsets.end(), "Could not find toolset '%s'", toolset_version);
|
2017-06-03 00:13:12 +08:00
|
|
|
return *toolset;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-28 09:08:52 +08:00
|
|
|
Files::Filesystem& VcpkgPaths::get_filesystem() const { return Files::get_real_filesystem(); }
|
2016-09-19 11:50:08 +08:00
|
|
|
}
|