forked from OSchip/llvm-project
[LV] Move code from widenInstruction to VPWidenRecipe. (NFC)
The code in widenInstruction has already been transitioned to only rely on information provided by VPWidenRecipe directly. Moving the code directly to VPWidenRecipe::execute completes the transition for the recipe. It provides the following advantages: 1. Less indirection, easier to see what's going on. 2. Removes accesses to fields of ILV. 2) in particular ensures that no dependencies on fields in ILV for vector code generation are re-introduced. Reviewed By: Ayal Differential Revision: https://reviews.llvm.org/D114322
This commit is contained in:
parent
1b3cd35ade
commit
fd71159f64
|
@ -473,10 +473,6 @@ public:
|
|||
/// handle the more complex control flow around the loops.
|
||||
virtual BasicBlock *createVectorizedLoopSkeleton();
|
||||
|
||||
/// Widen a single instruction within the innermost loop.
|
||||
void widenInstruction(Instruction &I, VPWidenRecipe *WidenRec,
|
||||
VPTransformState &State);
|
||||
|
||||
/// Widen a single call instruction within the innermost loop.
|
||||
void widenCallInstruction(CallInst &I, VPValue *Def, VPUser &ArgOperands,
|
||||
VPTransformState &State);
|
||||
|
@ -4866,123 +4862,6 @@ static bool mayDivideByZero(Instruction &I) {
|
|||
return !CInt || CInt->isZero();
|
||||
}
|
||||
|
||||
void InnerLoopVectorizer::widenInstruction(Instruction &I,
|
||||
VPWidenRecipe *WidenRec,
|
||||
VPTransformState &State) {
|
||||
switch (I.getOpcode()) {
|
||||
case Instruction::Call:
|
||||
case Instruction::Br:
|
||||
case Instruction::PHI:
|
||||
case Instruction::GetElementPtr:
|
||||
case Instruction::Select:
|
||||
llvm_unreachable("This instruction is handled by a different recipe.");
|
||||
case Instruction::UDiv:
|
||||
case Instruction::SDiv:
|
||||
case Instruction::SRem:
|
||||
case Instruction::URem:
|
||||
case Instruction::Add:
|
||||
case Instruction::FAdd:
|
||||
case Instruction::Sub:
|
||||
case Instruction::FSub:
|
||||
case Instruction::FNeg:
|
||||
case Instruction::Mul:
|
||||
case Instruction::FMul:
|
||||
case Instruction::FDiv:
|
||||
case Instruction::FRem:
|
||||
case Instruction::Shl:
|
||||
case Instruction::LShr:
|
||||
case Instruction::AShr:
|
||||
case Instruction::And:
|
||||
case Instruction::Or:
|
||||
case Instruction::Xor: {
|
||||
// Just widen unops and binops.
|
||||
setDebugLocFromInst(&I);
|
||||
|
||||
for (unsigned Part = 0; Part < UF; ++Part) {
|
||||
SmallVector<Value *, 2> Ops;
|
||||
for (VPValue *VPOp : WidenRec->operands())
|
||||
Ops.push_back(State.get(VPOp, Part));
|
||||
|
||||
Value *V = Builder.CreateNAryOp(I.getOpcode(), Ops);
|
||||
|
||||
if (auto *VecOp = dyn_cast<Instruction>(V)) {
|
||||
VecOp->copyIRFlags(&I);
|
||||
|
||||
// If the instruction is vectorized and was in a basic block that needed
|
||||
// predication, we can't propagate poison-generating flags (nuw/nsw,
|
||||
// exact, etc.). The control flow has been linearized and the
|
||||
// instruction is no longer guarded by the predicate, which could make
|
||||
// the flag properties to no longer hold.
|
||||
if (State.MayGeneratePoisonRecipes.count(WidenRec) > 0)
|
||||
VecOp->dropPoisonGeneratingFlags();
|
||||
}
|
||||
|
||||
// Use this vector value for all users of the original instruction.
|
||||
State.set(WidenRec, V, Part);
|
||||
addMetadata(V, &I);
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
case Instruction::ICmp:
|
||||
case Instruction::FCmp: {
|
||||
// Widen compares. Generate vector compares.
|
||||
bool FCmp = (I.getOpcode() == Instruction::FCmp);
|
||||
auto *Cmp = cast<CmpInst>(&I);
|
||||
setDebugLocFromInst(Cmp);
|
||||
for (unsigned Part = 0; Part < UF; ++Part) {
|
||||
Value *A = State.get(WidenRec->getOperand(0), Part);
|
||||
Value *B = State.get(WidenRec->getOperand(1), Part);
|
||||
Value *C = nullptr;
|
||||
if (FCmp) {
|
||||
// Propagate fast math flags.
|
||||
IRBuilder<>::FastMathFlagGuard FMFG(Builder);
|
||||
Builder.setFastMathFlags(Cmp->getFastMathFlags());
|
||||
C = Builder.CreateFCmp(Cmp->getPredicate(), A, B);
|
||||
} else {
|
||||
C = Builder.CreateICmp(Cmp->getPredicate(), A, B);
|
||||
}
|
||||
State.set(WidenRec, C, Part);
|
||||
addMetadata(C, &I);
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
case Instruction::ZExt:
|
||||
case Instruction::SExt:
|
||||
case Instruction::FPToUI:
|
||||
case Instruction::FPToSI:
|
||||
case Instruction::FPExt:
|
||||
case Instruction::PtrToInt:
|
||||
case Instruction::IntToPtr:
|
||||
case Instruction::SIToFP:
|
||||
case Instruction::UIToFP:
|
||||
case Instruction::Trunc:
|
||||
case Instruction::FPTrunc:
|
||||
case Instruction::BitCast: {
|
||||
auto *CI = cast<CastInst>(&I);
|
||||
setDebugLocFromInst(CI);
|
||||
|
||||
/// Vectorize casts.
|
||||
Type *DestTy =
|
||||
(VF.isScalar()) ? CI->getType() : VectorType::get(CI->getType(), VF);
|
||||
|
||||
for (unsigned Part = 0; Part < UF; ++Part) {
|
||||
Value *A = State.get(WidenRec->getOperand(0), Part);
|
||||
Value *Cast = Builder.CreateCast(CI->getOpcode(), A, DestTy);
|
||||
State.set(WidenRec, Cast, Part);
|
||||
addMetadata(Cast, &I);
|
||||
}
|
||||
break;
|
||||
}
|
||||
default:
|
||||
// This instruction is not vectorized by simple widening.
|
||||
LLVM_DEBUG(dbgs() << "LV: Found an unhandled instruction: " << I);
|
||||
llvm_unreachable("Unhandled instruction!");
|
||||
} // end of switch.
|
||||
}
|
||||
|
||||
void InnerLoopVectorizer::widenCallInstruction(CallInst &I, VPValue *Def,
|
||||
VPUser &ArgOperands,
|
||||
VPTransformState &State) {
|
||||
|
@ -9784,7 +9663,121 @@ void VPWidenSelectRecipe::execute(VPTransformState &State) {
|
|||
}
|
||||
|
||||
void VPWidenRecipe::execute(VPTransformState &State) {
|
||||
State.ILV->widenInstruction(*getUnderlyingInstr(), this, State);
|
||||
auto &I = *cast<Instruction>(getUnderlyingValue());
|
||||
auto &Builder = State.Builder;
|
||||
switch (I.getOpcode()) {
|
||||
case Instruction::Call:
|
||||
case Instruction::Br:
|
||||
case Instruction::PHI:
|
||||
case Instruction::GetElementPtr:
|
||||
case Instruction::Select:
|
||||
llvm_unreachable("This instruction is handled by a different recipe.");
|
||||
case Instruction::UDiv:
|
||||
case Instruction::SDiv:
|
||||
case Instruction::SRem:
|
||||
case Instruction::URem:
|
||||
case Instruction::Add:
|
||||
case Instruction::FAdd:
|
||||
case Instruction::Sub:
|
||||
case Instruction::FSub:
|
||||
case Instruction::FNeg:
|
||||
case Instruction::Mul:
|
||||
case Instruction::FMul:
|
||||
case Instruction::FDiv:
|
||||
case Instruction::FRem:
|
||||
case Instruction::Shl:
|
||||
case Instruction::LShr:
|
||||
case Instruction::AShr:
|
||||
case Instruction::And:
|
||||
case Instruction::Or:
|
||||
case Instruction::Xor: {
|
||||
// Just widen unops and binops.
|
||||
State.ILV->setDebugLocFromInst(&I);
|
||||
|
||||
for (unsigned Part = 0; Part < State.UF; ++Part) {
|
||||
SmallVector<Value *, 2> Ops;
|
||||
for (VPValue *VPOp : operands())
|
||||
Ops.push_back(State.get(VPOp, Part));
|
||||
|
||||
Value *V = Builder.CreateNAryOp(I.getOpcode(), Ops);
|
||||
|
||||
if (auto *VecOp = dyn_cast<Instruction>(V)) {
|
||||
VecOp->copyIRFlags(&I);
|
||||
|
||||
// If the instruction is vectorized and was in a basic block that needed
|
||||
// predication, we can't propagate poison-generating flags (nuw/nsw,
|
||||
// exact, etc.). The control flow has been linearized and the
|
||||
// instruction is no longer guarded by the predicate, which could make
|
||||
// the flag properties to no longer hold.
|
||||
if (State.MayGeneratePoisonRecipes.count(this) > 0)
|
||||
VecOp->dropPoisonGeneratingFlags();
|
||||
}
|
||||
|
||||
// Use this vector value for all users of the original instruction.
|
||||
State.set(this, V, Part);
|
||||
State.ILV->addMetadata(V, &I);
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
case Instruction::ICmp:
|
||||
case Instruction::FCmp: {
|
||||
// Widen compares. Generate vector compares.
|
||||
bool FCmp = (I.getOpcode() == Instruction::FCmp);
|
||||
auto *Cmp = cast<CmpInst>(&I);
|
||||
State.ILV->setDebugLocFromInst(Cmp);
|
||||
for (unsigned Part = 0; Part < State.UF; ++Part) {
|
||||
Value *A = State.get(getOperand(0), Part);
|
||||
Value *B = State.get(getOperand(1), Part);
|
||||
Value *C = nullptr;
|
||||
if (FCmp) {
|
||||
// Propagate fast math flags.
|
||||
IRBuilder<>::FastMathFlagGuard FMFG(Builder);
|
||||
Builder.setFastMathFlags(Cmp->getFastMathFlags());
|
||||
C = Builder.CreateFCmp(Cmp->getPredicate(), A, B);
|
||||
} else {
|
||||
C = Builder.CreateICmp(Cmp->getPredicate(), A, B);
|
||||
}
|
||||
State.set(this, C, Part);
|
||||
State.ILV->addMetadata(C, &I);
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
case Instruction::ZExt:
|
||||
case Instruction::SExt:
|
||||
case Instruction::FPToUI:
|
||||
case Instruction::FPToSI:
|
||||
case Instruction::FPExt:
|
||||
case Instruction::PtrToInt:
|
||||
case Instruction::IntToPtr:
|
||||
case Instruction::SIToFP:
|
||||
case Instruction::UIToFP:
|
||||
case Instruction::Trunc:
|
||||
case Instruction::FPTrunc:
|
||||
case Instruction::BitCast: {
|
||||
auto *CI = cast<CastInst>(&I);
|
||||
State.ILV->setDebugLocFromInst(CI);
|
||||
|
||||
/// Vectorize casts.
|
||||
Type *DestTy = (State.VF.isScalar())
|
||||
? CI->getType()
|
||||
: VectorType::get(CI->getType(), State.VF);
|
||||
|
||||
for (unsigned Part = 0; Part < State.UF; ++Part) {
|
||||
Value *A = State.get(getOperand(0), Part);
|
||||
Value *Cast = Builder.CreateCast(CI->getOpcode(), A, DestTy);
|
||||
State.set(this, Cast, Part);
|
||||
State.ILV->addMetadata(Cast, &I);
|
||||
}
|
||||
break;
|
||||
}
|
||||
default:
|
||||
// This instruction is not vectorized by simple widening.
|
||||
LLVM_DEBUG(dbgs() << "LV: Found an unhandled instruction: " << I);
|
||||
llvm_unreachable("Unhandled instruction!");
|
||||
} // end of switch.
|
||||
}
|
||||
|
||||
void VPWidenGEPRecipe::execute(VPTransformState &State) {
|
||||
|
|
Loading…
Reference in New Issue