/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying file Copyright.txt or https://cmake.org/licensing for details. */ #include "cmLinkLineComputer.h" #include #include #include #include "cmComputeLinkInformation.h" #include "cmGeneratorTarget.h" #include "cmLinkItem.h" #include "cmListFileCache.h" #include "cmOutputConverter.h" #include "cmStateDirectory.h" #include "cmStateTypes.h" #include "cmStringAlgorithms.h" #include "cmSystemTools.h" cmLinkLineComputer::cmLinkLineComputer(cmOutputConverter* outputConverter, cmStateDirectory const& stateDir) : StateDir(stateDir) , OutputConverter(outputConverter) , ForResponse(false) , UseWatcomQuote(false) , Relink(false) { } cmLinkLineComputer::~cmLinkLineComputer() = default; void cmLinkLineComputer::SetUseWatcomQuote(bool useWatcomQuote) { this->UseWatcomQuote = useWatcomQuote; } void cmLinkLineComputer::SetForResponse(bool forResponse) { this->ForResponse = forResponse; } void cmLinkLineComputer::SetRelink(bool relink) { this->Relink = relink; } std::string cmLinkLineComputer::ConvertToLinkReference( std::string const& lib) const { std::string relLib = lib; if (this->StateDir.ContainsBoth(this->StateDir.GetCurrentBinary(), lib)) { relLib = cmSystemTools::ForceToRelativePath( this->StateDir.GetCurrentBinary(), lib); } return relLib; } std::string cmLinkLineComputer::ComputeLinkLibs(cmComputeLinkInformation& cli) { std::string linkLibs; std::vector> linkLibsList; this->ComputeLinkLibs(cli, linkLibsList); cli.AppendValues(linkLibs, linkLibsList); return linkLibs; } void cmLinkLineComputer::ComputeLinkLibs( cmComputeLinkInformation& cli, std::vector>& linkLibraries) { using ItemVector = cmComputeLinkInformation::ItemVector; ItemVector const& items = cli.GetItems(); for (auto const& item : items) { if (item.Target && item.Target->GetType() == cmStateEnums::INTERFACE_LIBRARY) { continue; } BT linkLib; if (item.IsPath) { linkLib.Value += cli.GetLibLinkFileFlag(); linkLib.Value += this->ConvertToOutputFormat(this->ConvertToLinkReference(item.Value)); } else { linkLib.Value += item.Value; } linkLib.Value += " "; const cmLinkImplementation* linkImpl = cli.GetTarget()->GetLinkImplementation(cli.GetConfig()); for (const cmLinkImplItem& iter : linkImpl->Libraries) { if (iter.Target != nullptr && iter.Target->GetType() != cmStateEnums::INTERFACE_LIBRARY) { std::string libPath = iter.Target->GetLocation(cli.GetConfig()); if (item.Value == libPath) { linkLib.Backtrace = iter.Backtrace; break; } } } linkLibraries.emplace_back(linkLib); } } std::string cmLinkLineComputer::ConvertToOutputFormat(std::string const& input) { cmOutputConverter::OutputFormat shellFormat = (this->ForResponse) ? cmOutputConverter::RESPONSE : ((this->UseWatcomQuote) ? cmOutputConverter::WATCOMQUOTE : cmOutputConverter::SHELL); return this->OutputConverter->ConvertToOutputFormat(input, shellFormat); } std::string cmLinkLineComputer::ConvertToOutputForExisting( std::string const& input) { cmOutputConverter::OutputFormat shellFormat = (this->ForResponse) ? cmOutputConverter::RESPONSE : ((this->UseWatcomQuote) ? cmOutputConverter::WATCOMQUOTE : cmOutputConverter::SHELL); return this->OutputConverter->ConvertToOutputForExisting(input, shellFormat); } std::string cmLinkLineComputer::ComputeLinkPath( cmComputeLinkInformation& cli, std::string const& libPathFlag, std::string const& libPathTerminator) { std::string linkPath; std::vector> linkPathList; this->ComputeLinkPath(cli, libPathFlag, libPathTerminator, linkPathList); cli.AppendValues(linkPath, linkPathList); return linkPath; } void cmLinkLineComputer::ComputeLinkPath( cmComputeLinkInformation& cli, std::string const& libPathFlag, std::string const& libPathTerminator, std::vector>& linkPath) { if (cli.GetLinkLanguage() == "Swift") { std::string linkPathNoBT; for (const cmComputeLinkInformation::Item& item : cli.GetItems()) { const cmGeneratorTarget* target = item.Target; if (!target) { continue; } if (target->GetType() == cmStateEnums::STATIC_LIBRARY || target->GetType() == cmStateEnums::SHARED_LIBRARY) { cmStateEnums::ArtifactType type = cmStateEnums::RuntimeBinaryArtifact; if (target->HasImportLibrary(cli.GetConfig())) { type = cmStateEnums::ImportLibraryArtifact; } linkPathNoBT += cmStrCat( " ", libPathFlag, item.Target->GetDirectory(cli.GetConfig(), type), libPathTerminator, " "); } } if (!linkPathNoBT.empty()) { linkPath.emplace_back(std::move(linkPathNoBT)); } } for (BT libDir : cli.GetDirectoriesWithBacktraces()) { libDir.Value = cmStrCat(" ", libPathFlag, this->ConvertToOutputForExisting(libDir.Value), libPathTerminator, " "); linkPath.emplace_back(libDir); } } std::string cmLinkLineComputer::ComputeRPath(cmComputeLinkInformation& cli) { std::string rpath; // Check what kind of rpath flags to use. if (cli.GetRuntimeSep().empty()) { // Each rpath entry gets its own option ("-R a -R b -R c") std::vector runtimeDirs; cli.GetRPath(runtimeDirs, this->Relink); for (std::string const& rd : runtimeDirs) { rpath += cli.GetRuntimeFlag(); rpath += this->ConvertToOutputFormat(rd); rpath += " "; } } else { // All rpath entries are combined ("-Wl,-rpath,a:b:c"). std::string rpathString = cli.GetRPathString(this->Relink); // Store the rpath option in the stream. if (!rpathString.empty()) { rpath += cli.GetRuntimeFlag(); rpath += this->OutputConverter->EscapeForShell(rpathString, !this->ForResponse); rpath += " "; } } return rpath; } std::string cmLinkLineComputer::ComputeFrameworkPath( cmComputeLinkInformation& cli, std::string const& fwSearchFlag) { std::string frameworkPath; if (!fwSearchFlag.empty()) { std::vector const& fwDirs = cli.GetFrameworkPaths(); for (std::string const& fd : fwDirs) { frameworkPath += fwSearchFlag; frameworkPath += this->ConvertToOutputFormat(fd); frameworkPath += " "; } } return frameworkPath; } std::string cmLinkLineComputer::ComputeLinkLibraries( cmComputeLinkInformation& cli, std::string const& stdLibString) { std::ostringstream fout; fout << this->ComputeRPath(cli); // Write the library flags to the build rule. fout << this->ComputeLinkLibs(cli); // Add the linker runtime search path if any. std::string rpath_link = cli.GetRPathLinkString(); if (!cli.GetRPathLinkFlag().empty() && !rpath_link.empty()) { fout << cli.GetRPathLinkFlag(); fout << this->OutputConverter->EscapeForShell(rpath_link, !this->ForResponse); fout << " "; } if (!stdLibString.empty()) { fout << stdLibString << " "; } return fout.str(); } std::string cmLinkLineComputer::GetLinkerLanguage(cmGeneratorTarget* target, std::string const& config) { return target->GetLinkerLanguage(config); }