2015-04-20 12:38:33 +08:00
|
|
|
//===-- LoopUtils.cpp - Loop Utility functions -------------------------===//
|
|
|
|
//
|
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
|
2015-04-20 12:38:33 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file defines common loop utility functions.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2016-07-27 01:52:02 +08:00
|
|
|
#include "llvm/Transforms/Utils/LoopUtils.h"
|
2017-06-26 06:45:31 +08:00
|
|
|
#include "llvm/ADT/ScopeExit.h"
|
[LPM] Factor all of the loop analysis usage updates into a common helper
routine.
We were getting this wrong in small ways and generally being very
inconsistent about it across loop passes. Instead, let's have a common
place where we do this. One minor downside is that this will require
some analyses like SCEV in more places than they are strictly needed.
However, this seems benign as these analyses are complete no-ops, and
without this consistency we can in many cases end up with the legacy
pass manager scheduling deciding to split up a loop pass pipeline in
order to run the function analysis half-way through. It is very, very
annoying to fix these without just being very pedantic across the board.
The only loop passes I've not updated here are ones that use
AU.setPreservesAll() such as IVUsers (an analysis) and the pass printer.
They seemed less relevant.
With this patch, almost all of the problems in PR24804 around loop pass
pipelines are fixed. The one remaining issue is that we run simplify-cfg
and instcombine in the middle of the loop pass pipeline. We've recently
added some loop variants of these passes that would seem substantially
cleaner to use, but this at least gets us much closer to the previous
state. Notably, the seven loop pass managers is down to three.
I've not updated the loop passes using LoopAccessAnalysis because that
analysis hasn't been fully wired into LoopSimplify/LCSSA, and it isn't
clear that those transforms want to support those forms anyways. They
all run late anyways, so this is harmless. Similarly, LSR is left alone
because it already carefully manages its forms and doesn't need to get
fused into a single loop pass manager with a bunch of other loop passes.
LoopReroll didn't use loop simplified form previously, and I've updated
the test case to match the trivially different output.
Finally, I've also factored all the pass initialization for the passes
that use this technique as well, so that should be done regularly and
reliably.
Thanks to James for the help reviewing and thinking about this stuff,
and Ben for help thinking about it as well!
Differential Revision: http://reviews.llvm.org/D17435
llvm-svn: 261316
2016-02-19 18:45:18 +08:00
|
|
|
#include "llvm/Analysis/AliasAnalysis.h"
|
|
|
|
#include "llvm/Analysis/BasicAliasAnalysis.h"
|
2019-02-06 10:52:52 +08:00
|
|
|
#include "llvm/Analysis/DomTreeUpdater.h"
|
[LPM] Factor all of the loop analysis usage updates into a common helper
routine.
We were getting this wrong in small ways and generally being very
inconsistent about it across loop passes. Instead, let's have a common
place where we do this. One minor downside is that this will require
some analyses like SCEV in more places than they are strictly needed.
However, this seems benign as these analyses are complete no-ops, and
without this consistency we can in many cases end up with the legacy
pass manager scheduling deciding to split up a loop pass pipeline in
order to run the function analysis half-way through. It is very, very
annoying to fix these without just being very pedantic across the board.
The only loop passes I've not updated here are ones that use
AU.setPreservesAll() such as IVUsers (an analysis) and the pass printer.
They seemed less relevant.
With this patch, almost all of the problems in PR24804 around loop pass
pipelines are fixed. The one remaining issue is that we run simplify-cfg
and instcombine in the middle of the loop pass pipeline. We've recently
added some loop variants of these passes that would seem substantially
cleaner to use, but this at least gets us much closer to the previous
state. Notably, the seven loop pass managers is down to three.
I've not updated the loop passes using LoopAccessAnalysis because that
analysis hasn't been fully wired into LoopSimplify/LCSSA, and it isn't
clear that those transforms want to support those forms anyways. They
all run late anyways, so this is harmless. Similarly, LSR is left alone
because it already carefully manages its forms and doesn't need to get
fused into a single loop pass manager with a bunch of other loop passes.
LoopReroll didn't use loop simplified form previously, and I've updated
the test case to match the trivially different output.
Finally, I've also factored all the pass initialization for the passes
that use this technique as well, so that should be done regularly and
reliably.
Thanks to James for the help reviewing and thinking about this stuff,
and Ben for help thinking about it as well!
Differential Revision: http://reviews.llvm.org/D17435
llvm-svn: 261316
2016-02-19 18:45:18 +08:00
|
|
|
#include "llvm/Analysis/GlobalsModRef.h"
|
2018-03-16 05:04:28 +08:00
|
|
|
#include "llvm/Analysis/InstructionSimplify.h"
|
2016-07-27 01:52:02 +08:00
|
|
|
#include "llvm/Analysis/LoopInfo.h"
|
2016-10-28 20:57:20 +08:00
|
|
|
#include "llvm/Analysis/LoopPass.h"
|
2019-09-05 03:16:04 +08:00
|
|
|
#include "llvm/Analysis/MemorySSA.h"
|
2019-02-22 05:13:34 +08:00
|
|
|
#include "llvm/Analysis/MemorySSAUpdater.h"
|
2018-03-21 06:45:23 +08:00
|
|
|
#include "llvm/Analysis/MustExecute.h"
|
2015-11-25 02:57:06 +08:00
|
|
|
#include "llvm/Analysis/ScalarEvolution.h"
|
2016-07-27 01:52:02 +08:00
|
|
|
#include "llvm/Analysis/ScalarEvolutionAliasAnalysis.h"
|
2016-05-10 15:33:35 +08:00
|
|
|
#include "llvm/Analysis/ScalarEvolutionExpander.h"
|
2015-11-25 02:57:06 +08:00
|
|
|
#include "llvm/Analysis/ScalarEvolutionExpressions.h"
|
2017-06-06 19:49:48 +08:00
|
|
|
#include "llvm/Analysis/TargetTransformInfo.h"
|
2018-02-04 23:42:24 +08:00
|
|
|
#include "llvm/Analysis/ValueTracking.h"
|
2018-12-13 07:32:35 +08:00
|
|
|
#include "llvm/IR/DIBuilder.h"
|
[LPM] Factor all of the loop analysis usage updates into a common helper
routine.
We were getting this wrong in small ways and generally being very
inconsistent about it across loop passes. Instead, let's have a common
place where we do this. One minor downside is that this will require
some analyses like SCEV in more places than they are strictly needed.
However, this seems benign as these analyses are complete no-ops, and
without this consistency we can in many cases end up with the legacy
pass manager scheduling deciding to split up a loop pass pipeline in
order to run the function analysis half-way through. It is very, very
annoying to fix these without just being very pedantic across the board.
The only loop passes I've not updated here are ones that use
AU.setPreservesAll() such as IVUsers (an analysis) and the pass printer.
They seemed less relevant.
With this patch, almost all of the problems in PR24804 around loop pass
pipelines are fixed. The one remaining issue is that we run simplify-cfg
and instcombine in the middle of the loop pass pipeline. We've recently
added some loop variants of these passes that would seem substantially
cleaner to use, but this at least gets us much closer to the previous
state. Notably, the seven loop pass managers is down to three.
I've not updated the loop passes using LoopAccessAnalysis because that
analysis hasn't been fully wired into LoopSimplify/LCSSA, and it isn't
clear that those transforms want to support those forms anyways. They
all run late anyways, so this is harmless. Similarly, LSR is left alone
because it already carefully manages its forms and doesn't need to get
fused into a single loop pass manager with a bunch of other loop passes.
LoopReroll didn't use loop simplified form previously, and I've updated
the test case to match the trivially different output.
Finally, I've also factored all the pass initialization for the passes
that use this technique as well, so that should be done regularly and
reliably.
Thanks to James for the help reviewing and thinking about this stuff,
and Ben for help thinking about it as well!
Differential Revision: http://reviews.llvm.org/D17435
llvm-svn: 261316
2016-02-19 18:45:18 +08:00
|
|
|
#include "llvm/IR/Dominators.h"
|
2015-04-20 12:38:33 +08:00
|
|
|
#include "llvm/IR/Instructions.h"
|
2018-12-13 07:32:35 +08:00
|
|
|
#include "llvm/IR/IntrinsicInst.h"
|
2015-11-25 02:57:06 +08:00
|
|
|
#include "llvm/IR/Module.h"
|
2015-04-20 12:38:33 +08:00
|
|
|
#include "llvm/IR/PatternMatch.h"
|
|
|
|
#include "llvm/IR/ValueHandle.h"
|
[LPM] Factor all of the loop analysis usage updates into a common helper
routine.
We were getting this wrong in small ways and generally being very
inconsistent about it across loop passes. Instead, let's have a common
place where we do this. One minor downside is that this will require
some analyses like SCEV in more places than they are strictly needed.
However, this seems benign as these analyses are complete no-ops, and
without this consistency we can in many cases end up with the legacy
pass manager scheduling deciding to split up a loop pass pipeline in
order to run the function analysis half-way through. It is very, very
annoying to fix these without just being very pedantic across the board.
The only loop passes I've not updated here are ones that use
AU.setPreservesAll() such as IVUsers (an analysis) and the pass printer.
They seemed less relevant.
With this patch, almost all of the problems in PR24804 around loop pass
pipelines are fixed. The one remaining issue is that we run simplify-cfg
and instcombine in the middle of the loop pass pipeline. We've recently
added some loop variants of these passes that would seem substantially
cleaner to use, but this at least gets us much closer to the previous
state. Notably, the seven loop pass managers is down to three.
I've not updated the loop passes using LoopAccessAnalysis because that
analysis hasn't been fully wired into LoopSimplify/LCSSA, and it isn't
clear that those transforms want to support those forms anyways. They
all run late anyways, so this is harmless. Similarly, LSR is left alone
because it already carefully manages its forms and doesn't need to get
fused into a single loop pass manager with a bunch of other loop passes.
LoopReroll didn't use loop simplified form previously, and I've updated
the test case to match the trivially different output.
Finally, I've also factored all the pass initialization for the passes
that use this technique as well, so that should be done regularly and
reliably.
Thanks to James for the help reviewing and thinking about this stuff,
and Ben for help thinking about it as well!
Differential Revision: http://reviews.llvm.org/D17435
llvm-svn: 261316
2016-02-19 18:45:18 +08:00
|
|
|
#include "llvm/Pass.h"
|
2015-04-20 12:38:33 +08:00
|
|
|
#include "llvm/Support/Debug.h"
|
2018-02-04 23:42:24 +08:00
|
|
|
#include "llvm/Support/KnownBits.h"
|
2017-06-26 06:45:31 +08:00
|
|
|
#include "llvm/Transforms/Utils/BasicBlockUtils.h"
|
2015-04-20 12:38:33 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
using namespace llvm::PatternMatch;
|
|
|
|
|
|
|
|
#define DEBUG_TYPE "loop-utils"
|
|
|
|
|
[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
|
|
|
static const char *LLVMLoopDisableNonforced = "llvm.loop.disable_nonforced";
|
2019-08-08 21:46:17 +08:00
|
|
|
static const char *LLVMLoopDisableLICM = "llvm.licm.disable";
|
[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
|
|
|
|
2017-06-26 06:45:31 +08:00
|
|
|
bool llvm::formDedicatedExitBlocks(Loop *L, DominatorTree *DT, LoopInfo *LI,
|
2019-02-22 05:13:34 +08:00
|
|
|
MemorySSAUpdater *MSSAU,
|
2017-06-26 06:45:31 +08:00
|
|
|
bool PreserveLCSSA) {
|
|
|
|
bool Changed = false;
|
|
|
|
|
|
|
|
// We re-use a vector for the in-loop predecesosrs.
|
|
|
|
SmallVector<BasicBlock *, 4> InLoopPredecessors;
|
|
|
|
|
|
|
|
auto RewriteExit = [&](BasicBlock *BB) {
|
|
|
|
assert(InLoopPredecessors.empty() &&
|
|
|
|
"Must start with an empty predecessors list!");
|
|
|
|
auto Cleanup = make_scope_exit([&] { InLoopPredecessors.clear(); });
|
|
|
|
|
|
|
|
// See if there are any non-loop predecessors of this exit block and
|
|
|
|
// keep track of the in-loop predecessors.
|
|
|
|
bool IsDedicatedExit = true;
|
|
|
|
for (auto *PredBB : predecessors(BB))
|
|
|
|
if (L->contains(PredBB)) {
|
|
|
|
if (isa<IndirectBrInst>(PredBB->getTerminator()))
|
|
|
|
// We cannot rewrite exiting edges from an indirectbr.
|
|
|
|
return false;
|
2019-02-09 04:48:56 +08:00
|
|
|
if (isa<CallBrInst>(PredBB->getTerminator()))
|
|
|
|
// We cannot rewrite exiting edges from a callbr.
|
|
|
|
return false;
|
2017-06-26 06:45:31 +08:00
|
|
|
|
|
|
|
InLoopPredecessors.push_back(PredBB);
|
|
|
|
} else {
|
|
|
|
IsDedicatedExit = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(!InLoopPredecessors.empty() && "Must have *some* loop predecessor!");
|
|
|
|
|
|
|
|
// Nothing to do if this is already a dedicated exit.
|
|
|
|
if (IsDedicatedExit)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
auto *NewExitBB = SplitBlockPredecessors(
|
2019-02-22 05:13:34 +08:00
|
|
|
BB, InLoopPredecessors, ".loopexit", DT, LI, MSSAU, PreserveLCSSA);
|
2017-06-26 06:45:31 +08:00
|
|
|
|
|
|
|
if (!NewExitBB)
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(
|
|
|
|
dbgs() << "WARNING: Can't create a dedicated exit block for loop: "
|
|
|
|
<< *L << "\n");
|
2017-06-26 06:45:31 +08:00
|
|
|
else
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "LoopSimplify: Creating dedicated exit block "
|
|
|
|
<< NewExitBB->getName() << "\n");
|
2017-06-26 06:45:31 +08:00
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Walk the exit blocks directly rather than building up a data structure for
|
|
|
|
// them, but only visit each one once.
|
|
|
|
SmallPtrSet<BasicBlock *, 4> Visited;
|
|
|
|
for (auto *BB : L->blocks())
|
|
|
|
for (auto *SuccBB : successors(BB)) {
|
|
|
|
// We're looking for exit blocks so skip in-loop successors.
|
|
|
|
if (L->contains(SuccBB))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Visit each exit block exactly once.
|
|
|
|
if (!Visited.insert(SuccBB).second)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
Changed |= RewriteExit(SuccBB);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Changed;
|
|
|
|
}
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// Returns the instructions that use values defined in the loop.
|
2015-08-19 13:40:42 +08:00
|
|
|
SmallVector<Instruction *, 8> llvm::findDefsUsedOutsideOfLoop(Loop *L) {
|
|
|
|
SmallVector<Instruction *, 8> UsedOutside;
|
|
|
|
|
|
|
|
for (auto *Block : L->getBlocks())
|
|
|
|
// FIXME: I believe that this could use copy_if if the Inst reference could
|
|
|
|
// be adapted into a pointer.
|
|
|
|
for (auto &Inst : *Block) {
|
|
|
|
auto Users = Inst.users();
|
2016-08-12 05:15:00 +08:00
|
|
|
if (any_of(Users, [&](User *U) {
|
2015-08-19 13:40:42 +08:00
|
|
|
auto *Use = cast<Instruction>(U);
|
|
|
|
return !L->contains(Use->getParent());
|
|
|
|
}))
|
|
|
|
UsedOutside.push_back(&Inst);
|
|
|
|
}
|
|
|
|
|
|
|
|
return UsedOutside;
|
|
|
|
}
|
[LPM] Factor all of the loop analysis usage updates into a common helper
routine.
We were getting this wrong in small ways and generally being very
inconsistent about it across loop passes. Instead, let's have a common
place where we do this. One minor downside is that this will require
some analyses like SCEV in more places than they are strictly needed.
However, this seems benign as these analyses are complete no-ops, and
without this consistency we can in many cases end up with the legacy
pass manager scheduling deciding to split up a loop pass pipeline in
order to run the function analysis half-way through. It is very, very
annoying to fix these without just being very pedantic across the board.
The only loop passes I've not updated here are ones that use
AU.setPreservesAll() such as IVUsers (an analysis) and the pass printer.
They seemed less relevant.
With this patch, almost all of the problems in PR24804 around loop pass
pipelines are fixed. The one remaining issue is that we run simplify-cfg
and instcombine in the middle of the loop pass pipeline. We've recently
added some loop variants of these passes that would seem substantially
cleaner to use, but this at least gets us much closer to the previous
state. Notably, the seven loop pass managers is down to three.
I've not updated the loop passes using LoopAccessAnalysis because that
analysis hasn't been fully wired into LoopSimplify/LCSSA, and it isn't
clear that those transforms want to support those forms anyways. They
all run late anyways, so this is harmless. Similarly, LSR is left alone
because it already carefully manages its forms and doesn't need to get
fused into a single loop pass manager with a bunch of other loop passes.
LoopReroll didn't use loop simplified form previously, and I've updated
the test case to match the trivially different output.
Finally, I've also factored all the pass initialization for the passes
that use this technique as well, so that should be done regularly and
reliably.
Thanks to James for the help reviewing and thinking about this stuff,
and Ben for help thinking about it as well!
Differential Revision: http://reviews.llvm.org/D17435
llvm-svn: 261316
2016-02-19 18:45:18 +08:00
|
|
|
|
|
|
|
void llvm::getLoopAnalysisUsage(AnalysisUsage &AU) {
|
|
|
|
// By definition, all loop passes need the LoopInfo analysis and the
|
|
|
|
// Dominator tree it depends on. Because they all participate in the loop
|
|
|
|
// pass manager, they must also preserve these.
|
|
|
|
AU.addRequired<DominatorTreeWrapperPass>();
|
|
|
|
AU.addPreserved<DominatorTreeWrapperPass>();
|
|
|
|
AU.addRequired<LoopInfoWrapperPass>();
|
|
|
|
AU.addPreserved<LoopInfoWrapperPass>();
|
|
|
|
|
|
|
|
// We must also preserve LoopSimplify and LCSSA. We locally access their IDs
|
|
|
|
// here because users shouldn't directly get them from this header.
|
|
|
|
extern char &LoopSimplifyID;
|
|
|
|
extern char &LCSSAID;
|
|
|
|
AU.addRequiredID(LoopSimplifyID);
|
|
|
|
AU.addPreservedID(LoopSimplifyID);
|
|
|
|
AU.addRequiredID(LCSSAID);
|
|
|
|
AU.addPreservedID(LCSSAID);
|
2016-10-28 20:57:20 +08:00
|
|
|
// This is used in the LPPassManager to perform LCSSA verification on passes
|
|
|
|
// which preserve lcssa form
|
|
|
|
AU.addRequired<LCSSAVerificationPass>();
|
|
|
|
AU.addPreserved<LCSSAVerificationPass>();
|
[LPM] Factor all of the loop analysis usage updates into a common helper
routine.
We were getting this wrong in small ways and generally being very
inconsistent about it across loop passes. Instead, let's have a common
place where we do this. One minor downside is that this will require
some analyses like SCEV in more places than they are strictly needed.
However, this seems benign as these analyses are complete no-ops, and
without this consistency we can in many cases end up with the legacy
pass manager scheduling deciding to split up a loop pass pipeline in
order to run the function analysis half-way through. It is very, very
annoying to fix these without just being very pedantic across the board.
The only loop passes I've not updated here are ones that use
AU.setPreservesAll() such as IVUsers (an analysis) and the pass printer.
They seemed less relevant.
With this patch, almost all of the problems in PR24804 around loop pass
pipelines are fixed. The one remaining issue is that we run simplify-cfg
and instcombine in the middle of the loop pass pipeline. We've recently
added some loop variants of these passes that would seem substantially
cleaner to use, but this at least gets us much closer to the previous
state. Notably, the seven loop pass managers is down to three.
I've not updated the loop passes using LoopAccessAnalysis because that
analysis hasn't been fully wired into LoopSimplify/LCSSA, and it isn't
clear that those transforms want to support those forms anyways. They
all run late anyways, so this is harmless. Similarly, LSR is left alone
because it already carefully manages its forms and doesn't need to get
fused into a single loop pass manager with a bunch of other loop passes.
LoopReroll didn't use loop simplified form previously, and I've updated
the test case to match the trivially different output.
Finally, I've also factored all the pass initialization for the passes
that use this technique as well, so that should be done regularly and
reliably.
Thanks to James for the help reviewing and thinking about this stuff,
and Ben for help thinking about it as well!
Differential Revision: http://reviews.llvm.org/D17435
llvm-svn: 261316
2016-02-19 18:45:18 +08:00
|
|
|
|
|
|
|
// Loop passes are designed to run inside of a loop pass manager which means
|
|
|
|
// that any function analyses they require must be required by the first loop
|
|
|
|
// pass in the manager (so that it is computed before the loop pass manager
|
|
|
|
// runs) and preserved by all loop pasess in the manager. To make this
|
|
|
|
// reasonably robust, the set needed for most loop passes is maintained here.
|
|
|
|
// If your loop pass requires an analysis not listed here, you will need to
|
|
|
|
// carefully audit the loop pass manager nesting structure that results.
|
|
|
|
AU.addRequired<AAResultsWrapperPass>();
|
|
|
|
AU.addPreserved<AAResultsWrapperPass>();
|
|
|
|
AU.addPreserved<BasicAAWrapperPass>();
|
|
|
|
AU.addPreserved<GlobalsAAWrapperPass>();
|
|
|
|
AU.addPreserved<SCEVAAWrapperPass>();
|
|
|
|
AU.addRequired<ScalarEvolutionWrapperPass>();
|
|
|
|
AU.addPreserved<ScalarEvolutionWrapperPass>();
|
2019-09-05 03:16:04 +08:00
|
|
|
// FIXME: When all loop passes preserve MemorySSA, it can be required and
|
|
|
|
// preserved here instead of the individual handling in each pass.
|
[LPM] Factor all of the loop analysis usage updates into a common helper
routine.
We were getting this wrong in small ways and generally being very
inconsistent about it across loop passes. Instead, let's have a common
place where we do this. One minor downside is that this will require
some analyses like SCEV in more places than they are strictly needed.
However, this seems benign as these analyses are complete no-ops, and
without this consistency we can in many cases end up with the legacy
pass manager scheduling deciding to split up a loop pass pipeline in
order to run the function analysis half-way through. It is very, very
annoying to fix these without just being very pedantic across the board.
The only loop passes I've not updated here are ones that use
AU.setPreservesAll() such as IVUsers (an analysis) and the pass printer.
They seemed less relevant.
With this patch, almost all of the problems in PR24804 around loop pass
pipelines are fixed. The one remaining issue is that we run simplify-cfg
and instcombine in the middle of the loop pass pipeline. We've recently
added some loop variants of these passes that would seem substantially
cleaner to use, but this at least gets us much closer to the previous
state. Notably, the seven loop pass managers is down to three.
I've not updated the loop passes using LoopAccessAnalysis because that
analysis hasn't been fully wired into LoopSimplify/LCSSA, and it isn't
clear that those transforms want to support those forms anyways. They
all run late anyways, so this is harmless. Similarly, LSR is left alone
because it already carefully manages its forms and doesn't need to get
fused into a single loop pass manager with a bunch of other loop passes.
LoopReroll didn't use loop simplified form previously, and I've updated
the test case to match the trivially different output.
Finally, I've also factored all the pass initialization for the passes
that use this technique as well, so that should be done regularly and
reliably.
Thanks to James for the help reviewing and thinking about this stuff,
and Ben for help thinking about it as well!
Differential Revision: http://reviews.llvm.org/D17435
llvm-svn: 261316
2016-02-19 18:45:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Manually defined generic "LoopPass" dependency initialization. This is used
|
|
|
|
/// to initialize the exact set of passes from above in \c
|
|
|
|
/// getLoopAnalysisUsage. It can be used within a loop pass's initialization
|
|
|
|
/// with:
|
|
|
|
///
|
|
|
|
/// INITIALIZE_PASS_DEPENDENCY(LoopPass)
|
|
|
|
///
|
|
|
|
/// As-if "LoopPass" were a pass.
|
|
|
|
void llvm::initializeLoopPassPass(PassRegistry &Registry) {
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(DominatorTreeWrapperPass)
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(LoopInfoWrapperPass)
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(LoopSimplify)
|
2016-06-10 03:44:46 +08:00
|
|
|
INITIALIZE_PASS_DEPENDENCY(LCSSAWrapperPass)
|
[LPM] Factor all of the loop analysis usage updates into a common helper
routine.
We were getting this wrong in small ways and generally being very
inconsistent about it across loop passes. Instead, let's have a common
place where we do this. One minor downside is that this will require
some analyses like SCEV in more places than they are strictly needed.
However, this seems benign as these analyses are complete no-ops, and
without this consistency we can in many cases end up with the legacy
pass manager scheduling deciding to split up a loop pass pipeline in
order to run the function analysis half-way through. It is very, very
annoying to fix these without just being very pedantic across the board.
The only loop passes I've not updated here are ones that use
AU.setPreservesAll() such as IVUsers (an analysis) and the pass printer.
They seemed less relevant.
With this patch, almost all of the problems in PR24804 around loop pass
pipelines are fixed. The one remaining issue is that we run simplify-cfg
and instcombine in the middle of the loop pass pipeline. We've recently
added some loop variants of these passes that would seem substantially
cleaner to use, but this at least gets us much closer to the previous
state. Notably, the seven loop pass managers is down to three.
I've not updated the loop passes using LoopAccessAnalysis because that
analysis hasn't been fully wired into LoopSimplify/LCSSA, and it isn't
clear that those transforms want to support those forms anyways. They
all run late anyways, so this is harmless. Similarly, LSR is left alone
because it already carefully manages its forms and doesn't need to get
fused into a single loop pass manager with a bunch of other loop passes.
LoopReroll didn't use loop simplified form previously, and I've updated
the test case to match the trivially different output.
Finally, I've also factored all the pass initialization for the passes
that use this technique as well, so that should be done regularly and
reliably.
Thanks to James for the help reviewing and thinking about this stuff,
and Ben for help thinking about it as well!
Differential Revision: http://reviews.llvm.org/D17435
llvm-svn: 261316
2016-02-19 18:45:18 +08:00
|
|
|
INITIALIZE_PASS_DEPENDENCY(AAResultsWrapperPass)
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(BasicAAWrapperPass)
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(GlobalsAAWrapperPass)
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(SCEVAAWrapperPass)
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(ScalarEvolutionWrapperPass)
|
2019-09-05 03:16:04 +08:00
|
|
|
INITIALIZE_PASS_DEPENDENCY(MemorySSAWrapperPass)
|
[LPM] Factor all of the loop analysis usage updates into a common helper
routine.
We were getting this wrong in small ways and generally being very
inconsistent about it across loop passes. Instead, let's have a common
place where we do this. One minor downside is that this will require
some analyses like SCEV in more places than they are strictly needed.
However, this seems benign as these analyses are complete no-ops, and
without this consistency we can in many cases end up with the legacy
pass manager scheduling deciding to split up a loop pass pipeline in
order to run the function analysis half-way through. It is very, very
annoying to fix these without just being very pedantic across the board.
The only loop passes I've not updated here are ones that use
AU.setPreservesAll() such as IVUsers (an analysis) and the pass printer.
They seemed less relevant.
With this patch, almost all of the problems in PR24804 around loop pass
pipelines are fixed. The one remaining issue is that we run simplify-cfg
and instcombine in the middle of the loop pass pipeline. We've recently
added some loop variants of these passes that would seem substantially
cleaner to use, but this at least gets us much closer to the previous
state. Notably, the seven loop pass managers is down to three.
I've not updated the loop passes using LoopAccessAnalysis because that
analysis hasn't been fully wired into LoopSimplify/LCSSA, and it isn't
clear that those transforms want to support those forms anyways. They
all run late anyways, so this is harmless. Similarly, LSR is left alone
because it already carefully manages its forms and doesn't need to get
fused into a single loop pass manager with a bunch of other loop passes.
LoopReroll didn't use loop simplified form previously, and I've updated
the test case to match the trivially different output.
Finally, I've also factored all the pass initialization for the passes
that use this technique as well, so that should be done regularly and
reliably.
Thanks to James for the help reviewing and thinking about this stuff,
and Ben for help thinking about it as well!
Differential Revision: http://reviews.llvm.org/D17435
llvm-svn: 261316
2016-02-19 18:45:18 +08:00
|
|
|
}
|
2016-04-22 01:33:17 +08:00
|
|
|
|
2019-07-26 14:10:08 +08:00
|
|
|
/// Create MDNode for input string.
|
|
|
|
static MDNode *createStringMetadata(Loop *TheLoop, StringRef Name, unsigned V) {
|
|
|
|
LLVMContext &Context = TheLoop->getHeader()->getContext();
|
|
|
|
Metadata *MDs[] = {
|
|
|
|
MDString::get(Context, Name),
|
|
|
|
ConstantAsMetadata::get(ConstantInt::get(Type::getInt32Ty(Context), V))};
|
|
|
|
return MDNode::get(Context, MDs);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Set input string into loop metadata by keeping other values intact.
|
2019-07-26 15:04:34 +08:00
|
|
|
/// If the string is already in loop metadata update value if it is
|
|
|
|
/// different.
|
|
|
|
void llvm::addStringMetadataToLoop(Loop *TheLoop, const char *StringMD,
|
2019-07-26 14:10:08 +08:00
|
|
|
unsigned V) {
|
|
|
|
SmallVector<Metadata *, 4> MDs(1);
|
|
|
|
// If the loop already has metadata, retain it.
|
|
|
|
MDNode *LoopID = TheLoop->getLoopID();
|
|
|
|
if (LoopID) {
|
|
|
|
for (unsigned i = 1, ie = LoopID->getNumOperands(); i < ie; ++i) {
|
|
|
|
MDNode *Node = cast<MDNode>(LoopID->getOperand(i));
|
2019-07-26 15:04:34 +08:00
|
|
|
// If it is of form key = value, try to parse it.
|
|
|
|
if (Node->getNumOperands() == 2) {
|
|
|
|
MDString *S = dyn_cast<MDString>(Node->getOperand(0));
|
|
|
|
if (S && S->getString().equals(StringMD)) {
|
|
|
|
ConstantInt *IntMD =
|
|
|
|
mdconst::extract_or_null<ConstantInt>(Node->getOperand(1));
|
|
|
|
if (IntMD && IntMD->getSExtValue() == V)
|
|
|
|
// It is already in place. Do nothing.
|
|
|
|
return;
|
|
|
|
// We need to update the value, so just skip it here and it will
|
|
|
|
// be added after copying other existed nodes.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2019-07-26 14:10:08 +08:00
|
|
|
MDs.push_back(Node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Add new metadata.
|
2019-07-26 15:04:34 +08:00
|
|
|
MDs.push_back(createStringMetadata(TheLoop, StringMD, V));
|
2019-07-26 14:10:08 +08:00
|
|
|
// Replace current metadata node with new one.
|
|
|
|
LLVMContext &Context = TheLoop->getHeader()->getContext();
|
|
|
|
MDNode *NewLoopID = MDNode::get(Context, MDs);
|
|
|
|
// Set operand 0 to refer to the loop id itself.
|
|
|
|
NewLoopID->replaceOperandWith(0, NewLoopID);
|
|
|
|
TheLoop->setLoopID(NewLoopID);
|
|
|
|
}
|
|
|
|
|
[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
|
|
|
/// Find string metadata for loop
|
|
|
|
///
|
|
|
|
/// If it has a value (e.g. {"llvm.distribute", 1} return the value as an
|
|
|
|
/// operand or null otherwise. If the string metadata is not found return
|
|
|
|
/// Optional's not-a-value.
|
2018-12-20 12:58:07 +08:00
|
|
|
Optional<const MDOperand *> llvm::findStringMetadataForLoop(const Loop *TheLoop,
|
[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
|
|
|
StringRef Name) {
|
2018-12-20 12:58:07 +08:00
|
|
|
MDNode *MD = findOptionMDForLoop(TheLoop, Name);
|
[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 (!MD)
|
|
|
|
return None;
|
|
|
|
switch (MD->getNumOperands()) {
|
|
|
|
case 1:
|
|
|
|
return nullptr;
|
|
|
|
case 2:
|
|
|
|
return &MD->getOperand(1);
|
|
|
|
default:
|
|
|
|
llvm_unreachable("loop metadata has 0 or 1 operand");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static Optional<bool> getOptionalBoolLoopAttribute(const Loop *TheLoop,
|
|
|
|
StringRef Name) {
|
2018-12-20 12:58:07 +08:00
|
|
|
MDNode *MD = findOptionMDForLoop(TheLoop, Name);
|
|
|
|
if (!MD)
|
[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
|
|
|
return None;
|
2018-12-20 12:58:07 +08:00
|
|
|
switch (MD->getNumOperands()) {
|
[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
|
|
|
case 1:
|
|
|
|
// When the value is absent it is interpreted as 'attribute set'.
|
|
|
|
return true;
|
|
|
|
case 2:
|
2019-01-30 06:33:20 +08:00
|
|
|
if (ConstantInt *IntMD =
|
|
|
|
mdconst::extract_or_null<ConstantInt>(MD->getOperand(1).get()))
|
|
|
|
return IntMD->getZExtValue();
|
|
|
|
return true;
|
[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
|
|
|
}
|
|
|
|
llvm_unreachable("unexpected number of options");
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool getBooleanLoopAttribute(const Loop *TheLoop, StringRef Name) {
|
|
|
|
return getOptionalBoolLoopAttribute(TheLoop, Name).getValueOr(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::Optional<int> llvm::getOptionalIntLoopAttribute(Loop *TheLoop,
|
|
|
|
StringRef Name) {
|
|
|
|
const MDOperand *AttrMD =
|
|
|
|
findStringMetadataForLoop(TheLoop, Name).getValueOr(nullptr);
|
|
|
|
if (!AttrMD)
|
|
|
|
return None;
|
|
|
|
|
|
|
|
ConstantInt *IntMD = mdconst::extract_or_null<ConstantInt>(AttrMD->get());
|
|
|
|
if (!IntMD)
|
|
|
|
return None;
|
|
|
|
|
|
|
|
return IntMD->getSExtValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
Optional<MDNode *> llvm::makeFollowupLoopID(
|
|
|
|
MDNode *OrigLoopID, ArrayRef<StringRef> FollowupOptions,
|
|
|
|
const char *InheritOptionsExceptPrefix, bool AlwaysNew) {
|
|
|
|
if (!OrigLoopID) {
|
|
|
|
if (AlwaysNew)
|
|
|
|
return nullptr;
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(OrigLoopID->getOperand(0) == OrigLoopID);
|
|
|
|
|
|
|
|
bool InheritAllAttrs = !InheritOptionsExceptPrefix;
|
|
|
|
bool InheritSomeAttrs =
|
|
|
|
InheritOptionsExceptPrefix && InheritOptionsExceptPrefix[0] != '\0';
|
|
|
|
SmallVector<Metadata *, 8> MDs;
|
|
|
|
MDs.push_back(nullptr);
|
|
|
|
|
|
|
|
bool Changed = false;
|
|
|
|
if (InheritAllAttrs || InheritSomeAttrs) {
|
|
|
|
for (const MDOperand &Existing : drop_begin(OrigLoopID->operands(), 1)) {
|
|
|
|
MDNode *Op = cast<MDNode>(Existing.get());
|
|
|
|
|
|
|
|
auto InheritThisAttribute = [InheritSomeAttrs,
|
|
|
|
InheritOptionsExceptPrefix](MDNode *Op) {
|
|
|
|
if (!InheritSomeAttrs)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Skip malformatted attribute metadata nodes.
|
|
|
|
if (Op->getNumOperands() == 0)
|
|
|
|
return true;
|
|
|
|
Metadata *NameMD = Op->getOperand(0).get();
|
|
|
|
if (!isa<MDString>(NameMD))
|
|
|
|
return true;
|
|
|
|
StringRef AttrName = cast<MDString>(NameMD)->getString();
|
|
|
|
|
|
|
|
// Do not inherit excluded attributes.
|
|
|
|
return !AttrName.startswith(InheritOptionsExceptPrefix);
|
|
|
|
};
|
|
|
|
|
|
|
|
if (InheritThisAttribute(Op))
|
|
|
|
MDs.push_back(Op);
|
|
|
|
else
|
|
|
|
Changed = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Modified if we dropped at least one attribute.
|
|
|
|
Changed = OrigLoopID->getNumOperands() > 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool HasAnyFollowup = false;
|
|
|
|
for (StringRef OptionName : FollowupOptions) {
|
2018-12-20 12:58:07 +08:00
|
|
|
MDNode *FollowupNode = findOptionMDForLoopID(OrigLoopID, OptionName);
|
[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 (!FollowupNode)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
HasAnyFollowup = true;
|
|
|
|
for (const MDOperand &Option : drop_begin(FollowupNode->operands(), 1)) {
|
|
|
|
MDs.push_back(Option.get());
|
|
|
|
Changed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attributes of the followup loop not specified explicity, so signal to the
|
|
|
|
// transformation pass to add suitable attributes.
|
|
|
|
if (!AlwaysNew && !HasAnyFollowup)
|
|
|
|
return None;
|
|
|
|
|
|
|
|
// If no attributes were added or remove, the previous loop Id can be reused.
|
|
|
|
if (!AlwaysNew && !Changed)
|
|
|
|
return OrigLoopID;
|
|
|
|
|
|
|
|
// No attributes is equivalent to having no !llvm.loop metadata at all.
|
|
|
|
if (MDs.size() == 1)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
// Build the new loop ID.
|
|
|
|
MDTuple *FollowupLoopID = MDNode::get(OrigLoopID->getContext(), MDs);
|
|
|
|
FollowupLoopID->replaceOperandWith(0, FollowupLoopID);
|
|
|
|
return FollowupLoopID;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool llvm::hasDisableAllTransformsHint(const Loop *L) {
|
|
|
|
return getBooleanLoopAttribute(L, LLVMLoopDisableNonforced);
|
|
|
|
}
|
|
|
|
|
2019-08-08 21:46:17 +08:00
|
|
|
bool llvm::hasDisableLICMTransformsHint(const Loop *L) {
|
|
|
|
return getBooleanLoopAttribute(L, LLVMLoopDisableLICM);
|
|
|
|
}
|
|
|
|
|
[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
|
|
|
TransformationMode llvm::hasUnrollTransformation(Loop *L) {
|
|
|
|
if (getBooleanLoopAttribute(L, "llvm.loop.unroll.disable"))
|
|
|
|
return TM_SuppressedByUser;
|
|
|
|
|
|
|
|
Optional<int> Count =
|
|
|
|
getOptionalIntLoopAttribute(L, "llvm.loop.unroll.count");
|
|
|
|
if (Count.hasValue())
|
|
|
|
return Count.getValue() == 1 ? TM_SuppressedByUser : TM_ForcedByUser;
|
|
|
|
|
|
|
|
if (getBooleanLoopAttribute(L, "llvm.loop.unroll.enable"))
|
|
|
|
return TM_ForcedByUser;
|
|
|
|
|
|
|
|
if (getBooleanLoopAttribute(L, "llvm.loop.unroll.full"))
|
|
|
|
return TM_ForcedByUser;
|
|
|
|
|
|
|
|
if (hasDisableAllTransformsHint(L))
|
|
|
|
return TM_Disable;
|
|
|
|
|
|
|
|
return TM_Unspecified;
|
|
|
|
}
|
|
|
|
|
|
|
|
TransformationMode llvm::hasUnrollAndJamTransformation(Loop *L) {
|
|
|
|
if (getBooleanLoopAttribute(L, "llvm.loop.unroll_and_jam.disable"))
|
|
|
|
return TM_SuppressedByUser;
|
|
|
|
|
|
|
|
Optional<int> Count =
|
|
|
|
getOptionalIntLoopAttribute(L, "llvm.loop.unroll_and_jam.count");
|
|
|
|
if (Count.hasValue())
|
|
|
|
return Count.getValue() == 1 ? TM_SuppressedByUser : TM_ForcedByUser;
|
|
|
|
|
|
|
|
if (getBooleanLoopAttribute(L, "llvm.loop.unroll_and_jam.enable"))
|
|
|
|
return TM_ForcedByUser;
|
|
|
|
|
|
|
|
if (hasDisableAllTransformsHint(L))
|
|
|
|
return TM_Disable;
|
|
|
|
|
|
|
|
return TM_Unspecified;
|
|
|
|
}
|
|
|
|
|
|
|
|
TransformationMode llvm::hasVectorizeTransformation(Loop *L) {
|
|
|
|
Optional<bool> Enable =
|
|
|
|
getOptionalBoolLoopAttribute(L, "llvm.loop.vectorize.enable");
|
|
|
|
|
|
|
|
if (Enable == false)
|
|
|
|
return TM_SuppressedByUser;
|
|
|
|
|
|
|
|
Optional<int> VectorizeWidth =
|
|
|
|
getOptionalIntLoopAttribute(L, "llvm.loop.vectorize.width");
|
|
|
|
Optional<int> InterleaveCount =
|
|
|
|
getOptionalIntLoopAttribute(L, "llvm.loop.interleave.count");
|
|
|
|
|
2019-02-05 03:55:59 +08:00
|
|
|
// 'Forcing' vector width and interleave count to one effectively disables
|
|
|
|
// this tranformation.
|
|
|
|
if (Enable == true && VectorizeWidth == 1 && InterleaveCount == 1)
|
|
|
|
return TM_SuppressedByUser;
|
[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 (getBooleanLoopAttribute(L, "llvm.loop.isvectorized"))
|
|
|
|
return TM_Disable;
|
|
|
|
|
2019-02-05 03:55:59 +08:00
|
|
|
if (Enable == true)
|
|
|
|
return TM_ForcedByUser;
|
|
|
|
|
[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 (VectorizeWidth == 1 && InterleaveCount == 1)
|
|
|
|
return TM_Disable;
|
|
|
|
|
|
|
|
if (VectorizeWidth > 1 || InterleaveCount > 1)
|
|
|
|
return TM_Enable;
|
|
|
|
|
|
|
|
if (hasDisableAllTransformsHint(L))
|
|
|
|
return TM_Disable;
|
|
|
|
|
|
|
|
return TM_Unspecified;
|
|
|
|
}
|
|
|
|
|
|
|
|
TransformationMode llvm::hasDistributeTransformation(Loop *L) {
|
|
|
|
if (getBooleanLoopAttribute(L, "llvm.loop.distribute.enable"))
|
|
|
|
return TM_ForcedByUser;
|
|
|
|
|
|
|
|
if (hasDisableAllTransformsHint(L))
|
|
|
|
return TM_Disable;
|
|
|
|
|
|
|
|
return TM_Unspecified;
|
|
|
|
}
|
|
|
|
|
|
|
|
TransformationMode llvm::hasLICMVersioningTransformation(Loop *L) {
|
|
|
|
if (getBooleanLoopAttribute(L, "llvm.loop.licm_versioning.disable"))
|
|
|
|
return TM_SuppressedByUser;
|
|
|
|
|
|
|
|
if (hasDisableAllTransformsHint(L))
|
|
|
|
return TM_Disable;
|
|
|
|
|
|
|
|
return TM_Unspecified;
|
|
|
|
}
|
|
|
|
|
2017-09-15 08:04:16 +08:00
|
|
|
/// Does a BFS from a given node to all of its children inside a given loop.
|
|
|
|
/// The returned vector of nodes includes the starting point.
|
|
|
|
SmallVector<DomTreeNode *, 16>
|
|
|
|
llvm::collectChildrenInLoop(DomTreeNode *N, const Loop *CurLoop) {
|
|
|
|
SmallVector<DomTreeNode *, 16> Worklist;
|
|
|
|
auto AddRegionToWorklist = [&](DomTreeNode *DTN) {
|
|
|
|
// Only include subregions in the top level loop.
|
|
|
|
BasicBlock *BB = DTN->getBlock();
|
|
|
|
if (CurLoop->contains(BB))
|
|
|
|
Worklist.push_back(DTN);
|
|
|
|
};
|
|
|
|
|
|
|
|
AddRegionToWorklist(N);
|
|
|
|
|
|
|
|
for (size_t I = 0; I < Worklist.size(); I++)
|
|
|
|
for (DomTreeNode *Child : Worklist[I]->getChildren())
|
|
|
|
AddRegionToWorklist(Child);
|
|
|
|
|
|
|
|
return Worklist;
|
|
|
|
}
|
|
|
|
|
2017-10-05 04:42:46 +08:00
|
|
|
void llvm::deleteDeadLoop(Loop *L, DominatorTree *DT = nullptr,
|
|
|
|
ScalarEvolution *SE = nullptr,
|
|
|
|
LoopInfo *LI = nullptr) {
|
2017-10-05 05:14:07 +08:00
|
|
|
assert((!DT || L->isLCSSAForm(*DT)) && "Expected LCSSA!");
|
2017-10-05 04:42:46 +08:00
|
|
|
auto *Preheader = L->getLoopPreheader();
|
|
|
|
assert(Preheader && "Preheader should exist!");
|
|
|
|
|
|
|
|
// Now that we know the removal is safe, remove the loop by changing the
|
|
|
|
// branch from the preheader to go to the single exit block.
|
|
|
|
//
|
|
|
|
// Because we're deleting a large chunk of code at once, the sequence in which
|
|
|
|
// we remove things is very important to avoid invalidation issues.
|
|
|
|
|
|
|
|
// Tell ScalarEvolution that the loop is deleted. Do this before
|
|
|
|
// deleting the loop so that ScalarEvolution can look at the loop
|
|
|
|
// to determine what it needs to clean up.
|
|
|
|
if (SE)
|
|
|
|
SE->forgetLoop(L);
|
|
|
|
|
|
|
|
auto *ExitBlock = L->getUniqueExitBlock();
|
|
|
|
assert(ExitBlock && "Should have a unique exit block!");
|
|
|
|
assert(L->hasDedicatedExits() && "Loop should have dedicated exits!");
|
|
|
|
|
|
|
|
auto *OldBr = dyn_cast<BranchInst>(Preheader->getTerminator());
|
|
|
|
assert(OldBr && "Preheader must end with a branch");
|
|
|
|
assert(OldBr->isUnconditional() && "Preheader must have a single successor");
|
|
|
|
// Connect the preheader to the exit block. Keep the old edge to the header
|
|
|
|
// around to perform the dominator tree update in two separate steps
|
|
|
|
// -- #1 insertion of the edge preheader -> exit and #2 deletion of the edge
|
|
|
|
// preheader -> header.
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// 0. Preheader 1. Preheader 2. Preheader
|
|
|
|
// | | | |
|
|
|
|
// V | V |
|
|
|
|
// Header <--\ | Header <--\ | Header <--\
|
|
|
|
// | | | | | | | | | | |
|
|
|
|
// | V | | | V | | | V |
|
|
|
|
// | Body --/ | | Body --/ | | Body --/
|
|
|
|
// V V V V V
|
|
|
|
// Exit Exit Exit
|
|
|
|
//
|
|
|
|
// By doing this is two separate steps we can perform the dominator tree
|
|
|
|
// update without using the batch update API.
|
|
|
|
//
|
|
|
|
// Even when the loop is never executed, we cannot remove the edge from the
|
|
|
|
// source block to the exit block. Consider the case where the unexecuted loop
|
|
|
|
// branches back to an outer loop. If we deleted the loop and removed the edge
|
|
|
|
// coming to this inner loop, this will break the outer loop structure (by
|
|
|
|
// deleting the backedge of the outer loop). If the outer loop is indeed a
|
|
|
|
// non-loop, it will be deleted in a future iteration of loop deletion pass.
|
|
|
|
IRBuilder<> Builder(OldBr);
|
|
|
|
Builder.CreateCondBr(Builder.getFalse(), L->getHeader(), ExitBlock);
|
|
|
|
// Remove the old branch. The conditional branch becomes a new terminator.
|
|
|
|
OldBr->eraseFromParent();
|
|
|
|
|
|
|
|
// Rewrite phis in the exit block to get their inputs from the Preheader
|
|
|
|
// instead of the exiting block.
|
2017-12-30 23:27:33 +08:00
|
|
|
for (PHINode &P : ExitBlock->phis()) {
|
2017-10-05 04:42:46 +08:00
|
|
|
// Set the zero'th element of Phi to be from the preheader and remove all
|
|
|
|
// other incoming values. Given the loop has dedicated exits, all other
|
|
|
|
// incoming values must be from the exiting blocks.
|
|
|
|
int PredIndex = 0;
|
2017-12-30 23:27:33 +08:00
|
|
|
P.setIncomingBlock(PredIndex, Preheader);
|
2017-10-05 04:42:46 +08:00
|
|
|
// Removes all incoming values from all other exiting blocks (including
|
|
|
|
// duplicate values from an exiting block).
|
|
|
|
// Nuke all entries except the zero'th entry which is the preheader entry.
|
|
|
|
// NOTE! We need to remove Incoming Values in the reverse order as done
|
|
|
|
// below, to keep the indices valid for deletion (removeIncomingValues
|
|
|
|
// updates getNumIncomingValues and shifts all values down into the operand
|
|
|
|
// being deleted).
|
2017-12-30 23:27:33 +08:00
|
|
|
for (unsigned i = 0, e = P.getNumIncomingValues() - 1; i != e; ++i)
|
|
|
|
P.removeIncomingValue(e - i, false);
|
2017-10-05 04:42:46 +08:00
|
|
|
|
2017-12-30 23:27:33 +08:00
|
|
|
assert((P.getNumIncomingValues() == 1 &&
|
|
|
|
P.getIncomingBlock(PredIndex) == Preheader) &&
|
2017-10-05 04:42:46 +08:00
|
|
|
"Should have exactly one value and that's from the preheader!");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Disconnect the loop body by branching directly to its exit.
|
|
|
|
Builder.SetInsertPoint(Preheader->getTerminator());
|
|
|
|
Builder.CreateBr(ExitBlock);
|
|
|
|
// Remove the old branch.
|
|
|
|
Preheader->getTerminator()->eraseFromParent();
|
|
|
|
|
2018-08-03 13:08:17 +08:00
|
|
|
DomTreeUpdater DTU(DT, DomTreeUpdater::UpdateStrategy::Eager);
|
2017-10-05 04:42:46 +08:00
|
|
|
if (DT) {
|
|
|
|
// Update the dominator tree by informing it about the new edge from the
|
2019-02-22 13:41:43 +08:00
|
|
|
// preheader to the exit and the removed edge.
|
|
|
|
DTU.applyUpdates({{DominatorTree::Insert, Preheader, ExitBlock},
|
|
|
|
{DominatorTree::Delete, Preheader, L->getHeader()}});
|
2017-10-05 04:42:46 +08:00
|
|
|
}
|
|
|
|
|
2018-12-13 07:32:35 +08:00
|
|
|
// Use a map to unique and a vector to guarantee deterministic ordering.
|
2018-12-13 09:11:52 +08:00
|
|
|
llvm::SmallDenseSet<std::pair<DIVariable *, DIExpression *>, 4> DeadDebugSet;
|
2018-12-13 07:32:35 +08:00
|
|
|
llvm::SmallVector<DbgVariableIntrinsic *, 4> DeadDebugInst;
|
|
|
|
|
2018-01-12 15:24:43 +08:00
|
|
|
// Given LCSSA form is satisfied, we should not have users of instructions
|
|
|
|
// within the dead loop outside of the loop. However, LCSSA doesn't take
|
|
|
|
// unreachable uses into account. We handle them here.
|
|
|
|
// We could do it after drop all references (in this case all users in the
|
|
|
|
// loop will be already eliminated and we have less work to do but according
|
|
|
|
// to API doc of User::dropAllReferences only valid operation after dropping
|
|
|
|
// references, is deletion. So let's substitute all usages of
|
|
|
|
// instruction from the loop with undef value of corresponding type first.
|
|
|
|
for (auto *Block : L->blocks())
|
|
|
|
for (Instruction &I : *Block) {
|
|
|
|
auto *Undef = UndefValue::get(I.getType());
|
|
|
|
for (Value::use_iterator UI = I.use_begin(), E = I.use_end(); UI != E;) {
|
|
|
|
Use &U = *UI;
|
|
|
|
++UI;
|
|
|
|
if (auto *Usr = dyn_cast<Instruction>(U.getUser()))
|
|
|
|
if (L->contains(Usr->getParent()))
|
|
|
|
continue;
|
|
|
|
// If we have a DT then we can check that uses outside a loop only in
|
|
|
|
// unreachable block.
|
|
|
|
if (DT)
|
|
|
|
assert(!DT->isReachableFromEntry(U) &&
|
|
|
|
"Unexpected user in reachable block");
|
|
|
|
U.set(Undef);
|
|
|
|
}
|
2018-12-13 07:32:35 +08:00
|
|
|
auto *DVI = dyn_cast<DbgVariableIntrinsic>(&I);
|
|
|
|
if (!DVI)
|
|
|
|
continue;
|
2018-12-13 09:11:52 +08:00
|
|
|
auto Key = DeadDebugSet.find({DVI->getVariable(), DVI->getExpression()});
|
|
|
|
if (Key != DeadDebugSet.end())
|
2018-12-13 07:32:35 +08:00
|
|
|
continue;
|
2018-12-13 09:11:52 +08:00
|
|
|
DeadDebugSet.insert({DVI->getVariable(), DVI->getExpression()});
|
2018-12-13 07:32:35 +08:00
|
|
|
DeadDebugInst.push_back(DVI);
|
2018-01-12 15:24:43 +08:00
|
|
|
}
|
|
|
|
|
2018-12-13 07:32:35 +08:00
|
|
|
// After the loop has been deleted all the values defined and modified
|
|
|
|
// inside the loop are going to be unavailable.
|
|
|
|
// Since debug values in the loop have been deleted, inserting an undef
|
|
|
|
// dbg.value truncates the range of any dbg.value before the loop where the
|
|
|
|
// loop used to be. This is particularly important for constant values.
|
|
|
|
DIBuilder DIB(*ExitBlock->getModule());
|
2019-05-06 02:59:12 +08:00
|
|
|
Instruction *InsertDbgValueBefore = ExitBlock->getFirstNonPHI();
|
|
|
|
assert(InsertDbgValueBefore &&
|
|
|
|
"There should be a non-PHI instruction in exit block, else these "
|
|
|
|
"instructions will have no parent.");
|
2018-12-13 07:32:35 +08:00
|
|
|
for (auto *DVI : DeadDebugInst)
|
2019-05-06 02:59:12 +08:00
|
|
|
DIB.insertDbgValueIntrinsic(UndefValue::get(Builder.getInt32Ty()),
|
|
|
|
DVI->getVariable(), DVI->getExpression(),
|
|
|
|
DVI->getDebugLoc(), InsertDbgValueBefore);
|
2018-12-13 07:32:35 +08:00
|
|
|
|
2017-10-05 04:42:46 +08:00
|
|
|
// Remove the block from the reference counting scheme, so that we can
|
|
|
|
// delete it freely later.
|
|
|
|
for (auto *Block : L->blocks())
|
|
|
|
Block->dropAllReferences();
|
|
|
|
|
|
|
|
if (LI) {
|
|
|
|
// Erase the instructions and the blocks without having to worry
|
|
|
|
// about ordering because we already dropped the references.
|
|
|
|
// NOTE: This iteration is safe because erasing the block does not remove
|
|
|
|
// its entry from the loop's block list. We do that in the next section.
|
|
|
|
for (Loop::block_iterator LpI = L->block_begin(), LpE = L->block_end();
|
|
|
|
LpI != LpE; ++LpI)
|
|
|
|
(*LpI)->eraseFromParent();
|
|
|
|
|
|
|
|
// Finally, the blocks from loopinfo. This has to happen late because
|
|
|
|
// otherwise our loop iterators won't work.
|
|
|
|
|
|
|
|
SmallPtrSet<BasicBlock *, 8> blocks;
|
|
|
|
blocks.insert(L->block_begin(), L->block_end());
|
|
|
|
for (BasicBlock *BB : blocks)
|
|
|
|
LI->removeBlock(BB);
|
|
|
|
|
|
|
|
// The last step is to update LoopInfo now that we've eliminated this loop.
|
|
|
|
LI->erase(L);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-17 09:17:02 +08:00
|
|
|
Optional<unsigned> llvm::getLoopEstimatedTripCount(Loop *L) {
|
2019-07-15 14:42:39 +08:00
|
|
|
// Support loops with an exiting latch and other existing exists only
|
|
|
|
// deoptimize.
|
2016-11-17 09:17:02 +08:00
|
|
|
|
2018-01-19 18:55:29 +08:00
|
|
|
// Get the branch weights for the loop's backedge.
|
2019-07-15 14:42:39 +08:00
|
|
|
BasicBlock *Latch = L->getLoopLatch();
|
|
|
|
if (!Latch)
|
|
|
|
return None;
|
|
|
|
BranchInst *LatchBR = dyn_cast<BranchInst>(Latch->getTerminator());
|
|
|
|
if (!LatchBR || LatchBR->getNumSuccessors() != 2 || !L->isLoopExiting(Latch))
|
2016-11-17 09:17:02 +08:00
|
|
|
return None;
|
|
|
|
|
|
|
|
assert((LatchBR->getSuccessor(0) == L->getHeader() ||
|
|
|
|
LatchBR->getSuccessor(1) == L->getHeader()) &&
|
|
|
|
"At least one edge out of the latch must go to the header");
|
|
|
|
|
2019-07-15 14:42:39 +08:00
|
|
|
SmallVector<BasicBlock *, 4> ExitBlocks;
|
|
|
|
L->getUniqueNonLatchExitBlocks(ExitBlocks);
|
|
|
|
if (any_of(ExitBlocks, [](const BasicBlock *EB) {
|
|
|
|
return !EB->getTerminatingDeoptimizeCall();
|
|
|
|
}))
|
|
|
|
return None;
|
|
|
|
|
2016-11-17 09:17:02 +08:00
|
|
|
// To estimate the number of times the loop body was executed, we want to
|
|
|
|
// know the number of times the backedge was taken, vs. the number of times
|
|
|
|
// we exited the loop.
|
|
|
|
uint64_t TrueVal, FalseVal;
|
2016-12-01 05:13:57 +08:00
|
|
|
if (!LatchBR->extractProfMetadata(TrueVal, FalseVal))
|
2016-11-17 09:17:02 +08:00
|
|
|
return None;
|
|
|
|
|
2016-12-01 05:13:57 +08:00
|
|
|
if (!TrueVal || !FalseVal)
|
|
|
|
return 0;
|
2016-11-17 09:17:02 +08:00
|
|
|
|
2016-12-01 05:13:57 +08:00
|
|
|
// Divide the count of the backedge by the count of the edge exiting the loop,
|
|
|
|
// rounding to nearest.
|
2016-11-17 09:17:02 +08:00
|
|
|
if (LatchBR->getSuccessor(0) == L->getHeader())
|
2016-12-01 05:13:57 +08:00
|
|
|
return (TrueVal + (FalseVal / 2)) / FalseVal;
|
2016-11-17 09:17:02 +08:00
|
|
|
else
|
2016-12-01 05:13:57 +08:00
|
|
|
return (FalseVal + (TrueVal / 2)) / TrueVal;
|
2016-11-17 09:17:02 +08:00
|
|
|
}
|
2017-05-09 18:43:25 +08:00
|
|
|
|
2018-08-15 18:59:41 +08:00
|
|
|
bool llvm::hasIterationCountInvariantInParent(Loop *InnerLoop,
|
|
|
|
ScalarEvolution &SE) {
|
2018-08-11 14:57:28 +08:00
|
|
|
Loop *OuterL = InnerLoop->getParentLoop();
|
|
|
|
if (!OuterL)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Get the backedge taken count for the inner loop
|
|
|
|
BasicBlock *InnerLoopLatch = InnerLoop->getLoopLatch();
|
|
|
|
const SCEV *InnerLoopBECountSC = SE.getExitCount(InnerLoop, InnerLoopLatch);
|
|
|
|
if (isa<SCEVCouldNotCompute>(InnerLoopBECountSC) ||
|
|
|
|
!InnerLoopBECountSC->getType()->isIntegerTy())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Get whether count is invariant to the outer loop
|
|
|
|
ScalarEvolution::LoopDisposition LD =
|
|
|
|
SE.getLoopDisposition(InnerLoopBECountSC, OuterL);
|
|
|
|
if (LD != ScalarEvolution::LoopInvariant)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-09-10 13:05:08 +08:00
|
|
|
Value *llvm::createMinMaxOp(IRBuilder<> &Builder,
|
|
|
|
RecurrenceDescriptor::MinMaxRecurrenceKind RK,
|
|
|
|
Value *Left, Value *Right) {
|
|
|
|
CmpInst::Predicate P = CmpInst::ICMP_NE;
|
|
|
|
switch (RK) {
|
|
|
|
default:
|
|
|
|
llvm_unreachable("Unknown min/max recurrence kind");
|
|
|
|
case RecurrenceDescriptor::MRK_UIntMin:
|
|
|
|
P = CmpInst::ICMP_ULT;
|
|
|
|
break;
|
|
|
|
case RecurrenceDescriptor::MRK_UIntMax:
|
|
|
|
P = CmpInst::ICMP_UGT;
|
|
|
|
break;
|
|
|
|
case RecurrenceDescriptor::MRK_SIntMin:
|
|
|
|
P = CmpInst::ICMP_SLT;
|
|
|
|
break;
|
|
|
|
case RecurrenceDescriptor::MRK_SIntMax:
|
|
|
|
P = CmpInst::ICMP_SGT;
|
|
|
|
break;
|
|
|
|
case RecurrenceDescriptor::MRK_FloatMin:
|
|
|
|
P = CmpInst::FCMP_OLT;
|
|
|
|
break;
|
|
|
|
case RecurrenceDescriptor::MRK_FloatMax:
|
|
|
|
P = CmpInst::FCMP_OGT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We only match FP sequences that are 'fast', so we can unconditionally
|
|
|
|
// set it on any generated instructions.
|
|
|
|
IRBuilder<>::FastMathFlagGuard FMFG(Builder);
|
|
|
|
FastMathFlags FMF;
|
|
|
|
FMF.setFast();
|
|
|
|
Builder.setFastMathFlags(FMF);
|
|
|
|
|
|
|
|
Value *Cmp;
|
|
|
|
if (RK == RecurrenceDescriptor::MRK_FloatMin ||
|
|
|
|
RK == RecurrenceDescriptor::MRK_FloatMax)
|
|
|
|
Cmp = Builder.CreateFCmp(P, Left, Right, "rdx.minmax.cmp");
|
|
|
|
else
|
|
|
|
Cmp = Builder.CreateICmp(P, Left, Right, "rdx.minmax.cmp");
|
|
|
|
|
|
|
|
Value *Select = Builder.CreateSelect(Cmp, Left, Right, "rdx.minmax.select");
|
|
|
|
return Select;
|
|
|
|
}
|
|
|
|
|
2018-04-09 23:44:20 +08:00
|
|
|
// Helper to generate an ordered reduction.
|
|
|
|
Value *
|
|
|
|
llvm::getOrderedReduction(IRBuilder<> &Builder, Value *Acc, Value *Src,
|
|
|
|
unsigned Op,
|
|
|
|
RecurrenceDescriptor::MinMaxRecurrenceKind MinMaxKind,
|
|
|
|
ArrayRef<Value *> RedOps) {
|
|
|
|
unsigned VF = Src->getType()->getVectorNumElements();
|
|
|
|
|
|
|
|
// Extract and apply reduction ops in ascending order:
|
|
|
|
// e.g. ((((Acc + Scl[0]) + Scl[1]) + Scl[2]) + ) ... + Scl[VF-1]
|
|
|
|
Value *Result = Acc;
|
|
|
|
for (unsigned ExtractIdx = 0; ExtractIdx != VF; ++ExtractIdx) {
|
|
|
|
Value *Ext =
|
|
|
|
Builder.CreateExtractElement(Src, Builder.getInt32(ExtractIdx));
|
|
|
|
|
|
|
|
if (Op != Instruction::ICmp && Op != Instruction::FCmp) {
|
|
|
|
Result = Builder.CreateBinOp((Instruction::BinaryOps)Op, Result, Ext,
|
|
|
|
"bin.rdx");
|
|
|
|
} else {
|
|
|
|
assert(MinMaxKind != RecurrenceDescriptor::MRK_Invalid &&
|
|
|
|
"Invalid min/max");
|
2018-09-10 13:05:08 +08:00
|
|
|
Result = createMinMaxOp(Builder, MinMaxKind, Result, Ext);
|
2018-04-09 23:44:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!RedOps.empty())
|
|
|
|
propagateIRFlags(Result, RedOps);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2017-05-09 18:43:25 +08:00
|
|
|
// Helper to generate a log2 shuffle reduction.
|
2017-05-10 17:42:49 +08:00
|
|
|
Value *
|
|
|
|
llvm::getShuffleReduction(IRBuilder<> &Builder, Value *Src, unsigned Op,
|
|
|
|
RecurrenceDescriptor::MinMaxRecurrenceKind MinMaxKind,
|
2019-06-05 22:58:04 +08:00
|
|
|
ArrayRef<Value *> RedOps) {
|
2017-05-09 18:43:25 +08:00
|
|
|
unsigned VF = Src->getType()->getVectorNumElements();
|
|
|
|
// VF is a power of 2 so we can emit the reduction using log2(VF) shuffles
|
|
|
|
// and vector ops, reducing the set of values being computed by half each
|
|
|
|
// round.
|
|
|
|
assert(isPowerOf2_32(VF) &&
|
|
|
|
"Reduction emission only supported for pow2 vectors!");
|
|
|
|
Value *TmpVec = Src;
|
|
|
|
SmallVector<Constant *, 32> ShuffleMask(VF, nullptr);
|
|
|
|
for (unsigned i = VF; i != 1; i >>= 1) {
|
|
|
|
// Move the upper half of the vector to the lower half.
|
|
|
|
for (unsigned j = 0; j != i / 2; ++j)
|
|
|
|
ShuffleMask[j] = Builder.getInt32(i / 2 + j);
|
|
|
|
|
|
|
|
// Fill the rest of the mask with undef.
|
|
|
|
std::fill(&ShuffleMask[i / 2], ShuffleMask.end(),
|
|
|
|
UndefValue::get(Builder.getInt32Ty()));
|
|
|
|
|
|
|
|
Value *Shuf = Builder.CreateShuffleVector(
|
|
|
|
TmpVec, UndefValue::get(TmpVec->getType()),
|
|
|
|
ConstantVector::get(ShuffleMask), "rdx.shuf");
|
|
|
|
|
|
|
|
if (Op != Instruction::ICmp && Op != Instruction::FCmp) {
|
2019-06-05 22:58:04 +08:00
|
|
|
// The builder propagates its fast-math-flags setting.
|
|
|
|
TmpVec = Builder.CreateBinOp((Instruction::BinaryOps)Op, TmpVec, Shuf,
|
|
|
|
"bin.rdx");
|
2017-05-09 18:43:25 +08:00
|
|
|
} else {
|
|
|
|
assert(MinMaxKind != RecurrenceDescriptor::MRK_Invalid &&
|
|
|
|
"Invalid min/max");
|
2018-09-10 13:05:08 +08:00
|
|
|
TmpVec = createMinMaxOp(Builder, MinMaxKind, TmpVec, Shuf);
|
2017-05-09 18:43:25 +08:00
|
|
|
}
|
|
|
|
if (!RedOps.empty())
|
|
|
|
propagateIRFlags(TmpVec, RedOps);
|
|
|
|
}
|
|
|
|
// The result is in the first element of the vector.
|
|
|
|
return Builder.CreateExtractElement(TmpVec, Builder.getInt32(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a simple vector reduction specified by an opcode and some
|
|
|
|
/// flags (if generating min/max reductions).
|
|
|
|
Value *llvm::createSimpleTargetReduction(
|
|
|
|
IRBuilder<> &Builder, const TargetTransformInfo *TTI, unsigned Opcode,
|
2019-06-05 22:58:04 +08:00
|
|
|
Value *Src, TargetTransformInfo::ReductionFlags Flags,
|
2017-05-09 18:43:25 +08:00
|
|
|
ArrayRef<Value *> RedOps) {
|
|
|
|
assert(isa<VectorType>(Src->getType()) && "Type must be a vector");
|
|
|
|
|
2018-09-12 09:59:43 +08:00
|
|
|
std::function<Value *()> BuildFunc;
|
2017-05-09 18:43:25 +08:00
|
|
|
using RD = RecurrenceDescriptor;
|
|
|
|
RD::MinMaxRecurrenceKind MinMaxKind = RD::MRK_Invalid;
|
|
|
|
|
|
|
|
switch (Opcode) {
|
|
|
|
case Instruction::Add:
|
|
|
|
BuildFunc = [&]() { return Builder.CreateAddReduce(Src); };
|
|
|
|
break;
|
|
|
|
case Instruction::Mul:
|
|
|
|
BuildFunc = [&]() { return Builder.CreateMulReduce(Src); };
|
|
|
|
break;
|
|
|
|
case Instruction::And:
|
|
|
|
BuildFunc = [&]() { return Builder.CreateAndReduce(Src); };
|
|
|
|
break;
|
|
|
|
case Instruction::Or:
|
|
|
|
BuildFunc = [&]() { return Builder.CreateOrReduce(Src); };
|
|
|
|
break;
|
|
|
|
case Instruction::Xor:
|
|
|
|
BuildFunc = [&]() { return Builder.CreateXorReduce(Src); };
|
|
|
|
break;
|
|
|
|
case Instruction::FAdd:
|
|
|
|
BuildFunc = [&]() {
|
2019-06-11 16:22:10 +08:00
|
|
|
auto Rdx = Builder.CreateFAddReduce(
|
|
|
|
Constant::getNullValue(Src->getType()->getVectorElementType()), Src);
|
2017-05-09 18:43:25 +08:00
|
|
|
return Rdx;
|
|
|
|
};
|
|
|
|
break;
|
|
|
|
case Instruction::FMul:
|
|
|
|
BuildFunc = [&]() {
|
2019-06-11 16:22:10 +08:00
|
|
|
Type *Ty = Src->getType()->getVectorElementType();
|
|
|
|
auto Rdx = Builder.CreateFMulReduce(ConstantFP::get(Ty, 1.0), Src);
|
2017-05-09 18:43:25 +08:00
|
|
|
return Rdx;
|
|
|
|
};
|
|
|
|
break;
|
|
|
|
case Instruction::ICmp:
|
|
|
|
if (Flags.IsMaxOp) {
|
|
|
|
MinMaxKind = Flags.IsSigned ? RD::MRK_SIntMax : RD::MRK_UIntMax;
|
|
|
|
BuildFunc = [&]() {
|
|
|
|
return Builder.CreateIntMaxReduce(Src, Flags.IsSigned);
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
MinMaxKind = Flags.IsSigned ? RD::MRK_SIntMin : RD::MRK_UIntMin;
|
|
|
|
BuildFunc = [&]() {
|
|
|
|
return Builder.CreateIntMinReduce(Src, Flags.IsSigned);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case Instruction::FCmp:
|
|
|
|
if (Flags.IsMaxOp) {
|
|
|
|
MinMaxKind = RD::MRK_FloatMax;
|
|
|
|
BuildFunc = [&]() { return Builder.CreateFPMaxReduce(Src, Flags.NoNaN); };
|
|
|
|
} else {
|
|
|
|
MinMaxKind = RD::MRK_FloatMin;
|
|
|
|
BuildFunc = [&]() { return Builder.CreateFPMinReduce(Src, Flags.NoNaN); };
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
llvm_unreachable("Unhandled opcode");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (TTI->useReductionIntrinsic(Opcode, Src->getType(), Flags))
|
|
|
|
return BuildFunc();
|
2019-06-05 22:58:04 +08:00
|
|
|
return getShuffleReduction(Builder, Src, Opcode, MinMaxKind, RedOps);
|
2017-05-09 18:43:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a vector reduction using a given recurrence descriptor.
|
2017-12-07 03:37:00 +08:00
|
|
|
Value *llvm::createTargetReduction(IRBuilder<> &B,
|
2017-05-09 18:43:25 +08:00
|
|
|
const TargetTransformInfo *TTI,
|
|
|
|
RecurrenceDescriptor &Desc, Value *Src,
|
|
|
|
bool NoNaN) {
|
|
|
|
// TODO: Support in-order reductions based on the recurrence descriptor.
|
2017-12-07 03:37:00 +08:00
|
|
|
using RD = RecurrenceDescriptor;
|
|
|
|
RD::RecurrenceKind RecKind = Desc.getRecurrenceKind();
|
2017-05-09 18:43:25 +08:00
|
|
|
TargetTransformInfo::ReductionFlags Flags;
|
|
|
|
Flags.NoNaN = NoNaN;
|
2019-06-05 22:58:04 +08:00
|
|
|
|
|
|
|
// All ops in the reduction inherit fast-math-flags from the recurrence
|
|
|
|
// descriptor.
|
|
|
|
IRBuilder<>::FastMathFlagGuard FMFGuard(B);
|
|
|
|
B.setFastMathFlags(Desc.getFastMathFlags());
|
|
|
|
|
2017-05-09 18:43:25 +08:00
|
|
|
switch (RecKind) {
|
2017-12-07 03:37:00 +08:00
|
|
|
case RD::RK_FloatAdd:
|
2019-06-05 22:58:04 +08:00
|
|
|
return createSimpleTargetReduction(B, TTI, Instruction::FAdd, Src, Flags);
|
2017-12-07 03:37:00 +08:00
|
|
|
case RD::RK_FloatMult:
|
2019-06-05 22:58:04 +08:00
|
|
|
return createSimpleTargetReduction(B, TTI, Instruction::FMul, Src, Flags);
|
2017-12-07 03:37:00 +08:00
|
|
|
case RD::RK_IntegerAdd:
|
2019-06-05 22:58:04 +08:00
|
|
|
return createSimpleTargetReduction(B, TTI, Instruction::Add, Src, Flags);
|
2017-12-07 03:37:00 +08:00
|
|
|
case RD::RK_IntegerMult:
|
2019-06-05 22:58:04 +08:00
|
|
|
return createSimpleTargetReduction(B, TTI, Instruction::Mul, Src, Flags);
|
2017-12-07 03:37:00 +08:00
|
|
|
case RD::RK_IntegerAnd:
|
2019-06-05 22:58:04 +08:00
|
|
|
return createSimpleTargetReduction(B, TTI, Instruction::And, Src, Flags);
|
2017-12-07 03:37:00 +08:00
|
|
|
case RD::RK_IntegerOr:
|
2019-06-05 22:58:04 +08:00
|
|
|
return createSimpleTargetReduction(B, TTI, Instruction::Or, Src, Flags);
|
2017-12-07 03:37:00 +08:00
|
|
|
case RD::RK_IntegerXor:
|
2019-06-05 22:58:04 +08:00
|
|
|
return createSimpleTargetReduction(B, TTI, Instruction::Xor, Src, Flags);
|
2017-12-07 03:37:00 +08:00
|
|
|
case RD::RK_IntegerMinMax: {
|
|
|
|
RD::MinMaxRecurrenceKind MMKind = Desc.getMinMaxRecurrenceKind();
|
|
|
|
Flags.IsMaxOp = (MMKind == RD::MRK_SIntMax || MMKind == RD::MRK_UIntMax);
|
|
|
|
Flags.IsSigned = (MMKind == RD::MRK_SIntMax || MMKind == RD::MRK_SIntMin);
|
2019-06-05 22:58:04 +08:00
|
|
|
return createSimpleTargetReduction(B, TTI, Instruction::ICmp, Src, Flags);
|
2017-05-09 18:43:25 +08:00
|
|
|
}
|
2017-12-07 03:37:00 +08:00
|
|
|
case RD::RK_FloatMinMax: {
|
|
|
|
Flags.IsMaxOp = Desc.getMinMaxRecurrenceKind() == RD::MRK_FloatMax;
|
2019-06-05 22:58:04 +08:00
|
|
|
return createSimpleTargetReduction(B, TTI, Instruction::FCmp, Src, Flags);
|
2017-05-09 18:43:25 +08:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
llvm_unreachable("Unhandled RecKind");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-19 18:02:07 +08:00
|
|
|
void llvm::propagateIRFlags(Value *I, ArrayRef<Value *> VL, Value *OpValue) {
|
|
|
|
auto *VecOp = dyn_cast<Instruction>(I);
|
|
|
|
if (!VecOp)
|
|
|
|
return;
|
|
|
|
auto *Intersection = (OpValue == nullptr) ? dyn_cast<Instruction>(VL[0])
|
|
|
|
: dyn_cast<Instruction>(OpValue);
|
|
|
|
if (!Intersection)
|
|
|
|
return;
|
|
|
|
const unsigned Opcode = Intersection->getOpcode();
|
|
|
|
VecOp->copyIRFlags(Intersection);
|
|
|
|
for (auto *V : VL) {
|
|
|
|
auto *Instr = dyn_cast<Instruction>(V);
|
|
|
|
if (!Instr)
|
|
|
|
continue;
|
|
|
|
if (OpValue == nullptr || Opcode == Instr->getOpcode())
|
|
|
|
VecOp->andIRFlags(V);
|
2017-05-09 18:43:25 +08:00
|
|
|
}
|
|
|
|
}
|
2019-01-15 17:51:34 +08:00
|
|
|
|
|
|
|
bool llvm::isKnownNegativeInLoop(const SCEV *S, const Loop *L,
|
|
|
|
ScalarEvolution &SE) {
|
|
|
|
const SCEV *Zero = SE.getZero(S->getType());
|
|
|
|
return SE.isAvailableAtLoopEntry(S, L) &&
|
|
|
|
SE.isLoopEntryGuardedByCond(L, ICmpInst::ICMP_SLT, S, Zero);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool llvm::isKnownNonNegativeInLoop(const SCEV *S, const Loop *L,
|
|
|
|
ScalarEvolution &SE) {
|
|
|
|
const SCEV *Zero = SE.getZero(S->getType());
|
|
|
|
return SE.isAvailableAtLoopEntry(S, L) &&
|
|
|
|
SE.isLoopEntryGuardedByCond(L, ICmpInst::ICMP_SGE, S, Zero);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool llvm::cannotBeMinInLoop(const SCEV *S, const Loop *L, ScalarEvolution &SE,
|
|
|
|
bool Signed) {
|
|
|
|
unsigned BitWidth = cast<IntegerType>(S->getType())->getBitWidth();
|
|
|
|
APInt Min = Signed ? APInt::getSignedMinValue(BitWidth) :
|
|
|
|
APInt::getMinValue(BitWidth);
|
|
|
|
auto Predicate = Signed ? ICmpInst::ICMP_SGT : ICmpInst::ICMP_UGT;
|
|
|
|
return SE.isAvailableAtLoopEntry(S, L) &&
|
|
|
|
SE.isLoopEntryGuardedByCond(L, Predicate, S,
|
|
|
|
SE.getConstant(Min));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool llvm::cannotBeMaxInLoop(const SCEV *S, const Loop *L, ScalarEvolution &SE,
|
|
|
|
bool Signed) {
|
|
|
|
unsigned BitWidth = cast<IntegerType>(S->getType())->getBitWidth();
|
|
|
|
APInt Max = Signed ? APInt::getSignedMaxValue(BitWidth) :
|
|
|
|
APInt::getMaxValue(BitWidth);
|
|
|
|
auto Predicate = Signed ? ICmpInst::ICMP_SLT : ICmpInst::ICMP_ULT;
|
|
|
|
return SE.isAvailableAtLoopEntry(S, L) &&
|
|
|
|
SE.isLoopEntryGuardedByCond(L, Predicate, S,
|
|
|
|
SE.getConstant(Max));
|
|
|
|
}
|