2017-08-27 20:55:46 +08:00
|
|
|
//===- VPlan.cpp - Vectorizer Plan ----------------------------------------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2017-08-27 20:55:46 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
///
|
|
|
|
/// \file
|
|
|
|
/// This is the LLVM vectorization plan. It represents a candidate for
|
|
|
|
/// vectorization, allowing to plan and optimize how to vectorize a given loop
|
|
|
|
/// before generating LLVM-IR.
|
|
|
|
/// The vectorizer uses vectorization plans to estimate the costs of potential
|
|
|
|
/// candidates and if profitable to execute the desired plan, generating vector
|
|
|
|
/// LLVM-IR code.
|
|
|
|
///
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "VPlan.h"
|
2018-07-31 05:33:31 +08:00
|
|
|
#include "VPlanDominatorTree.h"
|
2017-10-18 05:27:42 +08:00
|
|
|
#include "llvm/ADT/DepthFirstIterator.h"
|
2017-08-27 20:55:46 +08:00
|
|
|
#include "llvm/ADT/PostOrderIterator.h"
|
2017-10-18 05:27:42 +08:00
|
|
|
#include "llvm/ADT/SmallVector.h"
|
|
|
|
#include "llvm/ADT/Twine.h"
|
2017-08-27 20:55:46 +08:00
|
|
|
#include "llvm/Analysis/LoopInfo.h"
|
|
|
|
#include "llvm/IR/BasicBlock.h"
|
2017-10-18 05:27:42 +08:00
|
|
|
#include "llvm/IR/CFG.h"
|
|
|
|
#include "llvm/IR/InstrTypes.h"
|
|
|
|
#include "llvm/IR/Instruction.h"
|
|
|
|
#include "llvm/IR/Instructions.h"
|
|
|
|
#include "llvm/IR/Type.h"
|
|
|
|
#include "llvm/IR/Value.h"
|
|
|
|
#include "llvm/Support/Casting.h"
|
2019-11-15 07:15:48 +08:00
|
|
|
#include "llvm/Support/CommandLine.h"
|
2017-10-18 05:27:42 +08:00
|
|
|
#include "llvm/Support/Debug.h"
|
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2018-07-31 05:33:31 +08:00
|
|
|
#include "llvm/Support/GenericDomTreeConstruction.h"
|
2017-08-27 20:55:46 +08:00
|
|
|
#include "llvm/Support/GraphWriter.h"
|
2017-10-18 05:27:42 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2017-08-27 20:55:46 +08:00
|
|
|
#include "llvm/Transforms/Utils/BasicBlockUtils.h"
|
2017-10-18 05:27:42 +08:00
|
|
|
#include <cassert>
|
|
|
|
#include <iterator>
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2017-08-27 20:55:46 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
[VPlan] Implement initial vector code generation support for simple outer loops.
Summary:
[VPlan] Implement vector code generation support for simple outer loops.
Context: Patch Series #1 for outer loop vectorization support in LV using VPlan. (RFC: http://lists.llvm.org/pipermail/llvm-dev/2017-December/119523.html).
This patch introduces vector code generation support for simple outer loops that are currently supported in the VPlanNativePath. Changes here essentially do the following:
- force vector code generation using explicit vectorize_width
- add conservative early returns in cost model and other places for VPlanNativePath
- add code for setting up outer loop inductions
- support for widening non-induction PHIs that can result from inner loops and uniform conditional branches
- support for generating uniform inner branches
We plan to add a handful C outer loop executable tests once the initial code generation support is committed. This patch is expected to be NFC for the inner loop vectorizer path. Since we are moving in the direction of supporting outer loop vectorization in LV, it may also be time to rename classes such as InnerLoopVectorizer.
Reviewers: fhahn, rengolin, hsaito, dcaballe, mkuper, hfinkel, Ayal
Reviewed By: fhahn, hsaito
Subscribers: dmgreen, bollu, tschuett, rkruppe, rogfer01, llvm-commits
Differential Revision: https://reviews.llvm.org/D50820
llvm-svn: 342197
2018-09-14 08:36:00 +08:00
|
|
|
extern cl::opt<bool> EnableVPlanNativePath;
|
2017-08-27 20:55:46 +08:00
|
|
|
|
|
|
|
#define DEBUG_TYPE "vplan"
|
|
|
|
|
2017-11-20 20:01:47 +08:00
|
|
|
raw_ostream &llvm::operator<<(raw_ostream &OS, const VPValue &V) {
|
|
|
|
if (const VPInstruction *Instr = dyn_cast<VPInstruction>(&V))
|
|
|
|
Instr->print(OS);
|
|
|
|
else
|
|
|
|
V.printAsOperand(OS);
|
|
|
|
return OS;
|
|
|
|
}
|
|
|
|
|
2017-08-27 20:55:46 +08:00
|
|
|
/// \return the VPBasicBlock that is the entry of Block, possibly indirectly.
|
|
|
|
const VPBasicBlock *VPBlockBase::getEntryBasicBlock() const {
|
|
|
|
const VPBlockBase *Block = this;
|
|
|
|
while (const VPRegionBlock *Region = dyn_cast<VPRegionBlock>(Block))
|
|
|
|
Block = Region->getEntry();
|
|
|
|
return cast<VPBasicBlock>(Block);
|
|
|
|
}
|
|
|
|
|
|
|
|
VPBasicBlock *VPBlockBase::getEntryBasicBlock() {
|
|
|
|
VPBlockBase *Block = this;
|
|
|
|
while (VPRegionBlock *Region = dyn_cast<VPRegionBlock>(Block))
|
|
|
|
Block = Region->getEntry();
|
|
|
|
return cast<VPBasicBlock>(Block);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \return the VPBasicBlock that is the exit of Block, possibly indirectly.
|
|
|
|
const VPBasicBlock *VPBlockBase::getExitBasicBlock() const {
|
|
|
|
const VPBlockBase *Block = this;
|
|
|
|
while (const VPRegionBlock *Region = dyn_cast<VPRegionBlock>(Block))
|
|
|
|
Block = Region->getExit();
|
|
|
|
return cast<VPBasicBlock>(Block);
|
|
|
|
}
|
|
|
|
|
|
|
|
VPBasicBlock *VPBlockBase::getExitBasicBlock() {
|
|
|
|
VPBlockBase *Block = this;
|
|
|
|
while (VPRegionBlock *Region = dyn_cast<VPRegionBlock>(Block))
|
|
|
|
Block = Region->getExit();
|
|
|
|
return cast<VPBasicBlock>(Block);
|
|
|
|
}
|
|
|
|
|
|
|
|
VPBlockBase *VPBlockBase::getEnclosingBlockWithSuccessors() {
|
|
|
|
if (!Successors.empty() || !Parent)
|
|
|
|
return this;
|
|
|
|
assert(Parent->getExit() == this &&
|
|
|
|
"Block w/o successors not the exit of its parent.");
|
|
|
|
return Parent->getEnclosingBlockWithSuccessors();
|
|
|
|
}
|
|
|
|
|
|
|
|
VPBlockBase *VPBlockBase::getEnclosingBlockWithPredecessors() {
|
|
|
|
if (!Predecessors.empty() || !Parent)
|
|
|
|
return this;
|
|
|
|
assert(Parent->getEntry() == this &&
|
|
|
|
"Block w/o predecessors not the entry of its parent.");
|
|
|
|
return Parent->getEnclosingBlockWithPredecessors();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPBlockBase::deleteCFG(VPBlockBase *Entry) {
|
|
|
|
SmallVector<VPBlockBase *, 8> Blocks;
|
|
|
|
for (VPBlockBase *Block : depth_first(Entry))
|
|
|
|
Blocks.push_back(Block);
|
|
|
|
|
|
|
|
for (VPBlockBase *Block : Blocks)
|
|
|
|
delete Block;
|
|
|
|
}
|
|
|
|
|
|
|
|
BasicBlock *
|
|
|
|
VPBasicBlock::createEmptyBasicBlock(VPTransformState::CFGState &CFG) {
|
|
|
|
// BB stands for IR BasicBlocks. VPBB stands for VPlan VPBasicBlocks.
|
|
|
|
// Pred stands for Predessor. Prev stands for Previous - last visited/created.
|
|
|
|
BasicBlock *PrevBB = CFG.PrevBB;
|
|
|
|
BasicBlock *NewBB = BasicBlock::Create(PrevBB->getContext(), getName(),
|
|
|
|
PrevBB->getParent(), CFG.LastBB);
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "LV: created " << NewBB->getName() << '\n');
|
2017-08-27 20:55:46 +08:00
|
|
|
|
|
|
|
// Hook up the new basic block to its predecessors.
|
|
|
|
for (VPBlockBase *PredVPBlock : getHierarchicalPredecessors()) {
|
|
|
|
VPBasicBlock *PredVPBB = PredVPBlock->getExitBasicBlock();
|
|
|
|
auto &PredVPSuccessors = PredVPBB->getSuccessors();
|
|
|
|
BasicBlock *PredBB = CFG.VPBB2IRBB[PredVPBB];
|
[VPlan] Implement initial vector code generation support for simple outer loops.
Summary:
[VPlan] Implement vector code generation support for simple outer loops.
Context: Patch Series #1 for outer loop vectorization support in LV using VPlan. (RFC: http://lists.llvm.org/pipermail/llvm-dev/2017-December/119523.html).
This patch introduces vector code generation support for simple outer loops that are currently supported in the VPlanNativePath. Changes here essentially do the following:
- force vector code generation using explicit vectorize_width
- add conservative early returns in cost model and other places for VPlanNativePath
- add code for setting up outer loop inductions
- support for widening non-induction PHIs that can result from inner loops and uniform conditional branches
- support for generating uniform inner branches
We plan to add a handful C outer loop executable tests once the initial code generation support is committed. This patch is expected to be NFC for the inner loop vectorizer path. Since we are moving in the direction of supporting outer loop vectorization in LV, it may also be time to rename classes such as InnerLoopVectorizer.
Reviewers: fhahn, rengolin, hsaito, dcaballe, mkuper, hfinkel, Ayal
Reviewed By: fhahn, hsaito
Subscribers: dmgreen, bollu, tschuett, rkruppe, rogfer01, llvm-commits
Differential Revision: https://reviews.llvm.org/D50820
llvm-svn: 342197
2018-09-14 08:36:00 +08:00
|
|
|
|
|
|
|
// In outer loop vectorization scenario, the predecessor BBlock may not yet
|
|
|
|
// be visited(backedge). Mark the VPBasicBlock for fixup at the end of
|
|
|
|
// vectorization. We do not encounter this case in inner loop vectorization
|
|
|
|
// as we start out by building a loop skeleton with the vector loop header
|
|
|
|
// and latch blocks. As a result, we never enter this function for the
|
|
|
|
// header block in the non VPlan-native path.
|
|
|
|
if (!PredBB) {
|
|
|
|
assert(EnableVPlanNativePath &&
|
|
|
|
"Unexpected null predecessor in non VPlan-native path");
|
|
|
|
CFG.VPBBsToFix.push_back(PredVPBB);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-08-27 20:55:46 +08:00
|
|
|
assert(PredBB && "Predecessor basic-block not found building successor.");
|
|
|
|
auto *PredBBTerminator = PredBB->getTerminator();
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "LV: draw edge from" << PredBB->getName() << '\n');
|
2017-08-27 20:55:46 +08:00
|
|
|
if (isa<UnreachableInst>(PredBBTerminator)) {
|
|
|
|
assert(PredVPSuccessors.size() == 1 &&
|
|
|
|
"Predecessor ending w/o branch must have single successor.");
|
|
|
|
PredBBTerminator->eraseFromParent();
|
|
|
|
BranchInst::Create(NewBB, PredBB);
|
|
|
|
} else {
|
|
|
|
assert(PredVPSuccessors.size() == 2 &&
|
|
|
|
"Predecessor ending with branch must have two successors.");
|
|
|
|
unsigned idx = PredVPSuccessors.front() == this ? 0 : 1;
|
|
|
|
assert(!PredBBTerminator->getSuccessor(idx) &&
|
|
|
|
"Trying to reset an existing successor block.");
|
|
|
|
PredBBTerminator->setSuccessor(idx, NewBB);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NewBB;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPBasicBlock::execute(VPTransformState *State) {
|
|
|
|
bool Replica = State->Instance &&
|
|
|
|
!(State->Instance->Part == 0 && State->Instance->Lane == 0);
|
|
|
|
VPBasicBlock *PrevVPBB = State->CFG.PrevVPBB;
|
|
|
|
VPBlockBase *SingleHPred = nullptr;
|
|
|
|
BasicBlock *NewBB = State->CFG.PrevBB; // Reuse it if possible.
|
|
|
|
|
|
|
|
// 1. Create an IR basic block, or reuse the last one if possible.
|
|
|
|
// The last IR basic block is reused, as an optimization, in three cases:
|
|
|
|
// A. the first VPBB reuses the loop header BB - when PrevVPBB is null;
|
|
|
|
// B. when the current VPBB has a single (hierarchical) predecessor which
|
|
|
|
// is PrevVPBB and the latter has a single (hierarchical) successor; and
|
|
|
|
// C. when the current VPBB is an entry of a region replica - where PrevVPBB
|
|
|
|
// is the exit of this region from a previous instance, or the predecessor
|
|
|
|
// of this region.
|
|
|
|
if (PrevVPBB && /* A */
|
|
|
|
!((SingleHPred = getSingleHierarchicalPredecessor()) &&
|
|
|
|
SingleHPred->getExitBasicBlock() == PrevVPBB &&
|
|
|
|
PrevVPBB->getSingleHierarchicalSuccessor()) && /* B */
|
|
|
|
!(Replica && getPredecessors().empty())) { /* C */
|
|
|
|
NewBB = createEmptyBasicBlock(State->CFG);
|
|
|
|
State->Builder.SetInsertPoint(NewBB);
|
|
|
|
// Temporarily terminate with unreachable until CFG is rewired.
|
|
|
|
UnreachableInst *Terminator = State->Builder.CreateUnreachable();
|
|
|
|
State->Builder.SetInsertPoint(Terminator);
|
|
|
|
// Register NewBB in its loop. In innermost loops its the same for all BB's.
|
|
|
|
Loop *L = State->LI->getLoopFor(State->CFG.LastBB);
|
|
|
|
L->addBasicBlockToLoop(NewBB, *State->LI);
|
|
|
|
State->CFG.PrevBB = NewBB;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 2. Fill the IR basic block with IR instructions.
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "LV: vectorizing VPBB:" << getName()
|
|
|
|
<< " in BB:" << NewBB->getName() << '\n');
|
2017-08-27 20:55:46 +08:00
|
|
|
|
|
|
|
State->CFG.VPBB2IRBB[this] = NewBB;
|
|
|
|
State->CFG.PrevVPBB = this;
|
|
|
|
|
|
|
|
for (VPRecipeBase &Recipe : Recipes)
|
|
|
|
Recipe.execute(*State);
|
|
|
|
|
[VPlan] Implement initial vector code generation support for simple outer loops.
Summary:
[VPlan] Implement vector code generation support for simple outer loops.
Context: Patch Series #1 for outer loop vectorization support in LV using VPlan. (RFC: http://lists.llvm.org/pipermail/llvm-dev/2017-December/119523.html).
This patch introduces vector code generation support for simple outer loops that are currently supported in the VPlanNativePath. Changes here essentially do the following:
- force vector code generation using explicit vectorize_width
- add conservative early returns in cost model and other places for VPlanNativePath
- add code for setting up outer loop inductions
- support for widening non-induction PHIs that can result from inner loops and uniform conditional branches
- support for generating uniform inner branches
We plan to add a handful C outer loop executable tests once the initial code generation support is committed. This patch is expected to be NFC for the inner loop vectorizer path. Since we are moving in the direction of supporting outer loop vectorization in LV, it may also be time to rename classes such as InnerLoopVectorizer.
Reviewers: fhahn, rengolin, hsaito, dcaballe, mkuper, hfinkel, Ayal
Reviewed By: fhahn, hsaito
Subscribers: dmgreen, bollu, tschuett, rkruppe, rogfer01, llvm-commits
Differential Revision: https://reviews.llvm.org/D50820
llvm-svn: 342197
2018-09-14 08:36:00 +08:00
|
|
|
VPValue *CBV;
|
|
|
|
if (EnableVPlanNativePath && (CBV = getCondBit())) {
|
|
|
|
Value *IRCBV = CBV->getUnderlyingValue();
|
|
|
|
assert(IRCBV && "Unexpected null underlying value for condition bit");
|
|
|
|
|
|
|
|
// Condition bit value in a VPBasicBlock is used as the branch selector. In
|
|
|
|
// the VPlan-native path case, since all branches are uniform we generate a
|
|
|
|
// branch instruction using the condition value from vector lane 0 and dummy
|
|
|
|
// successors. The successors are fixed later when the successor blocks are
|
|
|
|
// visited.
|
|
|
|
Value *NewCond = State->Callback.getOrCreateVectorValues(IRCBV, 0);
|
|
|
|
NewCond = State->Builder.CreateExtractElement(NewCond,
|
|
|
|
State->Builder.getInt32(0));
|
|
|
|
|
|
|
|
// Replace the temporary unreachable terminator with the new conditional
|
|
|
|
// branch.
|
|
|
|
auto *CurrentTerminator = NewBB->getTerminator();
|
|
|
|
assert(isa<UnreachableInst>(CurrentTerminator) &&
|
|
|
|
"Expected to replace unreachable terminator with conditional "
|
|
|
|
"branch.");
|
|
|
|
auto *CondBr = BranchInst::Create(NewBB, nullptr, NewCond);
|
|
|
|
CondBr->setSuccessor(0, nullptr);
|
|
|
|
ReplaceInstWithInst(CurrentTerminator, CondBr);
|
|
|
|
}
|
|
|
|
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "LV: filled BB:" << *NewBB);
|
2017-08-27 20:55:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void VPRegionBlock::execute(VPTransformState *State) {
|
|
|
|
ReversePostOrderTraversal<VPBlockBase *> RPOT(Entry);
|
|
|
|
|
|
|
|
if (!isReplicator()) {
|
|
|
|
// Visit the VPBlocks connected to "this", starting from it.
|
|
|
|
for (VPBlockBase *Block : RPOT) {
|
[VPlan] Implement initial vector code generation support for simple outer loops.
Summary:
[VPlan] Implement vector code generation support for simple outer loops.
Context: Patch Series #1 for outer loop vectorization support in LV using VPlan. (RFC: http://lists.llvm.org/pipermail/llvm-dev/2017-December/119523.html).
This patch introduces vector code generation support for simple outer loops that are currently supported in the VPlanNativePath. Changes here essentially do the following:
- force vector code generation using explicit vectorize_width
- add conservative early returns in cost model and other places for VPlanNativePath
- add code for setting up outer loop inductions
- support for widening non-induction PHIs that can result from inner loops and uniform conditional branches
- support for generating uniform inner branches
We plan to add a handful C outer loop executable tests once the initial code generation support is committed. This patch is expected to be NFC for the inner loop vectorizer path. Since we are moving in the direction of supporting outer loop vectorization in LV, it may also be time to rename classes such as InnerLoopVectorizer.
Reviewers: fhahn, rengolin, hsaito, dcaballe, mkuper, hfinkel, Ayal
Reviewed By: fhahn, hsaito
Subscribers: dmgreen, bollu, tschuett, rkruppe, rogfer01, llvm-commits
Differential Revision: https://reviews.llvm.org/D50820
llvm-svn: 342197
2018-09-14 08:36:00 +08:00
|
|
|
if (EnableVPlanNativePath) {
|
|
|
|
// The inner loop vectorization path does not represent loop preheader
|
|
|
|
// and exit blocks as part of the VPlan. In the VPlan-native path, skip
|
|
|
|
// vectorizing loop preheader block. In future, we may replace this
|
|
|
|
// check with the check for loop preheader.
|
|
|
|
if (Block->getNumPredecessors() == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Skip vectorizing loop exit block. In future, we may replace this
|
|
|
|
// check with the check for loop exit.
|
|
|
|
if (Block->getNumSuccessors() == 0)
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "LV: VPBlock in RPO " << Block->getName() << '\n');
|
2017-08-27 20:55:46 +08:00
|
|
|
Block->execute(State);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(!State->Instance && "Replicating a Region with non-null instance.");
|
|
|
|
|
|
|
|
// Enter replicating mode.
|
|
|
|
State->Instance = {0, 0};
|
|
|
|
|
|
|
|
for (unsigned Part = 0, UF = State->UF; Part < UF; ++Part) {
|
|
|
|
State->Instance->Part = Part;
|
|
|
|
for (unsigned Lane = 0, VF = State->VF; Lane < VF; ++Lane) {
|
|
|
|
State->Instance->Lane = Lane;
|
|
|
|
// Visit the VPBlocks connected to \p this, starting from it.
|
|
|
|
for (VPBlockBase *Block : RPOT) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "LV: VPBlock in RPO " << Block->getName() << '\n');
|
2017-08-27 20:55:46 +08:00
|
|
|
Block->execute(State);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exit replicating mode.
|
|
|
|
State->Instance.reset();
|
|
|
|
}
|
|
|
|
|
2018-06-18 19:34:17 +08:00
|
|
|
void VPRecipeBase::insertBefore(VPRecipeBase *InsertPos) {
|
2019-10-07 22:24:33 +08:00
|
|
|
assert(!Parent && "Recipe already in some VPBasicBlock");
|
|
|
|
assert(InsertPos->getParent() &&
|
|
|
|
"Insertion position not in any VPBasicBlock");
|
2018-06-18 21:51:28 +08:00
|
|
|
Parent = InsertPos->getParent();
|
|
|
|
Parent->getRecipeList().insert(InsertPos->getIterator(), this);
|
2018-06-18 19:34:17 +08:00
|
|
|
}
|
|
|
|
|
2019-10-07 22:24:33 +08:00
|
|
|
void VPRecipeBase::insertAfter(VPRecipeBase *InsertPos) {
|
|
|
|
assert(!Parent && "Recipe already in some VPBasicBlock");
|
|
|
|
assert(InsertPos->getParent() &&
|
|
|
|
"Insertion position not in any VPBasicBlock");
|
|
|
|
Parent = InsertPos->getParent();
|
|
|
|
Parent->getRecipeList().insertAfter(InsertPos->getIterator(), this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPRecipeBase::removeFromParent() {
|
|
|
|
assert(getParent() && "Recipe not in any VPBasicBlock");
|
|
|
|
getParent()->getRecipeList().remove(getIterator());
|
|
|
|
Parent = nullptr;
|
|
|
|
}
|
|
|
|
|
2018-06-18 23:18:48 +08:00
|
|
|
iplist<VPRecipeBase>::iterator VPRecipeBase::eraseFromParent() {
|
2019-10-07 22:24:33 +08:00
|
|
|
assert(getParent() && "Recipe not in any VPBasicBlock");
|
2018-06-18 23:18:48 +08:00
|
|
|
return getParent()->getRecipeList().erase(getIterator());
|
|
|
|
}
|
|
|
|
|
2019-10-11 23:36:55 +08:00
|
|
|
void VPRecipeBase::moveAfter(VPRecipeBase *InsertPos) {
|
2019-10-07 22:24:33 +08:00
|
|
|
removeFromParent();
|
|
|
|
insertAfter(InsertPos);
|
2019-10-11 23:36:55 +08:00
|
|
|
}
|
|
|
|
|
2017-11-20 20:01:47 +08:00
|
|
|
void VPInstruction::generateInstruction(VPTransformState &State,
|
|
|
|
unsigned Part) {
|
|
|
|
IRBuilder<> &Builder = State.Builder;
|
|
|
|
|
|
|
|
if (Instruction::isBinaryOp(getOpcode())) {
|
|
|
|
Value *A = State.get(getOperand(0), Part);
|
|
|
|
Value *B = State.get(getOperand(1), Part);
|
|
|
|
Value *V = Builder.CreateBinOp((Instruction::BinaryOps)getOpcode(), A, B);
|
|
|
|
State.set(this, V, Part);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (getOpcode()) {
|
|
|
|
case VPInstruction::Not: {
|
|
|
|
Value *A = State.get(getOperand(0), Part);
|
|
|
|
Value *V = Builder.CreateNot(A);
|
|
|
|
State.set(this, V, Part);
|
|
|
|
break;
|
|
|
|
}
|
2018-10-18 23:03:15 +08:00
|
|
|
case VPInstruction::ICmpULE: {
|
|
|
|
Value *IV = State.get(getOperand(0), Part);
|
|
|
|
Value *TC = State.get(getOperand(1), Part);
|
|
|
|
Value *V = Builder.CreateICmpULE(IV, TC);
|
|
|
|
State.set(this, V, Part);
|
|
|
|
break;
|
|
|
|
}
|
2019-08-28 17:02:23 +08:00
|
|
|
case Instruction::Select: {
|
|
|
|
Value *Cond = State.get(getOperand(0), Part);
|
|
|
|
Value *Op1 = State.get(getOperand(1), Part);
|
|
|
|
Value *Op2 = State.get(getOperand(2), Part);
|
|
|
|
Value *V = Builder.CreateSelect(Cond, Op1, Op2);
|
|
|
|
State.set(this, V, Part);
|
|
|
|
break;
|
|
|
|
}
|
2017-11-20 20:01:47 +08:00
|
|
|
default:
|
|
|
|
llvm_unreachable("Unsupported opcode for instruction");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPInstruction::execute(VPTransformState &State) {
|
|
|
|
assert(!State.Instance && "VPInstruction executing an Instance");
|
|
|
|
for (unsigned Part = 0; Part < State.UF; ++Part)
|
|
|
|
generateInstruction(State, Part);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPInstruction::print(raw_ostream &O, const Twine &Indent) const {
|
|
|
|
O << " +\n" << Indent << "\"EMIT ";
|
|
|
|
print(O);
|
|
|
|
O << "\\l\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPInstruction::print(raw_ostream &O) const {
|
|
|
|
printAsOperand(O);
|
|
|
|
O << " = ";
|
|
|
|
|
|
|
|
switch (getOpcode()) {
|
|
|
|
case VPInstruction::Not:
|
|
|
|
O << "not";
|
|
|
|
break;
|
2018-10-18 23:03:15 +08:00
|
|
|
case VPInstruction::ICmpULE:
|
|
|
|
O << "icmp ule";
|
|
|
|
break;
|
[VPlan, SLP] Add simple SLP analysis on top of VPlan.
This patch adds an initial implementation of the look-ahead SLP tree
construction described in 'Look-Ahead SLP: Auto-vectorization in the Presence
of Commutative Operations, CGO 2018 by Vasileios Porpodas, Rodrigo C. O. Rocha,
Luís F. W. Góes'.
It returns an SLP tree represented as VPInstructions, with combined
instructions represented as a single, wider VPInstruction.
This initial version does not support instructions with multiple
different users (either inside or outside the SLP tree) or
non-instruction operands; it won't generate any shuffles or
insertelement instructions.
It also just adds the analysis that builds an SLP tree rooted in a set
of stores. It does not include any cost modeling or memory legality
checks. The plan is to integrate it with VPlan based cost modeling, once
available and to only apply it to operations that can be widened.
A follow-up patch will add a support for replacing instructions in a
VPlan with their SLP counter parts.
Reviewers: Ayal, mssimpso, rengolin, mkuper, hfinkel, hsaito, dcaballe, vporpo, RKSimon, ABataev
Reviewed By: rengolin
Differential Revision: https://reviews.llvm.org/D4949
llvm-svn: 346857
2018-11-14 21:11:49 +08:00
|
|
|
case VPInstruction::SLPLoad:
|
|
|
|
O << "combined load";
|
|
|
|
break;
|
|
|
|
case VPInstruction::SLPStore:
|
|
|
|
O << "combined store";
|
|
|
|
break;
|
2017-11-20 20:01:47 +08:00
|
|
|
default:
|
|
|
|
O << Instruction::getOpcodeName(getOpcode());
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const VPValue *Operand : operands()) {
|
|
|
|
O << " ";
|
|
|
|
Operand->printAsOperand(O);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-27 20:55:46 +08:00
|
|
|
/// Generate the code inside the body of the vectorized loop. Assumes a single
|
|
|
|
/// LoopVectorBody basic-block was created for this. Introduce additional
|
|
|
|
/// basic-blocks as needed, and fill them all.
|
|
|
|
void VPlan::execute(VPTransformState *State) {
|
2018-10-18 23:03:15 +08:00
|
|
|
// -1. Check if the backedge taken count is needed, and if so build it.
|
|
|
|
if (BackedgeTakenCount && BackedgeTakenCount->getNumUsers()) {
|
|
|
|
Value *TC = State->TripCount;
|
|
|
|
IRBuilder<> Builder(State->CFG.PrevBB->getTerminator());
|
|
|
|
auto *TCMO = Builder.CreateSub(TC, ConstantInt::get(TC->getType(), 1),
|
|
|
|
"trip.count.minus.1");
|
|
|
|
Value2VPValue[TCMO] = BackedgeTakenCount;
|
|
|
|
}
|
|
|
|
|
2017-11-20 20:01:47 +08:00
|
|
|
// 0. Set the reverse mapping from VPValues to Values for code generation.
|
|
|
|
for (auto &Entry : Value2VPValue)
|
|
|
|
State->VPValue2Value[Entry.second] = Entry.first;
|
|
|
|
|
2017-08-27 20:55:46 +08:00
|
|
|
BasicBlock *VectorPreHeaderBB = State->CFG.PrevBB;
|
|
|
|
BasicBlock *VectorHeaderBB = VectorPreHeaderBB->getSingleSuccessor();
|
|
|
|
assert(VectorHeaderBB && "Loop preheader does not have a single successor.");
|
|
|
|
|
|
|
|
// 1. Make room to generate basic-blocks inside loop body if needed.
|
2019-05-08 18:52:26 +08:00
|
|
|
BasicBlock *VectorLatchBB = VectorHeaderBB->splitBasicBlock(
|
2017-08-27 20:55:46 +08:00
|
|
|
VectorHeaderBB->getFirstInsertionPt(), "vector.body.latch");
|
|
|
|
Loop *L = State->LI->getLoopFor(VectorHeaderBB);
|
|
|
|
L->addBasicBlockToLoop(VectorLatchBB, *State->LI);
|
|
|
|
// Remove the edge between Header and Latch to allow other connections.
|
|
|
|
// Temporarily terminate with unreachable until CFG is rewired.
|
|
|
|
// Note: this asserts the generated code's assumption that
|
|
|
|
// getFirstInsertionPt() can be dereferenced into an Instruction.
|
|
|
|
VectorHeaderBB->getTerminator()->eraseFromParent();
|
|
|
|
State->Builder.SetInsertPoint(VectorHeaderBB);
|
|
|
|
UnreachableInst *Terminator = State->Builder.CreateUnreachable();
|
|
|
|
State->Builder.SetInsertPoint(Terminator);
|
|
|
|
|
|
|
|
// 2. Generate code in loop body.
|
|
|
|
State->CFG.PrevVPBB = nullptr;
|
|
|
|
State->CFG.PrevBB = VectorHeaderBB;
|
|
|
|
State->CFG.LastBB = VectorLatchBB;
|
|
|
|
|
|
|
|
for (VPBlockBase *Block : depth_first(Entry))
|
|
|
|
Block->execute(State);
|
|
|
|
|
[VPlan] Implement initial vector code generation support for simple outer loops.
Summary:
[VPlan] Implement vector code generation support for simple outer loops.
Context: Patch Series #1 for outer loop vectorization support in LV using VPlan. (RFC: http://lists.llvm.org/pipermail/llvm-dev/2017-December/119523.html).
This patch introduces vector code generation support for simple outer loops that are currently supported in the VPlanNativePath. Changes here essentially do the following:
- force vector code generation using explicit vectorize_width
- add conservative early returns in cost model and other places for VPlanNativePath
- add code for setting up outer loop inductions
- support for widening non-induction PHIs that can result from inner loops and uniform conditional branches
- support for generating uniform inner branches
We plan to add a handful C outer loop executable tests once the initial code generation support is committed. This patch is expected to be NFC for the inner loop vectorizer path. Since we are moving in the direction of supporting outer loop vectorization in LV, it may also be time to rename classes such as InnerLoopVectorizer.
Reviewers: fhahn, rengolin, hsaito, dcaballe, mkuper, hfinkel, Ayal
Reviewed By: fhahn, hsaito
Subscribers: dmgreen, bollu, tschuett, rkruppe, rogfer01, llvm-commits
Differential Revision: https://reviews.llvm.org/D50820
llvm-svn: 342197
2018-09-14 08:36:00 +08:00
|
|
|
// Setup branch terminator successors for VPBBs in VPBBsToFix based on
|
|
|
|
// VPBB's successors.
|
|
|
|
for (auto VPBB : State->CFG.VPBBsToFix) {
|
|
|
|
assert(EnableVPlanNativePath &&
|
|
|
|
"Unexpected VPBBsToFix in non VPlan-native path");
|
|
|
|
BasicBlock *BB = State->CFG.VPBB2IRBB[VPBB];
|
|
|
|
assert(BB && "Unexpected null basic block for VPBB");
|
|
|
|
|
|
|
|
unsigned Idx = 0;
|
|
|
|
auto *BBTerminator = BB->getTerminator();
|
|
|
|
|
|
|
|
for (VPBlockBase *SuccVPBlock : VPBB->getHierarchicalSuccessors()) {
|
|
|
|
VPBasicBlock *SuccVPBB = SuccVPBlock->getEntryBasicBlock();
|
|
|
|
BBTerminator->setSuccessor(Idx, State->CFG.VPBB2IRBB[SuccVPBB]);
|
|
|
|
++Idx;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-27 20:55:46 +08:00
|
|
|
// 3. Merge the temporary latch created with the last basic-block filled.
|
|
|
|
BasicBlock *LastBB = State->CFG.PrevBB;
|
|
|
|
// Connect LastBB to VectorLatchBB to facilitate their merge.
|
[VPlan] Implement initial vector code generation support for simple outer loops.
Summary:
[VPlan] Implement vector code generation support for simple outer loops.
Context: Patch Series #1 for outer loop vectorization support in LV using VPlan. (RFC: http://lists.llvm.org/pipermail/llvm-dev/2017-December/119523.html).
This patch introduces vector code generation support for simple outer loops that are currently supported in the VPlanNativePath. Changes here essentially do the following:
- force vector code generation using explicit vectorize_width
- add conservative early returns in cost model and other places for VPlanNativePath
- add code for setting up outer loop inductions
- support for widening non-induction PHIs that can result from inner loops and uniform conditional branches
- support for generating uniform inner branches
We plan to add a handful C outer loop executable tests once the initial code generation support is committed. This patch is expected to be NFC for the inner loop vectorizer path. Since we are moving in the direction of supporting outer loop vectorization in LV, it may also be time to rename classes such as InnerLoopVectorizer.
Reviewers: fhahn, rengolin, hsaito, dcaballe, mkuper, hfinkel, Ayal
Reviewed By: fhahn, hsaito
Subscribers: dmgreen, bollu, tschuett, rkruppe, rogfer01, llvm-commits
Differential Revision: https://reviews.llvm.org/D50820
llvm-svn: 342197
2018-09-14 08:36:00 +08:00
|
|
|
assert((EnableVPlanNativePath ||
|
|
|
|
isa<UnreachableInst>(LastBB->getTerminator())) &&
|
|
|
|
"Expected InnerLoop VPlan CFG to terminate with unreachable");
|
|
|
|
assert((!EnableVPlanNativePath || isa<BranchInst>(LastBB->getTerminator())) &&
|
|
|
|
"Expected VPlan CFG to terminate with branch in NativePath");
|
2017-08-27 20:55:46 +08:00
|
|
|
LastBB->getTerminator()->eraseFromParent();
|
|
|
|
BranchInst::Create(VectorLatchBB, LastBB);
|
|
|
|
|
|
|
|
// Merge LastBB with Latch.
|
|
|
|
bool Merged = MergeBlockIntoPredecessor(VectorLatchBB, nullptr, State->LI);
|
|
|
|
(void)Merged;
|
|
|
|
assert(Merged && "Could not merge last basic block with latch.");
|
|
|
|
VectorLatchBB = LastBB;
|
|
|
|
|
[VPlan] Implement initial vector code generation support for simple outer loops.
Summary:
[VPlan] Implement vector code generation support for simple outer loops.
Context: Patch Series #1 for outer loop vectorization support in LV using VPlan. (RFC: http://lists.llvm.org/pipermail/llvm-dev/2017-December/119523.html).
This patch introduces vector code generation support for simple outer loops that are currently supported in the VPlanNativePath. Changes here essentially do the following:
- force vector code generation using explicit vectorize_width
- add conservative early returns in cost model and other places for VPlanNativePath
- add code for setting up outer loop inductions
- support for widening non-induction PHIs that can result from inner loops and uniform conditional branches
- support for generating uniform inner branches
We plan to add a handful C outer loop executable tests once the initial code generation support is committed. This patch is expected to be NFC for the inner loop vectorizer path. Since we are moving in the direction of supporting outer loop vectorization in LV, it may also be time to rename classes such as InnerLoopVectorizer.
Reviewers: fhahn, rengolin, hsaito, dcaballe, mkuper, hfinkel, Ayal
Reviewed By: fhahn, hsaito
Subscribers: dmgreen, bollu, tschuett, rkruppe, rogfer01, llvm-commits
Differential Revision: https://reviews.llvm.org/D50820
llvm-svn: 342197
2018-09-14 08:36:00 +08:00
|
|
|
// We do not attempt to preserve DT for outer loop vectorization currently.
|
|
|
|
if (!EnableVPlanNativePath)
|
2019-12-09 16:11:30 +08:00
|
|
|
updateDominatorTree(State->DT, VectorPreHeaderBB, VectorLatchBB,
|
|
|
|
L->getExitBlock());
|
2017-08-27 20:55:46 +08:00
|
|
|
}
|
|
|
|
|
2019-12-03 19:58:31 +08:00
|
|
|
#if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
|
|
|
|
LLVM_DUMP_METHOD
|
|
|
|
void VPlan::dump() const { dbgs() << *this << '\n'; }
|
|
|
|
#endif
|
|
|
|
|
2017-08-27 20:55:46 +08:00
|
|
|
void VPlan::updateDominatorTree(DominatorTree *DT, BasicBlock *LoopPreHeaderBB,
|
2019-12-09 16:11:30 +08:00
|
|
|
BasicBlock *LoopLatchBB,
|
|
|
|
BasicBlock *LoopExitBB) {
|
2017-08-27 20:55:46 +08:00
|
|
|
BasicBlock *LoopHeaderBB = LoopPreHeaderBB->getSingleSuccessor();
|
|
|
|
assert(LoopHeaderBB && "Loop preheader does not have a single successor.");
|
|
|
|
// The vector body may be more than a single basic-block by this point.
|
|
|
|
// Update the dominator tree information inside the vector body by propagating
|
|
|
|
// it from header to latch, expecting only triangular control-flow, if any.
|
|
|
|
BasicBlock *PostDomSucc = nullptr;
|
|
|
|
for (auto *BB = LoopHeaderBB; BB != LoopLatchBB; BB = PostDomSucc) {
|
|
|
|
// Get the list of successors of this block.
|
|
|
|
std::vector<BasicBlock *> Succs(succ_begin(BB), succ_end(BB));
|
|
|
|
assert(Succs.size() <= 2 &&
|
|
|
|
"Basic block in vector loop has more than 2 successors.");
|
|
|
|
PostDomSucc = Succs[0];
|
|
|
|
if (Succs.size() == 1) {
|
|
|
|
assert(PostDomSucc->getSinglePredecessor() &&
|
|
|
|
"PostDom successor has more than one predecessor.");
|
|
|
|
DT->addNewBlock(PostDomSucc, BB);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
BasicBlock *InterimSucc = Succs[1];
|
|
|
|
if (PostDomSucc->getSingleSuccessor() == InterimSucc) {
|
|
|
|
PostDomSucc = Succs[1];
|
|
|
|
InterimSucc = Succs[0];
|
|
|
|
}
|
|
|
|
assert(InterimSucc->getSingleSuccessor() == PostDomSucc &&
|
|
|
|
"One successor of a basic block does not lead to the other.");
|
|
|
|
assert(InterimSucc->getSinglePredecessor() &&
|
|
|
|
"Interim successor has more than one predecessor.");
|
2018-11-20 03:54:27 +08:00
|
|
|
assert(PostDomSucc->hasNPredecessors(2) &&
|
2017-08-27 20:55:46 +08:00
|
|
|
"PostDom successor has more than two predecessors.");
|
|
|
|
DT->addNewBlock(InterimSucc, BB);
|
|
|
|
DT->addNewBlock(PostDomSucc, BB);
|
|
|
|
}
|
2019-12-09 16:11:30 +08:00
|
|
|
// Latch block is a new dominator for the loop exit.
|
|
|
|
DT->changeImmediateDominator(LoopExitBB, LoopLatchBB);
|
|
|
|
assert(DT->verify(DominatorTree::VerificationLevel::Fast));
|
2017-08-27 20:55:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const Twine VPlanPrinter::getUID(const VPBlockBase *Block) {
|
|
|
|
return (isa<VPRegionBlock>(Block) ? "cluster_N" : "N") +
|
|
|
|
Twine(getOrCreateBID(Block));
|
|
|
|
}
|
|
|
|
|
|
|
|
const Twine VPlanPrinter::getOrCreateName(const VPBlockBase *Block) {
|
|
|
|
const std::string &Name = Block->getName();
|
|
|
|
if (!Name.empty())
|
|
|
|
return Name;
|
|
|
|
return "VPB" + Twine(getOrCreateBID(Block));
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPlanPrinter::dump() {
|
|
|
|
Depth = 1;
|
|
|
|
bumpIndent(0);
|
|
|
|
OS << "digraph VPlan {\n";
|
|
|
|
OS << "graph [labelloc=t, fontsize=30; label=\"Vectorization Plan";
|
|
|
|
if (!Plan.getName().empty())
|
|
|
|
OS << "\\n" << DOT::EscapeString(Plan.getName());
|
2018-10-18 23:03:15 +08:00
|
|
|
if (!Plan.Value2VPValue.empty() || Plan.BackedgeTakenCount) {
|
2017-11-20 20:01:47 +08:00
|
|
|
OS << ", where:";
|
2018-10-18 23:03:15 +08:00
|
|
|
if (Plan.BackedgeTakenCount)
|
2019-12-03 19:58:31 +08:00
|
|
|
OS << "\\n" << *Plan.BackedgeTakenCount << " := BackedgeTakenCount";
|
2017-11-20 20:01:47 +08:00
|
|
|
for (auto Entry : Plan.Value2VPValue) {
|
|
|
|
OS << "\\n" << *Entry.second;
|
|
|
|
OS << DOT::EscapeString(" := ");
|
|
|
|
Entry.first->printAsOperand(OS, false);
|
|
|
|
}
|
|
|
|
}
|
2017-08-27 20:55:46 +08:00
|
|
|
OS << "\"]\n";
|
|
|
|
OS << "node [shape=rect, fontname=Courier, fontsize=30]\n";
|
|
|
|
OS << "edge [fontname=Courier, fontsize=30]\n";
|
|
|
|
OS << "compound=true\n";
|
|
|
|
|
2019-12-03 19:58:31 +08:00
|
|
|
for (const VPBlockBase *Block : depth_first(Plan.getEntry()))
|
2017-08-27 20:55:46 +08:00
|
|
|
dumpBlock(Block);
|
|
|
|
|
|
|
|
OS << "}\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPlanPrinter::dumpBlock(const VPBlockBase *Block) {
|
|
|
|
if (const VPBasicBlock *BasicBlock = dyn_cast<VPBasicBlock>(Block))
|
|
|
|
dumpBasicBlock(BasicBlock);
|
|
|
|
else if (const VPRegionBlock *Region = dyn_cast<VPRegionBlock>(Block))
|
|
|
|
dumpRegion(Region);
|
|
|
|
else
|
|
|
|
llvm_unreachable("Unsupported kind of VPBlock.");
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPlanPrinter::drawEdge(const VPBlockBase *From, const VPBlockBase *To,
|
|
|
|
bool Hidden, const Twine &Label) {
|
|
|
|
// Due to "dot" we print an edge between two regions as an edge between the
|
|
|
|
// exit basic block and the entry basic of the respective regions.
|
|
|
|
const VPBlockBase *Tail = From->getExitBasicBlock();
|
|
|
|
const VPBlockBase *Head = To->getEntryBasicBlock();
|
|
|
|
OS << Indent << getUID(Tail) << " -> " << getUID(Head);
|
|
|
|
OS << " [ label=\"" << Label << '\"';
|
|
|
|
if (Tail != From)
|
|
|
|
OS << " ltail=" << getUID(From);
|
|
|
|
if (Head != To)
|
|
|
|
OS << " lhead=" << getUID(To);
|
|
|
|
if (Hidden)
|
|
|
|
OS << "; splines=none";
|
|
|
|
OS << "]\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPlanPrinter::dumpEdges(const VPBlockBase *Block) {
|
|
|
|
auto &Successors = Block->getSuccessors();
|
|
|
|
if (Successors.size() == 1)
|
|
|
|
drawEdge(Block, Successors.front(), false, "");
|
|
|
|
else if (Successors.size() == 2) {
|
|
|
|
drawEdge(Block, Successors.front(), false, "T");
|
|
|
|
drawEdge(Block, Successors.back(), false, "F");
|
|
|
|
} else {
|
|
|
|
unsigned SuccessorNumber = 0;
|
|
|
|
for (auto *Successor : Successors)
|
|
|
|
drawEdge(Block, Successor, false, Twine(SuccessorNumber++));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPlanPrinter::dumpBasicBlock(const VPBasicBlock *BasicBlock) {
|
|
|
|
OS << Indent << getUID(BasicBlock) << " [label =\n";
|
|
|
|
bumpIndent(1);
|
|
|
|
OS << Indent << "\"" << DOT::EscapeString(BasicBlock->getName()) << ":\\n\"";
|
|
|
|
bumpIndent(1);
|
2019-01-24 06:43:12 +08:00
|
|
|
|
|
|
|
// Dump the block predicate.
|
|
|
|
const VPValue *Pred = BasicBlock->getPredicate();
|
|
|
|
if (Pred) {
|
|
|
|
OS << " +\n" << Indent << " \"BlockPredicate: ";
|
|
|
|
if (const VPInstruction *PredI = dyn_cast<VPInstruction>(Pred)) {
|
|
|
|
PredI->printAsOperand(OS);
|
|
|
|
OS << " (" << DOT::EscapeString(PredI->getParent()->getName())
|
|
|
|
<< ")\\l\"";
|
|
|
|
} else
|
|
|
|
Pred->printAsOperand(OS);
|
|
|
|
}
|
|
|
|
|
2017-08-27 20:55:46 +08:00
|
|
|
for (const VPRecipeBase &Recipe : *BasicBlock)
|
|
|
|
Recipe.print(OS, Indent);
|
2018-07-09 23:57:09 +08:00
|
|
|
|
|
|
|
// Dump the condition bit.
|
|
|
|
const VPValue *CBV = BasicBlock->getCondBit();
|
|
|
|
if (CBV) {
|
|
|
|
OS << " +\n" << Indent << " \"CondBit: ";
|
|
|
|
if (const VPInstruction *CBI = dyn_cast<VPInstruction>(CBV)) {
|
|
|
|
CBI->printAsOperand(OS);
|
|
|
|
OS << " (" << DOT::EscapeString(CBI->getParent()->getName()) << ")\\l\"";
|
2018-10-11 01:55:21 +08:00
|
|
|
} else {
|
2018-07-09 23:57:09 +08:00
|
|
|
CBV->printAsOperand(OS);
|
2018-10-11 01:55:21 +08:00
|
|
|
OS << "\"";
|
|
|
|
}
|
2018-07-09 23:57:09 +08:00
|
|
|
}
|
|
|
|
|
2017-08-27 20:55:46 +08:00
|
|
|
bumpIndent(-2);
|
|
|
|
OS << "\n" << Indent << "]\n";
|
|
|
|
dumpEdges(BasicBlock);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPlanPrinter::dumpRegion(const VPRegionBlock *Region) {
|
|
|
|
OS << Indent << "subgraph " << getUID(Region) << " {\n";
|
|
|
|
bumpIndent(1);
|
|
|
|
OS << Indent << "fontname=Courier\n"
|
|
|
|
<< Indent << "label=\""
|
|
|
|
<< DOT::EscapeString(Region->isReplicator() ? "<xVFxUF> " : "<x1> ")
|
|
|
|
<< DOT::EscapeString(Region->getName()) << "\"\n";
|
|
|
|
// Dump the blocks of the region.
|
|
|
|
assert(Region->getEntry() && "Region contains no inner blocks.");
|
|
|
|
for (const VPBlockBase *Block : depth_first(Region->getEntry()))
|
|
|
|
dumpBlock(Block);
|
|
|
|
bumpIndent(-1);
|
|
|
|
OS << Indent << "}\n";
|
|
|
|
dumpEdges(Region);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPlanPrinter::printAsIngredient(raw_ostream &O, Value *V) {
|
|
|
|
std::string IngredientString;
|
|
|
|
raw_string_ostream RSO(IngredientString);
|
|
|
|
if (auto *Inst = dyn_cast<Instruction>(V)) {
|
|
|
|
if (!Inst->getType()->isVoidTy()) {
|
|
|
|
Inst->printAsOperand(RSO, false);
|
|
|
|
RSO << " = ";
|
|
|
|
}
|
|
|
|
RSO << Inst->getOpcodeName() << " ";
|
|
|
|
unsigned E = Inst->getNumOperands();
|
|
|
|
if (E > 0) {
|
|
|
|
Inst->getOperand(0)->printAsOperand(RSO, false);
|
|
|
|
for (unsigned I = 1; I < E; ++I)
|
|
|
|
Inst->getOperand(I)->printAsOperand(RSO << ", ", false);
|
|
|
|
}
|
|
|
|
} else // !Inst
|
|
|
|
V->printAsOperand(RSO, false);
|
|
|
|
RSO.flush();
|
|
|
|
O << DOT::EscapeString(IngredientString);
|
|
|
|
}
|
2017-12-16 09:12:50 +08:00
|
|
|
|
|
|
|
void VPWidenRecipe::print(raw_ostream &O, const Twine &Indent) const {
|
|
|
|
O << " +\n" << Indent << "\"WIDEN\\l\"";
|
|
|
|
for (auto &Instr : make_range(Begin, End))
|
|
|
|
O << " +\n" << Indent << "\" " << VPlanIngredient(&Instr) << "\\l\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPWidenIntOrFpInductionRecipe::print(raw_ostream &O,
|
|
|
|
const Twine &Indent) const {
|
|
|
|
O << " +\n" << Indent << "\"WIDEN-INDUCTION";
|
|
|
|
if (Trunc) {
|
|
|
|
O << "\\l\"";
|
|
|
|
O << " +\n" << Indent << "\" " << VPlanIngredient(IV) << "\\l\"";
|
|
|
|
O << " +\n" << Indent << "\" " << VPlanIngredient(Trunc) << "\\l\"";
|
|
|
|
} else
|
|
|
|
O << " " << VPlanIngredient(IV) << "\\l\"";
|
|
|
|
}
|
|
|
|
|
2019-10-17 02:39:53 +08:00
|
|
|
void VPWidenGEPRecipe::print(raw_ostream &O, const Twine &Indent) const {
|
|
|
|
O << " +\n" << Indent << "\"WIDEN-GEP ";
|
|
|
|
O << (IsPtrLoopInvariant ? "Inv" : "Var");
|
|
|
|
size_t IndicesNumber = IsIndexLoopInvariant.size();
|
|
|
|
for (size_t I = 0; I < IndicesNumber; ++I)
|
|
|
|
O << "[" << (IsIndexLoopInvariant[I] ? "Inv" : "Var") << "]";
|
|
|
|
O << "\\l\"";
|
|
|
|
O << " +\n" << Indent << "\" " << VPlanIngredient(GEP) << "\\l\"";
|
|
|
|
}
|
|
|
|
|
2017-12-16 09:12:50 +08:00
|
|
|
void VPWidenPHIRecipe::print(raw_ostream &O, const Twine &Indent) const {
|
|
|
|
O << " +\n" << Indent << "\"WIDEN-PHI " << VPlanIngredient(Phi) << "\\l\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPBlendRecipe::print(raw_ostream &O, const Twine &Indent) const {
|
|
|
|
O << " +\n" << Indent << "\"BLEND ";
|
|
|
|
Phi->printAsOperand(O, false);
|
|
|
|
O << " =";
|
|
|
|
if (!User) {
|
|
|
|
// Not a User of any mask: not really blending, this is a
|
|
|
|
// single-predecessor phi.
|
|
|
|
O << " ";
|
|
|
|
Phi->getIncomingValue(0)->printAsOperand(O, false);
|
|
|
|
} else {
|
|
|
|
for (unsigned I = 0, E = User->getNumOperands(); I < E; ++I) {
|
|
|
|
O << " ";
|
|
|
|
Phi->getIncomingValue(I)->printAsOperand(O, false);
|
|
|
|
O << "/";
|
|
|
|
User->getOperand(I)->printAsOperand(O);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
O << "\\l\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPReplicateRecipe::print(raw_ostream &O, const Twine &Indent) const {
|
|
|
|
O << " +\n"
|
|
|
|
<< Indent << "\"" << (IsUniform ? "CLONE " : "REPLICATE ")
|
|
|
|
<< VPlanIngredient(Ingredient);
|
|
|
|
if (AlsoPack)
|
|
|
|
O << " (S->V)";
|
|
|
|
O << "\\l\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPPredInstPHIRecipe::print(raw_ostream &O, const Twine &Indent) const {
|
|
|
|
O << " +\n"
|
|
|
|
<< Indent << "\"PHI-PREDICATED-INSTRUCTION " << VPlanIngredient(PredInst)
|
|
|
|
<< "\\l\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPWidenMemoryInstructionRecipe::print(raw_ostream &O,
|
|
|
|
const Twine &Indent) const {
|
|
|
|
O << " +\n" << Indent << "\"WIDEN " << VPlanIngredient(&Instr);
|
2019-11-28 19:54:38 +08:00
|
|
|
O << ", ";
|
|
|
|
getAddr()->printAsOperand(O);
|
2019-12-29 01:59:31 +08:00
|
|
|
VPValue *Mask = getMask();
|
|
|
|
if (Mask) {
|
2017-12-16 09:12:50 +08:00
|
|
|
O << ", ";
|
2019-12-29 01:59:31 +08:00
|
|
|
Mask->printAsOperand(O);
|
2017-12-16 09:12:50 +08:00
|
|
|
}
|
|
|
|
O << "\\l\"";
|
|
|
|
}
|
2018-07-31 05:33:31 +08:00
|
|
|
|
|
|
|
template void DomTreeBuilder::Calculate<VPDominatorTree>(VPDominatorTree &DT);
|
2018-11-13 23:58:18 +08:00
|
|
|
|
[VPlan, SLP] Add simple SLP analysis on top of VPlan.
This patch adds an initial implementation of the look-ahead SLP tree
construction described in 'Look-Ahead SLP: Auto-vectorization in the Presence
of Commutative Operations, CGO 2018 by Vasileios Porpodas, Rodrigo C. O. Rocha,
Luís F. W. Góes'.
It returns an SLP tree represented as VPInstructions, with combined
instructions represented as a single, wider VPInstruction.
This initial version does not support instructions with multiple
different users (either inside or outside the SLP tree) or
non-instruction operands; it won't generate any shuffles or
insertelement instructions.
It also just adds the analysis that builds an SLP tree rooted in a set
of stores. It does not include any cost modeling or memory legality
checks. The plan is to integrate it with VPlan based cost modeling, once
available and to only apply it to operations that can be widened.
A follow-up patch will add a support for replacing instructions in a
VPlan with their SLP counter parts.
Reviewers: Ayal, mssimpso, rengolin, mkuper, hfinkel, hsaito, dcaballe, vporpo, RKSimon, ABataev
Reviewed By: rengolin
Differential Revision: https://reviews.llvm.org/D4949
llvm-svn: 346857
2018-11-14 21:11:49 +08:00
|
|
|
void VPValue::replaceAllUsesWith(VPValue *New) {
|
|
|
|
for (VPUser *User : users())
|
|
|
|
for (unsigned I = 0, E = User->getNumOperands(); I < E; ++I)
|
|
|
|
if (User->getOperand(I) == this)
|
|
|
|
User->setOperand(I, New);
|
|
|
|
}
|
|
|
|
|
2018-11-13 23:58:18 +08:00
|
|
|
void VPInterleavedAccessInfo::visitRegion(VPRegionBlock *Region,
|
|
|
|
Old2NewTy &Old2New,
|
|
|
|
InterleavedAccessInfo &IAI) {
|
|
|
|
ReversePostOrderTraversal<VPBlockBase *> RPOT(Region->getEntry());
|
|
|
|
for (VPBlockBase *Base : RPOT) {
|
|
|
|
visitBlock(Base, Old2New, IAI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPInterleavedAccessInfo::visitBlock(VPBlockBase *Block, Old2NewTy &Old2New,
|
|
|
|
InterleavedAccessInfo &IAI) {
|
|
|
|
if (VPBasicBlock *VPBB = dyn_cast<VPBasicBlock>(Block)) {
|
|
|
|
for (VPRecipeBase &VPI : *VPBB) {
|
|
|
|
assert(isa<VPInstruction>(&VPI) && "Can only handle VPInstructions");
|
|
|
|
auto *VPInst = cast<VPInstruction>(&VPI);
|
|
|
|
auto *Inst = cast<Instruction>(VPInst->getUnderlyingValue());
|
|
|
|
auto *IG = IAI.getInterleaveGroup(Inst);
|
|
|
|
if (!IG)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
auto NewIGIter = Old2New.find(IG);
|
|
|
|
if (NewIGIter == Old2New.end())
|
|
|
|
Old2New[IG] = new InterleaveGroup<VPInstruction>(
|
2019-10-10 20:35:04 +08:00
|
|
|
IG->getFactor(), IG->isReverse(), Align(IG->getAlignment()));
|
2018-11-13 23:58:18 +08:00
|
|
|
|
|
|
|
if (Inst == IG->getInsertPos())
|
|
|
|
Old2New[IG]->setInsertPos(VPInst);
|
|
|
|
|
|
|
|
InterleaveGroupMap[VPInst] = Old2New[IG];
|
|
|
|
InterleaveGroupMap[VPInst]->insertMember(
|
|
|
|
VPInst, IG->getIndex(Inst),
|
2019-10-10 20:35:04 +08:00
|
|
|
Align(IG->isReverse() ? (-1) * int(IG->getFactor())
|
|
|
|
: IG->getFactor()));
|
2018-11-13 23:58:18 +08:00
|
|
|
}
|
|
|
|
} else if (VPRegionBlock *Region = dyn_cast<VPRegionBlock>(Block))
|
|
|
|
visitRegion(Region, Old2New, IAI);
|
|
|
|
else
|
|
|
|
llvm_unreachable("Unsupported kind of VPBlock.");
|
|
|
|
}
|
|
|
|
|
|
|
|
VPInterleavedAccessInfo::VPInterleavedAccessInfo(VPlan &Plan,
|
|
|
|
InterleavedAccessInfo &IAI) {
|
|
|
|
Old2NewTy Old2New;
|
|
|
|
visitRegion(cast<VPRegionBlock>(Plan.getEntry()), Old2New, IAI);
|
|
|
|
}
|