| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376 |
- /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
- file Copyright.txt or https://cmake.org/licensing for details. */
- #include "cmRuntimeDependencyArchive.h"
- #include "cmBinUtilsLinuxELFLinker.h"
- #include "cmBinUtilsMacOSMachOLinker.h"
- #include "cmBinUtilsWindowsPELinker.h"
- #include "cmExecutionStatus.h"
- #include "cmMakefile.h"
- #include "cmStateTypes.h"
- #include "cmSystemTools.h"
- #if defined(_WIN32)
- # include "cmGlobalGenerator.h"
- # ifndef CMAKE_BOOTSTRAP
- # include "cmGlobalVisualStudioVersionedGenerator.h"
- # endif
- # include "cmVSSetupHelper.h"
- # include "cmsys/Glob.hxx"
- #endif
- #include <algorithm>
- #include <sstream>
- #include <string>
- #include <utility>
- #include <vector>
- #include "cm_memory.hxx"
- #if defined(_WIN32)
- static void AddVisualStudioPath(std::vector<std::string>& paths,
- const std::string& prefix,
- unsigned int version, cmGlobalGenerator* gg)
- {
- // If generating for the VS IDE, use the same instance.
- std::string vsloc;
- bool found = false;
- # ifndef CMAKE_BOOTSTRAP
- if (gg->GetName().find(prefix) == 0) {
- cmGlobalVisualStudioVersionedGenerator* vsgen =
- static_cast<cmGlobalVisualStudioVersionedGenerator*>(gg);
- if (vsgen->GetVSInstance(vsloc)) {
- found = true;
- }
- }
- # endif
- // Otherwise, find a VS instance ourselves.
- if (!found) {
- cmVSSetupAPIHelper vsSetupAPIHelper(version);
- if (vsSetupAPIHelper.GetVSInstanceInfo(vsloc)) {
- cmSystemTools::ConvertToUnixSlashes(vsloc);
- found = true;
- }
- }
- if (found) {
- cmsys::Glob glob;
- glob.SetListDirs(true);
- glob.FindFiles(vsloc + "/VC/Tools/MSVC/*");
- for (auto const& vcdir : glob.GetFiles()) {
- paths.push_back(vcdir + "/bin/Hostx64/x64");
- paths.push_back(vcdir + "/bin/Hostx86/x64");
- paths.push_back(vcdir + "/bin/Hostx64/x86");
- paths.push_back(vcdir + "/bin/Hostx86/x86");
- }
- }
- }
- static void AddRegistryPath(std::vector<std::string>& paths,
- const std::string& path, cmMakefile* mf)
- {
- // We should view the registry as the target application would view
- // it.
- cmSystemTools::KeyWOW64 view = cmSystemTools::KeyWOW64_32;
- cmSystemTools::KeyWOW64 other_view = cmSystemTools::KeyWOW64_64;
- if (mf->PlatformIs64Bit()) {
- view = cmSystemTools::KeyWOW64_64;
- other_view = cmSystemTools::KeyWOW64_32;
- }
- // Expand using the view of the target application.
- std::string expanded = path;
- cmSystemTools::ExpandRegistryValues(expanded, view);
- cmSystemTools::GlobDirs(expanded, paths);
- // Executables can be either 32-bit or 64-bit, so expand using the
- // alternative view.
- expanded = path;
- cmSystemTools::ExpandRegistryValues(expanded, other_view);
- cmSystemTools::GlobDirs(expanded, paths);
- }
- static void AddEnvPath(std::vector<std::string>& paths, const std::string& var,
- const std::string& suffix)
- {
- std::string value;
- if (cmSystemTools::GetEnv(var, value)) {
- paths.push_back(value + suffix);
- }
- }
- #endif
- static cmsys::RegularExpression TransformCompile(const std::string& str)
- {
- return cmsys::RegularExpression(str);
- }
- cmRuntimeDependencyArchive::cmRuntimeDependencyArchive(
- cmExecutionStatus& status, std::vector<std::string> searchDirectories,
- std::string bundleExecutable,
- const std::vector<std::string>& preIncludeRegexes,
- const std::vector<std::string>& preExcludeRegexes,
- const std::vector<std::string>& postIncludeRegexes,
- const std::vector<std::string>& postExcludeRegexes)
- : Status(status)
- , SearchDirectories(std::move(searchDirectories))
- , BundleExecutable(std::move(bundleExecutable))
- , PreIncludeRegexes(preIncludeRegexes.size())
- , PreExcludeRegexes(preExcludeRegexes.size())
- , PostIncludeRegexes(postIncludeRegexes.size())
- , PostExcludeRegexes(postExcludeRegexes.size())
- {
- std::transform(preIncludeRegexes.begin(), preIncludeRegexes.end(),
- this->PreIncludeRegexes.begin(), TransformCompile);
- std::transform(preExcludeRegexes.begin(), preExcludeRegexes.end(),
- this->PreExcludeRegexes.begin(), TransformCompile);
- std::transform(postIncludeRegexes.begin(), postIncludeRegexes.end(),
- this->PostIncludeRegexes.begin(), TransformCompile);
- std::transform(postExcludeRegexes.begin(), postExcludeRegexes.end(),
- this->PostExcludeRegexes.begin(), TransformCompile);
- }
- bool cmRuntimeDependencyArchive::Prepare()
- {
- std::string platform = this->GetMakefile()->GetSafeDefinition(
- "CMAKE_GET_RUNTIME_DEPENDENCIES_PLATFORM");
- if (platform.empty()) {
- std::string systemName =
- this->GetMakefile()->GetSafeDefinition("CMAKE_HOST_SYSTEM_NAME");
- if (systemName == "Windows") {
- platform = "windows+pe";
- } else if (systemName == "Darwin") {
- platform = "macos+macho";
- } else if (systemName == "Linux") {
- platform = "linux+elf";
- }
- }
- if (platform == "linux+elf") {
- this->Linker = cm::make_unique<cmBinUtilsLinuxELFLinker>(this);
- } else if (platform == "windows+pe") {
- this->Linker = cm::make_unique<cmBinUtilsWindowsPELinker>(this);
- } else if (platform == "macos+macho") {
- this->Linker = cm::make_unique<cmBinUtilsMacOSMachOLinker>(this);
- } else {
- std::ostringstream e;
- e << "Invalid value for CMAKE_GET_RUNTIME_DEPENDENCIES_PLATFORM: "
- << platform;
- this->SetError(e.str());
- return false;
- }
- return this->Linker->Prepare();
- }
- bool cmRuntimeDependencyArchive::GetRuntimeDependencies(
- const std::vector<std::string>& executables,
- const std::vector<std::string>& libraries,
- const std::vector<std::string>& modules)
- {
- for (auto const& exe : executables) {
- if (!this->Linker->ScanDependencies(exe, cmStateEnums::EXECUTABLE)) {
- return false;
- }
- }
- for (auto const& lib : libraries) {
- if (!this->Linker->ScanDependencies(lib, cmStateEnums::SHARED_LIBRARY)) {
- return false;
- }
- }
- for (auto const& mod : modules) {
- if (!this->Linker->ScanDependencies(mod, cmStateEnums::MODULE_LIBRARY)) {
- return false;
- }
- }
- return true;
- }
- void cmRuntimeDependencyArchive::SetError(const std::string& e)
- {
- this->Status.SetError(e);
- }
- std::string cmRuntimeDependencyArchive::GetBundleExecutable()
- {
- return this->BundleExecutable;
- }
- const std::vector<std::string>&
- cmRuntimeDependencyArchive::GetSearchDirectories()
- {
- return this->SearchDirectories;
- }
- std::string cmRuntimeDependencyArchive::GetGetRuntimeDependenciesTool()
- {
- return this->GetMakefile()->GetSafeDefinition(
- "CMAKE_GET_RUNTIME_DEPENDENCIES_TOOL");
- }
- bool cmRuntimeDependencyArchive::GetGetRuntimeDependenciesCommand(
- const std::string& search, std::vector<std::string>& command)
- {
- // First see if it was supplied by the user
- std::string toolCommand = this->GetMakefile()->GetSafeDefinition(
- "CMAKE_GET_RUNTIME_DEPENDENCIES_COMMAND");
- if (!toolCommand.empty()) {
- cmSystemTools::ExpandListArgument(toolCommand, command);
- return true;
- }
- // Now go searching for it
- std::vector<std::string> paths;
- #ifdef _WIN32
- cmGlobalGenerator* gg = this->GetMakefile()->GetGlobalGenerator();
- // Add newer Visual Studio paths
- AddVisualStudioPath(paths, "Visual Studio 16 ", 16, gg);
- AddVisualStudioPath(paths, "Visual Studio 15 ", 15, gg);
- // Add older Visual Studio paths
- AddRegistryPath(
- paths,
- "[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\VisualStudio\\14.0;InstallDir]/"
- "../../VC/bin",
- this->GetMakefile());
- AddEnvPath(paths, "VS140COMNTOOLS", "/../../VC/bin");
- paths.push_back(
- "C:/Program Files (x86)/Microsoft Visual Studio 14.0/VC/bin");
- AddRegistryPath(
- paths,
- "[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\VisualStudio\\12.0;InstallDir]/"
- "../../VC/bin",
- this->GetMakefile());
- AddEnvPath(paths, "VS120COMNTOOLS", "/../../VC/bin");
- paths.push_back(
- "C:/Program Files (x86)/Microsoft Visual Studio 12.0/VC/bin");
- AddRegistryPath(
- paths,
- "[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\VisualStudio\\11.0;InstallDir]/"
- "../../VC/bin",
- this->GetMakefile());
- AddEnvPath(paths, "VS110COMNTOOLS", "/../../VC/bin");
- paths.push_back(
- "C:/Program Files (x86)/Microsoft Visual Studio 11.0/VC/bin");
- AddRegistryPath(
- paths,
- "[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\VisualStudio\\10.0;InstallDir]/"
- "../../VC/bin",
- this->GetMakefile());
- AddEnvPath(paths, "VS100COMNTOOLS", "/../../VC/bin");
- paths.push_back(
- "C:/Program Files (x86)/Microsoft Visual Studio 10.0/VC/bin");
- AddRegistryPath(
- paths,
- "[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\VisualStudio\\9.0;InstallDir]/"
- "../../VC/bin",
- this->GetMakefile());
- AddEnvPath(paths, "VS90COMNTOOLS", "/../../VC/bin");
- paths.push_back("C:/Program Files/Microsoft Visual Studio 9.0/VC/bin");
- paths.push_back("C:/Program Files (x86)/Microsoft Visual Studio 9.0/VC/bin");
- AddRegistryPath(
- paths,
- "[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\VisualStudio\\8.0;InstallDir]/"
- "../../VC/bin",
- this->GetMakefile());
- AddEnvPath(paths, "VS80COMNTOOLS", "/../../VC/bin");
- paths.push_back("C:/Program Files/Microsoft Visual Studio 8/VC/BIN");
- paths.push_back("C:/Program Files (x86)/Microsoft Visual Studio 8/VC/BIN");
- AddRegistryPath(
- paths,
- "[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\VisualStudio\\7.1;InstallDir]/"
- "../../VC7/bin",
- this->GetMakefile());
- AddEnvPath(paths, "VS71COMNTOOLS", "/../../VC7/bin");
- paths.push_back(
- "C:/Program Files/Microsoft Visual Studio .NET 2003/VC7/BIN");
- paths.push_back(
- "C:/Program Files (x86)/Microsoft Visual Studio .NET 2003/VC7/BIN");
- #endif
- std::string program = cmSystemTools::FindProgram(search, paths);
- if (!program.empty()) {
- command = { program };
- return true;
- }
- // Couldn't find it
- return false;
- }
- bool cmRuntimeDependencyArchive::IsPreExcluded(const std::string& name)
- {
- cmsys::RegularExpressionMatch match;
- for (auto const& regex : this->PreIncludeRegexes) {
- if (regex.find(name.c_str(), match)) {
- return false;
- }
- }
- for (auto const& regex : this->PreExcludeRegexes) {
- if (regex.find(name.c_str(), match)) {
- return true;
- }
- }
- return false;
- }
- bool cmRuntimeDependencyArchive::IsPostExcluded(const std::string& name)
- {
- cmsys::RegularExpressionMatch match;
- for (auto const& regex : this->PostIncludeRegexes) {
- if (regex.find(name.c_str(), match)) {
- return false;
- }
- }
- for (auto const& regex : this->PostExcludeRegexes) {
- if (regex.find(name.c_str(), match)) {
- return true;
- }
- }
- return false;
- }
- void cmRuntimeDependencyArchive::AddResolvedPath(const std::string& name,
- const std::string& path,
- bool& unique)
- {
- auto it = this->ResolvedPaths.emplace(name, std::set<std::string>{}).first;
- unique = true;
- for (auto const& other : it->second) {
- if (cmSystemTools::SameFile(path, other)) {
- unique = false;
- break;
- }
- }
- it->second.insert(path);
- }
- void cmRuntimeDependencyArchive::AddUnresolvedPath(const std::string& name)
- {
- this->UnresolvedPaths.insert(name);
- }
- cmMakefile* cmRuntimeDependencyArchive::GetMakefile()
- {
- return &this->Status.GetMakefile();
- }
- const std::map<std::string, std::set<std::string>>&
- cmRuntimeDependencyArchive::GetResolvedPaths()
- {
- return this->ResolvedPaths;
- }
- const std::set<std::string>& cmRuntimeDependencyArchive::GetUnresolvedPaths()
- {
- return this->UnresolvedPaths;
- }
|