/* 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 "cmAlgorithms.h" #include "cmBinUtilsLinuxELFLinker.h" #include "cmBinUtilsMacOSMachOLinker.h" #include "cmBinUtilsWindowsPELinker.h" #include "cmCommand.h" #include "cmMakefile.h" #include "cmStateTypes.h" #include "cmSystemTools.h" #if defined(_WIN32) # include "cmGlobalGenerator.h" # ifdef CMAKE_BUILD_WITH_CMAKE # include "cmGlobalVisualStudioVersionedGenerator.h" # endif # include "cmVSSetupHelper.h" # include "cmsys/Glob.hxx" #endif #include #include #include #include #include #if defined(_WIN32) static void AddVisualStudioPath(std::vector& 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; # ifdef CMAKE_BUILD_WITH_CMAKE if (gg->GetName().find(prefix) == 0) { cmGlobalVisualStudioVersionedGenerator* vsgen = static_cast(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& 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& 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( cmCommand* command, std::vector searchDirectories, std::string bundleExecutable, const std::vector& preIncludeRegexes, const std::vector& preExcludeRegexes, const std::vector& postIncludeRegexes, const std::vector& postExcludeRegexes) : Command(command) , 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(this); } else if (platform == "windows+pe") { this->Linker = cm::make_unique(this); } else if (platform == "macos+macho") { this->Linker = cm::make_unique(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& executables, const std::vector& libraries, const std::vector& 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->Command->SetError(e); } std::string cmRuntimeDependencyArchive::GetBundleExecutable() { return this->BundleExecutable; } const std::vector& 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& 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 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{}).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->Command->GetMakefile(); } const std::map>& cmRuntimeDependencyArchive::GetResolvedPaths() { return this->ResolvedPaths; } const std::set& cmRuntimeDependencyArchive::GetUnresolvedPaths() { return this->UnresolvedPaths; }