/* 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 not specified; defaulting to \""; message += tsn; message += "\""; cmSystemTools::Message(message.c_str()); /* store the toolset for later use * -- already done if -T 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__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 not specified; defaulting to \"arm\""); std::string arch = "arm"; /* store the platform name for later use * -- already done if -A 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 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 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& 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& 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& 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& 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 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 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 { if (targetName.compare(targetName.size() - 4, 4, ".gpj") == 0) { makeCommand.push_back(targetName); } else { makeCommand.push_back(targetName + ".gpj"); } } } } void cmGlobalGhsMultiGenerator::WriteMacros(std::ostream& fout) { char const* ghsGpjMacros = this->GetCMakeInstance()->GetCacheDefinition("GHS_GPJ_MACROS"); if (NULL != ghsGpjMacros) { std::vector expandedList; cmSystemTools::ExpandListArgument(std::string(ghsGpjMacros), expandedList); for (std::vector::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()); }