From 99b97dece82ccfc940b60e3cdb01a0368464629f Mon Sep 17 00:00:00 2001 From: Brad King Date: Sun, 27 Jan 2008 13:42:49 -0500 Subject: ENH: Created cmComputeLinkDepends to compute link dependencies. - This will be useful for imported library dependencies - Replaces old cmTarget analyze-lib-depends stuff for linking - Formalizes graph construction and dump - Explicitly represents dependency inferral sets - Use BFS of initial dependencies to preserve order --- Source/CMakeLists.txt | 2 + Source/cmComputeLinkDepends.cxx | 506 ++++++++++++++++++++++++++++++++++++ Source/cmComputeLinkDepends.h | 106 ++++++++ Source/cmComputeLinkInformation.cxx | 38 +-- Source/cmComputeLinkInformation.h | 2 +- Source/cmTarget.cxx | 7 +- bootstrap | 1 + 7 files changed, 631 insertions(+), 31 deletions(-) create mode 100644 Source/cmComputeLinkDepends.cxx create mode 100644 Source/cmComputeLinkDepends.h diff --git a/Source/CMakeLists.txt b/Source/CMakeLists.txt index aacdfcb..63ecda3 100644 --- a/Source/CMakeLists.txt +++ b/Source/CMakeLists.txt @@ -87,6 +87,8 @@ SET(SRCS cmCommandArgumentLexer.cxx cmCommandArgumentParser.cxx cmCommandArgumentParserHelper.cxx + cmComputeLinkDepends.cxx + cmComputeLinkDepends.h cmComputeLinkInformation.cxx cmComputeLinkInformation.h cmCustomCommand.cxx diff --git a/Source/cmComputeLinkDepends.cxx b/Source/cmComputeLinkDepends.cxx new file mode 100644 index 0000000..bf2b1c5 --- /dev/null +++ b/Source/cmComputeLinkDepends.cxx @@ -0,0 +1,506 @@ +/*========================================================================= + + Program: CMake - Cross-Platform Makefile Generator + Module: $RCSfile$ + Language: C++ + Date: $Date$ + Version: $Revision$ + + Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved. + See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details. + + This software is distributed WITHOUT ANY WARRANTY; without even + the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR + PURPOSE. See the above copyright notices for more information. + +=========================================================================*/ +#include "cmComputeLinkDepends.h" + +#include "cmGlobalGenerator.h" +#include "cmLocalGenerator.h" +#include "cmMakefile.h" +#include "cmTarget.h" + +#include + +/* + +This file computes an ordered list of link items to use when linking a +single target in one configuration. Each link item is identified by +the string naming it. A graph of dependencies is created in which +each node corresponds to one item and directed eges lead from nodes to +those which must *precede* them on the link line. For example, the +graph + + C -> B -> A + +will lead to the link line order + + A B C + +The set of items placed in the graph is formed with a breadth-first +search of the link dependencies starting from the main target. + +There are two types of items: those with known direct dependencies and +those without known dependencies. We will call the two types "known +items" and "unknown items", respecitvely. Known items are those whose +names correspond to targets (built or imported) and those for which an +old-style _LIB_DEPENDS variable is defined. All other items are +unknown and we must infer dependencies for them. + +Known items have dependency lists ordered based on how the user +specified them. We can use this order to infer potential dependencies +of unknown items. For example, if link items A and B are unknown and +items X and Y are known, then we might have the following dependency +lists: + + X: Y A B + Y: A B + +The explicitly known dependencies form graph edges + + X <- Y , X <- A , X <- B , Y <- A , Y <- B + +We can also infer the edge + + A <- B + +because *every* time A appears B is seen on its right. We do not know +whether A really needs symbols from B to link, but it *might* so we +must preserve their order. This is the case also for the following +explict lists: + + X: A B Y + Y: A B + +Here, A is followed by the set {B,Y} in one list, and {B} in the other +list. The intersection of these sets is {B}, so we can infer that A +depends on at most B. Meanwhile B is followed by the set {Y} in one +list and {} in the other. The intersection is {} so we can infer that +B has no dependencies. + +Let's make a more complex example by adding unknown item C and +considering these dependency lists: + + X: A B Y C + Y: A C B + +The explicit edges are + + X <- Y , X <- A , X <- B , X <- C , Y <- A , Y <- B , Y <- C + +For the unknown items, we infer dependencies by looking at the +"follow" sets: + + A: intersect( {B,Y,C} , {C,B} ) = {B,C} ; infer edges A <- B , A <- C + B: intersect( {Y,C} , {} ) = {} ; infer no edges + C: intersect( {} , {B} ) = {} ; infer no edges + +Once the complete graph is formed from all known and inferred +dependencies, we walk the graph with a series of depth-first-searches +in order to emit link items. When visiting a node all edges are +followed first because the neighbors must precede the item. Once +neighbors across all edges have been emitted it is safe to emit the +current node. + +If a single DFS returns to a node it previously reached then a cycle +is present. Cyclic link dependencies are resolved simply by repeating +one of the cycle entries at the beginning and end of the cycle +members. For example, the graph + + A <- B , B <- C , C <- A + +can be satisfied with the link item list + + A B C A + +When a node is reached a second time during the same DFS we make sure +its item has been emitted and then skip following its outgoing edges +again. + +The initial exploration of dependencies using a BFS associates an +integer index with each link item. When the graph is built outgoing +edges are sorted by this index. This preserves the original link +order as much as possible subject to the dependencies. + +*/ + +//---------------------------------------------------------------------------- +cmComputeLinkDepends +::cmComputeLinkDepends(cmTarget* target, const char* config) +{ + // Store context information. + this->Target = target; + this->Makefile = this->Target->GetMakefile(); + this->LocalGenerator = this->Makefile->GetLocalGenerator(); + this->GlobalGenerator = this->LocalGenerator->GetGlobalGenerator(); + + // The configuration being linked. + this->Config = config; + + // Enable debug mode if requested. + this->DebugMode = this->Makefile->IsOn("CMAKE_LINK_DEPENDS_DEBUG_MODE"); +} + +//---------------------------------------------------------------------------- +cmComputeLinkDepends::~cmComputeLinkDepends() +{ + for(std::vector::iterator + i = this->InferredDependSets.begin(); + i != this->InferredDependSets.end(); ++i) + { + delete *i; + } +} + +//---------------------------------------------------------------------------- +std::vector const& +cmComputeLinkDepends::Compute() +{ + // Follow the link dependencies of the target to be linked. + this->AddLinkEntries(-1, this->Target->GetOriginalLinkLibraries()); + + // Complete the breadth-first search of dependencies. + while(!this->BFSQueue.empty()) + { + // Get the next entry. + BFSEntry qe = this->BFSQueue.front(); + this->BFSQueue.pop(); + + // Follow the entry's dependencies. + this->FollowLinkEntry(qe); + } + + // Infer dependencies of targets for which they were not known. + this->InferDependencies(); + + // Display the constraint graph. + if(this->DebugMode) + { + fprintf(stderr, + "---------------------------------------" + "---------------------------------------\n"); + fprintf(stderr, "Link dependency analysis for target %s, config %s\n", + this->Target->GetName(), this->Config?this->Config:"noconfig"); + this->DisplayConstraintGraph(); + } + + // Compute the final set of link entries. + this->OrderLinkEntires(); + + // Display the final set. + if(this->DebugMode) + { + this->DisplayFinalEntries(); + } + + return this->FinalLinkEntries; +} + +//---------------------------------------------------------------------------- +int cmComputeLinkDepends::AddLinkEntry(std::string const& item) +{ + // Check if the item entry has already been added. + std::map::iterator lei = this->LinkEntryIndex.find(item); + if(lei != this->LinkEntryIndex.end()) + { + // Yes. We do not need to follow the item's dependencies again. + return lei->second; + } + + // Allocate a spot for the item entry. + { + std::map::value_type + index_entry(item, static_cast(this->EntryList.size())); + lei = this->LinkEntryIndex.insert(index_entry).first; + this->EntryList.push_back(LinkEntry()); + this->InferredDependSets.push_back(0); + this->EntryConstraintGraph.push_back(EntryConstraintSet()); + } + + // Initialize the item entry. + int index = lei->second; + LinkEntry& entry = this->EntryList[index]; + entry.Item = item; + entry.Target = + this->GlobalGenerator->FindTarget(0, entry.Item.c_str(), false); + + // If the item has dependencies queue it to follow them. + if(entry.Target) + { + // Target dependencies are always known. Follow them. + BFSEntry qe = {index, 0}; + this->BFSQueue.push(qe); + } + else + { + // Look for an old-style _LIB_DEPENDS variable. + std::string var = entry.Item; + var += "_LIB_DEPENDS"; + if(const char* val = this->Makefile->GetDefinition(var.c_str())) + { + // The item dependencies are known. Follow them. + BFSEntry qe = {index, val}; + this->BFSQueue.push(qe); + } + else + { + // The item dependencies are not known. We need to infer them. + this->InferredDependSets[index] = new DependSetList; + } + } + + return index; +} + +//---------------------------------------------------------------------------- +void cmComputeLinkDepends::FollowLinkEntry(BFSEntry const& qe) +{ + // Get this entry representation. + int depender_index = qe.Index; + LinkEntry const& entry = this->EntryList[depender_index]; + + // Follow the item's dependencies. + if(entry.Target) + { + // Follow the target dependencies. + this->AddLinkEntries(depender_index, + entry.Target->GetOriginalLinkLibraries()); + } + else + { + // Follow the old-style dependency list. + this->AddVarLinkEntries(depender_index, qe.LibDepends); + } +} + +void cmComputeLinkDepends::AddVarLinkEntries(int depender_index, + const char* value) +{ + // This is called to add the dependencies named by + // _LIB_DEPENDS. The variable contains a semicolon-separated + // list. The list contains link-type;item pairs and just items. + std::vector deplist; + cmSystemTools::ExpandListArgument(value, deplist); + + // Construct the vector of type/value pairs from the variable. + LinkLibraryVectorType libs; + cmTarget::LinkLibraryType linkType = cmTarget::GENERAL; + for(std::vector::const_iterator di = deplist.begin(); + di != deplist.end(); ++di) + { + if(*di == "debug") + { + linkType = cmTarget::DEBUG; + } + else if(*di == "optimized") + { + linkType = cmTarget::OPTIMIZED; + } + else if(*di == "general") + { + linkType = cmTarget::GENERAL; + } + else if(!di->empty()) + { + cmTarget::LibraryID lib(*di, linkType); + libs.push_back(lib); + linkType = cmTarget::GENERAL; + } + } + + // Add the entries from this list. + this->AddLinkEntries(depender_index, libs); +} + +//---------------------------------------------------------------------------- +void cmComputeLinkDepends::AddLinkEntries(int depender_index, + LinkLibraryVectorType const& libs) +{ + // Compute which library configuration to link. + cmTarget::LinkLibraryType linkType = cmTarget::OPTIMIZED; + if(this->Config && cmSystemTools::UpperCase(this->Config) == "DEBUG") + { + linkType = cmTarget::DEBUG; + } + + // Track inferred dependency sets implied by this list. + std::map dependSets; + + // Loop over the libraries linked directly by the target. + for(cmTarget::LinkLibraryVectorType::const_iterator li = libs.begin(); + li != libs.end(); ++li) + { + // Skip entries that will resolve to the target getting linked. + // Skip libraries not meant for the current configuration. + if(li->first == this->Target->GetName() || li->first.empty() || + !(li->second == cmTarget::GENERAL || li->second == linkType)) + { + continue; + } + + // Add a link entry for this item. + int dependee_index = this->AddLinkEntry(li->first); + + // The depender must come before the dependee. + if(depender_index >= 0) + { + this->EntryConstraintGraph[dependee_index].insert(depender_index); + } + + // Update the inferred dependencies for earlier items. + for(std::map::iterator dsi = dependSets.begin(); + dsi != dependSets.end(); ++dsi) + { + if(dependee_index != dsi->first) + { + dsi->second.insert(dependee_index); + } + } + + // If this item needs to have dependencies inferred, do so. + if(this->InferredDependSets[dependee_index]) + { + // Make sure an entry exists to hold the set for the item. + dependSets[dependee_index]; + } + } + + // Store the inferred dependency sets discovered for this list. + for(std::map::iterator dsi = dependSets.begin(); + dsi != dependSets.end(); ++dsi) + { + this->InferredDependSets[dsi->first]->push_back(dsi->second); + } +} + +//---------------------------------------------------------------------------- +void cmComputeLinkDepends::InferDependencies() +{ + // The inferred dependency sets for each item list the possible + // dependencies. The intersection of the sets for one item form its + // inferred dependencies. + for(unsigned int depender_index=0; + depender_index < this->InferredDependSets.size(); ++depender_index) + { + // Skip items for which dependencies do not need to be inferred or + // for which the inferred dependency sets are empty. + DependSetList* sets = this->InferredDependSets[depender_index]; + if(!sets || sets->empty()) + { + continue; + } + + // Intersect the sets for this item. + DependSetList::const_iterator i = sets->begin(); + DependSet common = *i; + for(++i; i != sets->end(); ++i) + { + DependSet intersection; + set_intersection(common.begin(), common.end(), + i->begin(), i->end(), + std::inserter(intersection, intersection.begin())); + common = intersection; + } + + // Add the inferred dependencies to the graph. + for(DependSet::const_iterator j = common.begin(); j != common.end(); ++j) + { + int dependee_index = *j; + this->EntryConstraintGraph[dependee_index].insert(depender_index); + } + } +} + +//---------------------------------------------------------------------------- +void cmComputeLinkDepends::DisplayConstraintGraph() +{ + // Display the conflict graph. + cmOStringStream e; + for(unsigned int i=0; i < this->EntryConstraintGraph.size(); ++i) + { + EntryConstraintSet const& cset = this->EntryConstraintGraph[i]; + e << "item " << i << " is [" << this->EntryList[i].Item << "]\n"; + for(EntryConstraintSet::const_iterator j = cset.begin(); + j != cset.end(); ++j) + { + e << " item " << *j << " must precede it\n"; + } + } + fprintf(stderr, "%s\n", e.str().c_str()); +} + +//---------------------------------------------------------------------------- +void cmComputeLinkDepends::OrderLinkEntires() +{ + // Setup visit tracking. + this->EntryVisited.resize(this->EntryList.size(), 0); + this->WalkId = 0; + + // Start a DFS from every entry. + for(unsigned int i=0; i < this->EntryList.size(); ++i) + { + ++this->WalkId; + this->VisitLinkEntry(i); + } +} + +//---------------------------------------------------------------------------- +void cmComputeLinkDepends::VisitLinkEntry(unsigned int i) +{ + // Check if the node has already been visited. + if(this->EntryVisited[i]) + { + if(this->EntryVisited[i] == this->WalkId) + { + // We have reached a node previously visited on this DFS. There + // is a cycle. In order to allow linking with cyclic + // dependencies we make sure the item is emitted but do not + // follow its outgoing edges again. + if(this->EntryEmitted.insert(i).second) + { + // The item has not been previously emitted so we emit it now. + // It will be emitted again when the stack unwinds back up to + // the beginning of the cycle. + this->FinalLinkEntries.push_back(this->EntryList[i]); + } + } + return; + } + + // We are now visiting this node so mark it. + this->EntryVisited[i] = this->WalkId; + + // Visit the neighbors of the node first. + EntryConstraintSet const& cset = this->EntryConstraintGraph[i]; + for(EntryConstraintSet::const_iterator j = cset.begin(); + j != cset.end(); ++j) + { + this->VisitLinkEntry(*j); + } + + // Now that all items required to come before this one have been + // emmitted, emit this item. + this->EntryEmitted.insert(i); + this->FinalLinkEntries.push_back(this->EntryList[i]); +} + +//---------------------------------------------------------------------------- +void cmComputeLinkDepends::DisplayFinalEntries() +{ + fprintf(stderr, "target [%s] links to:\n", this->Target->GetName()); + for(std::vector::const_iterator lei = + this->FinalLinkEntries.begin(); + lei != this->FinalLinkEntries.end(); ++lei) + { + if(lei->Target) + { + fprintf(stderr, " target [%s]\n", lei->Target->GetName()); + } + else + { + fprintf(stderr, " item [%s]\n", lei->Item.c_str()); + } + } + fprintf(stderr, "\n"); +} diff --git a/Source/cmComputeLinkDepends.h b/Source/cmComputeLinkDepends.h new file mode 100644 index 0000000..646adca --- /dev/null +++ b/Source/cmComputeLinkDepends.h @@ -0,0 +1,106 @@ +/*========================================================================= + + Program: CMake - Cross-Platform Makefile Generator + Module: $RCSfile$ + Language: C++ + Date: $Date$ + Version: $Revision$ + + Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved. + See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details. + + This software is distributed WITHOUT ANY WARRANTY; without even + the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR + PURPOSE. See the above copyright notices for more information. + +=========================================================================*/ +#ifndef cmComputeLinkDepends_h +#define cmComputeLinkDepends_h + +#include "cmStandardIncludes.h" +#include "cmTarget.h" + +#include + +class cmGlobalGenerator; +class cmLocalGenerator; +class cmMakefile; +class cmTarget; + +/** \class cmComputeLinkDepends + * \brief Compute link dependencies for targets. + */ +class cmComputeLinkDepends +{ +public: + cmComputeLinkDepends(cmTarget* target, const char* config); + ~cmComputeLinkDepends(); + + // Basic information about each link item. + struct LinkEntry + { + std::string Item; + cmTarget* Target; + LinkEntry(): Item(), Target(0) {} + LinkEntry(LinkEntry const& r): Item(r.Item), Target(r.Target) {} + }; + + typedef std::vector EntryVector; + EntryVector const& Compute(); + +private: + + // Context information. + cmTarget* Target; + cmMakefile* Makefile; + cmLocalGenerator* LocalGenerator; + cmGlobalGenerator* GlobalGenerator; + bool DebugMode; + + // Configuration information. + const char* Config; + + // Output information. + EntryVector FinalLinkEntries; + + typedef cmTarget::LinkLibraryVectorType LinkLibraryVectorType; + + int AddLinkEntry(std::string const& item); + void AddVarLinkEntries(int depender_index, const char* value); + void AddLinkEntries(int depender_index, + LinkLibraryVectorType const& libs); + + // One entry for each unique item. + std::vector EntryList; + std::map LinkEntryIndex; + + // BFS of initial dependencies. + struct BFSEntry + { + int Index; + const char* LibDepends; + }; + std::queue BFSQueue; + void FollowLinkEntry(BFSEntry const&); + + // Dependency inferral for each link item. + struct DependSet: public std::set {}; + struct DependSetList: public std::vector {}; + std::vector InferredDependSets; + void InferDependencies(); + + // Ordering constraint graph adjacency list. + struct EntryConstraintSet: public std::set {}; + std::vector EntryConstraintGraph; + void DisplayConstraintGraph(); + + // Ordering algorithm. + std::vector EntryVisited; + std::set EntryEmitted; + int WalkId; + void OrderLinkEntires(); + void VisitLinkEntry(unsigned int i); + void DisplayFinalEntries(); +}; + +#endif diff --git a/Source/cmComputeLinkInformation.cxx b/Source/cmComputeLinkInformation.cxx index d7db9b5..1ec1ba3 100644 --- a/Source/cmComputeLinkInformation.cxx +++ b/Source/cmComputeLinkInformation.cxx @@ -16,6 +16,8 @@ =========================================================================*/ #include "cmComputeLinkInformation.h" +#include "cmComputeLinkDepends.h" + #include "cmGlobalGenerator.h" #include "cmLocalGenerator.h" #include "cmMakefile.h" @@ -271,30 +273,17 @@ bool cmComputeLinkInformation::Compute() return false; } - // Compute which library configuration to link. - cmTarget::LinkLibraryType linkType = cmTarget::OPTIMIZED; - if(this->Config && cmSystemTools::UpperCase(this->Config) == "DEBUG") - { - linkType = cmTarget::DEBUG; - } + // Compute the ordered link line items. + cmComputeLinkDepends cld(this->Target, this->Config); + cmComputeLinkDepends::EntryVector const& linkEntries = cld.Compute(); - // Get the list of libraries against which this target wants to link. - { - const cmTarget::LinkLibraryVectorType& libs = - this->Target->GetLinkLibraries(); - for(cmTarget::LinkLibraryVectorType::const_iterator li = libs.begin(); - li != libs.end(); ++li) + // Add the link line items. + for(cmComputeLinkDepends::EntryVector::const_iterator + lei = linkEntries.begin(); + lei != linkEntries.end(); ++lei) { - // Link to a library if it is not the same target and is meant for - // this configuration type. - if((this->Target->GetType() == cmTarget::EXECUTABLE || - li->first != this->Target->GetName()) && - (li->second == cmTarget::GENERAL || li->second == linkType)) - { - this->AddItem(li->first); - } + this->AddItem(lei->Item, lei->Target); } - } // Restore the target link type so the correct system runtime // libraries are found. @@ -307,11 +296,10 @@ bool cmComputeLinkInformation::Compute() } //---------------------------------------------------------------------------- -void cmComputeLinkInformation::AddItem(std::string const& item) +void cmComputeLinkInformation::AddItem(std::string const& item, + cmTarget* tgt) { // Compute the proper name to use to link this library. - // TODO: Change third argument to support imported libraries. - cmTarget* tgt = this->GlobalGenerator->FindTarget(0, item.c_str(), false); const char* config = this->Config; bool implib = this->UseImportLibrary; bool impexe = (tgt && @@ -1226,7 +1214,7 @@ void cmComputeLinkInformation::DiagnoseCycle() << this->Target->GetName() << " because there is a cycle in the constraint graph:\n"; - // Clean up the conflict graph representation. + // Display the conflict graph. for(unsigned int i=0; i < this->RuntimeConflictGraph.size(); ++i) { RuntimeConflictList const& clist = this->RuntimeConflictGraph[i]; diff --git a/Source/cmComputeLinkInformation.h b/Source/cmComputeLinkInformation.h index 65a870a..993ad0f 100644 --- a/Source/cmComputeLinkInformation.h +++ b/Source/cmComputeLinkInformation.h @@ -51,7 +51,7 @@ public: const char* GetLinkLanguage() const { return this->LinkLanguage; } std::vector const& GetRuntimeSearchPath(); private: - void AddItem(std::string const& item); + void AddItem(std::string const& item, cmTarget* tgt); // Output information. ItemVector Items; diff --git a/Source/cmTarget.cxx b/Source/cmTarget.cxx index ab303ab..274c3af 100644 --- a/Source/cmTarget.cxx +++ b/Source/cmTarget.cxx @@ -891,6 +891,7 @@ void cmTarget::AddLinkLibrary(const std::string& lib, tmp.first = lib; tmp.second = llt; this->LinkLibraries.push_back(tmp); + this->OriginalLinkLibraries.push_back(tmp); } //---------------------------------------------------------------------------- @@ -936,6 +937,7 @@ void cmTarget::AddLinkLibrary(cmMakefile& mf, tmp.first = lib; tmp.second = llt; this->LinkLibraries.push_back( tmp ); + this->OriginalLinkLibraries.push_back(tmp); // Add the explicit dependency information for this target. This is // simply a set of libraries separated by ";". There should always @@ -1068,11 +1070,6 @@ cmTarget::AnalyzeLibDependencies( const cmMakefile& mf ) // The dependency map. DependencyMap dep_map; - if ( this->OriginalLinkLibraries.size() == 0 ) - { - this->OriginalLinkLibraries = this->LinkLibraries; - } - // 1. Build the dependency graph // for(LinkLibraryVectorType::reverse_iterator lib diff --git a/bootstrap b/bootstrap index cfcf19a..ad10eed 100755 --- a/bootstrap +++ b/bootstrap @@ -167,6 +167,7 @@ CMAKE_CXX_SOURCES="\ cmDocumentVariables \ cmCacheManager \ cmListFileCache \ + cmComputeLinkDepends \ cmComputeLinkInformation \ " -- cgit v0.12