| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498 |
- /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
- file Copyright.txt or https://cmake.org/licensing for details. */
- #include "cmGlobalGhsMultiGenerator.h"
- #include "cmsys/SystemTools.hxx"
- #include "cmAlgorithms.h"
- #include "cmDocumentationEntry.h"
- #include "cmGeneratedFileStream.h"
- #include "cmGeneratorTarget.h"
- #include "cmGhsMultiTargetGenerator.h"
- #include "cmLocalGhsMultiGenerator.h"
- #include "cmMakefile.h"
- #include "cmVersion.h"
- #include "cmake.h"
- const char* cmGlobalGhsMultiGenerator::FILE_EXTENSION = ".gpj";
- const char* cmGlobalGhsMultiGenerator::DEFAULT_BUILD_PROGRAM = "gbuild.exe";
- const char* cmGlobalGhsMultiGenerator::DEFAULT_TOOLSET_ROOT = "C:/ghs";
- cmGlobalGhsMultiGenerator::cmGlobalGhsMultiGenerator(cmake* cm)
- : cmGlobalGenerator(cm)
- {
- }
- cmGlobalGhsMultiGenerator::~cmGlobalGhsMultiGenerator()
- {
- }
- cmLocalGenerator* cmGlobalGhsMultiGenerator::CreateLocalGenerator(
- cmMakefile* mf)
- {
- return new cmLocalGhsMultiGenerator(this, mf);
- }
- void cmGlobalGhsMultiGenerator::GetDocumentation(cmDocumentationEntry& entry)
- {
- entry.Name = GetActualName();
- entry.Brief =
- "Generates Green Hills MULTI files (experimental, work-in-progress).";
- }
- void cmGlobalGhsMultiGenerator::ComputeTargetObjectDirectory(
- cmGeneratorTarget* gt) const
- {
- // Compute full path to object file directory for this target.
- std::string dir;
- dir += gt->LocalGenerator->GetCurrentBinaryDirectory();
- dir += "/";
- dir += gt->LocalGenerator->GetTargetDirectory(gt);
- dir += "/";
- gt->ObjectDirectory = dir;
- }
- bool cmGlobalGhsMultiGenerator::SetGeneratorToolset(std::string const& ts,
- cmMakefile* mf)
- {
- std::string tsp; /* toolset path */
- std::string tsn = ts; /* toolset name */
- GetToolset(mf, tsp, tsn);
- /* no toolset was found */
- if (tsn.empty()) {
- return false;
- } else if (ts.empty()) {
- std::string message;
- message =
- "Green Hills MULTI: -T <toolset> not specified; defaulting to \"";
- message += tsn;
- message += "\"";
- cmSystemTools::Message(message.c_str());
- /* store the toolset for later use
- * -- already done if -T<toolset> was specified
- */
- mf->AddCacheDefinition("CMAKE_GENERATOR_TOOLSET", tsn.c_str(),
- "Name of generator toolset.",
- cmStateEnums::INTERNAL);
- }
- /* set the build tool to use */
- const char* prevTool = mf->GetDefinition("CMAKE_MAKE_PROGRAM");
- std::string gbuild(tsp + "/" + tsn + "/" + DEFAULT_BUILD_PROGRAM);
- /* check if the toolset changed from last generate */
- if (prevTool != NULL && (gbuild != prevTool)) {
- std::string message = "generator toolset: ";
- message += gbuild;
- message += "\nDoes not match the toolset used previously: ";
- message += prevTool;
- message += "\nEither remove the CMakeCache.txt file and CMakeFiles "
- "directory or choose a different binary directory.";
- cmSystemTools::Error(message.c_str());
- } else {
- /* store the toolset that is being used for this build */
- mf->AddCacheDefinition("CMAKE_MAKE_PROGRAM", gbuild.c_str(),
- "build program to use", cmStateEnums::INTERNAL,
- true);
- }
- mf->AddDefinition("CMAKE_SYSTEM_VERSION", tsn.c_str());
- // FIXME: compiler detection not implemented
- // gbuild uses the primaryTarget setting in the top-level project
- // file to determine which compiler to use. Because compiler
- // detection is not implemented these variables must be
- // set to skip past these tests. However cmake will verify that
- // the executable pointed to by CMAKE_<LANG>_COMPILER exists.
- // To pass this additional check gbuild is used as a place holder for the
- // actual compiler.
- mf->AddDefinition("CMAKE_C_COMPILER", gbuild.c_str());
- mf->AddDefinition("CMAKE_C_COMPILER_ID_RUN", "TRUE");
- mf->AddDefinition("CMAKE_C_COMPILER_ID", "GHS");
- mf->AddDefinition("CMAKE_C_COMPILER_FORCED", "TRUE");
- mf->AddDefinition("CMAKE_CXX_COMPILER", gbuild.c_str());
- mf->AddDefinition("CMAKE_CXX_COMPILER_ID_RUN", "TRUE");
- mf->AddDefinition("CMAKE_CXX_COMPILER_ID", "GHS");
- mf->AddDefinition("CMAKE_CXX_COMPILER_FORCED", "TRUE");
- return true;
- }
- bool cmGlobalGhsMultiGenerator::SetGeneratorPlatform(std::string const& p,
- cmMakefile* mf)
- {
- if (p == "") {
- cmSystemTools::Message(
- "Green Hills MULTI: -A <arch> not specified; defaulting to \"arm\"");
- std::string arch = "arm";
- /* store the platform name for later use
- * -- already done if -A<arch> was specified
- */
- mf->AddCacheDefinition("CMAKE_GENERATOR_PLATFORM", arch.c_str(),
- "Name of generator platform.",
- cmStateEnums::INTERNAL);
- }
- const char* tgtPlatform = mf->GetDefinition("GHS_TARGET_PLATFORM");
- if (tgtPlatform == nullptr) {
- cmSystemTools::Message("Green Hills MULTI: GHS_TARGET_PLATFORM not "
- "specified; defaulting to \"integrity\"");
- tgtPlatform = "integrity";
- }
- /* store the platform name for later use */
- mf->AddCacheDefinition("GHS_TARGET_PLATFORM", tgtPlatform,
- "Name of GHS target platform.",
- cmStateEnums::INTERNAL);
- return true;
- }
- void cmGlobalGhsMultiGenerator::EnableLanguage(
- std::vector<std::string> const& l, cmMakefile* mf, bool optional)
- {
- mf->AddDefinition("CMAKE_SYSTEM_NAME", "GHS-MULTI");
- mf->AddDefinition("GHSMULTI", "1"); // identifier for user CMake files
- this->cmGlobalGenerator::EnableLanguage(l, mf, optional);
- }
- bool cmGlobalGhsMultiGenerator::FindMakeProgram(cmMakefile* /*mf*/)
- {
- // The GHS generator only knows how to lookup its build tool
- // during generation of the project files, but this
- // can only be done after the toolset is specified.
- return true;
- }
- void cmGlobalGhsMultiGenerator::GetToolset(cmMakefile* mf, std::string& tsd,
- std::string& ts)
- {
- const char* ghsRoot = mf->GetDefinition("GHS_TOOLSET_ROOT");
- if (!ghsRoot) {
- ghsRoot = DEFAULT_TOOLSET_ROOT;
- }
- tsd = ghsRoot;
- if (ts.empty()) {
- std::vector<std::string> output;
- // Use latest? version
- cmSystemTools::Glob(tsd, "comp_[^;]+", output);
- if (output.empty()) {
- cmSystemTools::Error("GHS toolset not found in ", tsd.c_str());
- ts = "";
- } else {
- ts = output.back();
- }
- } else {
- std::string tryPath = tsd + std::string("/") + ts;
- if (!cmSystemTools::FileExists(tryPath)) {
- cmSystemTools::Error("GHS toolset \"", ts.c_str(), "\" not found in ",
- tsd.c_str());
- ts = "";
- }
- }
- }
- void cmGlobalGhsMultiGenerator::WriteFileHeader(std::ostream& fout)
- {
- fout << "#!gbuild" << std::endl;
- fout << "#" << std::endl
- << "# CMAKE generated file: DO NOT EDIT!" << std::endl
- << "# Generated by \"" << this->GetActualName() << "\""
- << " Generator, CMake Version " << cmVersion::GetMajorVersion() << "."
- << cmVersion::GetMinorVersion() << std::endl
- << "#" << std::endl
- << std::endl;
- }
- void cmGlobalGhsMultiGenerator::WriteTopLevelProject(
- std::ostream& fout, cmLocalGenerator* root,
- std::vector<cmLocalGenerator*>& generators)
- {
- WriteFileHeader(fout);
- this->WriteMacros(fout);
- this->WriteHighLevelDirectives(fout);
- GhsMultiGpj::WriteGpjTag(GhsMultiGpj::PROJECT, fout);
- fout << "# Top Level Project File" << std::endl;
- // Specify BSP option if supplied by user
- // -- not all platforms require this entry in the project file
- // integrity platforms require this field; use default if needed
- std::string platform;
- if (const char* p =
- this->GetCMakeInstance()->GetCacheDefinition("GHS_TARGET_PLATFORM")) {
- platform = p;
- }
- std::string bspName;
- if (char const* bspCache =
- this->GetCMakeInstance()->GetCacheDefinition("GHS_BSP_NAME")) {
- bspName = bspCache;
- this->GetCMakeInstance()->MarkCliAsUsed("GHS_BSP_NAME");
- } else {
- bspName = "IGNORE";
- }
- if (platform.find("integrity") != std::string::npos &&
- cmSystemTools::IsOff(bspName.c_str())) {
- const char* a =
- this->GetCMakeInstance()->GetCacheDefinition("CMAKE_GENERATOR_PLATFORM");
- bspName = "sim";
- bspName += (a ? a : "");
- }
- if (!cmSystemTools::IsOff(bspName.c_str())) {
- fout << " -bsp " << bspName << std::endl;
- }
- // Specify OS DIR if supplied by user
- // -- not all platforms require this entry in the project file
- std::string osDir;
- std::string osDirOption;
- if (char const* osDirCache =
- this->GetCMakeInstance()->GetCacheDefinition("GHS_OS_DIR")) {
- osDir = osDirCache;
- }
- if (char const* osDirOptionCache =
- this->GetCMakeInstance()->GetCacheDefinition("GHS_OS_DIR_OPTION")) {
- osDirOption = osDirOptionCache;
- }
- if (!cmSystemTools::IsOff(osDir.c_str()) ||
- platform.find("integrity") != std::string::npos) {
- std::replace(osDir.begin(), osDir.end(), '\\', '/');
- fout << " " << osDirOption << "\"" << osDir << "\"" << std::endl;
- }
- WriteSubProjects(fout, root, generators);
- }
- void cmGlobalGhsMultiGenerator::WriteSubProjects(
- std::ostream& fout, cmLocalGenerator* root,
- std::vector<cmLocalGenerator*>& generators)
- {
- // Collect all targets under this root generator and the transitive
- // closure of their dependencies.
- TargetDependSet projectTargets;
- TargetDependSet originalTargets;
- this->GetTargetSets(projectTargets, originalTargets, root, generators);
- OrderedTargetDependSet orderedProjectTargets(projectTargets, "");
- // write out all the sub-projects
- std::string rootBinaryDir = root->GetCurrentBinaryDirectory();
- for (cmGeneratorTarget const* target : orderedProjectTargets) {
- if (target->GetType() == cmStateEnums::INTERFACE_LIBRARY) {
- continue;
- }
- const char* projName = target->GetProperty("GENERATOR_FILE_NAME");
- const char* projType = target->GetProperty("GENERATOR_FILE_NAME_EXT");
- if (projName && projType) {
- cmLocalGenerator* lg = target->GetLocalGenerator();
- std::string dir = lg->GetCurrentBinaryDirectory();
- dir = root->ConvertToRelativePath(rootBinaryDir, dir.c_str());
- if (dir == ".") {
- dir.clear();
- } else {
- if (dir.back() != '/') {
- dir += "/";
- }
- }
- if (cmSystemTools::IsOn(target->GetProperty("EXCLUDE_FROM_ALL"))) {
- fout << "{comment} ";
- }
- std::string projFile = dir + projName + FILE_EXTENSION;
- fout << projFile;
- fout << " " << projType << std::endl;
- if (cmSystemTools::IsOn(target->GetProperty("GHS_REFERENCE_PROJECT"))) {
- // create reference project
- std::string fname = dir;
- fname += target->GetName();
- fname += "REF";
- fname += FILE_EXTENSION;
- cmGeneratedFileStream fref(fname.c_str());
- fref.SetCopyIfDifferent(true);
- this->WriteFileHeader(fref);
- GhsMultiGpj::WriteGpjTag(GhsMultiGpj::REFERENCE, fref);
- fref << " :reference=" << projFile << std::endl;
- fref.Close();
- }
- }
- }
- }
- void cmGlobalGhsMultiGenerator::Generate()
- {
- // first do the superclass method
- this->cmGlobalGenerator::Generate();
- // output top-level projects
- for (auto& it : this->ProjectMap) {
- this->OutputTopLevelProject(it.second[0], it.second);
- }
- }
- void cmGlobalGhsMultiGenerator::OutputTopLevelProject(
- cmLocalGenerator* root, std::vector<cmLocalGenerator*>& generators)
- {
- if (generators.empty()) {
- return;
- }
- /* Name top-level projects as filename.top.gpj to avoid name clashes
- * with target projects. This avoid the issue where the project has
- * the same name as the executable target.
- */
- std::string fname = root->GetCurrentBinaryDirectory();
- fname += "/";
- fname += root->GetProjectName();
- fname += ".top";
- fname += FILE_EXTENSION;
- cmGeneratedFileStream fout(fname.c_str());
- fout.SetCopyIfDifferent(true);
- this->WriteTopLevelProject(fout, root, generators);
- fout.Close();
- }
- void cmGlobalGhsMultiGenerator::GenerateBuildCommand(
- std::vector<std::string>& makeCommand, const std::string& makeProgram,
- const std::string& projectName, const std::string& projectDir,
- const std::string& targetName, const std::string& /*config*/, bool /*fast*/,
- int jobs, bool /*verbose*/, std::vector<std::string> const& makeOptions)
- {
- const char* gbuild =
- this->CMakeInstance->GetCacheDefinition("CMAKE_MAKE_PROGRAM");
- makeCommand.push_back(
- this->SelectMakeProgram(makeProgram, (std::string)gbuild));
- if (jobs != cmake::NO_BUILD_PARALLEL_LEVEL) {
- makeCommand.push_back("-parallel");
- if (jobs != cmake::DEFAULT_BUILD_PARALLEL_LEVEL) {
- makeCommand.push_back(std::to_string(jobs));
- }
- }
- makeCommand.insert(makeCommand.end(), makeOptions.begin(),
- makeOptions.end());
- /* determine which top-project file to use */
- std::string proj = projectName + ".top" + FILE_EXTENSION;
- std::vector<std::string> files;
- cmSystemTools::Glob(projectDir, ".*\\.top\\.gpj", files);
- if (!files.empty()) {
- auto p = std::find(files.begin(), files.end(), proj);
- if (p == files.end()) {
- proj = files.at(0);
- }
- }
- makeCommand.push_back("-top");
- makeCommand.push_back(proj);
- if (!targetName.empty()) {
- if (targetName == "clean") {
- makeCommand.push_back("-clean");
- } else {
- makeCommand.push_back(targetName);
- }
- }
- }
- void cmGlobalGhsMultiGenerator::WriteMacros(std::ostream& fout)
- {
- char const* ghsGpjMacros =
- this->GetCMakeInstance()->GetCacheDefinition("GHS_GPJ_MACROS");
- if (NULL != ghsGpjMacros) {
- std::vector<std::string> expandedList;
- cmSystemTools::ExpandListArgument(std::string(ghsGpjMacros), expandedList);
- for (std::vector<std::string>::const_iterator expandedListI =
- expandedList.begin();
- expandedListI != expandedList.end(); ++expandedListI) {
- fout << "macro " << *expandedListI << std::endl;
- }
- }
- }
- void cmGlobalGhsMultiGenerator::WriteHighLevelDirectives(std::ostream& fout)
- {
- /* set primary target */
- std::string tgt;
- const char* t =
- this->GetCMakeInstance()->GetCacheDefinition("GHS_PRIMARY_TARGET");
- if (t) {
- tgt = t;
- this->GetCMakeInstance()->MarkCliAsUsed("GHS_PRIMARY_TARGET");
- } else {
- const char* a =
- this->GetCMakeInstance()->GetCacheDefinition("CMAKE_GENERATOR_PLATFORM");
- const char* p =
- this->GetCMakeInstance()->GetCacheDefinition("GHS_TARGET_PLATFORM");
- tgt = (a ? a : "");
- tgt += "_";
- tgt += (p ? p : "");
- tgt += ".tgt";
- }
- fout << "primaryTarget=" << tgt << std::endl;
- char const* const customization =
- this->GetCMakeInstance()->GetCacheDefinition("GHS_CUSTOMIZATION");
- if (NULL != customization && strlen(customization) > 0) {
- fout << "customization=" << trimQuotes(customization) << std::endl;
- this->GetCMakeInstance()->MarkCliAsUsed("GHS_CUSTOMIZATION");
- }
- }
- std::string cmGlobalGhsMultiGenerator::trimQuotes(std::string const& str)
- {
- std::string result;
- result.reserve(str.size());
- for (const char* ch = str.c_str(); *ch != '\0'; ++ch) {
- if (*ch != '"') {
- result += *ch;
- }
- }
- return result;
- }
- bool cmGlobalGhsMultiGenerator::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();
- }
- cmGlobalGhsMultiGenerator::OrderedTargetDependSet::OrderedTargetDependSet(
- TargetDependSet const& targets, std::string const& first)
- : derived(TargetCompare(first))
- {
- this->insert(targets.begin(), targets.end());
- }
|