2011-12-09 14:19:40 +08:00
|
|
|
//===-- UnrollLoopRuntime.cpp - Runtime Loop unrolling utilities ----------===//
|
|
|
|
//
|
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
|
2011-12-09 14:19:40 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements some loop unrolling utilities for loops with run-time
|
|
|
|
// trip counts. See LoopUnroll.cpp for unrolling loops with compile-time
|
|
|
|
// trip counts.
|
|
|
|
//
|
2011-12-19 05:52:30 +08:00
|
|
|
// The functions in this file are used to generate extra code when the
|
2011-12-09 14:19:40 +08:00
|
|
|
// run-time trip count modulo the unroll factor is not 0. When this is the
|
|
|
|
// case, we need to generate code to execute these 'left over' iterations.
|
|
|
|
//
|
2011-12-19 05:52:30 +08:00
|
|
|
// The current strategy generates an if-then-else sequence prior to the
|
2016-04-05 20:19:35 +08:00
|
|
|
// unrolled loop to execute the 'left over' iterations before or after the
|
|
|
|
// unrolled loop.
|
2011-12-09 14:19:40 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2018-06-12 19:16:56 +08:00
|
|
|
#include "llvm/ADT/SmallPtrSet.h"
|
2018-03-29 01:44:36 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
2015-01-18 17:21:15 +08:00
|
|
|
#include "llvm/Analysis/AliasAnalysis.h"
|
2011-12-09 14:19:40 +08:00
|
|
|
#include "llvm/Analysis/LoopIterator.h"
|
|
|
|
#include "llvm/Analysis/ScalarEvolution.h"
|
2020-01-05 02:44:38 +08:00
|
|
|
#include "llvm/Analysis/ScalarEvolutionExpander.h"
|
2013-01-02 19:36:10 +08:00
|
|
|
#include "llvm/IR/BasicBlock.h"
|
2015-01-18 10:39:37 +08:00
|
|
|
#include "llvm/IR/Dominators.h"
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
#include "llvm/IR/Metadata.h"
|
2015-03-10 10:37:25 +08:00
|
|
|
#include "llvm/IR/Module.h"
|
2019-11-15 07:15:48 +08:00
|
|
|
#include "llvm/Support/CommandLine.h"
|
2011-12-09 14:19:40 +08:00
|
|
|
#include "llvm/Support/Debug.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2018-03-29 01:44:36 +08:00
|
|
|
#include "llvm/Transforms/Utils.h"
|
2011-12-09 14:19:40 +08:00
|
|
|
#include "llvm/Transforms/Utils/BasicBlockUtils.h"
|
|
|
|
#include "llvm/Transforms/Utils/Cloning.h"
|
2017-07-01 01:57:07 +08:00
|
|
|
#include "llvm/Transforms/Utils/LoopUtils.h"
|
2017-06-06 19:49:48 +08:00
|
|
|
#include "llvm/Transforms/Utils/UnrollLoop.h"
|
2011-12-09 14:19:40 +08:00
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
2014-04-22 10:55:47 +08:00
|
|
|
#define DEBUG_TYPE "loop-unroll"
|
|
|
|
|
2011-12-19 05:52:30 +08:00
|
|
|
STATISTIC(NumRuntimeUnrolled,
|
2011-12-09 14:19:40 +08:00
|
|
|
"Number of loops unrolled with run-time trip counts");
|
2017-07-01 01:57:07 +08:00
|
|
|
static cl::opt<bool> UnrollRuntimeMultiExit(
|
|
|
|
"unroll-runtime-multi-exit", cl::init(false), cl::Hidden,
|
|
|
|
cl::desc("Allow runtime unrolling for loops with multiple exits, when "
|
|
|
|
"epilog is generated"));
|
2011-12-09 14:19:40 +08:00
|
|
|
|
|
|
|
/// Connect the unrolling prolog code to the original loop.
|
|
|
|
/// The unrolling prolog code contains code to execute the
|
|
|
|
/// 'extra' iterations if the run-time trip count modulo the
|
|
|
|
/// unroll count is non-zero.
|
|
|
|
///
|
|
|
|
/// This function performs the following:
|
|
|
|
/// - Create PHI nodes at prolog end block to combine values
|
|
|
|
/// that exit the prolog code and jump around the prolog.
|
|
|
|
/// - Add a PHI operand to a PHI node at the loop exit block
|
|
|
|
/// for values that exit the prolog and go around the loop.
|
|
|
|
/// - Branch around the original loop if the trip count is less
|
|
|
|
/// than the unroll factor.
|
|
|
|
///
|
2015-02-19 03:32:25 +08:00
|
|
|
static void ConnectProlog(Loop *L, Value *BECount, unsigned Count,
|
2017-07-08 02:05:28 +08:00
|
|
|
BasicBlock *PrologExit,
|
|
|
|
BasicBlock *OriginalLoopLatchExit,
|
|
|
|
BasicBlock *PreHeader, BasicBlock *NewPreHeader,
|
|
|
|
ValueToValueMapTy &VMap, DominatorTree *DT,
|
|
|
|
LoopInfo *LI, bool PreserveLCSSA) {
|
2018-12-22 03:45:05 +08:00
|
|
|
// Loop structure should be the following:
|
|
|
|
// Preheader
|
2018-12-29 02:52:16 +08:00
|
|
|
// PrologHeader
|
2018-12-22 03:45:05 +08:00
|
|
|
// ...
|
|
|
|
// PrologLatch
|
|
|
|
// PrologExit
|
|
|
|
// NewPreheader
|
|
|
|
// Header
|
|
|
|
// ...
|
|
|
|
// Latch
|
|
|
|
// LatchExit
|
2011-12-09 14:19:40 +08:00
|
|
|
BasicBlock *Latch = L->getLoopLatch();
|
2014-04-28 12:05:08 +08:00
|
|
|
assert(Latch && "Loop must have a latch");
|
2016-04-05 20:19:35 +08:00
|
|
|
BasicBlock *PrologLatch = cast<BasicBlock>(VMap[Latch]);
|
2011-12-09 14:19:40 +08:00
|
|
|
|
|
|
|
// Create a PHI node for each outgoing value from the original loop
|
|
|
|
// (which means it is an outgoing value from the prolog code too).
|
|
|
|
// The new PHI node is inserted in the prolog end basic block.
|
2016-04-05 20:19:35 +08:00
|
|
|
// The new PHI node value is added as an operand of a PHI node in either
|
2011-12-09 14:19:40 +08:00
|
|
|
// the loop header or the loop exit block.
|
2016-04-05 20:19:35 +08:00
|
|
|
for (BasicBlock *Succ : successors(Latch)) {
|
2017-12-30 23:27:33 +08:00
|
|
|
for (PHINode &PN : Succ->phis()) {
|
2011-12-09 14:19:40 +08:00
|
|
|
// Add a new PHI node to the prolog end block and add the
|
|
|
|
// appropriate incoming values.
|
2018-12-22 03:45:05 +08:00
|
|
|
// TODO: This code assumes that the PrologExit (or the LatchExit block for
|
|
|
|
// prolog loop) contains only one predecessor from the loop, i.e. the
|
|
|
|
// PrologLatch. When supporting multiple-exiting block loops, we can have
|
|
|
|
// two or more blocks that have the LatchExit as the target in the
|
|
|
|
// original loop.
|
2017-12-30 23:27:33 +08:00
|
|
|
PHINode *NewPN = PHINode::Create(PN.getType(), 2, PN.getName() + ".unr",
|
2016-04-05 20:19:35 +08:00
|
|
|
PrologExit->getFirstNonPHI());
|
2011-12-09 14:19:40 +08:00
|
|
|
// Adding a value to the new PHI node from the original loop preheader.
|
|
|
|
// This is the value that skips all the prolog code.
|
2017-12-30 23:27:33 +08:00
|
|
|
if (L->contains(&PN)) {
|
2018-12-22 03:45:05 +08:00
|
|
|
// Succ is loop header.
|
2017-12-30 23:27:33 +08:00
|
|
|
NewPN->addIncoming(PN.getIncomingValueForBlock(NewPreHeader),
|
2016-04-05 20:19:35 +08:00
|
|
|
PreHeader);
|
2011-12-09 14:19:40 +08:00
|
|
|
} else {
|
2018-12-29 02:52:16 +08:00
|
|
|
// Succ is LatchExit.
|
2017-12-30 23:27:33 +08:00
|
|
|
NewPN->addIncoming(UndefValue::get(PN.getType()), PreHeader);
|
2011-12-09 14:19:40 +08:00
|
|
|
}
|
2011-12-19 05:52:30 +08:00
|
|
|
|
2017-12-30 23:27:33 +08:00
|
|
|
Value *V = PN.getIncomingValueForBlock(Latch);
|
2011-12-09 14:19:40 +08:00
|
|
|
if (Instruction *I = dyn_cast<Instruction>(V)) {
|
|
|
|
if (L->contains(I)) {
|
2016-04-18 03:26:49 +08:00
|
|
|
V = VMap.lookup(I);
|
2011-12-09 14:19:40 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// Adding a value to the new PHI node from the last prolog block
|
|
|
|
// that was created.
|
2016-04-05 20:19:35 +08:00
|
|
|
NewPN->addIncoming(V, PrologLatch);
|
2011-12-09 14:19:40 +08:00
|
|
|
|
|
|
|
// Update the existing PHI node operand with the value from the
|
|
|
|
// new PHI node. How this is done depends on if the existing
|
|
|
|
// PHI node is in the original loop block, or the exit block.
|
PHINode: introduce setIncomingValueForBlock() function, and use it.
Summary:
There is PHINode::getBasicBlockIndex() and PHINode::setIncomingValue()
but no function to replace incoming value for a specified BasicBlock*
predecessor.
Clearly, there are a lot of places that could use that functionality.
Reviewer: craig.topper, lebedev.ri, Meinersbur, kbarton, fhahn
Reviewed By: Meinersbur, fhahn
Subscribers: fhahn, hiraditya, zzheng, jsji, llvm-commits
Tag: LLVM
Differential Revision: https://reviews.llvm.org/D63338
llvm-svn: 363566
2019-06-17 22:38:56 +08:00
|
|
|
if (L->contains(&PN))
|
|
|
|
PN.setIncomingValueForBlock(NewPreHeader, NewPN);
|
|
|
|
else
|
2017-12-30 23:27:33 +08:00
|
|
|
PN.addIncoming(NewPN, PrologExit);
|
2011-12-09 14:19:40 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-03 03:19:31 +08:00
|
|
|
// Make sure that created prolog loop is in simplified form
|
|
|
|
SmallVector<BasicBlock *, 4> PrologExitPreds;
|
|
|
|
Loop *PrologLoop = LI->getLoopFor(PrologLatch);
|
|
|
|
if (PrologLoop) {
|
|
|
|
for (BasicBlock *PredBB : predecessors(PrologExit))
|
|
|
|
if (PrologLoop->contains(PredBB))
|
|
|
|
PrologExitPreds.push_back(PredBB);
|
|
|
|
|
|
|
|
SplitBlockPredecessors(PrologExit, PrologExitPreds, ".unr-lcssa", DT, LI,
|
2018-08-22 07:32:03 +08:00
|
|
|
nullptr, PreserveLCSSA);
|
2016-08-03 03:19:31 +08:00
|
|
|
}
|
|
|
|
|
2016-02-09 03:27:33 +08:00
|
|
|
// Create a branch around the original loop, which is taken if there are no
|
2015-02-19 03:32:25 +08:00
|
|
|
// iterations remaining to be executed after running the prologue.
|
2016-04-05 20:19:35 +08:00
|
|
|
Instruction *InsertPt = PrologExit->getTerminator();
|
2015-06-12 02:25:44 +08:00
|
|
|
IRBuilder<> B(InsertPt);
|
2015-02-19 03:32:25 +08:00
|
|
|
|
|
|
|
assert(Count != 0 && "nonsensical Count!");
|
|
|
|
|
2016-03-25 22:24:52 +08:00
|
|
|
// If BECount <u (Count - 1) then (BECount + 1) % Count == (BECount + 1)
|
|
|
|
// This means %xtraiter is (BECount + 1) and all of the iterations of this
|
|
|
|
// loop were executed by the prologue. Note that if BECount <u (Count - 1)
|
|
|
|
// then (BECount + 1) cannot unsigned-overflow.
|
2015-06-12 02:25:44 +08:00
|
|
|
Value *BrLoopExit =
|
|
|
|
B.CreateICmpULT(BECount, ConstantInt::get(BECount->getType(), Count - 1));
|
2011-12-09 14:19:40 +08:00
|
|
|
// Split the exit to maintain loop canonicalization guarantees
|
2017-07-08 02:05:28 +08:00
|
|
|
SmallVector<BasicBlock *, 4> Preds(predecessors(OriginalLoopLatchExit));
|
|
|
|
SplitBlockPredecessors(OriginalLoopLatchExit, Preds, ".unr-lcssa", DT, LI,
|
2018-08-22 07:32:03 +08:00
|
|
|
nullptr, PreserveLCSSA);
|
2011-12-09 14:19:40 +08:00
|
|
|
// Add the branch to the exit block (around the unrolled loop)
|
2017-07-08 02:05:28 +08:00
|
|
|
B.CreateCondBr(BrLoopExit, OriginalLoopLatchExit, NewPreHeader);
|
2016-04-05 20:19:35 +08:00
|
|
|
InsertPt->eraseFromParent();
|
2017-01-19 07:26:37 +08:00
|
|
|
if (DT)
|
2017-07-08 02:05:28 +08:00
|
|
|
DT->changeImmediateDominator(OriginalLoopLatchExit, PrologExit);
|
2016-04-05 20:19:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Connect the unrolling epilog code to the original loop.
|
|
|
|
/// The unrolling epilog code contains code to execute the
|
|
|
|
/// 'extra' iterations if the run-time trip count modulo the
|
|
|
|
/// unroll count is non-zero.
|
|
|
|
///
|
|
|
|
/// This function performs the following:
|
|
|
|
/// - Update PHI nodes at the unrolling loop exit and epilog loop exit
|
|
|
|
/// - Create PHI nodes at the unrolling loop exit to combine
|
|
|
|
/// values that exit the unrolling loop code and jump around it.
|
|
|
|
/// - Update PHI operands in the epilog loop by the new PHI nodes
|
|
|
|
/// - Branch around the epilog loop if extra iters (ModVal) is zero.
|
|
|
|
///
|
|
|
|
static void ConnectEpilog(Loop *L, Value *ModVal, BasicBlock *NewExit,
|
|
|
|
BasicBlock *Exit, BasicBlock *PreHeader,
|
|
|
|
BasicBlock *EpilogPreHeader, BasicBlock *NewPreHeader,
|
|
|
|
ValueToValueMapTy &VMap, DominatorTree *DT,
|
|
|
|
LoopInfo *LI, bool PreserveLCSSA) {
|
|
|
|
BasicBlock *Latch = L->getLoopLatch();
|
|
|
|
assert(Latch && "Loop must have a latch");
|
|
|
|
BasicBlock *EpilogLatch = cast<BasicBlock>(VMap[Latch]);
|
|
|
|
|
|
|
|
// Loop structure should be the following:
|
|
|
|
//
|
|
|
|
// PreHeader
|
|
|
|
// NewPreHeader
|
|
|
|
// Header
|
|
|
|
// ...
|
|
|
|
// Latch
|
|
|
|
// NewExit (PN)
|
|
|
|
// EpilogPreHeader
|
|
|
|
// EpilogHeader
|
|
|
|
// ...
|
|
|
|
// EpilogLatch
|
|
|
|
// Exit (EpilogPN)
|
|
|
|
|
|
|
|
// Update PHI nodes at NewExit and Exit.
|
2017-12-30 23:27:33 +08:00
|
|
|
for (PHINode &PN : NewExit->phis()) {
|
2016-04-05 20:19:35 +08:00
|
|
|
// PN should be used in another PHI located in Exit block as
|
|
|
|
// Exit was split by SplitBlockPredecessors into Exit and NewExit
|
|
|
|
// Basicaly it should look like:
|
|
|
|
// NewExit:
|
|
|
|
// PN = PHI [I, Latch]
|
|
|
|
// ...
|
|
|
|
// Exit:
|
|
|
|
// EpilogPN = PHI [PN, EpilogPreHeader]
|
|
|
|
//
|
|
|
|
// There is EpilogPreHeader incoming block instead of NewExit as
|
|
|
|
// NewExit was spilt 1 more time to get EpilogPreHeader.
|
2017-12-30 23:27:33 +08:00
|
|
|
assert(PN.hasOneUse() && "The phi should have 1 use");
|
|
|
|
PHINode *EpilogPN = cast<PHINode>(PN.use_begin()->getUser());
|
2016-04-05 20:19:35 +08:00
|
|
|
assert(EpilogPN->getParent() == Exit && "EpilogPN should be in Exit block");
|
|
|
|
|
|
|
|
// Add incoming PreHeader from branch around the Loop
|
2017-12-30 23:27:33 +08:00
|
|
|
PN.addIncoming(UndefValue::get(PN.getType()), PreHeader);
|
2016-04-05 20:19:35 +08:00
|
|
|
|
2017-12-30 23:27:33 +08:00
|
|
|
Value *V = PN.getIncomingValueForBlock(Latch);
|
2016-04-05 20:19:35 +08:00
|
|
|
Instruction *I = dyn_cast<Instruction>(V);
|
|
|
|
if (I && L->contains(I))
|
|
|
|
// If value comes from an instruction in the loop add VMap value.
|
2016-04-18 03:26:49 +08:00
|
|
|
V = VMap.lookup(I);
|
2016-04-05 20:19:35 +08:00
|
|
|
// For the instruction out of the loop, constant or undefined value
|
|
|
|
// insert value itself.
|
|
|
|
EpilogPN->addIncoming(V, EpilogLatch);
|
|
|
|
|
|
|
|
assert(EpilogPN->getBasicBlockIndex(EpilogPreHeader) >= 0 &&
|
|
|
|
"EpilogPN should have EpilogPreHeader incoming block");
|
|
|
|
// Change EpilogPreHeader incoming block to NewExit.
|
|
|
|
EpilogPN->setIncomingBlock(EpilogPN->getBasicBlockIndex(EpilogPreHeader),
|
|
|
|
NewExit);
|
|
|
|
// Now PHIs should look like:
|
|
|
|
// NewExit:
|
|
|
|
// PN = PHI [I, Latch], [undef, PreHeader]
|
|
|
|
// ...
|
|
|
|
// Exit:
|
|
|
|
// EpilogPN = PHI [PN, NewExit], [VMap[I], EpilogLatch]
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create PHI nodes at NewExit (from the unrolling loop Latch and PreHeader).
|
|
|
|
// Update corresponding PHI nodes in epilog loop.
|
|
|
|
for (BasicBlock *Succ : successors(Latch)) {
|
|
|
|
// Skip this as we already updated phis in exit blocks.
|
|
|
|
if (!L->contains(Succ))
|
|
|
|
continue;
|
2017-12-30 23:27:33 +08:00
|
|
|
for (PHINode &PN : Succ->phis()) {
|
2016-04-05 20:19:35 +08:00
|
|
|
// Add new PHI nodes to the loop exit block and update epilog
|
|
|
|
// PHIs with the new PHI values.
|
2017-12-30 23:27:33 +08:00
|
|
|
PHINode *NewPN = PHINode::Create(PN.getType(), 2, PN.getName() + ".unr",
|
2016-04-05 20:19:35 +08:00
|
|
|
NewExit->getFirstNonPHI());
|
|
|
|
// Adding a value to the new PHI node from the unrolling loop preheader.
|
2017-12-30 23:27:33 +08:00
|
|
|
NewPN->addIncoming(PN.getIncomingValueForBlock(NewPreHeader), PreHeader);
|
2016-04-05 20:19:35 +08:00
|
|
|
// Adding a value to the new PHI node from the unrolling loop latch.
|
2017-12-30 23:27:33 +08:00
|
|
|
NewPN->addIncoming(PN.getIncomingValueForBlock(Latch), Latch);
|
2016-04-05 20:19:35 +08:00
|
|
|
|
|
|
|
// Update the existing PHI node operand with the value from the new PHI
|
|
|
|
// node. Corresponding instruction in epilog loop should be PHI.
|
2017-12-30 23:27:33 +08:00
|
|
|
PHINode *VPN = cast<PHINode>(VMap[&PN]);
|
PHINode: introduce setIncomingValueForBlock() function, and use it.
Summary:
There is PHINode::getBasicBlockIndex() and PHINode::setIncomingValue()
but no function to replace incoming value for a specified BasicBlock*
predecessor.
Clearly, there are a lot of places that could use that functionality.
Reviewer: craig.topper, lebedev.ri, Meinersbur, kbarton, fhahn
Reviewed By: Meinersbur, fhahn
Subscribers: fhahn, hiraditya, zzheng, jsji, llvm-commits
Tag: LLVM
Differential Revision: https://reviews.llvm.org/D63338
llvm-svn: 363566
2019-06-17 22:38:56 +08:00
|
|
|
VPN->setIncomingValueForBlock(EpilogPreHeader, NewPN);
|
2016-04-05 20:19:35 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Instruction *InsertPt = NewExit->getTerminator();
|
|
|
|
IRBuilder<> B(InsertPt);
|
2016-04-27 11:04:54 +08:00
|
|
|
Value *BrLoopExit = B.CreateIsNotNull(ModVal, "lcmp.mod");
|
2016-04-05 20:19:35 +08:00
|
|
|
assert(Exit && "Loop must have a single exit block only");
|
2017-01-19 07:26:37 +08:00
|
|
|
// Split the epilogue exit to maintain loop canonicalization guarantees
|
2016-04-05 20:19:35 +08:00
|
|
|
SmallVector<BasicBlock*, 4> Preds(predecessors(Exit));
|
2018-08-22 07:32:03 +08:00
|
|
|
SplitBlockPredecessors(Exit, Preds, ".epilog-lcssa", DT, LI, nullptr,
|
2016-04-05 20:19:35 +08:00
|
|
|
PreserveLCSSA);
|
|
|
|
// Add the branch to the exit block (around the unrolling loop)
|
|
|
|
B.CreateCondBr(BrLoopExit, EpilogPreHeader, Exit);
|
2011-12-09 14:19:40 +08:00
|
|
|
InsertPt->eraseFromParent();
|
2017-01-19 07:26:37 +08:00
|
|
|
if (DT)
|
|
|
|
DT->changeImmediateDominator(Exit, NewExit);
|
|
|
|
|
|
|
|
// Split the main loop exit to maintain canonicalization guarantees.
|
|
|
|
SmallVector<BasicBlock*, 4> NewExitPreds{Latch};
|
2018-08-22 07:32:03 +08:00
|
|
|
SplitBlockPredecessors(NewExit, NewExitPreds, ".loopexit", DT, LI, nullptr,
|
2017-01-19 07:26:37 +08:00
|
|
|
PreserveLCSSA);
|
2011-12-09 14:19:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a clone of the blocks in a loop and connect them together.
|
2016-04-05 20:19:35 +08:00
|
|
|
/// If CreateRemainderLoop is false, loop structure will not be cloned,
|
|
|
|
/// otherwise a new loop will be created including all cloned blocks, and the
|
|
|
|
/// iterator of it switches to count NewIter down to 0.
|
|
|
|
/// The cloned blocks should be inserted between InsertTop and InsertBot.
|
|
|
|
/// If loop structure is cloned InsertTop should be new preheader, InsertBot
|
|
|
|
/// new loop exit.
|
2017-07-01 01:57:07 +08:00
|
|
|
/// Return the new cloned loop that is created when CreateRemainderLoop is true.
|
|
|
|
static Loop *
|
|
|
|
CloneLoopBlocks(Loop *L, Value *NewIter, const bool CreateRemainderLoop,
|
2017-08-14 17:25:26 +08:00
|
|
|
const bool UseEpilogRemainder, const bool UnrollRemainder,
|
|
|
|
BasicBlock *InsertTop,
|
2017-07-01 01:57:07 +08:00
|
|
|
BasicBlock *InsertBot, BasicBlock *Preheader,
|
|
|
|
std::vector<BasicBlock *> &NewBlocks, LoopBlocksDFS &LoopBlocks,
|
|
|
|
ValueToValueMapTy &VMap, DominatorTree *DT, LoopInfo *LI) {
|
2016-04-05 20:19:35 +08:00
|
|
|
StringRef suffix = UseEpilogRemainder ? "epil" : "prol";
|
2011-12-09 14:19:40 +08:00
|
|
|
BasicBlock *Header = L->getHeader();
|
|
|
|
BasicBlock *Latch = L->getLoopLatch();
|
|
|
|
Function *F = Header->getParent();
|
|
|
|
LoopBlocksDFS::RPOIterator BlockBegin = LoopBlocks.beginRPO();
|
|
|
|
LoopBlocksDFS::RPOIterator BlockEnd = LoopBlocks.endRPO();
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
Loop *ParentLoop = L->getParentLoop();
|
2017-01-11 07:43:35 +08:00
|
|
|
NewLoopsMap NewLoops;
|
2017-01-31 19:13:44 +08:00
|
|
|
NewLoops[ParentLoop] = ParentLoop;
|
|
|
|
if (!CreateRemainderLoop)
|
2017-01-26 09:04:11 +08:00
|
|
|
NewLoops[L] = ParentLoop;
|
|
|
|
|
2011-12-09 14:19:40 +08:00
|
|
|
// For each block in the original loop, create a new copy,
|
|
|
|
// and update the value map with the newly created values.
|
|
|
|
for (LoopBlocksDFS::RPOIterator BB = BlockBegin; BB != BlockEnd; ++BB) {
|
2016-04-05 20:19:35 +08:00
|
|
|
BasicBlock *NewBB = CloneBasicBlock(*BB, VMap, "." + suffix, F);
|
2011-12-09 14:19:40 +08:00
|
|
|
NewBlocks.push_back(NewBB);
|
2017-01-31 19:13:44 +08:00
|
|
|
|
2017-01-26 09:04:11 +08:00
|
|
|
// If we're unrolling the outermost loop, there's no remainder loop,
|
|
|
|
// and this block isn't in a nested loop, then the new block is not
|
|
|
|
// in any loop. Otherwise, add it to loopinfo.
|
|
|
|
if (CreateRemainderLoop || LI->getLoopFor(*BB) != L || ParentLoop)
|
2017-01-11 07:43:35 +08:00
|
|
|
addClonedBlockToLoopInfo(*BB, NewBB, LI, NewLoops);
|
2011-12-09 14:19:40 +08:00
|
|
|
|
|
|
|
VMap[*BB] = NewBB;
|
|
|
|
if (Header == *BB) {
|
|
|
|
// For the first block, add a CFG connection to this newly
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
// created block.
|
2011-12-09 14:19:40 +08:00
|
|
|
InsertTop->getTerminator()->setSuccessor(0, NewBB);
|
|
|
|
}
|
2016-01-28 09:23:18 +08:00
|
|
|
|
2017-01-19 07:26:37 +08:00
|
|
|
if (DT) {
|
|
|
|
if (Header == *BB) {
|
|
|
|
// The header is dominated by the preheader.
|
|
|
|
DT->addNewBlock(NewBB, InsertTop);
|
|
|
|
} else {
|
|
|
|
// Copy information from original loop to unrolled loop.
|
|
|
|
BasicBlock *IDomBB = DT->getNode(*BB)->getIDom()->getBlock();
|
|
|
|
DT->addNewBlock(NewBB, cast<BasicBlock>(VMap[IDomBB]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-09 14:19:40 +08:00
|
|
|
if (Latch == *BB) {
|
2016-04-05 20:19:35 +08:00
|
|
|
// For the last block, if CreateRemainderLoop is false, create a direct
|
|
|
|
// jump to InsertBot. If not, create a loop back to cloned head.
|
2011-12-09 14:19:40 +08:00
|
|
|
VMap.erase((*BB)->getTerminator());
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
BasicBlock *FirstLoopBB = cast<BasicBlock>(VMap[Header]);
|
|
|
|
BranchInst *LatchBR = cast<BranchInst>(NewBB->getTerminator());
|
2015-06-12 02:25:44 +08:00
|
|
|
IRBuilder<> Builder(LatchBR);
|
2016-04-05 20:19:35 +08:00
|
|
|
if (!CreateRemainderLoop) {
|
2015-06-12 02:25:44 +08:00
|
|
|
Builder.CreateBr(InsertBot);
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
} else {
|
2016-04-05 20:19:35 +08:00
|
|
|
PHINode *NewIdx = PHINode::Create(NewIter->getType(), 2,
|
|
|
|
suffix + ".iter",
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
FirstLoopBB->getFirstNonPHI());
|
|
|
|
Value *IdxSub =
|
|
|
|
Builder.CreateSub(NewIdx, ConstantInt::get(NewIdx->getType(), 1),
|
|
|
|
NewIdx->getName() + ".sub");
|
|
|
|
Value *IdxCmp =
|
|
|
|
Builder.CreateIsNotNull(IdxSub, NewIdx->getName() + ".cmp");
|
2015-06-12 02:25:44 +08:00
|
|
|
Builder.CreateCondBr(IdxCmp, FirstLoopBB, InsertBot);
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
NewIdx->addIncoming(NewIter, InsertTop);
|
|
|
|
NewIdx->addIncoming(IdxSub, NewBB);
|
|
|
|
}
|
2015-06-12 02:25:44 +08:00
|
|
|
LatchBR->eraseFromParent();
|
2011-12-09 14:19:40 +08:00
|
|
|
}
|
|
|
|
}
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
|
|
|
|
// Change the incoming values to the ones defined in the preheader or
|
|
|
|
// cloned loop.
|
|
|
|
for (BasicBlock::iterator I = Header->begin(); isa<PHINode>(I); ++I) {
|
2015-10-13 10:39:05 +08:00
|
|
|
PHINode *NewPHI = cast<PHINode>(VMap[&*I]);
|
2016-04-05 20:19:35 +08:00
|
|
|
if (!CreateRemainderLoop) {
|
|
|
|
if (UseEpilogRemainder) {
|
|
|
|
unsigned idx = NewPHI->getBasicBlockIndex(Preheader);
|
|
|
|
NewPHI->setIncomingBlock(idx, InsertTop);
|
|
|
|
NewPHI->removeIncomingValue(Latch, false);
|
|
|
|
} else {
|
|
|
|
VMap[&*I] = NewPHI->getIncomingValueForBlock(Preheader);
|
|
|
|
cast<BasicBlock>(VMap[Header])->getInstList().erase(NewPHI);
|
|
|
|
}
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
} else {
|
|
|
|
unsigned idx = NewPHI->getBasicBlockIndex(Preheader);
|
|
|
|
NewPHI->setIncomingBlock(idx, InsertTop);
|
|
|
|
BasicBlock *NewLatch = cast<BasicBlock>(VMap[Latch]);
|
|
|
|
idx = NewPHI->getBasicBlockIndex(Latch);
|
|
|
|
Value *InVal = NewPHI->getIncomingValue(idx);
|
|
|
|
NewPHI->setIncomingBlock(idx, NewLatch);
|
2016-04-18 03:26:49 +08:00
|
|
|
if (Value *V = VMap.lookup(InVal))
|
|
|
|
NewPHI->setIncomingValue(idx, V);
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
}
|
|
|
|
}
|
2017-01-31 19:13:44 +08:00
|
|
|
if (CreateRemainderLoop) {
|
2019-11-04 03:03:54 +08:00
|
|
|
Loop *NewLoop = NewLoops[L];
|
2017-01-31 19:13:44 +08:00
|
|
|
assert(NewLoop && "L should have been cloned");
|
2019-11-04 03:03:54 +08:00
|
|
|
MDNode *LoopID = NewLoop->getLoopID();
|
2017-09-04 16:12:16 +08:00
|
|
|
|
|
|
|
// Only add loop metadata if the loop is not going to be completely
|
|
|
|
// unrolled.
|
|
|
|
if (UnrollRemainder)
|
|
|
|
return NewLoop;
|
|
|
|
|
[Unroll/UnrollAndJam/Vectorizer/Distribute] Add followup loop attributes.
When multiple loop transformation are defined in a loop's metadata, their order of execution is defined by the order of their respective passes in the pass pipeline. For instance, e.g.
#pragma clang loop unroll_and_jam(enable)
#pragma clang loop distribute(enable)
is the same as
#pragma clang loop distribute(enable)
#pragma clang loop unroll_and_jam(enable)
and will try to loop-distribute before Unroll-And-Jam because the LoopDistribute pass is scheduled after UnrollAndJam pass. UnrollAndJamPass only supports one inner loop, i.e. it will necessarily fail after loop distribution. It is not possible to specify another execution order. Also,t the order of passes in the pipeline is subject to change between versions of LLVM, optimization options and which pass manager is used.
This patch adds 'followup' attributes to various loop transformation passes. These attributes define which attributes the resulting loop of a transformation should have. For instance,
!0 = !{!0, !1, !2}
!1 = !{!"llvm.loop.unroll_and_jam.enable"}
!2 = !{!"llvm.loop.unroll_and_jam.followup_inner", !3}
!3 = !{!"llvm.loop.distribute.enable"}
defines a loop ID (!0) to be unrolled-and-jammed (!1) and then the attribute !3 to be added to the jammed inner loop, which contains the instruction to distribute the inner loop.
Currently, in both pass managers, pass execution is in a fixed order and UnrollAndJamPass will not execute again after LoopDistribute. We hope to fix this in the future by allowing pass managers to run passes until a fixpoint is reached, use Polly to perform these transformations, or add a loop transformation pass which takes the order issue into account.
For mandatory/forced transformations (e.g. by having been declared by #pragma omp simd), the user must be notified when a transformation could not be performed. It is not possible that the responsible pass emits such a warning because the transformation might be 'hidden' in a followup attribute when it is executed, or it is not present in the pipeline at all. For this reason, this patche introduces a WarnMissedTransformations pass, to warn about orphaned transformations.
Since this changes the user-visible diagnostic message when a transformation is applied, two test cases in the clang repository need to be updated.
To ensure that no other transformation is executed before the intended one, the attribute `llvm.loop.disable_nonforced` can be added which should disable transformation heuristics before the intended transformation is applied. E.g. it would be surprising if a loop is distributed before a #pragma unroll_and_jam is applied.
With more supported code transformations (loop fusion, interchange, stripmining, offloading, etc.), transformations can be used as building blocks for more complex transformations (e.g. stripmining+stripmining+interchange -> tiling).
Reviewed By: hfinkel, dmgreen
Differential Revision: https://reviews.llvm.org/D49281
Differential Revision: https://reviews.llvm.org/D55288
llvm-svn: 348944
2018-12-13 01:32:52 +08:00
|
|
|
Optional<MDNode *> NewLoopID = makeFollowupLoopID(
|
|
|
|
LoopID, {LLVMLoopUnrollFollowupAll, LLVMLoopUnrollFollowupRemainder});
|
|
|
|
if (NewLoopID.hasValue()) {
|
|
|
|
NewLoop->setLoopID(NewLoopID.getValue());
|
|
|
|
|
|
|
|
// Do not setLoopAlreadyUnrolled if loop attributes have been defined
|
|
|
|
// explicitly.
|
|
|
|
return NewLoop;
|
|
|
|
}
|
|
|
|
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
// Add unroll disable metadata to disable future unrolling for this loop.
|
2017-10-15 15:31:02 +08:00
|
|
|
NewLoop->setLoopAlreadyUnrolled();
|
2017-07-01 01:57:07 +08:00
|
|
|
return NewLoop;
|
2011-12-09 14:19:40 +08:00
|
|
|
}
|
2017-07-01 01:57:07 +08:00
|
|
|
else
|
|
|
|
return nullptr;
|
2011-12-09 14:19:40 +08:00
|
|
|
}
|
|
|
|
|
2017-07-13 04:55:43 +08:00
|
|
|
/// Returns true if we can safely unroll a multi-exit/exiting loop. OtherExits
|
|
|
|
/// is populated with all the loop exit blocks other than the LatchExit block.
|
2019-07-15 13:51:10 +08:00
|
|
|
static bool canSafelyUnrollMultiExitLoop(Loop *L, BasicBlock *LatchExit,
|
|
|
|
bool PreserveLCSSA,
|
|
|
|
bool UseEpilogRemainder) {
|
2017-07-13 04:55:43 +08:00
|
|
|
|
2017-07-22 00:30:38 +08:00
|
|
|
// We currently have some correctness constrains in unrolling a multi-exit
|
|
|
|
// loop. Check for these below.
|
|
|
|
|
2017-07-13 04:55:43 +08:00
|
|
|
// We rely on LCSSA form being preserved when the exit blocks are transformed.
|
|
|
|
if (!PreserveLCSSA)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// TODO: Support multiple exiting blocks jumping to the `LatchExit` when
|
|
|
|
// UnrollRuntimeMultiExit is true. This will need updating the logic in
|
|
|
|
// connectEpilog/connectProlog.
|
|
|
|
if (!LatchExit->getSinglePredecessor()) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(
|
|
|
|
dbgs() << "Bailout for multi-exit handling when latch exit has >1 "
|
|
|
|
"predecessor.\n");
|
2017-07-13 04:55:43 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// FIXME: We bail out of multi-exit unrolling when epilog loop is generated
|
|
|
|
// and L is an inner loop. This is because in presence of multiple exits, the
|
|
|
|
// outer loop is incorrect: we do not add the EpilogPreheader and exit to the
|
|
|
|
// outer loop. This is automatically handled in the prolog case, so we do not
|
|
|
|
// have that bug in prolog generation.
|
|
|
|
if (UseEpilogRemainder && L->getParentLoop())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// All constraints have been satisfied.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-07-22 00:30:38 +08:00
|
|
|
/// Returns true if we can profitably unroll the multi-exit loop L. Currently,
|
|
|
|
/// we return true only if UnrollRuntimeMultiExit is set to true.
|
|
|
|
static bool canProfitablyUnrollMultiExitLoop(
|
|
|
|
Loop *L, SmallVectorImpl<BasicBlock *> &OtherExits, BasicBlock *LatchExit,
|
|
|
|
bool PreserveLCSSA, bool UseEpilogRemainder) {
|
|
|
|
|
|
|
|
#if !defined(NDEBUG)
|
2019-07-15 13:51:10 +08:00
|
|
|
assert(canSafelyUnrollMultiExitLoop(L, LatchExit, PreserveLCSSA,
|
|
|
|
UseEpilogRemainder) &&
|
2017-07-22 00:30:38 +08:00
|
|
|
"Should be safe to unroll before checking profitability!");
|
|
|
|
#endif
|
2017-09-15 23:56:05 +08:00
|
|
|
|
2017-07-22 00:30:38 +08:00
|
|
|
// Priority goes to UnrollRuntimeMultiExit if it's supplied.
|
2017-09-15 23:56:05 +08:00
|
|
|
if (UnrollRuntimeMultiExit.getNumOccurrences())
|
|
|
|
return UnrollRuntimeMultiExit;
|
|
|
|
|
|
|
|
// The main pain point with multi-exit loop unrolling is that once unrolled,
|
|
|
|
// we will not be able to merge all blocks into a straight line code.
|
|
|
|
// There are branches within the unrolled loop that go to the OtherExits.
|
|
|
|
// The second point is the increase in code size, but this is true
|
|
|
|
// irrespective of multiple exits.
|
|
|
|
|
|
|
|
// Note: Both the heuristics below are coarse grained. We are essentially
|
|
|
|
// enabling unrolling of loops that have a single side exit other than the
|
|
|
|
// normal LatchExit (i.e. exiting into a deoptimize block).
|
|
|
|
// The heuristics considered are:
|
|
|
|
// 1. low number of branches in the unrolled version.
|
|
|
|
// 2. high predictability of these extra branches.
|
|
|
|
// We avoid unrolling loops that have more than two exiting blocks. This
|
|
|
|
// limits the total number of branches in the unrolled loop to be atmost
|
|
|
|
// the unroll factor (since one of the exiting blocks is the latch block).
|
|
|
|
SmallVector<BasicBlock*, 4> ExitingBlocks;
|
|
|
|
L->getExitingBlocks(ExitingBlocks);
|
|
|
|
if (ExitingBlocks.size() > 2)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// The second heuristic is that L has one exit other than the latchexit and
|
|
|
|
// that exit is a deoptimize block. We know that deoptimize blocks are rarely
|
|
|
|
// taken, which also implies the branch leading to the deoptimize block is
|
|
|
|
// highly predictable.
|
|
|
|
return (OtherExits.size() == 1 &&
|
|
|
|
OtherExits[0]->getTerminatingDeoptimizeCall());
|
|
|
|
// TODO: These can be fine-tuned further to consider code size or deopt states
|
|
|
|
// that are captured by the deoptimize exit block.
|
|
|
|
// Also, we can extend this to support more cases, if we actually
|
|
|
|
// know of kinds of multiexit loops that would benefit from unrolling.
|
2017-07-22 00:30:38 +08:00
|
|
|
}
|
2017-07-13 04:55:43 +08:00
|
|
|
|
2016-04-05 20:19:35 +08:00
|
|
|
/// Insert code in the prolog/epilog code when unrolling a loop with a
|
2011-12-09 14:19:40 +08:00
|
|
|
/// run-time trip-count.
|
|
|
|
///
|
|
|
|
/// This method assumes that the loop unroll factor is total number
|
2016-02-13 05:01:37 +08:00
|
|
|
/// of loop bodies in the loop after unrolling. (Some folks refer
|
2011-12-09 14:19:40 +08:00
|
|
|
/// to the unroll factor as the number of *extra* copies added).
|
|
|
|
/// We assume also that the loop unroll factor is a power-of-two. So, after
|
|
|
|
/// unrolling the loop, the number of loop bodies executed is 2,
|
2011-12-19 05:52:30 +08:00
|
|
|
/// 4, 8, etc. Note - LLVM converts the if-then-sequence to a switch
|
2011-12-09 14:19:40 +08:00
|
|
|
/// instruction in SimplifyCFG.cpp. Then, the backend decides how code for
|
|
|
|
/// the switch instruction is generated.
|
|
|
|
///
|
2016-04-05 20:19:35 +08:00
|
|
|
/// ***Prolog case***
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
/// extraiters = tripcount % loopfactor
|
|
|
|
/// if (extraiters == 0) jump Loop:
|
2016-04-09 04:20:38 +08:00
|
|
|
/// else jump Prol:
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
/// Prol: LoopBody;
|
|
|
|
/// extraiters -= 1 // Omitted if unroll factor is 2.
|
|
|
|
/// if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
|
2016-04-09 04:20:38 +08:00
|
|
|
/// if (tripcount < loopfactor) jump End:
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
/// Loop:
|
|
|
|
/// ...
|
|
|
|
/// End:
|
2011-12-09 14:19:40 +08:00
|
|
|
///
|
2016-04-05 20:19:35 +08:00
|
|
|
/// ***Epilog case***
|
|
|
|
/// extraiters = tripcount % loopfactor
|
2016-04-27 11:04:54 +08:00
|
|
|
/// if (tripcount < loopfactor) jump LoopExit:
|
2016-04-05 20:19:35 +08:00
|
|
|
/// unroll_iters = tripcount - extraiters
|
|
|
|
/// Loop: LoopBody; (executes unroll_iter times);
|
|
|
|
/// unroll_iter -= 1
|
|
|
|
/// if (unroll_iter != 0) jump Loop:
|
|
|
|
/// LoopExit:
|
|
|
|
/// if (extraiters == 0) jump EpilExit:
|
|
|
|
/// Epil: LoopBody; (executes extraiters times)
|
|
|
|
/// extraiters -= 1 // Omitted if unroll factor is 2.
|
|
|
|
/// if (extraiters != 0) jump Epil: // Omitted if unroll factor is 2.
|
|
|
|
/// EpilExit:
|
|
|
|
|
2020-02-26 02:50:55 +08:00
|
|
|
bool llvm::UnrollRuntimeLoopRemainder(
|
|
|
|
Loop *L, unsigned Count, bool AllowExpensiveTripCount,
|
|
|
|
bool UseEpilogRemainder, bool UnrollRemainder, bool ForgetAllSCEV,
|
|
|
|
LoopInfo *LI, ScalarEvolution *SE, DominatorTree *DT, AssumptionCache *AC,
|
|
|
|
const TargetTransformInfo *TTI, bool PreserveLCSSA, Loop **ResultLoop) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "Trying runtime unrolling on Loop: \n");
|
|
|
|
LLVM_DEBUG(L->dump());
|
|
|
|
LLVM_DEBUG(UseEpilogRemainder ? dbgs() << "Using epilog remainder.\n"
|
|
|
|
: dbgs() << "Using prolog remainder.\n");
|
2011-12-09 14:19:40 +08:00
|
|
|
|
2017-07-01 01:57:07 +08:00
|
|
|
// Make sure the loop is in canonical form.
|
2017-07-12 04:44:37 +08:00
|
|
|
if (!L->isLoopSimplifyForm()) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "Not in simplify form!\n");
|
2016-04-05 20:19:35 +08:00
|
|
|
return false;
|
2017-07-12 04:44:37 +08:00
|
|
|
}
|
2011-12-09 14:19:40 +08:00
|
|
|
|
2017-06-23 22:28:01 +08:00
|
|
|
// Guaranteed by LoopSimplifyForm.
|
|
|
|
BasicBlock *Latch = L->getLoopLatch();
|
2017-07-01 01:57:07 +08:00
|
|
|
BasicBlock *Header = L->getHeader();
|
2017-06-23 22:28:01 +08:00
|
|
|
|
2017-07-01 01:57:07 +08:00
|
|
|
BranchInst *LatchBR = cast<BranchInst>(Latch->getTerminator());
|
2018-09-25 18:08:47 +08:00
|
|
|
|
|
|
|
if (!LatchBR || LatchBR->isUnconditional()) {
|
|
|
|
// The loop-rotate pass can be helpful to avoid this in many cases.
|
|
|
|
LLVM_DEBUG(
|
|
|
|
dbgs()
|
|
|
|
<< "Loop latch not terminated by a conditional branch.\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-07-13 04:55:43 +08:00
|
|
|
unsigned ExitIndex = LatchBR->getSuccessor(0) == Header ? 1 : 0;
|
|
|
|
BasicBlock *LatchExit = LatchBR->getSuccessor(ExitIndex);
|
2018-09-25 18:08:47 +08:00
|
|
|
|
|
|
|
if (L->contains(LatchExit)) {
|
|
|
|
// Cloning the loop basic blocks (`CloneLoopBlocks`) requires that one of the
|
|
|
|
// targets of the Latch be an exit block out of the loop.
|
|
|
|
LLVM_DEBUG(
|
|
|
|
dbgs()
|
|
|
|
<< "One of the loop latch successors must be the exit block.\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-07-13 04:55:43 +08:00
|
|
|
// These are exit blocks other than the target of the latch exiting block.
|
|
|
|
SmallVector<BasicBlock *, 4> OtherExits;
|
2019-07-15 13:51:10 +08:00
|
|
|
L->getUniqueNonLatchExitBlocks(OtherExits);
|
2017-07-22 00:30:38 +08:00
|
|
|
bool isMultiExitUnrollingEnabled =
|
2019-07-15 13:51:10 +08:00
|
|
|
canSafelyUnrollMultiExitLoop(L, LatchExit, PreserveLCSSA,
|
2017-07-22 00:30:38 +08:00
|
|
|
UseEpilogRemainder) &&
|
|
|
|
canProfitablyUnrollMultiExitLoop(L, OtherExits, LatchExit, PreserveLCSSA,
|
|
|
|
UseEpilogRemainder);
|
2017-07-13 04:55:43 +08:00
|
|
|
// Support only single exit and exiting block unless multi-exit loop unrolling is enabled.
|
|
|
|
if (!isMultiExitUnrollingEnabled &&
|
|
|
|
(!L->getExitingBlock() || OtherExits.size())) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(
|
2017-07-13 04:55:43 +08:00
|
|
|
dbgs()
|
|
|
|
<< "Multiple exit/exiting blocks in loop and multi-exit unrolling not "
|
|
|
|
"enabled!\n");
|
2017-07-07 02:39:26 +08:00
|
|
|
return false;
|
2017-07-12 04:44:37 +08:00
|
|
|
}
|
2016-02-09 03:27:33 +08:00
|
|
|
// Use Scalar Evolution to compute the trip count. This allows more loops to
|
|
|
|
// be unrolled than relying on induction var simplification.
|
2015-12-16 03:40:57 +08:00
|
|
|
if (!SE)
|
2012-05-08 10:52:09 +08:00
|
|
|
return false;
|
2011-12-09 14:19:40 +08:00
|
|
|
|
2016-02-09 03:27:33 +08:00
|
|
|
// Only unroll loops with a computable trip count, and the trip count needs
|
|
|
|
// to be an int value (allowing a pointer type is a TODO item).
|
2017-06-27 22:14:35 +08:00
|
|
|
// We calculate the backedge count by using getExitCount on the Latch block,
|
|
|
|
// which is proven to be the only exiting block in this loop. This is same as
|
|
|
|
// calculating getBackedgeTakenCount on the loop (which computes SCEV for all
|
|
|
|
// exiting blocks).
|
|
|
|
const SCEV *BECountSC = SE->getExitCount(L, Latch);
|
2015-02-19 03:32:25 +08:00
|
|
|
if (isa<SCEVCouldNotCompute>(BECountSC) ||
|
2017-07-12 04:44:37 +08:00
|
|
|
!BECountSC->getType()->isIntegerTy()) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "Could not compute exit block SCEV\n");
|
2011-12-09 14:19:40 +08:00
|
|
|
return false;
|
2017-07-12 04:44:37 +08:00
|
|
|
}
|
2011-12-09 14:19:40 +08:00
|
|
|
|
2015-02-19 03:32:25 +08:00
|
|
|
unsigned BEWidth = cast<IntegerType>(BECountSC->getType())->getBitWidth();
|
2014-11-21 04:19:55 +08:00
|
|
|
|
2016-02-09 03:27:33 +08:00
|
|
|
// Add 1 since the backedge count doesn't include the first loop iteration.
|
2011-12-19 05:52:30 +08:00
|
|
|
const SCEV *TripCountSC =
|
2015-12-16 03:40:57 +08:00
|
|
|
SE->getAddExpr(BECountSC, SE->getConstant(BECountSC->getType(), 1));
|
2017-07-12 04:44:37 +08:00
|
|
|
if (isa<SCEVCouldNotCompute>(TripCountSC)) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "Could not compute trip count SCEV.\n");
|
2011-12-09 14:19:40 +08:00
|
|
|
return false;
|
2017-07-12 04:44:37 +08:00
|
|
|
}
|
2011-12-09 14:19:40 +08:00
|
|
|
|
2016-04-05 20:19:35 +08:00
|
|
|
BasicBlock *PreHeader = L->getLoopPreheader();
|
|
|
|
BranchInst *PreHeaderBR = cast<BranchInst>(PreHeader->getTerminator());
|
2015-04-14 11:20:38 +08:00
|
|
|
const DataLayout &DL = Header->getModule()->getDataLayout();
|
2015-12-16 03:40:57 +08:00
|
|
|
SCEVExpander Expander(*SE, DL, "loop-unroll");
|
2016-02-16 14:46:58 +08:00
|
|
|
if (!AllowExpensiveTripCount &&
|
2020-02-26 02:50:55 +08:00
|
|
|
Expander.isHighCostExpansion(TripCountSC, L, TTI, PreHeaderBR)) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "High cost for expanding trip count scev!\n");
|
2015-04-14 11:20:38 +08:00
|
|
|
return false;
|
2017-07-12 04:44:37 +08:00
|
|
|
}
|
2015-04-14 11:20:38 +08:00
|
|
|
|
2015-02-19 03:32:25 +08:00
|
|
|
// This constraint lets us deal with an overflowing trip count easily; see the
|
2015-04-12 09:24:01 +08:00
|
|
|
// comment on ModVal below.
|
2017-07-12 04:44:37 +08:00
|
|
|
if (Log2_32(Count) > BEWidth) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(
|
|
|
|
dbgs()
|
|
|
|
<< "Count failed constraint on overflow trip count calculation.\n");
|
2011-12-09 14:19:40 +08:00
|
|
|
return false;
|
2017-07-12 04:44:37 +08:00
|
|
|
}
|
2011-12-09 14:19:40 +08:00
|
|
|
|
2016-04-05 20:19:35 +08:00
|
|
|
// Loop structure is the following:
|
|
|
|
//
|
|
|
|
// PreHeader
|
|
|
|
// Header
|
|
|
|
// ...
|
|
|
|
// Latch
|
2017-06-23 22:28:01 +08:00
|
|
|
// LatchExit
|
2016-04-05 20:19:35 +08:00
|
|
|
|
|
|
|
BasicBlock *NewPreHeader;
|
|
|
|
BasicBlock *NewExit = nullptr;
|
|
|
|
BasicBlock *PrologExit = nullptr;
|
|
|
|
BasicBlock *EpilogPreHeader = nullptr;
|
|
|
|
BasicBlock *PrologPreHeader = nullptr;
|
|
|
|
|
|
|
|
if (UseEpilogRemainder) {
|
|
|
|
// If epilog remainder
|
|
|
|
// Split PreHeader to insert a branch around loop for unrolling.
|
|
|
|
NewPreHeader = SplitBlock(PreHeader, PreHeader->getTerminator(), DT, LI);
|
|
|
|
NewPreHeader->setName(PreHeader->getName() + ".new");
|
2017-06-23 22:28:01 +08:00
|
|
|
// Split LatchExit to create phi nodes from branch above.
|
|
|
|
SmallVector<BasicBlock*, 4> Preds(predecessors(LatchExit));
|
2018-08-22 07:32:03 +08:00
|
|
|
NewExit = SplitBlockPredecessors(LatchExit, Preds, ".unr-lcssa", DT, LI,
|
|
|
|
nullptr, PreserveLCSSA);
|
2017-12-27 07:31:21 +08:00
|
|
|
// NewExit gets its DebugLoc from LatchExit, which is not part of the
|
|
|
|
// original Loop.
|
|
|
|
// Fix this by setting Loop's DebugLoc to NewExit.
|
|
|
|
auto *NewExitTerminator = NewExit->getTerminator();
|
|
|
|
NewExitTerminator->setDebugLoc(Header->getTerminator()->getDebugLoc());
|
2016-04-05 20:19:35 +08:00
|
|
|
// Split NewExit to insert epilog remainder loop.
|
2017-12-27 07:31:21 +08:00
|
|
|
EpilogPreHeader = SplitBlock(NewExit, NewExitTerminator, DT, LI);
|
2016-04-05 20:19:35 +08:00
|
|
|
EpilogPreHeader->setName(Header->getName() + ".epil.preheader");
|
|
|
|
} else {
|
|
|
|
// If prolog remainder
|
|
|
|
// Split the original preheader twice to insert prolog remainder loop
|
|
|
|
PrologPreHeader = SplitEdge(PreHeader, Header, DT, LI);
|
|
|
|
PrologPreHeader->setName(Header->getName() + ".prol.preheader");
|
|
|
|
PrologExit = SplitBlock(PrologPreHeader, PrologPreHeader->getTerminator(),
|
|
|
|
DT, LI);
|
|
|
|
PrologExit->setName(Header->getName() + ".prol.loopexit");
|
|
|
|
// Split PrologExit to get NewPreHeader.
|
|
|
|
NewPreHeader = SplitBlock(PrologExit, PrologExit->getTerminator(), DT, LI);
|
|
|
|
NewPreHeader->setName(PreHeader->getName() + ".new");
|
|
|
|
}
|
|
|
|
// Loop structure should be the following:
|
|
|
|
// Epilog Prolog
|
|
|
|
//
|
|
|
|
// PreHeader PreHeader
|
|
|
|
// *NewPreHeader *PrologPreHeader
|
|
|
|
// Header *PrologExit
|
|
|
|
// ... *NewPreHeader
|
|
|
|
// Latch Header
|
|
|
|
// *NewExit ...
|
|
|
|
// *EpilogPreHeader Latch
|
2017-06-23 22:28:01 +08:00
|
|
|
// LatchExit LatchExit
|
2016-04-05 20:19:35 +08:00
|
|
|
|
|
|
|
// Calculate conditions for branch around loop for unrolling
|
|
|
|
// in epilog case and around prolog remainder loop in prolog case.
|
2011-12-09 14:19:40 +08:00
|
|
|
// Compute the number of extra iterations required, which is:
|
2016-04-05 20:19:35 +08:00
|
|
|
// extra iterations = run-time trip count % loop unroll factor
|
|
|
|
PreHeaderBR = cast<BranchInst>(PreHeader->getTerminator());
|
2011-12-09 14:19:40 +08:00
|
|
|
Value *TripCount = Expander.expandCodeFor(TripCountSC, TripCountSC->getType(),
|
|
|
|
PreHeaderBR);
|
2015-02-19 03:32:25 +08:00
|
|
|
Value *BECount = Expander.expandCodeFor(BECountSC, BECountSC->getType(),
|
|
|
|
PreHeaderBR);
|
2014-06-21 21:46:25 +08:00
|
|
|
IRBuilder<> B(PreHeaderBR);
|
2016-03-25 22:24:52 +08:00
|
|
|
Value *ModVal;
|
|
|
|
// Calculate ModVal = (BECount + 1) % Count.
|
|
|
|
// Note that TripCount is BECount + 1.
|
|
|
|
if (isPowerOf2_32(Count)) {
|
2016-04-05 20:19:35 +08:00
|
|
|
// When Count is power of 2 we don't BECount for epilog case, however we'll
|
|
|
|
// need it for a branch around unrolling loop for prolog case.
|
2016-03-25 22:24:52 +08:00
|
|
|
ModVal = B.CreateAnd(TripCount, Count - 1, "xtraiter");
|
2016-04-05 20:19:35 +08:00
|
|
|
// 1. There are no iterations to be run in the prolog/epilog loop.
|
2016-03-25 22:24:52 +08:00
|
|
|
// OR
|
|
|
|
// 2. The addition computing TripCount overflowed.
|
|
|
|
//
|
|
|
|
// If (2) is true, we know that TripCount really is (1 << BEWidth) and so
|
|
|
|
// the number of iterations that remain to be run in the original loop is a
|
|
|
|
// multiple Count == (1 << Log2(Count)) because Log2(Count) <= BEWidth (we
|
|
|
|
// explicitly check this above).
|
|
|
|
} else {
|
|
|
|
// As (BECount + 1) can potentially unsigned overflow we count
|
|
|
|
// (BECount % Count) + 1 which is overflow safe as BECount % Count < Count.
|
|
|
|
Value *ModValTmp = B.CreateURem(BECount,
|
|
|
|
ConstantInt::get(BECount->getType(),
|
|
|
|
Count));
|
|
|
|
Value *ModValAdd = B.CreateAdd(ModValTmp,
|
|
|
|
ConstantInt::get(ModValTmp->getType(), 1));
|
|
|
|
// At that point (BECount % Count) + 1 could be equal to Count.
|
|
|
|
// To handle this case we need to take mod by Count one more time.
|
|
|
|
ModVal = B.CreateURem(ModValAdd,
|
|
|
|
ConstantInt::get(BECount->getType(), Count),
|
|
|
|
"xtraiter");
|
|
|
|
}
|
2016-04-27 11:04:54 +08:00
|
|
|
Value *BranchVal =
|
|
|
|
UseEpilogRemainder ? B.CreateICmpULT(BECount,
|
|
|
|
ConstantInt::get(BECount->getType(),
|
|
|
|
Count - 1)) :
|
|
|
|
B.CreateIsNotNull(ModVal, "lcmp.mod");
|
|
|
|
BasicBlock *RemainderLoop = UseEpilogRemainder ? NewExit : PrologPreHeader;
|
|
|
|
BasicBlock *UnrollingLoop = UseEpilogRemainder ? NewPreHeader : PrologExit;
|
2016-04-05 20:19:35 +08:00
|
|
|
// Branch to either remainder (extra iterations) loop or unrolling loop.
|
2016-04-27 11:04:54 +08:00
|
|
|
B.CreateCondBr(BranchVal, RemainderLoop, UnrollingLoop);
|
2011-12-09 14:19:40 +08:00
|
|
|
PreHeaderBR->eraseFromParent();
|
2017-01-19 07:26:37 +08:00
|
|
|
if (DT) {
|
|
|
|
if (UseEpilogRemainder)
|
|
|
|
DT->changeImmediateDominator(NewExit, PreHeader);
|
|
|
|
else
|
|
|
|
DT->changeImmediateDominator(PrologExit, PreHeader);
|
|
|
|
}
|
2011-12-09 14:19:40 +08:00
|
|
|
Function *F = Header->getParent();
|
|
|
|
// Get an ordered list of blocks in the loop to help with the ordering of the
|
2016-04-05 20:19:35 +08:00
|
|
|
// cloned blocks in the prolog/epilog code
|
2011-12-09 14:19:40 +08:00
|
|
|
LoopBlocksDFS LoopBlocks(L);
|
|
|
|
LoopBlocks.perform(LI);
|
|
|
|
|
|
|
|
//
|
|
|
|
// For each extra loop iteration, create a copy of the loop's basic blocks
|
|
|
|
// and generate a condition that branches to the copy depending on the
|
|
|
|
// number of 'left over' iterations.
|
|
|
|
//
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
std::vector<BasicBlock *> NewBlocks;
|
|
|
|
ValueToValueMapTy VMap;
|
|
|
|
|
2016-04-05 20:19:35 +08:00
|
|
|
// For unroll factor 2 remainder loop will have 1 iterations.
|
|
|
|
// Do not create 1 iteration loop.
|
|
|
|
bool CreateRemainderLoop = (Count != 2);
|
2014-11-21 04:19:55 +08:00
|
|
|
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
// Clone all the basic blocks in the loop. If Count is 2, we don't clone
|
|
|
|
// the loop, otherwise we create a cloned loop to execute the extra
|
|
|
|
// iterations. This function adds the appropriate CFG connections.
|
2017-06-23 22:28:01 +08:00
|
|
|
BasicBlock *InsertBot = UseEpilogRemainder ? LatchExit : PrologExit;
|
2016-04-05 20:19:35 +08:00
|
|
|
BasicBlock *InsertTop = UseEpilogRemainder ? EpilogPreHeader : PrologPreHeader;
|
2017-07-01 01:57:07 +08:00
|
|
|
Loop *remainderLoop = CloneLoopBlocks(
|
2017-08-14 17:25:26 +08:00
|
|
|
L, ModVal, CreateRemainderLoop, UseEpilogRemainder, UnrollRemainder,
|
|
|
|
InsertTop, InsertBot,
|
2017-07-01 01:57:07 +08:00
|
|
|
NewPreHeader, NewBlocks, LoopBlocks, VMap, DT, LI);
|
2016-04-05 20:19:35 +08:00
|
|
|
|
|
|
|
// Insert the cloned blocks into the function.
|
|
|
|
F->getBasicBlockList().splice(InsertBot->getIterator(),
|
|
|
|
F->getBasicBlockList(),
|
|
|
|
NewBlocks[0]->getIterator(),
|
|
|
|
F->end());
|
|
|
|
|
2017-07-01 01:57:07 +08:00
|
|
|
// Now the loop blocks are cloned and the other exiting blocks from the
|
|
|
|
// remainder are connected to the original Loop's exit blocks. The remaining
|
|
|
|
// work is to update the phi nodes in the original loop, and take in the
|
2019-01-09 01:16:25 +08:00
|
|
|
// values from the cloned region.
|
2017-07-01 01:57:07 +08:00
|
|
|
for (auto *BB : OtherExits) {
|
|
|
|
for (auto &II : *BB) {
|
|
|
|
|
|
|
|
// Given we preserve LCSSA form, we know that the values used outside the
|
|
|
|
// loop will be used through these phi nodes at the exit blocks that are
|
|
|
|
// transformed below.
|
|
|
|
if (!isa<PHINode>(II))
|
|
|
|
break;
|
|
|
|
PHINode *Phi = cast<PHINode>(&II);
|
|
|
|
unsigned oldNumOperands = Phi->getNumIncomingValues();
|
|
|
|
// Add the incoming values from the remainder code to the end of the phi
|
|
|
|
// node.
|
|
|
|
for (unsigned i =0; i < oldNumOperands; i++){
|
2017-09-15 21:29:33 +08:00
|
|
|
Value *newVal = VMap.lookup(Phi->getIncomingValue(i));
|
2017-07-10 23:29:38 +08:00
|
|
|
// newVal can be a constant or derived from values outside the loop, and
|
2017-09-15 21:29:33 +08:00
|
|
|
// hence need not have a VMap value. Also, since lookup already generated
|
|
|
|
// a default "null" VMap entry for this value, we need to populate that
|
|
|
|
// VMap entry correctly, with the mapped entry being itself.
|
|
|
|
if (!newVal) {
|
2017-07-01 01:57:07 +08:00
|
|
|
newVal = Phi->getIncomingValue(i);
|
2017-09-15 21:29:33 +08:00
|
|
|
VMap[Phi->getIncomingValue(i)] = Phi->getIncomingValue(i);
|
|
|
|
}
|
2017-07-01 01:57:07 +08:00
|
|
|
Phi->addIncoming(newVal,
|
|
|
|
cast<BasicBlock>(VMap[Phi->getIncomingBlock(i)]));
|
|
|
|
}
|
|
|
|
}
|
2017-07-14 01:10:12 +08:00
|
|
|
#if defined(EXPENSIVE_CHECKS) && !defined(NDEBUG)
|
2017-07-13 21:21:23 +08:00
|
|
|
for (BasicBlock *SuccBB : successors(BB)) {
|
|
|
|
assert(!(any_of(OtherExits,
|
|
|
|
[SuccBB](BasicBlock *EB) { return EB == SuccBB; }) ||
|
|
|
|
SuccBB == LatchExit) &&
|
|
|
|
"Breaks the definition of dedicated exits!");
|
|
|
|
}
|
|
|
|
#endif
|
2019-01-09 01:16:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Update the immediate dominator of the exit blocks and blocks that are
|
|
|
|
// reachable from the exit blocks. This is needed because we now have paths
|
|
|
|
// from both the original loop and the remainder code reaching the exit
|
|
|
|
// blocks. While the IDom of these exit blocks were from the original loop,
|
|
|
|
// now the IDom is the preheader (which decides whether the original loop or
|
|
|
|
// remainder code should run).
|
|
|
|
if (DT && !L->getExitingBlock()) {
|
|
|
|
SmallVector<BasicBlock *, 16> ChildrenToUpdate;
|
|
|
|
// NB! We have to examine the dom children of all loop blocks, not just
|
|
|
|
// those which are the IDom of the exit blocks. This is because blocks
|
|
|
|
// reachable from the exit blocks can have their IDom as the nearest common
|
|
|
|
// dominator of the exit blocks.
|
|
|
|
for (auto *BB : L->blocks()) {
|
|
|
|
auto *DomNodeBB = DT->getNode(BB);
|
|
|
|
for (auto *DomChild : DomNodeBB->getChildren()) {
|
|
|
|
auto *DomChildBB = DomChild->getBlock();
|
|
|
|
if (!L->contains(LI->getLoopFor(DomChildBB)))
|
|
|
|
ChildrenToUpdate.push_back(DomChildBB);
|
|
|
|
}
|
2017-07-13 21:21:23 +08:00
|
|
|
}
|
2019-01-09 01:16:25 +08:00
|
|
|
for (auto *BB : ChildrenToUpdate)
|
|
|
|
DT->changeImmediateDominator(BB, PreHeader);
|
2017-07-01 01:57:07 +08:00
|
|
|
}
|
|
|
|
|
2016-04-05 20:19:35 +08:00
|
|
|
// Loop structure should be the following:
|
|
|
|
// Epilog Prolog
|
|
|
|
//
|
|
|
|
// PreHeader PreHeader
|
|
|
|
// NewPreHeader PrologPreHeader
|
|
|
|
// Header PrologHeader
|
|
|
|
// ... ...
|
|
|
|
// Latch PrologLatch
|
|
|
|
// NewExit PrologExit
|
|
|
|
// EpilogPreHeader NewPreHeader
|
|
|
|
// EpilogHeader Header
|
|
|
|
// ... ...
|
|
|
|
// EpilogLatch Latch
|
2017-06-23 22:28:01 +08:00
|
|
|
// LatchExit LatchExit
|
Use a loop to simplify the runtime unrolling prologue.
Runtime unrolling will create a prologue to execute the extra
iterations which is can't divided by the unroll factor. It
generates an if-then-else sequence to jump into a factor -1
times unrolled loop body, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
if (extraiters == loopfactor) jump L1
if (extraiters == loopfactor-1) jump L2
...
L1: LoopBody;
L2: LoopBody;
...
if tripcount < loopfactor jump End
Loop:
...
End:
It means if the unroll factor is 4, the loop body will be 7
times unrolled, 3 are in loop prologue, and 4 are in the loop.
This commit is to use a loop to execute the extra iterations
in prologue, like
extraiters = tripcount % loopfactor
if (extraiters == 0) jump Loop:
else jump Prol
Prol: LoopBody;
extraiters -= 1 // Omitted if unroll factor is 2.
if (extraiters != 0) jump Prol: // Omitted if unroll factor is 2.
if (tripcount < loopfactor) jump End
Loop:
...
End:
Then when unroll factor is 4, the loop body will be copied by
only 5 times, 1 in the prologue loop, 4 in the original loop.
And if the unroll factor is 2, new loop won't be created, just
as the original solution.
llvm-svn: 218604
2014-09-29 19:15:00 +08:00
|
|
|
|
2016-02-09 05:32:43 +08:00
|
|
|
// Rewrite the cloned instruction operands to use the values created when the
|
|
|
|
// clone is created.
|
|
|
|
for (BasicBlock *BB : NewBlocks) {
|
|
|
|
for (Instruction &I : *BB) {
|
|
|
|
RemapInstruction(&I, VMap,
|
2016-04-07 08:26:43 +08:00
|
|
|
RF_NoModuleLevelChanges | RF_IgnoreMissingLocals);
|
2011-12-09 14:19:40 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-05 20:19:35 +08:00
|
|
|
if (UseEpilogRemainder) {
|
|
|
|
// Connect the epilog code to the original loop and update the
|
|
|
|
// PHI functions.
|
2017-06-23 22:28:01 +08:00
|
|
|
ConnectEpilog(L, ModVal, NewExit, LatchExit, PreHeader,
|
2016-04-05 20:19:35 +08:00
|
|
|
EpilogPreHeader, NewPreHeader, VMap, DT, LI,
|
|
|
|
PreserveLCSSA);
|
|
|
|
|
|
|
|
// Update counter in loop for unrolling.
|
|
|
|
// I should be multiply of Count.
|
|
|
|
IRBuilder<> B2(NewPreHeader->getTerminator());
|
|
|
|
Value *TestVal = B2.CreateSub(TripCount, ModVal, "unroll_iter");
|
|
|
|
BranchInst *LatchBR = cast<BranchInst>(Latch->getTerminator());
|
|
|
|
B2.SetInsertPoint(LatchBR);
|
|
|
|
PHINode *NewIdx = PHINode::Create(TestVal->getType(), 2, "niter",
|
|
|
|
Header->getFirstNonPHI());
|
|
|
|
Value *IdxSub =
|
|
|
|
B2.CreateSub(NewIdx, ConstantInt::get(NewIdx->getType(), 1),
|
|
|
|
NewIdx->getName() + ".nsub");
|
|
|
|
Value *IdxCmp;
|
|
|
|
if (LatchBR->getSuccessor(0) == Header)
|
|
|
|
IdxCmp = B2.CreateIsNotNull(IdxSub, NewIdx->getName() + ".ncmp");
|
|
|
|
else
|
|
|
|
IdxCmp = B2.CreateIsNull(IdxSub, NewIdx->getName() + ".ncmp");
|
|
|
|
NewIdx->addIncoming(TestVal, NewPreHeader);
|
|
|
|
NewIdx->addIncoming(IdxSub, Latch);
|
|
|
|
LatchBR->setCondition(IdxCmp);
|
|
|
|
} else {
|
|
|
|
// Connect the prolog code to the original loop and update the
|
|
|
|
// PHI functions.
|
2017-07-08 02:05:28 +08:00
|
|
|
ConnectProlog(L, BECount, Count, PrologExit, LatchExit, PreHeader,
|
|
|
|
NewPreHeader, VMap, DT, LI, PreserveLCSSA);
|
2016-04-05 20:19:35 +08:00
|
|
|
}
|
2016-08-26 00:17:18 +08:00
|
|
|
|
2018-04-23 18:39:38 +08:00
|
|
|
// If this loop is nested, then the loop unroller changes the code in the any
|
|
|
|
// of its parent loops, so the Scalar Evolution pass needs to be run again.
|
|
|
|
SE->forgetTopmostLoop(L);
|
2016-08-26 00:17:18 +08:00
|
|
|
|
2019-01-04 01:44:44 +08:00
|
|
|
// Verify that the Dom Tree is correct.
|
2019-01-04 03:43:33 +08:00
|
|
|
#if defined(EXPENSIVE_CHECKS) && !defined(NDEBUG)
|
2019-01-04 01:44:44 +08:00
|
|
|
if (DT)
|
|
|
|
assert(DT->verify(DominatorTree::VerificationLevel::Full));
|
|
|
|
#endif
|
|
|
|
|
2017-07-01 01:57:07 +08:00
|
|
|
// Canonicalize to LoopSimplifyForm both original and remainder loops. We
|
|
|
|
// cannot rely on the LoopUnrollPass to do this because it only does
|
|
|
|
// canonicalization for parent/subloops and not the sibling loops.
|
|
|
|
if (OtherExits.size() > 0) {
|
|
|
|
// Generate dedicated exit blocks for the original loop, to preserve
|
|
|
|
// LoopSimplifyForm.
|
2019-02-22 05:13:34 +08:00
|
|
|
formDedicatedExitBlocks(L, DT, LI, nullptr, PreserveLCSSA);
|
2017-07-01 01:57:07 +08:00
|
|
|
// Generate dedicated exit blocks for the remainder loop if one exists, to
|
|
|
|
// preserve LoopSimplifyForm.
|
|
|
|
if (remainderLoop)
|
2019-02-22 05:13:34 +08:00
|
|
|
formDedicatedExitBlocks(remainderLoop, DT, LI, nullptr, PreserveLCSSA);
|
2017-07-01 01:57:07 +08:00
|
|
|
}
|
|
|
|
|
[Unroll/UnrollAndJam/Vectorizer/Distribute] Add followup loop attributes.
When multiple loop transformation are defined in a loop's metadata, their order of execution is defined by the order of their respective passes in the pass pipeline. For instance, e.g.
#pragma clang loop unroll_and_jam(enable)
#pragma clang loop distribute(enable)
is the same as
#pragma clang loop distribute(enable)
#pragma clang loop unroll_and_jam(enable)
and will try to loop-distribute before Unroll-And-Jam because the LoopDistribute pass is scheduled after UnrollAndJam pass. UnrollAndJamPass only supports one inner loop, i.e. it will necessarily fail after loop distribution. It is not possible to specify another execution order. Also,t the order of passes in the pipeline is subject to change between versions of LLVM, optimization options and which pass manager is used.
This patch adds 'followup' attributes to various loop transformation passes. These attributes define which attributes the resulting loop of a transformation should have. For instance,
!0 = !{!0, !1, !2}
!1 = !{!"llvm.loop.unroll_and_jam.enable"}
!2 = !{!"llvm.loop.unroll_and_jam.followup_inner", !3}
!3 = !{!"llvm.loop.distribute.enable"}
defines a loop ID (!0) to be unrolled-and-jammed (!1) and then the attribute !3 to be added to the jammed inner loop, which contains the instruction to distribute the inner loop.
Currently, in both pass managers, pass execution is in a fixed order and UnrollAndJamPass will not execute again after LoopDistribute. We hope to fix this in the future by allowing pass managers to run passes until a fixpoint is reached, use Polly to perform these transformations, or add a loop transformation pass which takes the order issue into account.
For mandatory/forced transformations (e.g. by having been declared by #pragma omp simd), the user must be notified when a transformation could not be performed. It is not possible that the responsible pass emits such a warning because the transformation might be 'hidden' in a followup attribute when it is executed, or it is not present in the pipeline at all. For this reason, this patche introduces a WarnMissedTransformations pass, to warn about orphaned transformations.
Since this changes the user-visible diagnostic message when a transformation is applied, two test cases in the clang repository need to be updated.
To ensure that no other transformation is executed before the intended one, the attribute `llvm.loop.disable_nonforced` can be added which should disable transformation heuristics before the intended transformation is applied. E.g. it would be surprising if a loop is distributed before a #pragma unroll_and_jam is applied.
With more supported code transformations (loop fusion, interchange, stripmining, offloading, etc.), transformations can be used as building blocks for more complex transformations (e.g. stripmining+stripmining+interchange -> tiling).
Reviewed By: hfinkel, dmgreen
Differential Revision: https://reviews.llvm.org/D49281
Differential Revision: https://reviews.llvm.org/D55288
llvm-svn: 348944
2018-12-13 01:32:52 +08:00
|
|
|
auto UnrollResult = LoopUnrollResult::Unmodified;
|
2017-08-14 17:25:26 +08:00
|
|
|
if (remainderLoop && UnrollRemainder) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "Unrolling remainder loop\n");
|
[Unroll/UnrollAndJam/Vectorizer/Distribute] Add followup loop attributes.
When multiple loop transformation are defined in a loop's metadata, their order of execution is defined by the order of their respective passes in the pass pipeline. For instance, e.g.
#pragma clang loop unroll_and_jam(enable)
#pragma clang loop distribute(enable)
is the same as
#pragma clang loop distribute(enable)
#pragma clang loop unroll_and_jam(enable)
and will try to loop-distribute before Unroll-And-Jam because the LoopDistribute pass is scheduled after UnrollAndJam pass. UnrollAndJamPass only supports one inner loop, i.e. it will necessarily fail after loop distribution. It is not possible to specify another execution order. Also,t the order of passes in the pipeline is subject to change between versions of LLVM, optimization options and which pass manager is used.
This patch adds 'followup' attributes to various loop transformation passes. These attributes define which attributes the resulting loop of a transformation should have. For instance,
!0 = !{!0, !1, !2}
!1 = !{!"llvm.loop.unroll_and_jam.enable"}
!2 = !{!"llvm.loop.unroll_and_jam.followup_inner", !3}
!3 = !{!"llvm.loop.distribute.enable"}
defines a loop ID (!0) to be unrolled-and-jammed (!1) and then the attribute !3 to be added to the jammed inner loop, which contains the instruction to distribute the inner loop.
Currently, in both pass managers, pass execution is in a fixed order and UnrollAndJamPass will not execute again after LoopDistribute. We hope to fix this in the future by allowing pass managers to run passes until a fixpoint is reached, use Polly to perform these transformations, or add a loop transformation pass which takes the order issue into account.
For mandatory/forced transformations (e.g. by having been declared by #pragma omp simd), the user must be notified when a transformation could not be performed. It is not possible that the responsible pass emits such a warning because the transformation might be 'hidden' in a followup attribute when it is executed, or it is not present in the pipeline at all. For this reason, this patche introduces a WarnMissedTransformations pass, to warn about orphaned transformations.
Since this changes the user-visible diagnostic message when a transformation is applied, two test cases in the clang repository need to be updated.
To ensure that no other transformation is executed before the intended one, the attribute `llvm.loop.disable_nonforced` can be added which should disable transformation heuristics before the intended transformation is applied. E.g. it would be surprising if a loop is distributed before a #pragma unroll_and_jam is applied.
With more supported code transformations (loop fusion, interchange, stripmining, offloading, etc.), transformations can be used as building blocks for more complex transformations (e.g. stripmining+stripmining+interchange -> tiling).
Reviewed By: hfinkel, dmgreen
Differential Revision: https://reviews.llvm.org/D49281
Differential Revision: https://reviews.llvm.org/D55288
llvm-svn: 348944
2018-12-13 01:32:52 +08:00
|
|
|
UnrollResult =
|
2019-04-19 07:43:49 +08:00
|
|
|
UnrollLoop(remainderLoop,
|
|
|
|
{/*Count*/ Count - 1, /*TripCount*/ Count - 1,
|
|
|
|
/*Force*/ false, /*AllowRuntime*/ false,
|
|
|
|
/*AllowExpensiveTripCount*/ false, /*PreserveCondBr*/ true,
|
|
|
|
/*PreserveOnlyFirst*/ false, /*TripMultiple*/ 1,
|
|
|
|
/*PeelCount*/ 0, /*UnrollRemainder*/ false, ForgetAllSCEV},
|
2020-02-26 02:50:55 +08:00
|
|
|
LI, SE, DT, AC, TTI, /*ORE*/ nullptr, PreserveLCSSA);
|
2017-08-14 17:25:26 +08:00
|
|
|
}
|
|
|
|
|
[Unroll/UnrollAndJam/Vectorizer/Distribute] Add followup loop attributes.
When multiple loop transformation are defined in a loop's metadata, their order of execution is defined by the order of their respective passes in the pass pipeline. For instance, e.g.
#pragma clang loop unroll_and_jam(enable)
#pragma clang loop distribute(enable)
is the same as
#pragma clang loop distribute(enable)
#pragma clang loop unroll_and_jam(enable)
and will try to loop-distribute before Unroll-And-Jam because the LoopDistribute pass is scheduled after UnrollAndJam pass. UnrollAndJamPass only supports one inner loop, i.e. it will necessarily fail after loop distribution. It is not possible to specify another execution order. Also,t the order of passes in the pipeline is subject to change between versions of LLVM, optimization options and which pass manager is used.
This patch adds 'followup' attributes to various loop transformation passes. These attributes define which attributes the resulting loop of a transformation should have. For instance,
!0 = !{!0, !1, !2}
!1 = !{!"llvm.loop.unroll_and_jam.enable"}
!2 = !{!"llvm.loop.unroll_and_jam.followup_inner", !3}
!3 = !{!"llvm.loop.distribute.enable"}
defines a loop ID (!0) to be unrolled-and-jammed (!1) and then the attribute !3 to be added to the jammed inner loop, which contains the instruction to distribute the inner loop.
Currently, in both pass managers, pass execution is in a fixed order and UnrollAndJamPass will not execute again after LoopDistribute. We hope to fix this in the future by allowing pass managers to run passes until a fixpoint is reached, use Polly to perform these transformations, or add a loop transformation pass which takes the order issue into account.
For mandatory/forced transformations (e.g. by having been declared by #pragma omp simd), the user must be notified when a transformation could not be performed. It is not possible that the responsible pass emits such a warning because the transformation might be 'hidden' in a followup attribute when it is executed, or it is not present in the pipeline at all. For this reason, this patche introduces a WarnMissedTransformations pass, to warn about orphaned transformations.
Since this changes the user-visible diagnostic message when a transformation is applied, two test cases in the clang repository need to be updated.
To ensure that no other transformation is executed before the intended one, the attribute `llvm.loop.disable_nonforced` can be added which should disable transformation heuristics before the intended transformation is applied. E.g. it would be surprising if a loop is distributed before a #pragma unroll_and_jam is applied.
With more supported code transformations (loop fusion, interchange, stripmining, offloading, etc.), transformations can be used as building blocks for more complex transformations (e.g. stripmining+stripmining+interchange -> tiling).
Reviewed By: hfinkel, dmgreen
Differential Revision: https://reviews.llvm.org/D49281
Differential Revision: https://reviews.llvm.org/D55288
llvm-svn: 348944
2018-12-13 01:32:52 +08:00
|
|
|
if (ResultLoop && UnrollResult != LoopUnrollResult::FullyUnrolled)
|
|
|
|
*ResultLoop = remainderLoop;
|
2011-12-09 14:19:40 +08:00
|
|
|
NumRuntimeUnrolled++;
|
|
|
|
return true;
|
|
|
|
}
|