When doing address-mode sinking, expand the base register first, rather

than the scaled register. This makes it more likely that subsequent
AddrModeMatcher queries will match the new address the same way as the
old, instead of accidentally matching what had been the base register
as the new scaled register, and then failing to match the scaled register.
This fixes some problems with address-mode sinking multiple muls into a
block, which will be a lot more common with some upcoming
LoopStrengthReduction changes.

llvm-svn: 93935
This commit is contained in:
Dan Gohman 2010-01-19 22:45:06 +00:00
parent 003b5c8472
commit ca19445d08
1 changed files with 17 additions and 12 deletions

View File

@ -617,7 +617,23 @@ bool CodeGenPrepare::OptimizeMemoryInst(Instruction *MemoryInst, Value *Addr,
TLI->getTargetData()->getIntPtrType(AccessTy->getContext()); TLI->getTargetData()->getIntPtrType(AccessTy->getContext());
Value *Result = 0; Value *Result = 0;
// Start with the scale value.
// Start with the base register. Do this first so that subsequent address
// matching finds it last, which will prevent it from trying to match it
// as the scaled value in case it happens to be a mul. That would be
// problematic if we've sunk a different mul for the scale, because then
// we'd end up sinking both muls.
if (AddrMode.BaseReg) {
Value *V = AddrMode.BaseReg;
if (isa<PointerType>(V->getType()))
V = new PtrToIntInst(V, IntPtrTy, "sunkaddr", InsertPt);
if (V->getType() != IntPtrTy)
V = CastInst::CreateIntegerCast(V, IntPtrTy, /*isSigned=*/true,
"sunkaddr", InsertPt);
Result = V;
}
// Add the scale value.
if (AddrMode.Scale) { if (AddrMode.Scale) {
Value *V = AddrMode.ScaledReg; Value *V = AddrMode.ScaledReg;
if (V->getType() == IntPtrTy) { if (V->getType() == IntPtrTy) {
@ -634,17 +650,6 @@ bool CodeGenPrepare::OptimizeMemoryInst(Instruction *MemoryInst, Value *Addr,
V = BinaryOperator::CreateMul(V, ConstantInt::get(IntPtrTy, V = BinaryOperator::CreateMul(V, ConstantInt::get(IntPtrTy,
AddrMode.Scale), AddrMode.Scale),
"sunkaddr", InsertPt); "sunkaddr", InsertPt);
Result = V;
}
// Add in the base register.
if (AddrMode.BaseReg) {
Value *V = AddrMode.BaseReg;
if (isa<PointerType>(V->getType()))
V = new PtrToIntInst(V, IntPtrTy, "sunkaddr", InsertPt);
if (V->getType() != IntPtrTy)
V = CastInst::CreateIntegerCast(V, IntPtrTy, /*isSigned=*/true,
"sunkaddr", InsertPt);
if (Result) if (Result)
Result = BinaryOperator::CreateAdd(Result, V, "sunkaddr", InsertPt); Result = BinaryOperator::CreateAdd(Result, V, "sunkaddr", InsertPt);
else else