Skip to content

Commit 0dbc708

Browse files
committedMar 19, 2015
GlobalDCE: Improve performance for large modules containing comdats.
When we encounter a global with a comdat, rather than iterating over every global in the module to find globals in the same comdat, store the members in a multimap. This effectively lowers the complexity to O(N log N), improving performance significantly for large modules such as might be encountered during LTO. It looks like we used to do something like this until r219191. No functional change. Differential Revision: http://reviews.llvm.org/D8431 llvm-svn: 232743
1 parent 7bf61d3 commit 0dbc708

File tree

1 file changed

+16
-10
lines changed

1 file changed

+16
-10
lines changed
 

‎llvm/lib/Transforms/IPO/GlobalDCE.cpp

+16-10
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424
#include "llvm/Transforms/Utils/CtorUtils.h"
2525
#include "llvm/Transforms/Utils/GlobalStatus.h"
2626
#include "llvm/Pass.h"
27+
#include <unordered_map>
2728
using namespace llvm;
2829

2930
#define DEBUG_TYPE "globaldce"
@@ -47,6 +48,7 @@ namespace {
4748
private:
4849
SmallPtrSet<GlobalValue*, 32> AliveGlobals;
4950
SmallPtrSet<Constant *, 8> SeenConstants;
51+
std::unordered_multimap<Comdat *, GlobalValue *> ComdatMembers;
5052

5153
/// GlobalIsNeeded - mark the specific global value as needed, and
5254
/// recursively mark anything that it uses as also needed.
@@ -78,6 +80,17 @@ bool GlobalDCE::runOnModule(Module &M) {
7880
// Remove empty functions from the global ctors list.
7981
Changed |= optimizeGlobalCtorsList(M, isEmptyFunction);
8082

83+
// Collect the set of members for each comdat.
84+
for (Function &F : M)
85+
if (Comdat *C = F.getComdat())
86+
ComdatMembers.insert(std::make_pair(C, &F));
87+
for (GlobalVariable &GV : M.globals())
88+
if (Comdat *C = GV.getComdat())
89+
ComdatMembers.insert(std::make_pair(C, &GV));
90+
for (GlobalAlias &GA : M.aliases())
91+
if (Comdat *C = GA.getComdat())
92+
ComdatMembers.insert(std::make_pair(C, &GA));
93+
8194
// Loop over the module, adding globals which are obviously necessary.
8295
for (Module::iterator I = M.begin(), E = M.end(); I != E; ++I) {
8396
Changed |= RemoveUnusedGlobalValue(*I);
@@ -177,6 +190,7 @@ bool GlobalDCE::runOnModule(Module &M) {
177190
// Make sure that all memory is released
178191
AliveGlobals.clear();
179192
SeenConstants.clear();
193+
ComdatMembers.clear();
180194

181195
return Changed;
182196
}
@@ -188,17 +202,9 @@ void GlobalDCE::GlobalIsNeeded(GlobalValue *G) {
188202
if (!AliveGlobals.insert(G).second)
189203
return;
190204

191-
Module *M = G->getParent();
192205
if (Comdat *C = G->getComdat()) {
193-
for (Function &F : *M)
194-
if (F.getComdat() == C)
195-
GlobalIsNeeded(&F);
196-
for (GlobalVariable &GV : M->globals())
197-
if (GV.getComdat() == C)
198-
GlobalIsNeeded(&GV);
199-
for (GlobalAlias &GA : M->aliases())
200-
if (GA.getComdat() == C)
201-
GlobalIsNeeded(&GA);
206+
for (auto &&CM : make_range(ComdatMembers.equal_range(C)))
207+
GlobalIsNeeded(CM.second);
202208
}
203209

204210
if (GlobalVariable *GV = dyn_cast<GlobalVariable>(G)) {

0 commit comments

Comments
 (0)
Please sign in to comment.