From d7118006deaf15c6e1b796b33cd883dd9bfa14fb Mon Sep 17 00:00:00 2001 From: Brad King Date: Fri, 3 Aug 2007 15:44:25 -0400 Subject: BUG: Target exclusion-from-all tests should always use the root local generator associated with the all target being tested. --- Source/cmGlobalGenerator.cxx | 150 ++++++++++++------------------ Source/cmGlobalGenerator.h | 5 +- Source/cmGlobalUnixMakefileGenerator3.cxx | 124 +++++------------------- Source/cmGlobalUnixMakefileGenerator3.h | 7 +- Source/cmMakefile.cxx | 8 +- Source/cmTarget.cxx | 4 +- 6 files changed, 92 insertions(+), 206 deletions(-) diff --git a/Source/cmGlobalGenerator.cxx b/Source/cmGlobalGenerator.cxx index d6532a8..63cdc9f 100644 --- a/Source/cmGlobalGenerator.cxx +++ b/Source/cmGlobalGenerator.cxx @@ -691,7 +691,7 @@ void cmGlobalGenerator::Configure() this->TargetDependencies.clear(); this->TotalTargets.clear(); this->ImportedTotalTargets.clear(); - this->ProjectToTargetMap.clear(); + this->LocalGeneratorToTargetMap.clear(); this->ProjectMap.clear(); // start with this directory @@ -779,11 +779,11 @@ void cmGlobalGenerator::Configure() } // at this point this->LocalGenerators has been filled, // so create the map from project name to vector of local generators - this->FillProjectMap(); - // now create project to target map - // This will make sure that targets have all the - // targets they depend on as part of the build. - this->FillProjectToTargetMap(); + this->FillProjectMap(); + + // Create a map from local generator to the complete set of targets + // it builds by default. + this->FillLocalGeneratorToTargetMap(); if ( !this->CMakeInstance->GetScriptMode() ) { @@ -1116,22 +1116,38 @@ void cmGlobalGenerator::GetDocumentation(cmDocumentationEntry& entry) const bool cmGlobalGenerator::IsExcluded(cmLocalGenerator* root, cmLocalGenerator* gen) { - if(gen == root) + if(!gen || gen == root) { + // No directory excludes itself. return false; } - cmLocalGenerator* cur = gen->GetParent(); - while(cur && cur != root) + + if(gen->GetMakefile()->GetPropertyAsBool("EXCLUDE_FROM_ALL")) { - if(cur->GetMakefile()->GetPropertyAsBool("EXCLUDE_FROM_ALL")) - { - return true; - } - cur = cur->GetParent(); + // This directory is excluded from its parent. + return true; } - return gen->GetMakefile()->GetPropertyAsBool("EXCLUDE_FROM_ALL"); + + // This directory is included in its parent. Check whether the + // parent is excluded. + return this->IsExcluded(root, gen->GetParent()); } +bool cmGlobalGenerator::IsExcluded(cmLocalGenerator* root, + cmTarget& target) +{ + if(target.GetPropertyAsBool("EXCLUDE_FROM_ALL")) + { + // This target is excluded from its directory. + return true; + } + else + { + // This target is included in its directory. Check whether the + // directory is excluded. + return this->IsExcluded(root, target.GetMakefile()->GetLocalGenerator()); + } +} void cmGlobalGenerator::GetEnabledLanguages(std::vector& lang) { @@ -1176,88 +1192,40 @@ void cmGlobalGenerator::FillProjectMap() } -// Build a map that contains a the set of targets used by each project -void cmGlobalGenerator::FillProjectToTargetMap() +// Build a map that contains a the set of targets used by each local +// generator directory level. +void cmGlobalGenerator::FillLocalGeneratorToTargetMap() { - // loop over each project in the build - for(std::map >::iterator m = - this->ProjectMap.begin(); - m != this->ProjectMap.end(); ++m) - { - std::vector& lgs = m->second; - if(lgs.size() == 0) + // Loop over all targets in all local generators. + for(std::vector::const_iterator + lgi = this->LocalGenerators.begin(); + lgi != this->LocalGenerators.end(); ++lgi) + { + cmLocalGenerator* lg = *lgi; + cmMakefile* mf = lg->GetMakefile(); + cmTargets& targets = mf->GetTargets(); + for(cmTargets::iterator t = targets.begin(); t != targets.end(); ++t) { - continue; - } - cmStdString const & projectName = m->first; - cmMakefile* projectMakefile = lgs[0]->GetMakefile(); - // get the current EXCLUDE_FROM_ALL value from projectMakefile - const char* exclude = 0; - std::string excludeSave; - bool chain = false; - exclude = - projectMakefile->GetProperties(). - GetPropertyValue("EXCLUDE_FROM_ALL", - cmProperty::DIRECTORY, chain); - if(exclude) - { - excludeSave = exclude; - } - // Set EXCLUDE_FROM_ALL to FALSE for the top level makefile because - // in the current project nothing is excluded yet - projectMakefile->SetProperty("EXCLUDE_FROM_ALL", "FALSE"); - // now loop over all cmLocalGenerators in this project and pull - // out all the targets that depend on each other, even if those - // targets come from a target that is excluded. - for(std::vector::iterator lg = - lgs.begin(); lg != lgs.end(); ++lg) - { - cmMakefile* mf = (*lg)->GetMakefile(); - cmTargets& targets = mf->GetTargets(); - for(cmTargets::iterator t = targets.begin(); - t != targets.end(); ++t) + cmTarget& target = t->second; + + // Consider the directory containing the target and all its + // parents until something excludes the target. + for(cmLocalGenerator* clg = lg; clg && !this->IsExcluded(clg, target); + clg = clg->GetParent()) { - cmTarget& target = t->second; - // if the target is in all then add it to the project - if(!target.GetPropertyAsBool("EXCLUDE_FROM_ALL")) - { - // add this target to the project - this->ProjectToTargetMap[projectName].insert(&target); - // get the target's dependencies - std::vector& tgtdeps = this->GetTargetDepends(target); - this->ProjectToTargetMap[projectName].insert(tgtdeps.begin(), - tgtdeps.end()); - } + // This local generator includes the target. + std::set& targetSet = + this->LocalGeneratorToTargetMap[clg]; + targetSet.insert(&target); + + // Add dependencies of the included target. An excluded + // target may still be included if it is a dependency of a + // non-excluded target. + std::vector& tgtdeps = this->GetTargetDepends(target); + targetSet.insert(tgtdeps.begin(), tgtdeps.end()); } } - // Now restore the EXCLUDE_FROM_ALL property on the project top - // makefile - if(exclude) - { - exclude = excludeSave.c_str(); - } - projectMakefile->SetProperty("EXCLUDE_FROM_ALL", exclude); } - // dump the map for debug purposes - // right now this map is not being used, but it was - // checked in to avoid constant conflicts. - // It is also the first step to creating sub projects - // that contain all of the targets they need. -#if 0 - std::map >::iterator i = - this->ProjectToTargetMap.begin(); - for(; i != this->ProjectToTargetMap.end(); ++i) - { - std::cerr << i->first << "\n"; - std::set::iterator t = i->second.begin(); - for(; t != i->second.end(); ++t) - { - cmTarget* target = *t; - std::cerr << "\t" << target->GetName() << "\n"; - } - } -#endif } diff --git a/Source/cmGlobalGenerator.h b/Source/cmGlobalGenerator.h index e0def62..88c1f69 100644 --- a/Source/cmGlobalGenerator.h +++ b/Source/cmGlobalGenerator.h @@ -230,7 +230,8 @@ protected: // has been populated. void FillProjectMap(); bool IsExcluded(cmLocalGenerator* root, cmLocalGenerator* gen); - void FillProjectToTargetMap(); + bool IsExcluded(cmLocalGenerator* root, cmTarget& target); + void FillLocalGeneratorToTargetMap(); void CreateDefaultGlobalTargets(cmTargets* targets); cmTarget CreateGlobalTarget(const char* name, const char* message, const cmCustomCommandLines* commandLines, @@ -247,7 +248,7 @@ protected: cmLocalGenerator* CurrentLocalGenerator; // map from project name to vector of local generators in that project std::map > ProjectMap; - std::map > ProjectToTargetMap; + std::map > LocalGeneratorToTargetMap; // Set of named installation components requested by the project. std::set InstallComponents; diff --git a/Source/cmGlobalUnixMakefileGenerator3.cxx b/Source/cmGlobalUnixMakefileGenerator3.cxx index 922cdb7..6b8d78d 100644 --- a/Source/cmGlobalUnixMakefileGenerator3.cxx +++ b/Source/cmGlobalUnixMakefileGenerator3.cxx @@ -229,19 +229,7 @@ void cmGlobalUnixMakefileGenerator3::WriteMainMakefile2() { lg = static_cast(this->LocalGenerators[i]); - // are any parents excluded - bool exclude = false; - cmLocalGenerator *lg3 = lg; - while (lg3) - { - if (lg3->GetMakefile()->GetPropertyAsBool("EXCLUDE_FROM_ALL")) - { - exclude = true; - break; - } - lg3 = lg3->GetParent(); - } - this->WriteConvenienceRules2(makefileStream,lg,exclude); + this->WriteConvenienceRules2(makefileStream,lg); } lg = static_cast(this->LocalGenerators[0]); @@ -699,8 +687,7 @@ cmGlobalUnixMakefileGenerator3 void cmGlobalUnixMakefileGenerator3 ::WriteConvenienceRules2(std::ostream& ruleFileStream, - cmLocalUnixMakefileGenerator3 *lg, - bool exclude) + cmLocalUnixMakefileGenerator3 *lg) { std::vector depends; std::vector commands; @@ -792,7 +779,7 @@ cmGlobalUnixMakefileGenerator3 localName.c_str(), depends, commands, true); // add the all/all dependency - if (!exclude && !t->second.GetPropertyAsBool("EXCLUDE_FROM_ALL")) + if(!this->IsExcluded(this->LocalGenerators[0], t->second)) { depends.clear(); depends.push_back(localName); @@ -861,11 +848,15 @@ cmGlobalUnixMakefileGenerator3 lg->WriteMakeRule(ruleFileStream, "Pre-install relink rule for target.", localName.c_str(), depends, commands, true); - depends.clear(); - depends.push_back(localName); - commands.clear(); - lg->WriteMakeRule(ruleFileStream, "Prepare target for install.", - "preinstall", depends, commands, true); + + if(!this->IsExcluded(this->LocalGenerators[0], t->second)) + { + depends.clear(); + depends.push_back(localName); + commands.clear(); + lg->WriteMakeRule(ruleFileStream, "Prepare target for install.", + "preinstall", depends, commands, true); + } } // add the clean rule @@ -917,89 +908,16 @@ unsigned long cmGlobalUnixMakefileGenerator3 ::GetNumberOfProgressActionsInAll(cmLocalUnixMakefileGenerator3 *lg) { unsigned long result = 0; - - // if this is a project - if (!lg->GetParent() || - strcmp(lg->GetMakefile()->GetProjectName(), - lg->GetParent()->GetMakefile()->GetProjectName())) + std::set& targets = this->LocalGeneratorToTargetMap[lg]; + for(std::set::iterator t = targets.begin(); + t != targets.end(); ++t) { - // use the new project to target map - std::set &targets = - this->ProjectToTargetMap[lg->GetMakefile()->GetProjectName()]; - std::set::iterator t = targets.begin(); - for(; t != targets.end(); ++t) - { - cmTarget* target = *t; - cmLocalUnixMakefileGenerator3 *lg3 = - static_cast - (target->GetMakefile()->GetLocalGenerator()); - std::vector &progFiles = lg3->ProgressFiles[target->GetName()]; - result += static_cast(progFiles.size()); - } - } - // for subdirectories - else - { - std::deque lg3Stack; - lg3Stack.push_back(lg); - std::vector targetsInAll; - std::set targets; - while (lg3Stack.size()) - { - cmLocalUnixMakefileGenerator3 *lg3 = lg3Stack.front(); - lg3Stack.pop_front(); - for(cmTargets::iterator l = lg3->GetMakefile()->GetTargets().begin(); - l != lg3->GetMakefile()->GetTargets().end(); ++l) - { - if((l->second.GetType() == cmTarget::EXECUTABLE) || - (l->second.GetType() == cmTarget::STATIC_LIBRARY) || - (l->second.GetType() == cmTarget::SHARED_LIBRARY) || - (l->second.GetType() == cmTarget::MODULE_LIBRARY) || - (l->second.GetType() == cmTarget::UTILITY)) - { - // Add this to the list of depends rules in this directory. - if (!l->second.GetPropertyAsBool("EXCLUDE_FROM_ALL") && - targets.find(&l->second) == targets.end()) - { - std::deque activeTgts; - activeTgts.push_back(&(l->second)); - // trace depth of target dependencies - while (activeTgts.size()) - { - if (targets.find(activeTgts.front()) == targets.end()) - { - targets.insert(activeTgts.front()); - cmLocalUnixMakefileGenerator3 *lg4 = - static_cast - (activeTgts.front()->GetMakefile()->GetLocalGenerator()); - std::vector &progFiles2 = - lg4->ProgressFiles[activeTgts.front()->GetName()]; - result += static_cast(progFiles2.size()); - std::vector deps2 = - this->GetTargetDepends(*activeTgts.front()); - for (std::vector::const_iterator di = - deps2.begin(); di != deps2.end(); ++di) - { - activeTgts.push_back(*di); - } - } - activeTgts.pop_front(); - } - } - } - } - - // The directory-level rule depends on the directory-level - // rules of the subdirectories. - for(std::vector::iterator sdi = - lg3->GetChildren().begin(); - sdi != lg3->GetChildren().end(); ++sdi) - { - cmLocalUnixMakefileGenerator3* slg = - static_cast(*sdi); - lg3Stack.push_back(slg); - } - } + cmTarget* target = *t; + cmLocalUnixMakefileGenerator3 *lg3 = + static_cast + (target->GetMakefile()->GetLocalGenerator()); + std::vector &progFiles = lg3->ProgressFiles[target->GetName()]; + result += static_cast(progFiles.size()); } return result; } diff --git a/Source/cmGlobalUnixMakefileGenerator3.h b/Source/cmGlobalUnixMakefileGenerator3.h index 56c4994..0ca930f 100644 --- a/Source/cmGlobalUnixMakefileGenerator3.h +++ b/Source/cmGlobalUnixMakefileGenerator3.h @@ -145,10 +145,9 @@ public: protected: void WriteMainMakefile2(); void WriteMainCMakefile(); - - void WriteConvenienceRules2(std::ostream& ruleFileStream, - cmLocalUnixMakefileGenerator3 *, - bool exclude); + + void WriteConvenienceRules2(std::ostream& ruleFileStream, + cmLocalUnixMakefileGenerator3*); void WriteDirectoryRule2(std::ostream& ruleFileStream, cmLocalUnixMakefileGenerator3* lg, diff --git a/Source/cmMakefile.cxx b/Source/cmMakefile.cxx index 132d2ae..feda29b 100644 --- a/Source/cmMakefile.cxx +++ b/Source/cmMakefile.cxx @@ -2847,10 +2847,10 @@ void cmMakefile::DefineProperties(cmake *cm) cm->DefineProperty ("EXCLUDE_FROM_ALL", cmProperty::DIRECTORY, - "Exclude the target from the all target.", - "A property on a target that indicates if the target is excluded " + "Exclude the directory from the all target of its parent.", + "A property on a directory that indicates if its targets are excluded " "from the default build target. If it is not, then with a Makefile " - "for example typing make will couse this target to be built as well. " + "for example typing make will cause the targets to be built. " "The same concept applies to the default build of other generators.", - true); + false); } diff --git a/Source/cmTarget.cxx b/Source/cmTarget.cxx index 18b0ad1..42bf043 100644 --- a/Source/cmTarget.cxx +++ b/Source/cmTarget.cxx @@ -122,9 +122,9 @@ void cmTarget::DefineProperties(cmake *cm) "Exclude the target from the all target.", "A property on a target that indicates if the target is excluded " "from the default build target. If it is not, then with a Makefile " - "for example typing make will couse this target to be built as well. " + "for example typing make will cause this target to be built. " "The same concept applies to the default build of other generators.", - true); + false); cm->DefineProperty ("INSTALL_NAME_DIR", cmProperty::TARGET, -- cgit v0.12