Chromium Code Reviews| Index: src/IceInst.cpp |
| diff --git a/src/IceInst.cpp b/src/IceInst.cpp |
| index 659d5e55d80f35ece874c05a1ac9f7c2d6c03d6c..5fd5b592794afbd4228099986f39aafe4fdfb61d 100644 |
| --- a/src/IceInst.cpp |
| +++ b/src/IceInst.cpp |
| @@ -112,6 +112,39 @@ bool Inst::isLastUse(const Operand *TestSrc) const { |
| return false; |
| } |
| +// Given an instruction like: |
| +// a = b + c + [x,y] + e |
| +// which was created from OrigInst: |
| +// a = b + c + d + e |
| +// with SpliceAssn spliced in: |
| +// d = [x,y] |
| +// |
| +// Reconstruct the LiveRangesEnded bitmask in this instruction by |
| +// combining the LiveRangesEnded values of OrigInst and SpliceAssn. |
| +// If operands d and [x,y] contain a different number of variables, |
| +// then the bitmask position for e may be different in OrigInst and |
| +// the current instruction, requiring extra shifts and masks in the |
| +// computation. In the example above, OrigInst has variable e in bit |
| +// position 3, whereas the current instruction has e in bit position 4 |
| +// because [x,y] consumes 2 bitmask slots while d only consumed 1. |
| +void Inst::spliceLastUseInfo(Inst *OrigInst, Inst *SpliceAssn) { |
| + // Find the bitmask index of SpliceAssn's dest within OrigInst. |
| + Variable *SpliceDest = SpliceAssn->getDest(); |
| + SizeT Index = 0; |
| + for (SizeT I = 0; I < OrigInst->getSrcSize(); ++I) { |
| + Operand *Src = OrigInst->getSrc(I); |
| + if (Src == SpliceDest) { |
| + LREndedBits LeftMask = OrigInst->LiveRangesEnded & ((1 << Index) - 1); |
| + LREndedBits RightMask = OrigInst->LiveRangesEnded >> (Index + 1); |
| + LiveRangesEnded = LeftMask | (SpliceAssn->LiveRangesEnded << Index) | |
| + (RightMask >> (Index + getSrc(I)->getNumVars())); |
|
jvoung (off chromium)
2015/06/03 18:18:15
Should this be RightMask << (Index + ...) ?
I not
Jim Stichnoth
2015/06/03 21:06:51
Oops, you're right! Done.
|
| + return; |
| + } |
| + Index += getSrc(I)->getNumVars(); |
| + } |
| + llvm::report_fatal_error("Failed to find splice operand"); |
| +} |
| + |
| void Inst::livenessLightweight(Cfg *Func, LivenessBV &Live) { |
| assert(!isDeleted()); |
| resetLastUses(); |