diff options
| author | Dan Gohman <gohman@apple.com> | 2009-05-22 07:14:20 +0000 |
|---|---|---|
| committer | Dan Gohman <gohman@apple.com> | 2009-05-22 07:14:20 +0000 |
| commit | 88e06db1160672d7152a2bd9db7fcf7977f6bbbd (patch) | |
| tree | 1dfbdd33687df97680b768405496b1f05bc66bdf /llvm/lib/Analysis/ScalarEvolutionExpander.cpp | |
| parent | 9c4dcfb4a7888f08b8438d332cbb269f9420085e (diff) | |
| download | bcm5719-llvm-88e06db1160672d7152a2bd9db7fcf7977f6bbbd.tar.gz bcm5719-llvm-88e06db1160672d7152a2bd9db7fcf7977f6bbbd.zip | |
Fix a thinko in the code that adapted SCEVMulExpr operands for
use in expanding SCEVAddExprs with GEPs. The operands of a
SCEVMulExpr need to be multiplied together, not added.
llvm-svn: 72250
Diffstat (limited to 'llvm/lib/Analysis/ScalarEvolutionExpander.cpp')
| -rw-r--r-- | llvm/lib/Analysis/ScalarEvolutionExpander.cpp | 16 |
1 files changed, 13 insertions, 3 deletions
diff --git a/llvm/lib/Analysis/ScalarEvolutionExpander.cpp b/llvm/lib/Analysis/ScalarEvolutionExpander.cpp index d110385fb3e..507ced74fd1 100644 --- a/llvm/lib/Analysis/ScalarEvolutionExpander.cpp +++ b/llvm/lib/Analysis/ScalarEvolutionExpander.cpp @@ -168,6 +168,8 @@ Value *SCEVExpander::expandAddToGEP(const SCEVAddExpr *S, std::vector<SCEVHandle> ScaledOps; for (unsigned i = 0, e = Ops.size(); i != e; ++i) { if (ElSize != 0) { + // For a Constant, check for a multiple of the pointer type's + // scale size. if (const SCEVConstant *C = dyn_cast<SCEVConstant>(Ops[i])) if (!C->getValue()->getValue().srem(ElSize)) { ConstantInt *CI = @@ -176,13 +178,19 @@ Value *SCEVExpander::expandAddToGEP(const SCEVAddExpr *S, ScaledOps.push_back(Div); continue; } + // In a Mul, check if there is a constant operand which is a multiple + // of the pointer type's scale size. if (const SCEVMulExpr *M = dyn_cast<SCEVMulExpr>(Ops[i])) if (const SCEVConstant *C = dyn_cast<SCEVConstant>(M->getOperand(0))) - if (C->getValue()->getValue() == ElSize) { - for (unsigned j = 1, f = M->getNumOperands(); j != f; ++j) - ScaledOps.push_back(M->getOperand(j)); + if (!C->getValue()->getValue().srem(ElSize)) { + std::vector<SCEVHandle> NewMulOps(M->getOperands()); + NewMulOps[0] = + SE.getConstant(C->getValue()->getValue().sdiv(ElSize)); + ScaledOps.push_back(SE.getMulExpr(NewMulOps)); continue; } + // In an Unknown, check if the underlying value is a Mul by a constant + // which is equal to the pointer type's scale size. if (const SCEVUnknown *U = dyn_cast<SCEVUnknown>(Ops[i])) if (BinaryOperator *BO = dyn_cast<BinaryOperator>(U->getValue())) if (BO->getOpcode() == Instruction::Mul) @@ -191,6 +199,8 @@ Value *SCEVExpander::expandAddToGEP(const SCEVAddExpr *S, ScaledOps.push_back(SE.getUnknown(BO->getOperand(0))); continue; } + // If the pointer type's scale size is 1, no scaling is necessary + // and any value can be used. if (ElSize == 1) { ScaledOps.push_back(Ops[i]); continue; |

