cmComputeLinkInformation.h 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. file Copyright.txt or https://cmake.org/licensing for details. */
  3. #pragma once
  4. #include "cmConfigure.h" // IWYU pragma: keep
  5. #include <map>
  6. #include <memory>
  7. #include <set>
  8. #include <string>
  9. #include <utility>
  10. #include <vector>
  11. #include "cmsys/RegularExpression.hxx"
  12. #include "cmComputeLinkDepends.h"
  13. #include "cmListFileCache.h"
  14. #include "cmValue.h"
  15. class cmGeneratorTarget;
  16. class cmGlobalGenerator;
  17. class cmMakefile;
  18. class cmOrderDirectories;
  19. class cmSourceFile;
  20. class cmake;
  21. /** \class cmComputeLinkInformation
  22. * \brief Compute link information for a target in one configuration.
  23. */
  24. class cmComputeLinkInformation
  25. {
  26. private:
  27. class FeatureDescriptor;
  28. public:
  29. cmComputeLinkInformation(cmGeneratorTarget const* target,
  30. const std::string& config);
  31. cmComputeLinkInformation(const cmComputeLinkInformation&) = delete;
  32. cmComputeLinkInformation& operator=(const cmComputeLinkInformation&) =
  33. delete;
  34. ~cmComputeLinkInformation();
  35. bool Compute();
  36. enum class ItemIsPath
  37. {
  38. No,
  39. Yes,
  40. };
  41. struct Item
  42. {
  43. Item(BT<std::string> v, ItemIsPath isPath,
  44. cmGeneratorTarget const* target = nullptr,
  45. cmSourceFile const* objectSource = nullptr,
  46. FeatureDescriptor const* feature = nullptr)
  47. : Value(std::move(v))
  48. , IsPath(isPath)
  49. , Target(target)
  50. , ObjectSource(objectSource)
  51. , Feature(feature)
  52. {
  53. }
  54. BT<std::string> Value;
  55. ItemIsPath IsPath = ItemIsPath::No;
  56. cmGeneratorTarget const* Target = nullptr;
  57. // The source file representing the external object (used when linking
  58. // `$<TARGET_OBJECTS>`)
  59. cmSourceFile const* ObjectSource = nullptr;
  60. bool HasFeature() const { return this->Feature != nullptr; }
  61. const std::string& GetFeatureName() const
  62. {
  63. return HasFeature() ? this->Feature->Name
  64. : cmComputeLinkDepends::LinkEntry::DEFAULT;
  65. }
  66. BT<std::string> GetFormattedItem(std::string const& path) const
  67. {
  68. return { this->Feature
  69. ? this->Feature->GetDecoratedItem(path, this->IsPath)
  70. : path,
  71. Value.Backtrace };
  72. }
  73. private:
  74. FeatureDescriptor const* Feature = nullptr;
  75. };
  76. using ItemVector = std::vector<Item>;
  77. void AppendValues(std::string& result, std::vector<BT<std::string>>& values);
  78. ItemVector const& GetItems() const;
  79. std::vector<std::string> const& GetDirectories() const;
  80. std::vector<BT<std::string>> GetDirectoriesWithBacktraces();
  81. std::vector<std::string> const& GetDepends() const;
  82. std::vector<std::string> const& GetFrameworkPaths() const;
  83. std::set<std::string> const& GetFrameworkPathsEmitted() const;
  84. std::vector<std::string> const& GetXcFrameworkHeaderPaths() const;
  85. std::string GetLinkLanguage() const { return this->LinkLanguage; }
  86. std::vector<std::string> const& GetRuntimeSearchPath() const;
  87. std::string const& GetRuntimeFlag() const { return this->RuntimeFlag; }
  88. std::string const& GetRuntimeSep() const { return this->RuntimeSep; }
  89. void GetRPath(std::vector<std::string>& runtimeDirs, bool for_install) const;
  90. std::string GetRPathString(bool for_install) const;
  91. std::string GetChrpathString() const;
  92. std::set<cmGeneratorTarget const*> const& GetSharedLibrariesLinked() const;
  93. std::vector<cmGeneratorTarget const*> const& GetExternalObjectTargets()
  94. const;
  95. std::vector<cmGeneratorTarget const*> const& GetRuntimeDLLs() const
  96. {
  97. return this->RuntimeDLLs;
  98. }
  99. std::string const& GetLibLinkFileFlag() const
  100. {
  101. return this->LibLinkFileFlag;
  102. }
  103. std::string const& GetObjLinkFileFlag() const
  104. {
  105. return this->ObjLinkFileFlag;
  106. }
  107. std::string const& GetRPathLinkFlag() const { return this->RPathLinkFlag; }
  108. std::string GetRPathLinkString() const;
  109. std::string GetConfig() const { return this->Config; }
  110. const cmGeneratorTarget* GetTarget() { return this->Target; }
  111. private:
  112. using LinkEntry = cmComputeLinkDepends::LinkEntry;
  113. void AddItem(LinkEntry const& entry);
  114. void AddSharedDepItem(LinkEntry const& entry);
  115. void AddRuntimeDLL(cmGeneratorTarget const* tgt);
  116. // Output information.
  117. ItemVector Items;
  118. std::vector<std::string> Directories;
  119. std::vector<std::string> Depends;
  120. std::vector<std::string> FrameworkPaths;
  121. std::vector<std::string> XcFrameworkHeaderPaths;
  122. std::vector<std::string> RuntimeSearchPath;
  123. std::set<cmGeneratorTarget const*> SharedLibrariesLinked;
  124. std::vector<cmGeneratorTarget const*> ExternalObjectTargets;
  125. std::vector<cmGeneratorTarget const*> RuntimeDLLs;
  126. // Context information.
  127. cmGeneratorTarget const* const Target;
  128. cmMakefile* const Makefile;
  129. cmGlobalGenerator* const GlobalGenerator;
  130. cmake* const CMakeInstance;
  131. // Configuration information.
  132. std::string const Config;
  133. std::string LinkLanguage;
  134. // Modes for dealing with dependent shared libraries.
  135. enum SharedDepMode
  136. {
  137. SharedDepModeNone, // Drop
  138. SharedDepModeDir, // List dir in -rpath-link flag
  139. SharedDepModeLibDir, // List dir in linker search path
  140. SharedDepModeLink // List file on link line
  141. };
  142. cmValue LoaderFlag;
  143. std::string LibLinkFlag;
  144. std::string LibLinkFileFlag;
  145. std::string ObjLinkFileFlag;
  146. std::string LibLinkSuffix;
  147. std::string RuntimeFlag;
  148. std::string RuntimeSep;
  149. std::string RuntimeAlways;
  150. std::string RPathLinkFlag;
  151. SharedDepMode SharedDependencyMode;
  152. enum LinkType
  153. {
  154. LinkUnknown,
  155. LinkStatic,
  156. LinkShared
  157. };
  158. void SetCurrentLinkType(LinkType lt);
  159. // Link type adjustment.
  160. void ComputeLinkTypeInfo();
  161. LinkType StartLinkType;
  162. LinkType CurrentLinkType;
  163. std::string StaticLinkTypeFlag;
  164. std::string SharedLinkTypeFlag;
  165. // Link item parsing.
  166. void ComputeItemParserInfo();
  167. std::vector<std::string> StaticLinkExtensions;
  168. std::vector<std::string> SharedLinkExtensions;
  169. std::vector<std::string> LinkExtensions;
  170. std::set<std::string> LinkPrefixes;
  171. cmsys::RegularExpression ExtractStaticLibraryName;
  172. cmsys::RegularExpression ExtractSharedLibraryName;
  173. cmsys::RegularExpression ExtractAnyLibraryName;
  174. std::string SharedRegexString;
  175. void AddLinkPrefix(std::string const& p);
  176. void AddLinkExtension(std::string const& e, LinkType type);
  177. std::string CreateExtensionRegex(std::vector<std::string> const& exts,
  178. LinkType type);
  179. std::string NoCaseExpression(std::string const& str);
  180. // Handling of link items.
  181. void AddTargetItem(LinkEntry const& entry);
  182. void AddFullItem(LinkEntry const& entry);
  183. bool CheckImplicitDirItem(LinkEntry const& entry);
  184. void AddUserItem(LinkEntry const& entry);
  185. void AddFrameworkItem(LinkEntry const& entry);
  186. void AddXcFrameworkItem(LinkEntry const& entry);
  187. void DropDirectoryItem(BT<std::string> const& item);
  188. bool CheckSharedLibNoSOName(LinkEntry const& entry);
  189. void AddSharedLibNoSOName(LinkEntry const& entry);
  190. // Framework info.
  191. void ComputeFrameworkInfo();
  192. void AddFrameworkPath(std::string const& p);
  193. std::set<std::string> FrameworkPathsEmitted;
  194. void AddXcFrameworkHeaderPath(std::string const& p);
  195. // Linker search path computation.
  196. std::unique_ptr<cmOrderDirectories> OrderLinkerSearchPath;
  197. void AddExternalObjectTargets();
  198. // Implicit link libraries and directories for linker language.
  199. void LoadImplicitLinkInfo();
  200. void AddImplicitLinkInfo();
  201. void AddImplicitLinkInfo(std::string const& lang);
  202. void AddRuntimeLinkLibrary(std::string const& lang);
  203. std::set<std::string> ImplicitLinkDirs;
  204. std::set<std::string> ImplicitLinkLibs;
  205. // Additional paths configured by the runtime linker
  206. std::vector<std::string> RuntimeLinkDirs;
  207. // Dependent library path computation.
  208. std::unique_ptr<cmOrderDirectories> OrderDependentRPath;
  209. // Runtime path computation.
  210. std::unique_ptr<cmOrderDirectories> OrderRuntimeSearchPath;
  211. bool IsOpenBSD;
  212. bool LinkDependsNoShared;
  213. bool RuntimeUseChrpath;
  214. bool NoSONameUsesPath;
  215. bool LinkWithRuntimePath;
  216. bool LinkTypeEnabled;
  217. bool ArchivesMayBeShared;
  218. void AddLibraryRuntimeInfo(std::string const& fullPath,
  219. const cmGeneratorTarget* target);
  220. void AddLibraryRuntimeInfo(std::string const& fullPath);
  221. class FeatureDescriptor
  222. {
  223. public:
  224. FeatureDescriptor() = default;
  225. const std::string Name;
  226. const bool Supported = false;
  227. const std::string Prefix;
  228. const std::string Suffix;
  229. std::string GetDecoratedItem(std::string const& library,
  230. ItemIsPath isPath) const;
  231. std::string GetDecoratedItem(std::string const& library,
  232. std::string const& linkItem,
  233. std::string const& defaultValue,
  234. ItemIsPath isPath) const;
  235. protected:
  236. FeatureDescriptor(std::string name, std::string itemFormat);
  237. FeatureDescriptor(std::string name, std::string itemPathFormat,
  238. std::string itemNameFormat);
  239. FeatureDescriptor(std::string name, std::string prefix,
  240. std::string itemPathFormat, std::string itemNameFormat,
  241. std::string suffix);
  242. FeatureDescriptor(std::string name, std::string prefix, std::string suffix,
  243. bool isGroup);
  244. private:
  245. std::string ItemPathFormat;
  246. std::string ItemNameFormat;
  247. };
  248. class LibraryFeatureDescriptor : public FeatureDescriptor
  249. {
  250. public:
  251. LibraryFeatureDescriptor(std::string name, std::string itemFormat);
  252. LibraryFeatureDescriptor(std::string name, std::string itemPathFormat,
  253. std::string itemNameFormat);
  254. LibraryFeatureDescriptor(std::string name, std::string prefix,
  255. std::string itemPathFormat,
  256. std::string itemNameFormat, std::string suffix);
  257. };
  258. std::map<std::string, FeatureDescriptor> LibraryFeatureDescriptors;
  259. bool AddLibraryFeature(std::string const& feature);
  260. FeatureDescriptor const& GetLibraryFeature(std::string const& feature) const;
  261. FeatureDescriptor const* FindLibraryFeature(
  262. std::string const& feature) const;
  263. class GroupFeatureDescriptor : public FeatureDescriptor
  264. {
  265. public:
  266. GroupFeatureDescriptor(std::string name, std::string prefix,
  267. std::string suffix);
  268. };
  269. std::map<std::string, FeatureDescriptor> GroupFeatureDescriptors;
  270. FeatureDescriptor const& GetGroupFeature(std::string const& feature);
  271. };