cmFileSet.cxx 6.3 KB

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