cmFileSet.cxx 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. file LICENSE.rst or https://cmake.org/licensing for details. */
  3. #include "cmFileSet.h"
  4. #include <sstream>
  5. #include <string>
  6. #include <unordered_map>
  7. #include <utility>
  8. #include <vector>
  9. #include <cm/optional>
  10. #include <cmext/algorithm>
  11. #include <cmext/string_view>
  12. #include "cmsys/RegularExpression.hxx"
  13. #include "cmGeneratorExpression.h"
  14. #include "cmList.h"
  15. #include "cmListFileCache.h"
  16. #include "cmLocalGenerator.h"
  17. #include "cmMakefile.h"
  18. #include "cmMessageType.h"
  19. #include "cmStringAlgorithms.h"
  20. #include "cmSystemTools.h"
  21. #include "cmake.h"
  22. cm::static_string_view cmFileSetVisibilityToName(cmFileSetVisibility vis)
  23. {
  24. switch (vis) {
  25. case cmFileSetVisibility::Interface:
  26. return "INTERFACE"_s;
  27. case cmFileSetVisibility::Public:
  28. return "PUBLIC"_s;
  29. case cmFileSetVisibility::Private:
  30. return "PRIVATE"_s;
  31. }
  32. return ""_s;
  33. }
  34. cmFileSetVisibility cmFileSetVisibilityFromName(cm::string_view name,
  35. cmMakefile* mf)
  36. {
  37. if (name == "INTERFACE"_s) {
  38. return cmFileSetVisibility::Interface;
  39. }
  40. if (name == "PUBLIC"_s) {
  41. return cmFileSetVisibility::Public;
  42. }
  43. if (name == "PRIVATE"_s) {
  44. return cmFileSetVisibility::Private;
  45. }
  46. auto msg = cmStrCat("File set visibility \"", name, "\" is not valid.");
  47. if (mf) {
  48. mf->IssueMessage(MessageType::FATAL_ERROR, msg);
  49. } else {
  50. cmSystemTools::Error(msg);
  51. }
  52. return cmFileSetVisibility::Private;
  53. }
  54. bool cmFileSetVisibilityIsForSelf(cmFileSetVisibility vis)
  55. {
  56. switch (vis) {
  57. case cmFileSetVisibility::Interface:
  58. return false;
  59. case cmFileSetVisibility::Public:
  60. case cmFileSetVisibility::Private:
  61. return true;
  62. }
  63. return false;
  64. }
  65. bool cmFileSetVisibilityIsForInterface(cmFileSetVisibility vis)
  66. {
  67. switch (vis) {
  68. case cmFileSetVisibility::Interface:
  69. case cmFileSetVisibility::Public:
  70. return true;
  71. case cmFileSetVisibility::Private:
  72. return false;
  73. }
  74. return false;
  75. }
  76. bool cmFileSetTypeCanBeIncluded(std::string const& type)
  77. {
  78. return type == "HEADERS"_s;
  79. }
  80. cmFileSet::cmFileSet(cmake& cmakeInstance, std::string name, std::string type,
  81. cmFileSetVisibility visibility)
  82. : CMakeInstance(cmakeInstance)
  83. , Name(std::move(name))
  84. , Type(std::move(type))
  85. , Visibility(visibility)
  86. {
  87. }
  88. void cmFileSet::CopyEntries(cmFileSet const* fs)
  89. {
  90. cm::append(this->DirectoryEntries, fs->DirectoryEntries);
  91. cm::append(this->FileEntries, fs->FileEntries);
  92. }
  93. void cmFileSet::ClearDirectoryEntries()
  94. {
  95. this->DirectoryEntries.clear();
  96. }
  97. void cmFileSet::AddDirectoryEntry(BT<std::string> directories)
  98. {
  99. this->DirectoryEntries.push_back(std::move(directories));
  100. }
  101. void cmFileSet::ClearFileEntries()
  102. {
  103. this->FileEntries.clear();
  104. }
  105. void cmFileSet::AddFileEntry(BT<std::string> files)
  106. {
  107. this->FileEntries.push_back(std::move(files));
  108. }
  109. std::vector<std::unique_ptr<cmCompiledGeneratorExpression>>
  110. cmFileSet::CompileFileEntries() const
  111. {
  112. std::vector<std::unique_ptr<cmCompiledGeneratorExpression>> result;
  113. for (auto const& entry : this->FileEntries) {
  114. for (auto const& ex : cmList{ entry.Value }) {
  115. cmGeneratorExpression ge(this->CMakeInstance, entry.Backtrace);
  116. auto cge = ge.Parse(ex);
  117. result.push_back(std::move(cge));
  118. }
  119. }
  120. return result;
  121. }
  122. std::vector<std::unique_ptr<cmCompiledGeneratorExpression>>
  123. cmFileSet::CompileDirectoryEntries() const
  124. {
  125. std::vector<std::unique_ptr<cmCompiledGeneratorExpression>> result;
  126. for (auto const& entry : this->DirectoryEntries) {
  127. for (auto const& ex : cmList{ entry.Value }) {
  128. cmGeneratorExpression ge(this->CMakeInstance, entry.Backtrace);
  129. auto cge = ge.Parse(ex);
  130. result.push_back(std::move(cge));
  131. }
  132. }
  133. return result;
  134. }
  135. std::vector<std::string> cmFileSet::EvaluateDirectoryEntries(
  136. std::vector<std::unique_ptr<cmCompiledGeneratorExpression>> const& cges,
  137. cmLocalGenerator const* lg, std::string const& config,
  138. cmGeneratorTarget const* target,
  139. cmGeneratorExpressionDAGChecker* dagChecker) const
  140. {
  141. struct DirCacheEntry
  142. {
  143. std::string collapsedDir;
  144. cm::optional<cmSystemTools::FileId> fileId;
  145. };
  146. std::unordered_map<std::string, DirCacheEntry> dirCache;
  147. std::vector<std::string> result;
  148. for (auto const& cge : cges) {
  149. auto entry = cge->Evaluate(lg, config, target, dagChecker);
  150. cmList dirs{ entry };
  151. for (std::string dir : dirs) {
  152. if (!cmSystemTools::FileIsFullPath(dir)) {
  153. dir = cmStrCat(lg->GetCurrentSourceDirectory(), '/', dir);
  154. }
  155. auto dirCacheResult = dirCache.emplace(dir, DirCacheEntry());
  156. auto& dirCacheEntry = dirCacheResult.first->second;
  157. auto const isNewCacheEntry = dirCacheResult.second;
  158. if (isNewCacheEntry) {
  159. cmSystemTools::FileId fileId;
  160. auto isFileIdValid = cmSystemTools::GetFileId(dir, fileId);
  161. dirCacheEntry.collapsedDir = cmSystemTools::CollapseFullPath(dir);
  162. dirCacheEntry.fileId =
  163. isFileIdValid ? cm::optional<decltype(fileId)>(fileId) : cm::nullopt;
  164. }
  165. for (auto const& priorDir : result) {
  166. auto priorDirCacheEntry = dirCache.at(priorDir);
  167. bool sameFile = dirCacheEntry.fileId.has_value() &&
  168. priorDirCacheEntry.fileId.has_value() &&
  169. (*dirCacheEntry.fileId == *priorDirCacheEntry.fileId);
  170. if (!sameFile &&
  171. (cmSystemTools::IsSubDirectory(dirCacheEntry.collapsedDir,
  172. priorDirCacheEntry.collapsedDir) ||
  173. cmSystemTools::IsSubDirectory(priorDirCacheEntry.collapsedDir,
  174. dirCacheEntry.collapsedDir))) {
  175. lg->GetCMakeInstance()->IssueMessage(
  176. MessageType::FATAL_ERROR,
  177. cmStrCat(
  178. "Base directories in file set cannot be subdirectories of each "
  179. "other:\n ",
  180. priorDir, "\n ", dir),
  181. cge->GetBacktrace());
  182. return {};
  183. }
  184. }
  185. result.push_back(dir);
  186. }
  187. }
  188. return result;
  189. }
  190. void cmFileSet::EvaluateFileEntry(
  191. std::vector<std::string> const& dirs,
  192. std::map<std::string, std::vector<std::string>>& filesPerDir,
  193. std::unique_ptr<cmCompiledGeneratorExpression> const& cge,
  194. cmLocalGenerator const* lg, std::string const& config,
  195. cmGeneratorTarget const* target,
  196. cmGeneratorExpressionDAGChecker* dagChecker) const
  197. {
  198. auto files = cge->Evaluate(lg, config, target, dagChecker);
  199. for (std::string file : cmList{ files }) {
  200. if (!cmSystemTools::FileIsFullPath(file)) {
  201. file = cmStrCat(lg->GetCurrentSourceDirectory(), '/', file);
  202. }
  203. auto collapsedFile = cmSystemTools::CollapseFullPath(file);
  204. bool found = false;
  205. std::string relDir;
  206. for (auto const& dir : dirs) {
  207. auto collapsedDir = cmSystemTools::CollapseFullPath(dir);
  208. if (cmSystemTools::IsSubDirectory(collapsedFile, collapsedDir)) {
  209. found = true;
  210. relDir = cmSystemTools::GetParentDirectory(
  211. cmSystemTools::RelativePath(collapsedDir, collapsedFile));
  212. break;
  213. }
  214. }
  215. if (!found) {
  216. std::ostringstream e;
  217. e << "File:\n " << file
  218. << "\nmust be in one of the file set's base directories:";
  219. for (auto const& dir : dirs) {
  220. e << "\n " << dir;
  221. }
  222. lg->GetCMakeInstance()->IssueMessage(MessageType::FATAL_ERROR, e.str(),
  223. cge->GetBacktrace());
  224. return;
  225. }
  226. filesPerDir[relDir].push_back(file);
  227. }
  228. }
  229. bool cmFileSet::IsValidName(std::string const& name)
  230. {
  231. static cmsys::RegularExpression const regex("^[a-z0-9][a-zA-Z0-9_]*$");
  232. cmsys::RegularExpressionMatch match;
  233. return regex.find(name.c_str(), match);
  234. }