diff options
author | Chris Lattner <sabre@nondot.org> | 2004-02-09 21:03:38 +0000 |
---|---|---|
committer | Chris Lattner <sabre@nondot.org> | 2004-02-09 21:03:38 +0000 |
commit | 16af11d9622cb0ae678df4e21bdb874261626163 (patch) | |
tree | fd43130fbe0fada277451a39547b0b55fe6707d2 /lib/AsmParser/llvmAsmParser.y | |
parent | a44fb0d3621aa818819a074d54d0e9e88fd51967 (diff) | |
download | external_llvm-16af11d9622cb0ae678df4e21bdb874261626163.zip external_llvm-16af11d9622cb0ae678df4e21bdb874261626163.tar.gz external_llvm-16af11d9622cb0ae678df4e21bdb874261626163.tar.bz2 |
It turns out that the two dimensional vectors were causing big slowdowns
in this for programs with lots of types (like the testcase in PR224).
The problem was that the type ID that the outer vector was using was not
very dense (as many types are getting resolved), so the vector is large
and gets reallocated a lot.
Since there are a lot of values in the program (the .ll file is 10M),
each reallocation has to copy the subvectors, which is also quite slow
(this wouldn't be a problem if C++ supported move semantics, but it
doesn't, at least not yet :(
Changing the outer data structure to a map speeds a release build of
llvm-as up from 11.21s to 5.13s on the testcase in PR224.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@11244 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/AsmParser/llvmAsmParser.y')
-rw-r--r-- | lib/AsmParser/llvmAsmParser.y | 66 |
1 files changed, 34 insertions, 32 deletions
diff --git a/lib/AsmParser/llvmAsmParser.y b/lib/AsmParser/llvmAsmParser.y index a1ad18a..08be9f4 100644 --- a/lib/AsmParser/llvmAsmParser.y +++ b/lib/AsmParser/llvmAsmParser.y @@ -57,14 +57,14 @@ static bool ObsoleteVarArgs; // destroyed when the function is completed. // typedef std::vector<Value *> ValueList; // Numbered defs -static void ResolveDefinitions(std::vector<ValueList> &LateResolvers, - std::vector<ValueList> *FutureLateResolvers = 0); +static void ResolveDefinitions(std::map<unsigned,ValueList> &LateResolvers, + std::map<unsigned,ValueList> *FutureLateResolvers = 0); static struct PerModuleInfo { Module *CurrentModule; - std::vector<ValueList> Values; // Module level numbered definitions - std::vector<ValueList> LateResolveValues; - std::vector<PATypeHolder> Types; + std::map<unsigned,ValueList> Values; // Module level numbered definitions + std::map<unsigned,ValueList> LateResolveValues; + std::vector<PATypeHolder> Types; std::map<ValID, PATypeHolder> LateResolveTypes; // GlobalRefs - This maintains a mapping between <Type, ValID>'s and forward @@ -142,8 +142,8 @@ static struct PerModuleInfo { static struct PerFunctionInfo { Function *CurrentFunction; // Pointer to current function being created - std::vector<ValueList> Values; // Keep track of numbered definitions - std::vector<ValueList> LateResolveValues; + std::map<unsigned,ValueList> Values; // Keep track of numbered definitions + std::map<unsigned,ValueList> LateResolveValues; std::vector<PATypeHolder> Types; std::map<ValID, PATypeHolder> LateResolveTypes; SymbolTable LocalSymtab; @@ -170,11 +170,11 @@ static struct PerFunctionInfo { FID = ValID::create((char*)CurrentFunction->getName().c_str()); } else { unsigned Slot = CurrentFunction->getType()->getUniqueID(); - assert(CurModule.Values.size() > Slot && "Function not inserted?"); // Figure out which slot number if is... + ValueList &List = CurModule.Values[Slot]; for (unsigned i = 0; ; ++i) { - assert(i < CurModule.Values[Slot].size() && "Function not found!"); - if (CurModule.Values[Slot][i] == CurrentFunction) { + assert(i < List.size() && "Function not found!"); + if (List[i] == CurrentFunction) { FID = ValID::create((int)i); break; } @@ -198,16 +198,15 @@ static bool inFunctionScope() { return CurFun.CurrentFunction != 0; } //===----------------------------------------------------------------------===// static int InsertValue(Value *D, - std::vector<ValueList> &ValueTab = CurFun.Values) { + std::map<unsigned,ValueList> &ValueTab = CurFun.Values) { if (D->hasName()) return -1; // Is this a numbered definition? // Yes, insert the value into the value table... unsigned type = D->getType()->getUniqueID(); - if (ValueTab.size() <= type) - ValueTab.resize(type+1, ValueList()); //printf("Values[%d][%d] = %d\n", type, ValueTab[type].size(), D); - ValueTab[type].push_back(D); - return ValueTab[type].size()-1; + ValueList &List = ValueTab[type]; + List.push_back(D); + return List.size()-1; } // TODO: FIXME when Type are not const @@ -297,20 +296,21 @@ static Value *getValNonImprovising(const Type *Ty, const ValID &D) { unsigned Num = (unsigned)D.Num; // Module constants occupy the lowest numbered slots... - if (type < CurModule.Values.size()) { - if (Num < CurModule.Values[type].size()) - return CurModule.Values[type][Num]; - - Num -= CurModule.Values[type].size(); + std::map<unsigned,ValueList>::iterator VI = CurModule.Values.find(type); + if (VI != CurModule.Values.end()) { + if (Num < VI->second.size()) + return VI->second[Num]; + Num -= VI->second.size(); } // Make sure that our type is within bounds - if (CurFun.Values.size() <= type) return 0; + VI = CurFun.Values.find(type); + if (VI == CurFun.Values.end()) return 0; // Check that the number is within bounds... - if (CurFun.Values[type].size() <= Num) return 0; + if (VI->second.size() <= Num) return 0; - return CurFun.Values[type][Num]; + return VI->second[Num]; } case ValID::NameVal: { // Is it a named definition? @@ -415,18 +415,20 @@ static Value *getVal(const Type *Ty, const ValID &D) { // time (forward branches, phi functions for loops, etc...) resolve the // defs now... // -static void ResolveDefinitions(std::vector<ValueList> &LateResolvers, - std::vector<ValueList> *FutureLateResolvers) { +static void ResolveDefinitions(std::map<unsigned,ValueList> &LateResolvers, + std::map<unsigned,ValueList> *FutureLateResolvers) { // Loop over LateResolveDefs fixing up stuff that couldn't be resolved - for (unsigned ty = 0; ty < LateResolvers.size(); ty++) { - while (!LateResolvers[ty].empty()) { - Value *V = LateResolvers[ty].back(); + for (std::map<unsigned,ValueList>::iterator LRI = LateResolvers.begin(), + E = LateResolvers.end(); LRI != E; ++LRI) { + ValueList &List = LRI->second; + while (!List.empty()) { + Value *V = List.back(); + List.pop_back(); assert(!isa<Type>(V) && "Types should be in LateResolveTypes!"); - - LateResolvers[ty].pop_back(); ValID &DID = getValIDFromPlaceHolder(V); - Value *TheRealValue = getValNonImprovising(Type::getUniqueIDType(ty),DID); + Value *TheRealValue = + getValNonImprovising(Type::getUniqueIDType(LRI->first), DID); if (TheRealValue) { V->replaceAllUsesWith(TheRealValue); delete V; @@ -662,7 +664,7 @@ static PATypeHolder HandleUpRefs(const Type *ty) { if (TypeToResolve) { UR_OUT(" * Resolving upreference for " << UpRefs[i].second->getDescription() << "\n"; - std::string OldName = UpRefs[i].UpRefTy->getDescription()); + std::string OldName = TypeToResolve->getDescription()); TypeToResolve->refineAbstractTypeTo(Ty); } |