From e0cbf970ac5e9636a3a635e1f3390aa5d93c827a Mon Sep 17 00:00:00 2001 From: Chris Lattner Date: Tue, 23 Aug 2005 23:40:41 +0000 Subject: [PATCH] Change live variables from using multimaps to using maps of vectors and rearrange some of the accessors to be more efficient. This makes it much more efficient to iterate over all of the things with the same value. This speeds up liveintervals analysis from 8.63s to 3.79s with a release build of llc on kc++ with -march=ia64. This also speeds up live var from 1.66s -> 0.87s as well, reducing total llc time from 20.1s->15.2s. This also speeds up other targets slightly, e.g. llc time on X86 from 16.84 -> 16.45s, and PPC from 17.64->17.03s. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@22990 91177308-0d34-0410-b5e6-96231b3b80d8 --- include/llvm/CodeGen/LiveVariables.h | 108 +++++++++++++++------------ 1 file changed, 62 insertions(+), 46 deletions(-) diff --git a/include/llvm/CodeGen/LiveVariables.h b/include/llvm/CodeGen/LiveVariables.h index b3911ec80b3..3debf347fc8 100644 --- a/include/llvm/CodeGen/LiveVariables.h +++ b/include/llvm/CodeGen/LiveVariables.h @@ -40,7 +40,8 @@ class LiveVariables : public MachineFunctionPass { public: struct VarInfo { /// DefInst - The machine instruction that defines this register. - MachineInstr *DefInst; + /// + MachineInstr *DefInst; /// AliveBlocks - Set of blocks of which this value is alive completely /// through. This is a bit set which uses the basic block number as an @@ -76,18 +77,22 @@ private: /// std::vector VirtRegInfo; - /// RegistersKilled - This multimap keeps track of all of the registers that + /// RegistersKilled - This map keeps track of all of the registers that /// are dead immediately after an instruction reads its operands. If an /// instruction does not have an entry in this map, it kills no registers. /// - std::multimap RegistersKilled; + std::map > RegistersKilled; - /// RegistersDead - This multimap keeps track of all of the registers that are + /// RegistersDead - This map keeps track of all of the registers that are /// dead immediately after an instruction executes, which are not dead after /// the operands are evaluated. In practice, this only contains registers /// which are defined by an instruction, but never used. /// - std::multimap RegistersDead; + std::map > RegistersDead; + + /// Dummy - An always empty vector used for instructions without dead or + /// killed operands. + std::vector Dummy; /// AllocatablePhysicalRegisters - This vector keeps track of which registers /// are actually register allocatable by the target machine. We can not track @@ -110,51 +115,67 @@ public: /// killed_iterator - Iterate over registers killed by a machine instruction /// - typedef std::multimap::iterator killed_iterator; + typedef std::vector::iterator killed_iterator; + std::vector &getKillsVector(MachineInstr *MI) { + std::map >::iterator I = + RegistersKilled.find(MI); + return I != RegistersKilled.end() ? I->second : Dummy; + } + std::vector &getDeadDefsVector(MachineInstr *MI) { + std::map >::iterator I = + RegistersDead.find(MI); + return I != RegistersDead.end() ? I->second : Dummy; + } + + /// killed_begin/end - Get access to the range of registers killed by a /// machine instruction. killed_iterator killed_begin(MachineInstr *MI) { - return RegistersKilled.lower_bound(MI); + return getKillsVector(MI).begin(); } killed_iterator killed_end(MachineInstr *MI) { - return RegistersKilled.upper_bound(MI); + return getKillsVector(MI).end(); } std::pair killed_range(MachineInstr *MI) { - return RegistersKilled.equal_range(MI); + std::vector &V = getKillsVector(MI); + return std::make_pair(V.begin(), V.end()); } /// KillsRegister - Return true if the specified instruction kills the /// specified register. bool KillsRegister(MachineInstr *MI, unsigned Reg) const { - typedef std::multimap::const_iterator cki; - std::pair KIP = RegistersKilled.equal_range(MI); - for (; KIP.first != KIP.second; ++KIP.first) - if (KIP.first->second == Reg) - return true; + std::map >::const_iterator I = + RegistersKilled.find(MI); + if (I != RegistersKilled.end()) + for (std::vector::const_iterator CI = I->second.begin(), + E = I->second.end(); CI != E; ++CI) + if (*CI == Reg) return true; return false; } killed_iterator dead_begin(MachineInstr *MI) { - return RegistersDead.lower_bound(MI); + return getDeadDefsVector(MI).begin(); } killed_iterator dead_end(MachineInstr *MI) { - return RegistersDead.upper_bound(MI); + return getDeadDefsVector(MI).end(); } std::pair dead_range(MachineInstr *MI) { - return RegistersDead.equal_range(MI); + std::vector &V = getDeadDefsVector(MI); + return std::make_pair(V.begin(), V.end()); } /// RegisterDefIsDead - Return true if the specified instruction defines the /// specified register, but that definition is dead. bool RegisterDefIsDead(MachineInstr *MI, unsigned Reg) const { - typedef std::multimap::const_iterator cki; - std::pair KIP = RegistersDead.equal_range(MI); - for (; KIP.first != KIP.second; ++KIP.first) - if (KIP.first->second == Reg) - return true; + std::map >::const_iterator I = + RegistersDead.find(MI); + if (I != RegistersDead.end()) + for (std::vector::const_iterator CI = I->second.begin(), + E = I->second.end(); CI != E; ++CI) + if (*CI == Reg) return true; return false; } @@ -185,23 +206,20 @@ public: MachineInstr *MI) { if (!getVarInfo(reg).removeKill(MI)) return false; - for (killed_iterator i = killed_begin(MI), e = killed_end(MI); i != e; ) { - if (i->second == reg) - RegistersKilled.erase(i++); - else - ++i; - } + + std::vector &V = getKillsVector(MI); + for (unsigned i = 0, e = V.size(); i != e; ++i) + if (V[i] == reg) { + V.erase(V.begin()+i); + return true; + } return true; } - /// removeVirtualRegistersKilled - Remove all of the specified killed - /// registers from the live variable information. - void removeVirtualRegistersKilled(killed_iterator B, killed_iterator E) { - for (killed_iterator I = B; I != E; ++I) { // Remove VarInfo entries... - bool removed = getVarInfo(I->second).removeKill(I->first); - assert(removed && "kill not in register's VarInfo?"); - } - RegistersKilled.erase(B, E); + /// removeVirtualRegistersKilled - Remove all killed info for the specified + /// instruction. + void removeVirtualRegistersKilled(MachineInstr *MI) { + RegistersKilled.erase(MI); } /// addVirtualRegisterDead - Add information about the fact that the specified @@ -222,21 +240,19 @@ public: if (!getVarInfo(reg).removeKill(MI)) return false; - for (killed_iterator i = killed_begin(MI), e = killed_end(MI); i != e; ) { - if (i->second == reg) - RegistersKilled.erase(i++); - else - ++i; - } + std::vector &V = getDeadDefsVector(MI); + for (unsigned i = 0, e = V.size(); i != e; ++i) + if (V[i] == reg) { + V.erase(V.begin()+i); + return true; + } return true; } /// removeVirtualRegistersDead - Remove all of the specified dead /// registers from the live variable information. - void removeVirtualRegistersDead(killed_iterator B, killed_iterator E) { - for (killed_iterator I = B; I != E; ++I) // Remove VarInfo entries... - getVarInfo(I->second).removeKill(I->first); - RegistersDead.erase(B, E); + void removeVirtualRegistersDead(MachineInstr *MI) { + RegistersDead.erase(MI); } virtual void getAnalysisUsage(AnalysisUsage &AU) const {