| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350 | /* Distributed under the OSI-approved BSD 3-Clause License.  See accompanying   file Copyright.txt or https://cmake.org/licensing for details.  */#include "cmFileInstaller.h"#include "cmFSPermissions.h"#include "cmFileCommand.h"#include "cmMakefile.h"#include "cmSystemTools.h"#include "cm_sys_stat.h"#include <sstream>using namespace cmFSPermissions;cmFileInstaller::cmFileInstaller(cmFileCommand* command)  : cmFileCopier(command, "INSTALL")  , InstallType(cmInstallType_FILES)  , Optional(false)  , MessageAlways(false)  , MessageLazy(false)  , MessageNever(false)  , DestDirLength(0){  // Installation does not use source permissions by default.  this->UseSourcePermissions = false;  // Check whether to copy files always or only if they have changed.  std::string install_always;  if (cmSystemTools::GetEnv("CMAKE_INSTALL_ALWAYS", install_always)) {    this->Always = cmSystemTools::IsOn(install_always);  }  // Get the current manifest.  this->Manifest =    this->Makefile->GetSafeDefinition("CMAKE_INSTALL_MANIFEST_FILES");}cmFileInstaller::~cmFileInstaller(){  // Save the updated install manifest.  this->Makefile->AddDefinition("CMAKE_INSTALL_MANIFEST_FILES",                                this->Manifest.c_str());}void cmFileInstaller::ManifestAppend(std::string const& file){  if (!this->Manifest.empty()) {    this->Manifest += ";";  }  this->Manifest += file.substr(this->DestDirLength);}std::string const& cmFileInstaller::ToName(std::string const& fromName){  return this->Rename.empty() ? fromName : this->Rename;}void cmFileInstaller::ReportCopy(const std::string& toFile, Type type,                                 bool copy){  if (!this->MessageNever && (copy || !this->MessageLazy)) {    std::string message = (copy ? "Installing: " : "Up-to-date: ");    message += toFile;    this->Makefile->DisplayStatus(message, -1);  }  if (type != TypeDir) {    // Add the file to the manifest.    this->ManifestAppend(toFile);  }}bool cmFileInstaller::ReportMissing(const std::string& fromFile){  return (this->Optional || this->cmFileCopier::ReportMissing(fromFile));}bool cmFileInstaller::Install(const std::string& fromFile,                              const std::string& toFile){  // Support installing from empty source to make a directory.  if (this->InstallType == cmInstallType_DIRECTORY && fromFile.empty()) {    return this->InstallDirectory(fromFile, toFile, MatchProperties());  }  return this->cmFileCopier::Install(fromFile, toFile);}void cmFileInstaller::DefaultFilePermissions(){  this->cmFileCopier::DefaultFilePermissions();  // Add execute permissions based on the target type.  switch (this->InstallType) {    case cmInstallType_SHARED_LIBRARY:    case cmInstallType_MODULE_LIBRARY:      if (this->Makefile->IsOn("CMAKE_INSTALL_SO_NO_EXE")) {        break;      }      CM_FALLTHROUGH;    case cmInstallType_EXECUTABLE:    case cmInstallType_PROGRAMS:      this->FilePermissions |= mode_owner_execute;      this->FilePermissions |= mode_group_execute;      this->FilePermissions |= mode_world_execute;      break;    default:      break;  }}bool cmFileInstaller::Parse(std::vector<std::string> const& args){  if (!this->cmFileCopier::Parse(args)) {    return false;  }  if (!this->Rename.empty()) {    if (!this->FilesFromDir.empty()) {      this->FileCommand->SetError("INSTALL option RENAME may not be "                                  "combined with FILES_FROM_DIR.");      return false;    }    if (this->InstallType != cmInstallType_FILES &&        this->InstallType != cmInstallType_PROGRAMS) {      this->FileCommand->SetError("INSTALL option RENAME may be used "                                  "only with FILES or PROGRAMS.");      return false;    }    if (this->Files.size() > 1) {      this->FileCommand->SetError("INSTALL option RENAME may be used "                                  "only with one file.");      return false;    }  }  if (!this->HandleInstallDestination()) {    return false;  }  if (((this->MessageAlways ? 1 : 0) + (this->MessageLazy ? 1 : 0) +       (this->MessageNever ? 1 : 0)) > 1) {    this->FileCommand->SetError("INSTALL options MESSAGE_ALWAYS, "                                "MESSAGE_LAZY, and MESSAGE_NEVER "                                "are mutually exclusive.");    return false;  }  return true;}bool cmFileInstaller::CheckKeyword(std::string const& arg){  if (arg == "TYPE") {    if (this->CurrentMatchRule) {      this->NotAfterMatch(arg);    } else {      this->Doing = DoingType;    }  } else if (arg == "FILES") {    if (this->CurrentMatchRule) {      this->NotAfterMatch(arg);    } else {      this->Doing = DoingFiles;    }  } else if (arg == "RENAME") {    if (this->CurrentMatchRule) {      this->NotAfterMatch(arg);    } else {      this->Doing = DoingRename;    }  } else if (arg == "OPTIONAL") {    if (this->CurrentMatchRule) {      this->NotAfterMatch(arg);    } else {      this->Doing = DoingNone;      this->Optional = true;    }  } else if (arg == "MESSAGE_ALWAYS") {    if (this->CurrentMatchRule) {      this->NotAfterMatch(arg);    } else {      this->Doing = DoingNone;      this->MessageAlways = true;    }  } else if (arg == "MESSAGE_LAZY") {    if (this->CurrentMatchRule) {      this->NotAfterMatch(arg);    } else {      this->Doing = DoingNone;      this->MessageLazy = true;    }  } else if (arg == "MESSAGE_NEVER") {    if (this->CurrentMatchRule) {      this->NotAfterMatch(arg);    } else {      this->Doing = DoingNone;      this->MessageNever = true;    }  } else if (arg == "PERMISSIONS") {    if (this->CurrentMatchRule) {      this->Doing = DoingPermissionsMatch;    } else {      // file(INSTALL) aliases PERMISSIONS to FILE_PERMISSIONS      this->Doing = DoingPermissionsFile;      this->UseGivenPermissionsFile = true;    }  } else if (arg == "DIR_PERMISSIONS") {    if (this->CurrentMatchRule) {      this->NotAfterMatch(arg);    } else {      // file(INSTALL) aliases DIR_PERMISSIONS to DIRECTORY_PERMISSIONS      this->Doing = DoingPermissionsDir;      this->UseGivenPermissionsDir = true;    }  } else if (arg == "COMPONENTS" || arg == "CONFIGURATIONS" ||             arg == "PROPERTIES") {    std::ostringstream e;    e << "INSTALL called with old-style " << arg << " argument.  "      << "This script was generated with an older version of CMake.  "      << "Re-run this cmake version on your build tree.";    this->FileCommand->SetError(e.str());    this->Doing = DoingError;  } else {    return this->cmFileCopier::CheckKeyword(arg);  }  return true;}bool cmFileInstaller::CheckValue(std::string const& arg){  switch (this->Doing) {    case DoingType:      if (!this->GetTargetTypeFromString(arg)) {        this->Doing = DoingError;      }      break;    case DoingRename:      this->Rename = arg;      break;    default:      return this->cmFileCopier::CheckValue(arg);  }  return true;}bool cmFileInstaller::GetTargetTypeFromString(const std::string& stype){  if (stype == "EXECUTABLE") {    this->InstallType = cmInstallType_EXECUTABLE;  } else if (stype == "FILE") {    this->InstallType = cmInstallType_FILES;  } else if (stype == "PROGRAM") {    this->InstallType = cmInstallType_PROGRAMS;  } else if (stype == "STATIC_LIBRARY") {    this->InstallType = cmInstallType_STATIC_LIBRARY;  } else if (stype == "SHARED_LIBRARY") {    this->InstallType = cmInstallType_SHARED_LIBRARY;  } else if (stype == "MODULE") {    this->InstallType = cmInstallType_MODULE_LIBRARY;  } else if (stype == "DIRECTORY") {    this->InstallType = cmInstallType_DIRECTORY;  } else {    std::ostringstream e;    e << "Option TYPE given unknown value \"" << stype << "\".";    this->FileCommand->SetError(e.str());    return false;  }  return true;}bool cmFileInstaller::HandleInstallDestination(){  std::string& destination = this->Destination;  // allow for / to be a valid destination  if (destination.size() < 2 && destination != "/") {    this->FileCommand->SetError("called with inappropriate arguments. "                                "No DESTINATION provided or .");    return false;  }  std::string sdestdir;  if (cmSystemTools::GetEnv("DESTDIR", sdestdir) && !sdestdir.empty()) {    cmSystemTools::ConvertToUnixSlashes(sdestdir);    char ch1 = destination[0];    char ch2 = destination[1];    char ch3 = 0;    if (destination.size() > 2) {      ch3 = destination[2];    }    int skip = 0;    if (ch1 != '/') {      int relative = 0;      if (((ch1 >= 'a' && ch1 <= 'z') || (ch1 >= 'A' && ch1 <= 'Z')) &&          ch2 == ':') {        // Assume windows        // let's do some destdir magic:        skip = 2;        if (ch3 != '/') {          relative = 1;        }      } else {        relative = 1;      }      if (relative) {        // This is relative path on unix or windows. Since we are doing        // destdir, this case does not make sense.        this->FileCommand->SetError(          "called with relative DESTINATION. This "          "does not make sense when using DESTDIR. Specify "          "absolute path or remove DESTDIR environment variable.");        return false;      }    } else {      if (ch2 == '/') {        // looks like a network path.        std::string message =          "called with network path DESTINATION. This "          "does not make sense when using DESTDIR. Specify local "          "absolute path or remove DESTDIR environment variable."          "\nDESTINATION=\n";        message += destination;        this->FileCommand->SetError(message);        return false;      }    }    destination = sdestdir + destination.substr(skip);    this->DestDirLength = int(sdestdir.size());  }  // check if default dir creation permissions were set  mode_t default_dir_mode_v = 0;  mode_t* default_dir_mode = &default_dir_mode_v;  if (!this->GetDefaultDirectoryPermissions(&default_dir_mode)) {    return false;  }  if (this->InstallType != cmInstallType_DIRECTORY) {    if (!cmSystemTools::FileExists(destination)) {      if (!cmSystemTools::MakeDirectory(destination, default_dir_mode)) {        std::string errstring = "cannot create directory: " + destination +          ". Maybe need administrative privileges.";        this->FileCommand->SetError(errstring);        return false;      }    }    if (!cmSystemTools::FileIsDirectory(destination)) {      std::string errstring =        "INSTALL destination: " + destination + " is not a directory.";      this->FileCommand->SetError(errstring);      return false;    }  }  return true;}
 |