| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983 | 
/* Distributed under the OSI-approved BSD 3-Clause License.  See accompanying   file Copyright.txt or https://cmake.org/licensing for details.  */#include "cmGlobalVisualStudioGenerator.h"#include <cassert>#include <future>#include <iostream>#include <sstream>#include <system_error>#include <utility>#include <cm/iterator>#include <cm/memory>#include <cmext/string_view>#include <windows.h>#include <objbase.h>#include <shellapi.h>#include "cmCallVisualStudioMacro.h"#include "cmCustomCommand.h"#include "cmCustomCommandLines.h"#include "cmGeneratedFileStream.h"#include "cmGeneratorTarget.h"#include "cmLocalGenerator.h"#include "cmMakefile.h"#include "cmMessageType.h"#include "cmPolicies.h"#include "cmSourceFile.h"#include "cmState.h"#include "cmStateTypes.h"#include "cmStringAlgorithms.h"#include "cmSystemTools.h"#include "cmTarget.h"#include "cmake.h"cmGlobalVisualStudioGenerator::cmGlobalVisualStudioGenerator(  cmake* cm, std::string const& platformInGeneratorName)  : cmGlobalGenerator(cm){  cm->GetState()->SetIsGeneratorMultiConfig(true);  cm->GetState()->SetWindowsShell(true);  cm->GetState()->SetWindowsVSIDE(true);  if (platformInGeneratorName.empty()) {    this->DefaultPlatformName = "Win32";  } else {    this->DefaultPlatformName = platformInGeneratorName;    this->PlatformInGeneratorName = true;  }}cmGlobalVisualStudioGenerator::~cmGlobalVisualStudioGenerator() = default;cmGlobalVisualStudioGenerator::VSVersioncmGlobalVisualStudioGenerator::GetVersion() const{  return this->Version;}void cmGlobalVisualStudioGenerator::SetVersion(VSVersion v){  this->Version = v;}void cmGlobalVisualStudioGenerator::EnableLanguage(  std::vector<std::string> const& lang, cmMakefile* mf, bool optional){  mf->AddDefinition("CMAKE_VS_PLATFORM_NAME_DEFAULT",                    this->DefaultPlatformName);  this->cmGlobalGenerator::EnableLanguage(lang, mf, optional);}bool cmGlobalVisualStudioGenerator::SetGeneratorPlatform(std::string const& p,                                                         cmMakefile* mf){  if (!this->InitializePlatform(mf)) {    return false;  }  if (this->GetPlatformName() == "x64"_s) {    mf->AddDefinition("CMAKE_FORCE_WIN64", "TRUE");  } else if (this->GetPlatformName() == "Itanium"_s) {    mf->AddDefinition("CMAKE_FORCE_IA64", "TRUE");  }  mf->AddDefinition("CMAKE_VS_PLATFORM_NAME", this->GetPlatformName());  return this->cmGlobalGenerator::SetGeneratorPlatform(p, mf);}bool cmGlobalVisualStudioGenerator::InitializePlatform(cmMakefile*){  return true;}std::string const& cmGlobalVisualStudioGenerator::GetPlatformName() const{  if (!this->GeneratorPlatform.empty()) {    return this->GeneratorPlatform;  }  return this->DefaultPlatformName;}const char* cmGlobalVisualStudioGenerator::GetIDEVersion() const{  switch (this->Version) {    case cmGlobalVisualStudioGenerator::VSVersion::VS12:      return "12.0";    case cmGlobalVisualStudioGenerator::VSVersion::VS14:      return "14.0";    case cmGlobalVisualStudioGenerator::VSVersion::VS15:      return "15.0";    case cmGlobalVisualStudioGenerator::VSVersion::VS16:      return "16.0";    case cmGlobalVisualStudioGenerator::VSVersion::VS17:      return "17.0";  }  return "";}void cmGlobalVisualStudioGenerator::WriteSLNHeader(std::ostream& fout){  char utf8bom[] = { char(0xEF), char(0xBB), char(0xBF) };  fout.write(utf8bom, 3);  fout << '\n';  switch (this->Version) {    case cmGlobalVisualStudioGenerator::VSVersion::VS12:      fout << "Microsoft Visual Studio Solution File, Format Version 12.00\n";      if (this->ExpressEdition) {        fout << "# Visual Studio Express 2013 for Windows Desktop\n";      } else {        fout << "# Visual Studio 2013\n";      }      break;    case cmGlobalVisualStudioGenerator::VSVersion::VS14:      // Visual Studio 14 writes .sln format 12.00      fout << "Microsoft Visual Studio Solution File, Format Version 12.00\n";      if (this->ExpressEdition) {        fout << "# Visual Studio Express 14 for Windows Desktop\n";      } else {        fout << "# Visual Studio 14\n";      }      break;    case cmGlobalVisualStudioGenerator::VSVersion::VS15:      // Visual Studio 15 writes .sln format 12.00      fout << "Microsoft Visual Studio Solution File, Format Version 12.00\n";      if (this->ExpressEdition) {        fout << "# Visual Studio Express 15 for Windows Desktop\n";      } else {        fout << "# Visual Studio 15\n";      }      break;    case cmGlobalVisualStudioGenerator::VSVersion::VS16:      // Visual Studio 16 writes .sln format 12.00      fout << "Microsoft Visual Studio Solution File, Format Version 12.00\n";      if (this->ExpressEdition) {        fout << "# Visual Studio Express 16 for Windows Desktop\n";      } else {        fout << "# Visual Studio Version 16\n";      }      break;    case cmGlobalVisualStudioGenerator::VSVersion::VS17:      // Visual Studio 17 writes .sln format 12.00      fout << "Microsoft Visual Studio Solution File, Format Version 12.00\n";      if (this->ExpressEdition) {        fout << "# Visual Studio Express 17 for Windows Desktop\n";      } else {        fout << "# Visual Studio Version 17\n";      }      break;  }}std::string cmGlobalVisualStudioGenerator::GetRegistryBase(){  return cmGlobalVisualStudioGenerator::GetRegistryBase(this->GetIDEVersion());}std::string cmGlobalVisualStudioGenerator::GetRegistryBase(const char* version){  return cmStrCat(R"(HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\VisualStudio\)",                  version);}void cmGlobalVisualStudioGenerator::AddExtraIDETargets(){  // Add a special target that depends on ALL projects for easy build  // of one configuration only.  for (auto const& it : this->ProjectMap) {    std::vector<cmLocalGenerator*> const& gen = it.second;    // add the ALL_BUILD to the first local generator of each project    if (!gen.empty()) {      // Use no actual command lines so that the target itself is not      // considered always out of date.      auto cc = cm::make_unique<cmCustomCommand>();      cc->SetEscapeOldStyle(false);      cc->SetComment("Build all projects");      cmTarget* allBuild =        gen[0]->AddUtilityCommand("ALL_BUILD", true, std::move(cc));      gen[0]->AddGeneratorTarget(        cm::make_unique<cmGeneratorTarget>(allBuild, gen[0]));      //      // Organize in the "predefined targets" folder:      //      if (this->UseFolderProperty()) {        allBuild->SetProperty("FOLDER", this->GetPredefinedTargetsFolder());      }      // Now make all targets depend on the ALL_BUILD target      for (cmLocalGenerator const* i : gen) {        for (const auto& tgt : i->GetGeneratorTargets()) {          if (tgt->GetType() == cmStateEnums::GLOBAL_TARGET ||              tgt->IsImported()) {            continue;          }          if (!this->IsExcluded(gen[0], tgt.get())) {            allBuild->AddUtility(tgt->GetName(), false);          }        }      }    }  }  // Configure CMake Visual Studio macros, for this user on this version  // of Visual Studio.  this->ConfigureCMakeVisualStudioMacros();}void cmGlobalVisualStudioGenerator::ComputeTargetObjectDirectory(  cmGeneratorTarget* gt) const{  std::string dir =    cmStrCat(gt->LocalGenerator->GetCurrentBinaryDirectory(), '/');  std::string tgtDir = gt->LocalGenerator->GetTargetDirectory(gt);  if (!tgtDir.empty()) {    dir += tgtDir;    dir += '/';  }  const char* cd = this->GetCMakeCFGIntDir();  if (cd && *cd) {    dir += cd;    dir += '/';  }  gt->ObjectDirectory = dir;}bool IsVisualStudioMacrosFileRegistered(const std::string& macrosFile,                                        const std::string& regKeyBase,                                        std::string& nextAvailableSubKeyName);void RegisterVisualStudioMacros(const std::string& macrosFile,                                const std::string& regKeyBase);#define CMAKE_VSMACROS_FILENAME "CMakeVSMacros2.vsmacros"#define CMAKE_VSMACROS_RELOAD_MACRONAME                                       \  "Macros.CMakeVSMacros2.Macros.ReloadProjects"#define CMAKE_VSMACROS_STOP_MACRONAME "Macros.CMakeVSMacros2.Macros.StopBuild"void cmGlobalVisualStudioGenerator::ConfigureCMakeVisualStudioMacros(){  std::string dir = this->GetUserMacrosDirectory();  if (!dir.empty()) {    std::string src = cmStrCat(cmSystemTools::GetCMakeRoot(),                               "/Templates/" CMAKE_VSMACROS_FILENAME);    std::string dst = cmStrCat(dir, "/CMakeMacros/" CMAKE_VSMACROS_FILENAME);    // Copy the macros file to the user directory only if the    // destination does not exist or the source location is newer.    // This will allow the user to edit the macros for development    // purposes but newer versions distributed with CMake will replace    // older versions in user directories.    int res;    if (!cmSystemTools::FileTimeCompare(src, dst, &res) || res > 0) {      if (!cmSystemTools::CopyFileAlways(src, dst)) {        std::ostringstream oss;        oss << "Could not copy from: " << src << std::endl            << "                 to: " << dst << std::endl;        cmSystemTools::Message(oss.str(), "Warning");      }    }    RegisterVisualStudioMacros(dst, this->GetUserMacrosRegKeyBase());  }}void cmGlobalVisualStudioGenerator::CallVisualStudioMacro(  MacroName m, const std::string& vsSolutionFile){  // If any solution or project files changed during the generation,  // tell Visual Studio to reload them...  std::string dir = this->GetUserMacrosDirectory();  // Only really try to call the macro if:  //  - there is a UserMacrosDirectory  //  - the CMake vsmacros file exists  //  - the CMake vsmacros file is registered  //  - there were .sln/.vcproj files changed during generation  //  if (!dir.empty()) {    std::string macrosFile =      cmStrCat(dir, "/CMakeMacros/" CMAKE_VSMACROS_FILENAME);    std::string nextSubkeyName;    if (cmSystemTools::FileExists(macrosFile) &&        IsVisualStudioMacrosFileRegistered(          macrosFile, this->GetUserMacrosRegKeyBase(), nextSubkeyName)) {      if (m == MacroReload) {        std::vector<std::string> filenames;        this->GetFilesReplacedDuringGenerate(filenames);        if (!filenames.empty()) {          std::string projects = cmJoin(filenames, ";");          cmCallVisualStudioMacro::CallMacro(            vsSolutionFile, CMAKE_VSMACROS_RELOAD_MACRONAME, projects,            this->GetCMakeInstance()->GetDebugOutput());        }      } else if (m == MacroStop) {        cmCallVisualStudioMacro::CallMacro(          vsSolutionFile, CMAKE_VSMACROS_STOP_MACRONAME, "",          this->GetCMakeInstance()->GetDebugOutput());      }    }  }}std::string cmGlobalVisualStudioGenerator::GetUserMacrosDirectory(){  return "";}std::string cmGlobalVisualStudioGenerator::GetUserMacrosRegKeyBase(){  return "";}void cmGlobalVisualStudioGenerator::FillLinkClosure(  const cmGeneratorTarget* target, TargetSet& linked){  if (linked.insert(target).second) {    TargetDependSet const& depends = this->GetTargetDirectDepends(target);    for (cmTargetDepend const& di : depends) {      if (di.IsLink()) {        this->FillLinkClosure(di, linked);      }    }  }}cmGlobalVisualStudioGenerator::TargetSet const&cmGlobalVisualStudioGenerator::GetTargetLinkClosure(cmGeneratorTarget* target){  auto i = this->TargetLinkClosure.find(target);  if (i == this->TargetLinkClosure.end()) {    TargetSetMap::value_type entry(target, TargetSet());    i = this->TargetLinkClosure.insert(entry).first;    this->FillLinkClosure(target, i->second);  }  return i->second;}void cmGlobalVisualStudioGenerator::FollowLinkDepends(  const cmGeneratorTarget* target, std::set<const cmGeneratorTarget*>& linked){  if (!target->IsInBuildSystem()) {    return;  }  if (linked.insert(target).second &&      target->GetType() == cmStateEnums::STATIC_LIBRARY) {    // Static library targets do not list their link dependencies so    // we must follow them transitively now.    TargetDependSet const& depends = this->GetTargetDirectDepends(target);    for (cmTargetDepend const& di : depends) {      if (di.IsLink()) {        this->FollowLinkDepends(di, linked);      }    }  }}bool cmGlobalVisualStudioGenerator::ComputeTargetDepends(){  if (!this->cmGlobalGenerator::ComputeTargetDepends()) {    return false;  }  for (auto const& it : this->ProjectMap) {    for (const cmLocalGenerator* i : it.second) {      for (const auto& ti : i->GetGeneratorTargets()) {        this->ComputeVSTargetDepends(ti.get());      }    }  }  return true;}static bool VSLinkable(cmGeneratorTarget const* t){  return t->IsLinkable() || t->GetType() == cmStateEnums::OBJECT_LIBRARY;}void cmGlobalVisualStudioGenerator::ComputeVSTargetDepends(  cmGeneratorTarget* target){  if (this->VSTargetDepends.find(target) != this->VSTargetDepends.end()) {    return;  }  VSDependSet& vsTargetDepend = this->VSTargetDepends[target];  // VS <= 7.1 has two behaviors that affect solution dependencies.  //  // (1) Solution-level dependencies between a linkable target and a  // library cause that library to be linked.  We use an intermedite  // empty utility target to express the dependency.  (VS 8 and above  // provide a project file "LinkLibraryDependencies" setting to  // choose whether to activate this behavior.  We disable it except  // when linking external project files.)  //  // (2) We cannot let static libraries depend directly on targets to  // which they "link" because the librarian tool will copy the  // targets into the static library.  While the work-around for  // behavior (1) would also avoid this, it would create a large  // number of extra utility targets for little gain.  Instead, use  // the above work-around only for dependencies explicitly added by  // the add_dependencies() command.  Approximate link dependencies by  // leaving them out for the static library itself but following them  // transitively for other targets.  bool allowLinkable = (target->GetType() != cmStateEnums::STATIC_LIBRARY &&                        target->GetType() != cmStateEnums::SHARED_LIBRARY &&                        target->GetType() != cmStateEnums::MODULE_LIBRARY &&                        target->GetType() != cmStateEnums::EXECUTABLE);  TargetDependSet const& depends = this->GetTargetDirectDepends(target);  // Collect implicit link dependencies (target_link_libraries).  // Static libraries cannot depend on their link implementation  // due to behavior (2), but they do not really need to.  std::set<cmGeneratorTarget const*> linkDepends;  if (target->GetType() != cmStateEnums::STATIC_LIBRARY) {    for (cmTargetDepend const& di : depends) {      if (di.IsLink()) {        this->FollowLinkDepends(di, linkDepends);      }    }  }  // Collect explicit util dependencies (add_dependencies).  std::set<cmGeneratorTarget const*> utilDepends;  for (cmTargetDepend const& di : depends) {    if (di.IsUtil()) {      this->FollowLinkDepends(di, utilDepends);    }  }  // Collect all targets linked by this target so we can avoid  // intermediate targets below.  TargetSet linked;  if (target->GetType() != cmStateEnums::STATIC_LIBRARY) {    linked = this->GetTargetLinkClosure(target);  }  // Emit link dependencies.  for (cmGeneratorTarget const* dep : linkDepends) {    vsTargetDepend.insert(dep->GetName());  }  // Emit util dependencies.  Possibly use intermediate targets.  for (cmGeneratorTarget const* dgt : utilDepends) {    if (allowLinkable || !VSLinkable(dgt) || linked.count(dgt)) {      // Direct dependency allowed.      vsTargetDepend.insert(dgt->GetName());    } else {      // Direct dependency on linkable target not allowed.      // Use an intermediate utility target.      vsTargetDepend.insert(this->GetUtilityDepend(dgt));    }  }}bool cmGlobalVisualStudioGenerator::FindMakeProgram(cmMakefile* mf){  // Visual Studio generators know how to lookup their build tool  // directly instead of needing a helper module to do it, so we  // do not actually need to put CMAKE_MAKE_PROGRAM into the cache.  if (mf->GetDefinition("CMAKE_MAKE_PROGRAM").IsOff()) {    mf->AddDefinition("CMAKE_MAKE_PROGRAM", this->GetVSMakeProgram());  }  return true;}std::string cmGlobalVisualStudioGenerator::GetUtilityDepend(  cmGeneratorTarget const* target){  auto i = this->UtilityDepends.find(target);  if (i == this->UtilityDepends.end()) {    std::string name = this->WriteUtilityDepend(target);    UtilityDependsMap::value_type entry(target, name);    i = this->UtilityDepends.insert(entry).first;  }  return i->second;}std::string cmGlobalVisualStudioGenerator::GetStartupProjectName(  cmLocalGenerator const* root) const{  cmValue n = root->GetMakefile()->GetProperty("VS_STARTUP_PROJECT");  if (cmNonempty(n)) {    std::string startup = *n;    if (this->FindTarget(startup)) {      return startup;    }    root->GetMakefile()->IssueMessage(      MessageType::AUTHOR_WARNING,      cmStrCat("Directory property VS_STARTUP_PROJECT specifies target "               "'",               startup, "' that does not exist.  Ignoring."));  }  // default, if not specified  return this->GetAllTargetName();}bool IsVisualStudioMacrosFileRegistered(const std::string& macrosFile,                                        const std::string& regKeyBase,                                        std::string& nextAvailableSubKeyName){  bool macrosRegistered = false;  std::string s1;  std::string s2;  // Make lowercase local copies, convert to Unix slashes, and  // see if the resulting strings are the same:  s1 = cmSystemTools::LowerCase(macrosFile);  cmSystemTools::ConvertToUnixSlashes(s1);  std::string keyname;  HKEY hkey = nullptr;  LONG result = ERROR_SUCCESS;  DWORD index = 0;  keyname = cmStrCat(regKeyBase, "\\OtherProjects7");  hkey = nullptr;  result =    RegOpenKeyExW(HKEY_CURRENT_USER, cmsys::Encoding::ToWide(keyname).c_str(),                  0, KEY_READ, &hkey);  if (ERROR_SUCCESS == result) {    // Iterate the subkeys and look for the values of interest in each subkey:    wchar_t subkeyname[256];    DWORD cch_subkeyname = cm::size(subkeyname);    wchar_t keyclass[256];    DWORD cch_keyclass = cm::size(keyclass);    FILETIME lastWriteTime;    lastWriteTime.dwHighDateTime = 0;    lastWriteTime.dwLowDateTime = 0;    while (ERROR_SUCCESS ==           RegEnumKeyExW(hkey, index, subkeyname, &cch_subkeyname, 0, keyclass,                         &cch_keyclass, &lastWriteTime)) {      // Open the subkey and query the values of interest:      HKEY hsubkey = nullptr;      result = RegOpenKeyExW(hkey, subkeyname, 0, KEY_READ, &hsubkey);      if (ERROR_SUCCESS == result) {        DWORD valueType = REG_SZ;        wchar_t data1[256];        DWORD cch_data1 = sizeof(data1);        RegQueryValueExW(hsubkey, L"Path", 0, &valueType, (LPBYTE)data1,                         &cch_data1);        DWORD data2 = 0;        DWORD cch_data2 = sizeof(data2);        RegQueryValueExW(hsubkey, L"Security", 0, &valueType, (LPBYTE)&data2,                         &cch_data2);        DWORD data3 = 0;        DWORD cch_data3 = sizeof(data3);        RegQueryValueExW(hsubkey, L"StorageFormat", 0, &valueType,                         (LPBYTE)&data3, &cch_data3);        s2 = cmSystemTools::LowerCase(cmsys::Encoding::ToNarrow(data1));        cmSystemTools::ConvertToUnixSlashes(s2);        if (s2 == s1) {          macrosRegistered = true;        }        std::string fullname = cmsys::Encoding::ToNarrow(data1);        std::string filename;        std::string filepath;        std::string filepathname;        std::string filepathpath;        if (cmSystemTools::FileExists(fullname)) {          filename = cmSystemTools::GetFilenameName(fullname);          filepath = cmSystemTools::GetFilenamePath(fullname);          filepathname = cmSystemTools::GetFilenameName(filepath);          filepathpath = cmSystemTools::GetFilenamePath(filepath);        }        // std::cout << keyname << "\\" << subkeyname << ":" << std::endl;        // std::cout << "  Path: " << data1 << std::endl;        // std::cout << "  Security: " << data2 << std::endl;        // std::cout << "  StorageFormat: " << data3 << std::endl;        // std::cout << "  filename: " << filename << std::endl;        // std::cout << "  filepath: " << filepath << std::endl;        // std::cout << "  filepathname: " << filepathname << std::endl;        // std::cout << "  filepathpath: " << filepathpath << std::endl;        // std::cout << std::endl;        RegCloseKey(hsubkey);      } else {        std::cout << "error opening subkey: "                  << cmsys::Encoding::ToNarrow(subkeyname) << std::endl;        std::cout << std::endl;      }      ++index;      cch_subkeyname = cm::size(subkeyname);      cch_keyclass = cm::size(keyclass);      lastWriteTime.dwHighDateTime = 0;      lastWriteTime.dwLowDateTime = 0;    }    RegCloseKey(hkey);  } else {    std::cout << "error opening key: " << keyname << std::endl;    std::cout << std::endl;  }  // Pass back next available sub key name, assuming sub keys always  // follow the expected naming scheme. Expected naming scheme is that  // the subkeys of OtherProjects7 is 0 to n-1, so it's ok to use "n"  // as the name of the next subkey.  nextAvailableSubKeyName = std::to_string(index);  keyname = cmStrCat(regKeyBase, "\\RecordingProject7");  hkey = nullptr;  result =    RegOpenKeyExW(HKEY_CURRENT_USER, cmsys::Encoding::ToWide(keyname).c_str(),                  0, KEY_READ, &hkey);  if (ERROR_SUCCESS == result) {    DWORD valueType = REG_SZ;    wchar_t data1[256];    DWORD cch_data1 = sizeof(data1);    RegQueryValueExW(hkey, L"Path", 0, &valueType, (LPBYTE)data1, &cch_data1);    DWORD data2 = 0;    DWORD cch_data2 = sizeof(data2);    RegQueryValueExW(hkey, L"Security", 0, &valueType, (LPBYTE)&data2,                     &cch_data2);    DWORD data3 = 0;    DWORD cch_data3 = sizeof(data3);    RegQueryValueExW(hkey, L"StorageFormat", 0, &valueType, (LPBYTE)&data3,                     &cch_data3);    s2 = cmSystemTools::LowerCase(cmsys::Encoding::ToNarrow(data1));    cmSystemTools::ConvertToUnixSlashes(s2);    if (s2 == s1) {      macrosRegistered = true;    }    // std::cout << keyname << ":" << std::endl;    // std::cout << "  Path: " << data1 << std::endl;    // std::cout << "  Security: " << data2 << std::endl;    // std::cout << "  StorageFormat: " << data3 << std::endl;    // std::cout << std::endl;    RegCloseKey(hkey);  } else {    std::cout << "error opening key: " << keyname << std::endl;    std::cout << std::endl;  }  return macrosRegistered;}void WriteVSMacrosFileRegistryEntry(const std::string& nextAvailableSubKeyName,                                    const std::string& macrosFile,                                    const std::string& regKeyBase){  std::string keyname = cmStrCat(regKeyBase, "\\OtherProjects7");  HKEY hkey = nullptr;  LONG result =    RegOpenKeyExW(HKEY_CURRENT_USER, cmsys::Encoding::ToWide(keyname).c_str(),                  0, KEY_READ | KEY_WRITE, &hkey);  if (ERROR_SUCCESS == result) {    // Create the subkey and set the values of interest:    HKEY hsubkey = nullptr;    wchar_t lpClass[] = L"";    result = RegCreateKeyExW(      hkey, cmsys::Encoding::ToWide(nextAvailableSubKeyName).c_str(), 0,      lpClass, 0, KEY_READ | KEY_WRITE, 0, &hsubkey, 0);    if (ERROR_SUCCESS == result) {      DWORD dw = 0;      std::string s(macrosFile);      std::replace(s.begin(), s.end(), '/', '\\');      std::wstring ws = cmsys::Encoding::ToWide(s);      result =        RegSetValueExW(hsubkey, L"Path", 0, REG_SZ, (LPBYTE)ws.c_str(),                       static_cast<DWORD>(ws.size() + 1) * sizeof(wchar_t));      if (ERROR_SUCCESS != result) {        std::cout << "error result 1: " << result << std::endl;        std::cout << std::endl;      }      // Security value is always "1" for sample macros files (seems to be "2"      // if you put the file somewhere outside the standard VSMacros folder)      dw = 1;      result = RegSetValueExW(hsubkey, L"Security", 0, REG_DWORD, (LPBYTE)&dw,                              sizeof(DWORD));      if (ERROR_SUCCESS != result) {        std::cout << "error result 2: " << result << std::endl;        std::cout << std::endl;      }      // StorageFormat value is always "0" for sample macros files      dw = 0;      result = RegSetValueExW(hsubkey, L"StorageFormat", 0, REG_DWORD,                              (LPBYTE)&dw, sizeof(DWORD));      if (ERROR_SUCCESS != result) {        std::cout << "error result 3: " << result << std::endl;        std::cout << std::endl;      }      RegCloseKey(hsubkey);    } else {      std::cout << "error creating subkey: " << nextAvailableSubKeyName                << std::endl;      std::cout << std::endl;    }    RegCloseKey(hkey);  } else {    std::cout << "error opening key: " << keyname << std::endl;    std::cout << std::endl;  }}void RegisterVisualStudioMacros(const std::string& macrosFile,                                const std::string& regKeyBase){  bool macrosRegistered;  std::string nextAvailableSubKeyName;  macrosRegistered = IsVisualStudioMacrosFileRegistered(    macrosFile, regKeyBase, nextAvailableSubKeyName);  if (!macrosRegistered) {    int count =      cmCallVisualStudioMacro::GetNumberOfRunningVisualStudioInstances("ALL");    // Only register the macros file if there are *no* instances of Visual    // Studio running. If we register it while one is running, first, it has    // no effect on the running instance; second, and worse, Visual Studio    // removes our newly added registration entry when it quits. Instead,    // emit a warning asking the user to exit all running Visual Studio    // instances...    //    if (0 != count) {      std::ostringstream oss;      oss << "Could not register CMake's Visual Studio macros file '"          << CMAKE_VSMACROS_FILENAME "' while Visual Studio is running."          << " Please exit all running instances of Visual Studio before"          << " continuing." << std::endl          << std::endl          << "CMake needs to register Visual Studio macros when its macros"          << " file is updated or when it detects that its current macros file"          << " is no longer registered with Visual Studio." << std::endl;      cmSystemTools::Message(oss.str(), "Warning");      // Count them again now that the warning is over. In the case of a GUI      // warning, the user may have gone to close Visual Studio and then come      // back to the CMake GUI and clicked ok on the above warning. If so,      // then register the macros *now* if the count is *now* 0...      //      count = cmCallVisualStudioMacro::GetNumberOfRunningVisualStudioInstances(        "ALL");      // Also re-get the nextAvailableSubKeyName in case Visual Studio      // wrote out new registered macros information as it was exiting:      //      if (0 == count) {        IsVisualStudioMacrosFileRegistered(macrosFile, regKeyBase,                                           nextAvailableSubKeyName);      }    }    // Do another if check - 'count' may have changed inside the above if:    //    if (0 == count) {      WriteVSMacrosFileRegistryEntry(nextAvailableSubKeyName, macrosFile,                                     regKeyBase);    }  }}bool cmGlobalVisualStudioGenerator::TargetIsFortranOnly(  cmGeneratorTarget const* gt){  // If there's only one source language, Fortran has to be used  // in order for the sources to compile.  std::set<std::string> languages = gt->GetAllConfigCompileLanguages();  // Consider an explicit linker language property, but *not* the  // computed linker language that may depend on linked targets.  // This allows the project to control the language choice in  // a target with none of its own sources, e.g. when also using  // object libraries.  cmValue linkLang = gt->GetProperty("LINKER_LANGUAGE");  if (cmNonempty(linkLang)) {    languages.insert(*linkLang);  }  // Intel Fortran .vfproj files do support the resource compiler.  languages.erase("RC");  return languages.size() == 1 && *languages.begin() == "Fortran"_s;}bool cmGlobalVisualStudioGenerator::IsInSolution(  const cmGeneratorTarget* gt) const{  return gt->IsInBuildSystem();}bool cmGlobalVisualStudioGenerator::IsDepInSolution(  const std::string& targetName) const{  return !targetName.empty();}bool cmGlobalVisualStudioGenerator::TargetCompare::operator()(  cmGeneratorTarget const* l, cmGeneratorTarget const* r) const{  // Make sure a given named target is ordered first,  // e.g. to set ALL_BUILD as the default active project.  // When the empty string is named this is a no-op.  if (r->GetName() == this->First) {    return false;  }  if (l->GetName() == this->First) {    return true;  }  return l->GetName() < r->GetName();}cmGlobalVisualStudioGenerator::OrderedTargetDependSet::OrderedTargetDependSet(  TargetDependSet const& targets, std::string const& first)  : derived(TargetCompare(first)){  this->insert(targets.begin(), targets.end());}cmGlobalVisualStudioGenerator::OrderedTargetDependSet::OrderedTargetDependSet(  TargetSet const& targets, std::string const& first)  : derived(TargetCompare(first)){  for (cmGeneratorTarget const* it : targets) {    this->insert(it);  }}std::string cmGlobalVisualStudioGenerator::ExpandCFGIntDir(  const std::string& str, const std::string& config) const{  std::string replace = GetCMakeCFGIntDir();  std::string tmp = str;  for (std::string::size_type i = tmp.find(replace); i != std::string::npos;       i = tmp.find(replace, i)) {    tmp.replace(i, replace.size(), config);    i += config.size();  }  return tmp;}void cmGlobalVisualStudioGenerator::AddSymbolExportCommand(  cmGeneratorTarget* gt, std::vector<cmCustomCommand>& commands,  std::string const& configName){  cmGeneratorTarget::ModuleDefinitionInfo const* mdi =    gt->GetModuleDefinitionInfo(configName);  if (!mdi || !mdi->DefFileGenerated) {    return;  }  std::vector<std::string> outputs;  outputs.push_back(mdi->DefFile);  std::vector<std::string> empty;  std::vector<cmSourceFile const*> objectSources;  gt->GetObjectSources(objectSources, configName);  std::map<cmSourceFile const*, std::string> mapping;  for (cmSourceFile const* it : objectSources) {    mapping[it];  }  gt->LocalGenerator->ComputeObjectFilenames(mapping, gt);  std::string obj_dir = gt->ObjectDirectory;  std::string cmakeCommand = cmSystemTools::GetCMakeCommand();  std::string obj_dir_expanded = obj_dir;  cmSystemTools::ReplaceString(obj_dir_expanded, this->GetCMakeCFGIntDir(),                               configName.c_str());  cmSystemTools::MakeDirectory(obj_dir_expanded);  std::string const objs_file = cmStrCat(obj_dir_expanded, "/objects.txt");  cmGeneratedFileStream fout(objs_file.c_str());  if (!fout) {    cmSystemTools::Error(cmStrCat("could not open ", objs_file));    return;  }  if (mdi->WindowsExportAllSymbols) {    std::vector<std::string> objs;    for (cmSourceFile const* it : objectSources) {      // Find the object file name corresponding to this source file.      // It must exist because we populated the mapping just above.      const auto& v = mapping[it];      assert(!v.empty());      std::string objFile = cmStrCat(obj_dir, v);      objs.push_back(objFile);    }    std::vector<cmSourceFile const*> externalObjectSources;    gt->GetExternalObjects(externalObjectSources, configName);    for (cmSourceFile const* it : externalObjectSources) {      objs.push_back(it->GetFullPath());    }    for (std::string const& it : objs) {      std::string objFile = it;      // replace $(ConfigurationName) in the object names      cmSystemTools::ReplaceString(objFile, this->GetCMakeCFGIntDir(),                                   configName);      if (cmHasLiteralSuffix(objFile, ".obj")) {        fout << objFile << "\n";      }    }  }  for (cmSourceFile const* i : mdi->Sources) {    fout << i->GetFullPath() << "\n";  }  cmCustomCommandLines commandLines = cmMakeSingleCommandLine(    { cmakeCommand, "-E", "__create_def", mdi->DefFile, objs_file });  cmCustomCommand command;  command.SetOutputs(outputs);  command.SetCommandLines(commandLines);  command.SetComment("Auto build dll exports");  command.SetBacktrace(gt->Target->GetMakefile()->GetBacktrace());  command.SetWorkingDirectory(".");  command.SetStdPipesUTF8(true);  commands.push_back(std::move(command));}static bool OpenSolution(std::string const& sln){  HRESULT comInitialized =    CoInitializeEx(nullptr, COINIT_APARTMENTTHREADED | COINIT_DISABLE_OLE1DDE);  if (FAILED(comInitialized)) {    return false;  }  HINSTANCE hi = ShellExecuteA(nullptr, "open", sln.c_str(), nullptr, nullptr,                               SW_SHOWNORMAL);  CoUninitialize();  return reinterpret_cast<intptr_t>(hi) > 32;}bool cmGlobalVisualStudioGenerator::Open(const std::string& bindir,                                         const std::string& projectName,                                         bool dryRun){  std::string sln = cmStrCat(bindir, '/', projectName, ".sln");  if (dryRun) {    return cmSystemTools::FileExists(sln, true);  }  sln = cmSystemTools::ConvertToOutputPath(sln);  return std::async(std::launch::async, OpenSolution, sln).get();}
 |