| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214 |
- /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
- file Copyright.txt or https://cmake.org/licensing for details. */
- #include "cmConfigureFileCommand.h"
- #include <set>
- #include <sstream>
- #include <cm/string_view>
- #include <cmext/string_view>
- #include <sys/types.h>
- #include "cmExecutionStatus.h"
- #include "cmFSPermissions.h"
- #include "cmMakefile.h"
- #include "cmMessageType.h"
- #include "cmNewLineStyle.h"
- #include "cmStringAlgorithms.h"
- #include "cmSystemTools.h"
- // cmConfigureFileCommand
- bool cmConfigureFileCommand(std::vector<std::string> const& args,
- cmExecutionStatus& status)
- {
- if (args.size() < 2) {
- status.SetError("called with incorrect number of arguments, expected 2");
- return false;
- }
- std::string const& inFile = args[0];
- const std::string inputFile = cmSystemTools::CollapseFullPath(
- inFile, status.GetMakefile().GetCurrentSourceDirectory());
- // If the input location is a directory, error out.
- if (cmSystemTools::FileIsDirectory(inputFile)) {
- status.SetError(cmStrCat("input location\n ", inputFile,
- "\n"
- "is a directory but a file was expected."));
- return false;
- }
- std::string const& outFile = args[1];
- std::string outputFile = cmSystemTools::CollapseFullPath(
- outFile, status.GetMakefile().GetCurrentBinaryDirectory());
- // If the output location is already a directory put the file in it.
- if (cmSystemTools::FileIsDirectory(outputFile)) {
- outputFile += "/";
- outputFile += cmSystemTools::GetFilenameName(inFile);
- }
- if (!status.GetMakefile().CanIWriteThisFile(outputFile)) {
- std::string e = "attempted to configure a file: " + outputFile +
- " into a source directory.";
- status.SetError(e);
- cmSystemTools::SetFatalErrorOccured();
- return false;
- }
- std::string errorMessage;
- cmNewLineStyle newLineStyle;
- if (!newLineStyle.ReadFromArguments(args, errorMessage)) {
- status.SetError(errorMessage);
- return false;
- }
- bool copyOnly = false;
- bool escapeQuotes = false;
- bool useSourcePermissions = false;
- bool noSourcePermissions = false;
- bool filePermissions = false;
- std::vector<std::string> filePermissionOptions;
- enum class Doing
- {
- DoingNone,
- DoingFilePermissions,
- DoneFilePermissions
- };
- Doing doing = Doing::DoingNone;
- static std::set<cm::string_view> noopOptions = {
- /* Legacy. */
- "IMMEDIATE"_s,
- /* Handled by NewLineStyle member. */
- "NEWLINE_STYLE"_s,
- "LF"_s,
- "UNIX"_s,
- "CRLF"_s,
- "WIN32"_s,
- "DOS"_s,
- };
- std::string unknown_args;
- bool atOnly = false;
- for (unsigned int i = 2; i < args.size(); ++i) {
- if (args[i] == "COPYONLY") {
- if (doing == Doing::DoingFilePermissions) {
- doing = Doing::DoneFilePermissions;
- }
- copyOnly = true;
- if (newLineStyle.IsValid()) {
- status.SetError("COPYONLY could not be used in combination "
- "with NEWLINE_STYLE");
- return false;
- }
- } else if (args[i] == "ESCAPE_QUOTES") {
- if (doing == Doing::DoingFilePermissions) {
- doing = Doing::DoneFilePermissions;
- }
- escapeQuotes = true;
- } else if (args[i] == "@ONLY") {
- if (doing == Doing::DoingFilePermissions) {
- doing = Doing::DoneFilePermissions;
- }
- atOnly = true;
- } else if (args[i] == "NO_SOURCE_PERMISSIONS") {
- if (doing == Doing::DoingFilePermissions) {
- status.SetError(" given both FILE_PERMISSIONS and "
- "NO_SOURCE_PERMISSIONS. Only one option allowed.");
- return false;
- }
- noSourcePermissions = true;
- } else if (args[i] == "USE_SOURCE_PERMISSIONS") {
- if (doing == Doing::DoingFilePermissions) {
- status.SetError(" given both FILE_PERMISSIONS and "
- "USE_SOURCE_PERMISSIONS. Only one option allowed.");
- return false;
- }
- useSourcePermissions = true;
- } else if (args[i] == "FILE_PERMISSIONS") {
- if (useSourcePermissions) {
- status.SetError(" given both FILE_PERMISSIONS and "
- "USE_SOURCE_PERMISSIONS. Only one option allowed.");
- return false;
- }
- if (noSourcePermissions) {
- status.SetError(" given both FILE_PERMISSIONS and "
- "NO_SOURCE_PERMISSIONS. Only one option allowed.");
- return false;
- }
- if (doing == Doing::DoingNone) {
- doing = Doing::DoingFilePermissions;
- filePermissions = true;
- }
- } else if (noopOptions.find(args[i]) != noopOptions.end()) {
- /* Ignore no-op options. */
- } else if (doing == Doing::DoingFilePermissions) {
- filePermissionOptions.push_back(args[i]);
- } else {
- unknown_args += " ";
- unknown_args += args[i];
- unknown_args += "\n";
- }
- }
- if (!unknown_args.empty()) {
- std::string msg = cmStrCat(
- "configure_file called with unknown argument(s):\n", unknown_args);
- status.GetMakefile().IssueMessage(MessageType::AUTHOR_WARNING, msg);
- }
- if (useSourcePermissions && noSourcePermissions) {
- status.SetError(" given both USE_SOURCE_PERMISSIONS and "
- "NO_SOURCE_PERMISSIONS. Only one option allowed.");
- return false;
- }
- mode_t permissions = 0;
- if (filePermissions) {
- if (filePermissionOptions.empty()) {
- status.SetError(" given FILE_PERMISSIONS without any options.");
- return false;
- }
- std::vector<std::string> invalidOptions;
- for (auto const& e : filePermissionOptions) {
- if (!cmFSPermissions::stringToModeT(e, permissions)) {
- invalidOptions.push_back(e);
- }
- }
- if (!invalidOptions.empty()) {
- std::ostringstream oss;
- oss << " given invalid permission ";
- for (auto i = 0u; i < invalidOptions.size(); i++) {
- if (i == 0u) {
- oss << "\"" << invalidOptions[i] << "\"";
- } else {
- oss << ",\"" << invalidOptions[i] << "\"";
- }
- }
- oss << ".";
- status.SetError(oss.str());
- return false;
- }
- }
- if (noSourcePermissions) {
- permissions |= cmFSPermissions::mode_owner_read;
- permissions |= cmFSPermissions::mode_owner_write;
- permissions |= cmFSPermissions::mode_group_read;
- permissions |= cmFSPermissions::mode_world_read;
- }
- if (!status.GetMakefile().ConfigureFile(inputFile, outputFile, copyOnly,
- atOnly, escapeQuotes, permissions,
- newLineStyle)) {
- status.SetError("Problem configuring file");
- return false;
- }
- return true;
- }
|