/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying file Copyright.txt or https://cmake.org/licensing for details. */ #include "cmSourceGroupCommand.h" #include #include #include #include #include "cmMakefile.h" #include "cmSourceGroup.h" #include "cmSystemTools.h" namespace { const size_t RootIndex = 1; const size_t FilesWithoutPrefixKeywordIndex = 2; const size_t FilesWithPrefixKeywordIndex = 4; const size_t PrefixKeywordIdex = 2; std::vector tokenizePath(const std::string& path) { return cmSystemTools::tokenize(path, "\\/"); } std::string getFullFilePath(const std::string& currentPath, const std::string& path) { std::string fullPath = path; if (!cmSystemTools::FileIsFullPath(path.c_str())) { fullPath = currentPath; fullPath += "/"; fullPath += path; } return cmSystemTools::CollapseFullPath(fullPath); } std::set getSourceGroupFilesPaths( const std::string& root, const std::vector& files) { std::set ret; const std::string::size_type rootLength = root.length(); for (size_t i = 0; i < files.size(); ++i) { ret.insert(files[i].substr(rootLength + 1)); // +1 to also omnit last '/' } return ret; } bool rootIsPrefix(const std::string& root, const std::vector& files, std::string& error) { for (size_t i = 0; i < files.size(); ++i) { if (!cmSystemTools::StringStartsWith(files[i], root.c_str())) { error = "ROOT: " + root + " is not a prefix of file: " + files[i]; return false; } } return true; } cmSourceGroup* addSourceGroup(const std::vector& tokenizedPath, cmMakefile& makefile) { cmSourceGroup* sg; sg = makefile.GetSourceGroup(tokenizedPath); if (!sg) { makefile.AddSourceGroup(tokenizedPath); sg = makefile.GetSourceGroup(tokenizedPath); if (!sg) { return CM_NULLPTR; } } return sg; } std::string prepareFilePathForTree(const std::string& path) { return cmSystemTools::CollapseFullPath(path); } std::vector prepareFilesPathsForTree( std::vector::const_iterator begin, std::vector::const_iterator end) { std::vector prepared(std::distance(begin, end)); std::transform(begin, end, prepared.begin(), prepareFilePathForTree); return prepared; } bool addFilesToItsSourceGroups(const std::string& root, const std::set& sgFilesPaths, const std::string& prefix, cmMakefile& makefile, std::string& errorMsg) { cmSourceGroup* sg; for (std::set::const_iterator it = sgFilesPaths.begin(); it != sgFilesPaths.end(); ++it) { std::vector tokenizedPath; if (!prefix.empty()) { tokenizedPath = tokenizePath(prefix + '/' + *it); } else { tokenizedPath = tokenizePath(*it); } if (tokenizedPath.size() > 1) { tokenizedPath.pop_back(); sg = addSourceGroup(tokenizedPath, makefile); if (!sg) { errorMsg = "Could not create source group for file: " + *it; return false; } const std::string fullPath = getFullFilePath(root, *it); sg->AddGroupFile(fullPath); } } return true; } } class cmExecutionStatus; // cmSourceGroupCommand bool cmSourceGroupCommand::InitialPass(std::vector const& args, cmExecutionStatus&) { if (args.empty()) { this->SetError("called with incorrect number of arguments"); return false; } if (args[0] == "TREE") { std::string error; if (!processTree(args, error)) { this->SetError(error); return false; } return true; } std::string delimiter = "\\"; if (this->Makefile->GetDefinition("SOURCE_GROUP_DELIMITER")) { delimiter = this->Makefile->GetDefinition("SOURCE_GROUP_DELIMITER"); } std::vector folders = cmSystemTools::tokenize(args[0], delimiter); cmSourceGroup* sg = CM_NULLPTR; sg = this->Makefile->GetSourceGroup(folders); if (!sg) { this->Makefile->AddSourceGroup(folders); sg = this->Makefile->GetSourceGroup(folders); } if (!sg) { this->SetError("Could not create or find source group"); return false; } // If only two arguments are given, the pre-1.8 version of the // command is being invoked. if (args.size() == 2 && args[1] != "FILES") { sg->SetGroupRegex(args[1].c_str()); return true; } // Process arguments. bool doingFiles = false; for (unsigned int i = 1; i < args.size(); ++i) { if (args[i] == "REGULAR_EXPRESSION") { // Next argument must specify the regex. if (i + 1 < args.size()) { ++i; sg->SetGroupRegex(args[i].c_str()); } else { this->SetError("REGULAR_EXPRESSION argument given without a regex."); return false; } doingFiles = false; } else if (args[i] == "FILES") { // Next arguments will specify files. doingFiles = true; } else if (doingFiles) { // Convert name to full path and add to the group's list. std::string src = args[i]; if (!cmSystemTools::FileIsFullPath(src.c_str())) { src = this->Makefile->GetCurrentSourceDirectory(); src += "/"; src += args[i]; } src = cmSystemTools::CollapseFullPath(src); sg->AddGroupFile(src); } else { std::ostringstream err; err << "Unknown argument \"" << args[i] << "\". " << "Perhaps the FILES keyword is missing.\n"; this->SetError(err.str()); return false; } } return true; } bool cmSourceGroupCommand::checkTreeArgumentsPreconditions( const std::vector& args, std::string& errorMsg) const { if (args.size() == 1) { errorMsg = "TREE argument given without a root."; return false; } if (args.size() < 3) { errorMsg = "Missing FILES arguments."; return false; } if (args[FilesWithoutPrefixKeywordIndex] != "FILES" && args[PrefixKeywordIdex] != "PREFIX") { errorMsg = "Unknown argument \"" + args[2] + "\". Perhaps the FILES keyword is missing.\n"; return false; } if (args[PrefixKeywordIdex] == "PREFIX" && (args.size() < 5 || args[FilesWithPrefixKeywordIndex] != "FILES")) { errorMsg = "Missing FILES arguments."; return false; } return true; } bool cmSourceGroupCommand::processTree(const std::vector& args, std::string& errorMsg) { if (!checkTreeArgumentsPreconditions(args, errorMsg)) { return false; } const std::string root = cmSystemTools::CollapseFullPath(args[RootIndex]); std::string prefix; size_t filesBegin = FilesWithoutPrefixKeywordIndex + 1; if (args[PrefixKeywordIdex] == "PREFIX") { prefix = args[PrefixKeywordIdex + 1]; filesBegin = FilesWithPrefixKeywordIndex + 1; } const std::vector filesVector = prepareFilesPathsForTree(args.begin() + filesBegin, args.end()); if (!rootIsPrefix(root, filesVector, errorMsg)) { return false; } std::set sourceGroupPaths = getSourceGroupFilesPaths(root, filesVector); if (!addFilesToItsSourceGroups(root, sourceGroupPaths, prefix, *(this->Makefile), errorMsg)) { return false; } return true; }