/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying file Copyright.txt or https://cmake.org/licensing for details. */ #include "cmBinUtilsMacOSMachOLinker.h" #include #include #include #include #include #include "cmBinUtilsMacOSMachOOToolGetRuntimeDependenciesTool.h" #include "cmRuntimeDependencyArchive.h" #include "cmStringAlgorithms.h" #include "cmSystemTools.h" namespace { bool IsMissingSystemDylib(std::string const& path) { // Starting on macOS 11, the dynamic loader has a builtin cache of // system-provided dylib files that do not exist on the filesystem. // Tell our caller that these are expected to be missing. return ((cmHasLiteralPrefix(path, "/System/Library/") || cmHasLiteralPrefix(path, "/usr/lib/")) && !cmSystemTools::PathExists(path)); } } cmBinUtilsMacOSMachOLinker::cmBinUtilsMacOSMachOLinker( cmRuntimeDependencyArchive* archive) : cmBinUtilsLinker(archive) { } bool cmBinUtilsMacOSMachOLinker::Prepare() { std::string tool = this->Archive->GetGetRuntimeDependenciesTool(); if (tool.empty()) { tool = "otool"; } if (tool == "otool") { this->Tool = cm::make_unique( this->Archive); } else { std::ostringstream e; e << "Invalid value for CMAKE_GET_RUNTIME_DEPENDENCIES_TOOL: " << tool; this->SetError(e.str()); return false; } return true; } auto cmBinUtilsMacOSMachOLinker::GetFileInfo(std::string const& file) -> const FileInfo* { // Memoize processed rpaths and library dependencies to reduce the number // of calls to otool, especially in the case of heavily recursive libraries auto iter = ScannedFileInfo.find(file); if (iter != ScannedFileInfo.end()) { return &iter->second; } FileInfo file_info; if (!this->Tool->GetFileInfo(file, file_info.libs, file_info.rpaths)) { // Call to otool failed return nullptr; } auto iter_inserted = ScannedFileInfo.insert({ file, std::move(file_info) }); return &iter_inserted.first->second; } bool cmBinUtilsMacOSMachOLinker::ScanDependencies( std::string const& file, cmStateEnums::TargetType type) { std::string executableFile; if (type == cmStateEnums::EXECUTABLE) { executableFile = file; } else { executableFile = this->Archive->GetBundleExecutable(); } std::string executablePath; if (!executableFile.empty()) { executablePath = cmSystemTools::GetFilenamePath(executableFile); } const FileInfo* file_info = this->GetFileInfo(file); if (!file_info) { return false; } return this->ScanDependencies(file, file_info->libs, file_info->rpaths, executablePath); } bool cmBinUtilsMacOSMachOLinker::ScanDependencies( std::string const& file, std::vector const& libs, std::vector const& rpaths, std::string const& executablePath) { std::string loaderPath = cmSystemTools::GetFilenamePath(file); return this->GetFileDependencies(libs, executablePath, loaderPath, rpaths); } bool cmBinUtilsMacOSMachOLinker::GetFileDependencies( std::vector const& names, std::string const& executablePath, std::string const& loaderPath, std::vector const& rpaths) { for (std::string const& name : names) { if (!this->Archive->IsPreExcluded(name)) { std::string path; bool resolved; if (!this->ResolveDependency(name, executablePath, loaderPath, rpaths, path, resolved)) { return false; } if (resolved) { if (!this->Archive->IsPostExcluded(path) && !IsMissingSystemDylib(path)) { auto filename = cmSystemTools::GetFilenameName(path); bool unique; const FileInfo* dep_file_info = this->GetFileInfo(path); if (!dep_file_info) { return false; } this->Archive->AddResolvedPath(filename, path, unique, dep_file_info->rpaths); if (unique) { std::vector combinedParentRpaths = dep_file_info->rpaths; combinedParentRpaths.insert(combinedParentRpaths.end(), rpaths.begin(), rpaths.end()); if (!this->ScanDependencies(path, dep_file_info->libs, combinedParentRpaths, executablePath)) { return false; } } } } else { this->Archive->AddUnresolvedPath(name); } } } return true; } bool cmBinUtilsMacOSMachOLinker::ResolveDependency( std::string const& name, std::string const& executablePath, std::string const& loaderPath, std::vector const& rpaths, std::string& path, bool& resolved) { resolved = false; if (cmHasLiteralPrefix(name, "@rpath/")) { if (!this->ResolveRPathDependency(name, executablePath, loaderPath, rpaths, path, resolved)) { return false; } } else if (cmHasLiteralPrefix(name, "@loader_path/")) { if (!this->ResolveLoaderPathDependency(name, loaderPath, path, resolved)) { return false; } } else if (cmHasLiteralPrefix(name, "@executable_path/")) { if (!this->ResolveExecutablePathDependency(name, executablePath, path, resolved)) { return false; } } else { resolved = true; path = name; } if (resolved && !cmSystemTools::FileIsFullPath(path)) { this->SetError("Resolved path is not absolute"); return false; } return true; } bool cmBinUtilsMacOSMachOLinker::ResolveExecutablePathDependency( std::string const& name, std::string const& executablePath, std::string& path, bool& resolved) { if (executablePath.empty()) { resolved = false; return true; } // 16 is == "@executable_path".length() path = name; path.replace(0, 16, executablePath); if (!cmSystemTools::PathExists(path)) { resolved = false; return true; } resolved = true; return true; } bool cmBinUtilsMacOSMachOLinker::ResolveLoaderPathDependency( std::string const& name, std::string const& loaderPath, std::string& path, bool& resolved) { if (loaderPath.empty()) { resolved = false; return true; } // 12 is "@loader_path".length(); path = name; path.replace(0, 12, loaderPath); if (!cmSystemTools::PathExists(path)) { resolved = false; return true; } resolved = true; return true; } bool cmBinUtilsMacOSMachOLinker::ResolveRPathDependency( std::string const& name, std::string const& executablePath, std::string const& loaderPath, std::vector const& rpaths, std::string& path, bool& resolved) { for (std::string const& rpath : rpaths) { std::string searchFile = name; searchFile.replace(0, 6, rpath); if (cmHasLiteralPrefix(searchFile, "@loader_path/")) { if (!this->ResolveLoaderPathDependency(searchFile, loaderPath, path, resolved)) { return false; } if (resolved) { return true; } } else if (cmHasLiteralPrefix(searchFile, "@executable_path/")) { if (!this->ResolveExecutablePathDependency(searchFile, executablePath, path, resolved)) { return false; } if (resolved) { return true; } } else if (cmSystemTools::PathExists(searchFile)) { /* * paraphrasing @ben.boeckel: * if /b/libB.dylib is supposed to be used, * /a/libbB.dylib will be found first if it exists. CMake tries to * sort rpath directories to avoid this, but sometimes there is no * right answer. * * I believe it is possible to resolve this using otools -l * then checking the LC_LOAD_DYLIB command whose name is * equal to the value of search_file, UNLESS the build * specifically sets the RPath to paths that will match * duplicate libs; at this point can we just point to * user error, or is there a reason why the advantages * to this scenario outweigh its disadvantages? * * Also priority seems to be the order as passed in when compiled * so as long as this method's resolution guarantees priority * in that manner further checking should not be necessary? */ path = searchFile; resolved = true; return true; } } resolved = false; return true; }