2017-09-23 07:46:57 +08:00
|
|
|
//===- IfConversion.cpp - Machine code if conversion pass -----------------===//
|
2007-05-16 10:00:57 +08:00
|
|
|
//
|
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
|
2007-05-16 10:00:57 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2016-04-01 09:09:03 +08:00
|
|
|
// This file implements the machine instruction level if-conversion pass, which
|
|
|
|
// tries to convert conditional branches into predicated instructions.
|
2007-05-16 10:00:57 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "BranchFolding.h"
|
|
|
|
#include "llvm/ADT/STLExtras.h"
|
2016-08-19 06:09:23 +08:00
|
|
|
#include "llvm/ADT/ScopeExit.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/ADT/SmallSet.h"
|
2017-09-23 07:46:57 +08:00
|
|
|
#include "llvm/ADT/SmallVector.h"
|
|
|
|
#include "llvm/ADT/SparseSet.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
2017-09-23 07:46:57 +08:00
|
|
|
#include "llvm/ADT/iterator_range.h"
|
2019-12-06 01:39:37 +08:00
|
|
|
#include "llvm/Analysis/ProfileSummaryInfo.h"
|
2014-01-07 19:48:04 +08:00
|
|
|
#include "llvm/CodeGen/LivePhysRegs.h"
|
2017-09-23 07:46:57 +08:00
|
|
|
#include "llvm/CodeGen/MachineBasicBlock.h"
|
2014-08-08 03:30:13 +08:00
|
|
|
#include "llvm/CodeGen/MachineBlockFrequencyInfo.h"
|
2011-08-04 06:53:41 +08:00
|
|
|
#include "llvm/CodeGen/MachineBranchProbabilityInfo.h"
|
2017-09-23 07:46:57 +08:00
|
|
|
#include "llvm/CodeGen/MachineFunction.h"
|
2007-05-16 10:00:57 +08:00
|
|
|
#include "llvm/CodeGen/MachineFunctionPass.h"
|
2017-09-23 07:46:57 +08:00
|
|
|
#include "llvm/CodeGen/MachineInstr.h"
|
2012-12-21 02:08:06 +08:00
|
|
|
#include "llvm/CodeGen/MachineInstrBuilder.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/CodeGen/MachineModuleInfo.h"
|
2017-09-23 07:46:57 +08:00
|
|
|
#include "llvm/CodeGen/MachineOperand.h"
|
2012-06-09 05:53:50 +08:00
|
|
|
#include "llvm/CodeGen/MachineRegisterInfo.h"
|
2017-11-08 09:01:31 +08:00
|
|
|
#include "llvm/CodeGen/TargetInstrInfo.h"
|
2017-11-17 09:07:10 +08:00
|
|
|
#include "llvm/CodeGen/TargetLowering.h"
|
|
|
|
#include "llvm/CodeGen/TargetRegisterInfo.h"
|
2013-09-30 23:28:56 +08:00
|
|
|
#include "llvm/CodeGen/TargetSchedule.h"
|
2017-11-17 09:07:10 +08:00
|
|
|
#include "llvm/CodeGen/TargetSubtargetInfo.h"
|
Sink all InitializePasses.h includes
This file lists every pass in LLVM, and is included by Pass.h, which is
very popular. Every time we add, remove, or rename a pass in LLVM, it
caused lots of recompilation.
I found this fact by looking at this table, which is sorted by the
number of times a file was changed over the last 100,000 git commits
multiplied by the number of object files that depend on it in the
current checkout:
recompiles touches affected_files header
342380 95 3604 llvm/include/llvm/ADT/STLExtras.h
314730 234 1345 llvm/include/llvm/InitializePasses.h
307036 118 2602 llvm/include/llvm/ADT/APInt.h
213049 59 3611 llvm/include/llvm/Support/MathExtras.h
170422 47 3626 llvm/include/llvm/Support/Compiler.h
162225 45 3605 llvm/include/llvm/ADT/Optional.h
158319 63 2513 llvm/include/llvm/ADT/Triple.h
140322 39 3598 llvm/include/llvm/ADT/StringRef.h
137647 59 2333 llvm/include/llvm/Support/Error.h
131619 73 1803 llvm/include/llvm/Support/FileSystem.h
Before this change, touching InitializePasses.h would cause 1345 files
to recompile. After this change, touching it only causes 550 compiles in
an incremental rebuild.
Reviewers: bkramer, asbirlea, bollu, jdoerfert
Differential Revision: https://reviews.llvm.org/D70211
2019-11-14 05:15:01 +08:00
|
|
|
#include "llvm/IR/Attributes.h"
|
2017-09-23 07:46:57 +08:00
|
|
|
#include "llvm/IR/DebugLoc.h"
|
Sink all InitializePasses.h includes
This file lists every pass in LLVM, and is included by Pass.h, which is
very popular. Every time we add, remove, or rename a pass in LLVM, it
caused lots of recompilation.
I found this fact by looking at this table, which is sorted by the
number of times a file was changed over the last 100,000 git commits
multiplied by the number of object files that depend on it in the
current checkout:
recompiles touches affected_files header
342380 95 3604 llvm/include/llvm/ADT/STLExtras.h
314730 234 1345 llvm/include/llvm/InitializePasses.h
307036 118 2602 llvm/include/llvm/ADT/APInt.h
213049 59 3611 llvm/include/llvm/Support/MathExtras.h
170422 47 3626 llvm/include/llvm/Support/Compiler.h
162225 45 3605 llvm/include/llvm/ADT/Optional.h
158319 63 2513 llvm/include/llvm/ADT/Triple.h
140322 39 3598 llvm/include/llvm/ADT/StringRef.h
137647 59 2333 llvm/include/llvm/Support/Error.h
131619 73 1803 llvm/include/llvm/Support/FileSystem.h
Before this change, touching InitializePasses.h would cause 1345 files
to recompile. After this change, touching it only causes 550 compiles in
an incremental rebuild.
Reviewers: bkramer, asbirlea, bollu, jdoerfert
Differential Revision: https://reviews.llvm.org/D70211
2019-11-14 05:15:01 +08:00
|
|
|
#include "llvm/InitializePasses.h"
|
2017-09-23 07:46:57 +08:00
|
|
|
#include "llvm/MC/MCRegisterInfo.h"
|
|
|
|
#include "llvm/Pass.h"
|
|
|
|
#include "llvm/Support/BranchProbability.h"
|
2007-06-09 03:10:51 +08:00
|
|
|
#include "llvm/Support/CommandLine.h"
|
2007-05-16 10:00:57 +08:00
|
|
|
#include "llvm/Support/Debug.h"
|
2009-07-11 21:10:19 +08:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2015-12-01 13:29:22 +08:00
|
|
|
#include <algorithm>
|
2017-09-23 07:46:57 +08:00
|
|
|
#include <cassert>
|
|
|
|
#include <functional>
|
|
|
|
#include <iterator>
|
|
|
|
#include <memory>
|
2016-05-27 22:27:24 +08:00
|
|
|
#include <utility>
|
2017-09-23 07:46:57 +08:00
|
|
|
#include <vector>
|
2013-09-30 23:28:56 +08:00
|
|
|
|
2007-05-16 10:00:57 +08:00
|
|
|
using namespace llvm;
|
|
|
|
|
2017-05-26 05:26:32 +08:00
|
|
|
#define DEBUG_TYPE "if-converter"
|
2014-04-22 10:02:50 +08:00
|
|
|
|
2008-01-07 13:40:58 +08:00
|
|
|
// Hidden options for help debugging.
|
|
|
|
static cl::opt<int> IfCvtFnStart("ifcvt-fn-start", cl::init(-1), cl::Hidden);
|
|
|
|
static cl::opt<int> IfCvtFnStop("ifcvt-fn-stop", cl::init(-1), cl::Hidden);
|
|
|
|
static cl::opt<int> IfCvtLimit("ifcvt-limit", cl::init(-1), cl::Hidden);
|
2010-06-16 06:18:54 +08:00
|
|
|
static cl::opt<bool> DisableSimple("disable-ifcvt-simple",
|
2008-01-07 13:40:58 +08:00
|
|
|
cl::init(false), cl::Hidden);
|
2010-06-16 06:18:54 +08:00
|
|
|
static cl::opt<bool> DisableSimpleF("disable-ifcvt-simple-false",
|
2008-01-07 13:40:58 +08:00
|
|
|
cl::init(false), cl::Hidden);
|
2010-06-16 06:18:54 +08:00
|
|
|
static cl::opt<bool> DisableTriangle("disable-ifcvt-triangle",
|
2008-01-07 13:40:58 +08:00
|
|
|
cl::init(false), cl::Hidden);
|
2010-06-16 06:18:54 +08:00
|
|
|
static cl::opt<bool> DisableTriangleR("disable-ifcvt-triangle-rev",
|
2008-01-07 13:40:58 +08:00
|
|
|
cl::init(false), cl::Hidden);
|
2010-06-16 06:18:54 +08:00
|
|
|
static cl::opt<bool> DisableTriangleF("disable-ifcvt-triangle-false",
|
2008-01-07 13:40:58 +08:00
|
|
|
cl::init(false), cl::Hidden);
|
2010-06-16 06:18:54 +08:00
|
|
|
static cl::opt<bool> DisableTriangleFR("disable-ifcvt-triangle-false-rev",
|
2008-01-07 13:40:58 +08:00
|
|
|
cl::init(false), cl::Hidden);
|
2010-06-16 06:18:54 +08:00
|
|
|
static cl::opt<bool> DisableDiamond("disable-ifcvt-diamond",
|
2008-01-07 13:40:58 +08:00
|
|
|
cl::init(false), cl::Hidden);
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
static cl::opt<bool> DisableForkedDiamond("disable-ifcvt-forked-diamond",
|
|
|
|
cl::init(false), cl::Hidden);
|
2010-06-16 15:35:02 +08:00
|
|
|
static cl::opt<bool> IfCvtBranchFold("ifcvt-branch-fold",
|
|
|
|
cl::init(true), cl::Hidden);
|
2007-06-09 03:10:51 +08:00
|
|
|
|
2007-06-09 09:03:43 +08:00
|
|
|
STATISTIC(NumSimple, "Number of simple if-conversions performed");
|
|
|
|
STATISTIC(NumSimpleFalse, "Number of simple (F) if-conversions performed");
|
|
|
|
STATISTIC(NumTriangle, "Number of triangle if-conversions performed");
|
2007-06-13 07:54:05 +08:00
|
|
|
STATISTIC(NumTriangleRev, "Number of triangle (R) if-conversions performed");
|
2007-06-09 09:03:43 +08:00
|
|
|
STATISTIC(NumTriangleFalse,"Number of triangle (F) if-conversions performed");
|
|
|
|
STATISTIC(NumTriangleFRev, "Number of triangle (F/R) if-conversions performed");
|
|
|
|
STATISTIC(NumDiamonds, "Number of diamond if-conversions performed");
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
STATISTIC(NumForkedDiamonds, "Number of forked-diamond if-conversions performed");
|
2007-06-09 09:03:43 +08:00
|
|
|
STATISTIC(NumIfConvBBs, "Number of if-converted blocks");
|
2007-06-15 15:36:12 +08:00
|
|
|
STATISTIC(NumDupBBs, "Number of duplicated blocks");
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
STATISTIC(NumUnpred, "Number of true blocks of diamonds unpredicated");
|
2007-05-16 10:00:57 +08:00
|
|
|
|
|
|
|
namespace {
|
2017-09-23 07:46:57 +08:00
|
|
|
|
2009-10-25 14:33:48 +08:00
|
|
|
class IfConverter : public MachineFunctionPass {
|
2007-06-16 17:34:52 +08:00
|
|
|
enum IfcvtKind {
|
2007-05-16 10:00:57 +08:00
|
|
|
ICNotClassfied, // BB data valid, but not classified.
|
2007-06-04 14:47:22 +08:00
|
|
|
ICSimpleFalse, // Same as ICSimple, but on the false path.
|
2007-06-16 17:34:52 +08:00
|
|
|
ICSimple, // BB is entry of an one split, no rejoin sub-CFG.
|
|
|
|
ICTriangleFRev, // Same as ICTriangleFalse, but false path rev condition.
|
2007-06-13 07:54:05 +08:00
|
|
|
ICTriangleRev, // Same as ICTriangle, but true path rev condition.
|
2007-06-09 09:03:43 +08:00
|
|
|
ICTriangleFalse, // Same as ICTriangle, but on the false path.
|
2007-06-16 17:34:52 +08:00
|
|
|
ICTriangle, // BB is entry of a triangle sub-CFG.
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
ICDiamond, // BB is entry of a diamond sub-CFG.
|
|
|
|
ICForkedDiamond // BB is entry of an almost diamond sub-CFG, with a
|
|
|
|
// common tail that can be shared.
|
2007-05-16 10:00:57 +08:00
|
|
|
};
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// One per MachineBasicBlock, this is used to cache the result
|
2007-05-16 10:00:57 +08:00
|
|
|
/// if-conversion feasibility analysis. This includes results from
|
2016-07-15 22:41:04 +08:00
|
|
|
/// TargetInstrInfo::analyzeBranch() (i.e. TBB, FBB, and Cond), and its
|
2007-05-18 08:20:58 +08:00
|
|
|
/// classification, and common tail block of its successors (if it's a
|
2007-05-19 02:14:37 +08:00
|
|
|
/// diamond shape), its size, whether it's predicable, and whether any
|
|
|
|
/// instruction can clobber the 'would-be' predicate.
|
2007-05-23 15:23:16 +08:00
|
|
|
///
|
2007-06-12 06:26:22 +08:00
|
|
|
/// IsDone - True if BB is not to be considered for ifcvt.
|
|
|
|
/// IsBeingAnalyzed - True if BB is currently being analyzed.
|
|
|
|
/// IsAnalyzed - True if BB has been analyzed (info is still valid).
|
|
|
|
/// IsEnqueued - True if BB has been enqueued to be ifcvt'ed.
|
2016-07-15 22:41:04 +08:00
|
|
|
/// IsBrAnalyzable - True if analyzeBranch() returns false.
|
2007-06-12 06:26:22 +08:00
|
|
|
/// HasFallThrough - True if BB may fallthrough to the following BB.
|
|
|
|
/// IsUnpredicable - True if BB is known to be unpredicable.
|
2007-07-11 01:50:43 +08:00
|
|
|
/// ClobbersPred - True if BB could modify predicates (e.g. has
|
2007-06-06 18:16:17 +08:00
|
|
|
/// cmp, call, etc.)
|
2007-06-12 06:26:22 +08:00
|
|
|
/// NonPredSize - Number of non-predicated instructions.
|
2010-11-03 08:45:17 +08:00
|
|
|
/// ExtraCost - Extra cost for multi-cycle instructions.
|
|
|
|
/// ExtraCost2 - Some instructions are slower when predicated
|
2007-05-23 15:23:16 +08:00
|
|
|
/// BB - Corresponding MachineBasicBlock.
|
2016-07-15 22:41:04 +08:00
|
|
|
/// TrueBB / FalseBB- See analyzeBranch().
|
2007-05-23 15:23:16 +08:00
|
|
|
/// BrCond - Conditions for end of block conditional branches.
|
|
|
|
/// Predicate - Predicate used in the BB.
|
2007-05-16 10:00:57 +08:00
|
|
|
struct BBInfo {
|
2007-06-12 06:26:22 +08:00
|
|
|
bool IsDone : 1;
|
|
|
|
bool IsBeingAnalyzed : 1;
|
|
|
|
bool IsAnalyzed : 1;
|
|
|
|
bool IsEnqueued : 1;
|
|
|
|
bool IsBrAnalyzable : 1;
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
bool IsBrReversible : 1;
|
2007-06-12 06:26:22 +08:00
|
|
|
bool HasFallThrough : 1;
|
|
|
|
bool IsUnpredicable : 1;
|
2007-06-16 05:18:05 +08:00
|
|
|
bool CannotBeCopied : 1;
|
2007-06-12 06:26:22 +08:00
|
|
|
bool ClobbersPred : 1;
|
2017-09-23 07:46:57 +08:00
|
|
|
unsigned NonPredSize = 0;
|
|
|
|
unsigned ExtraCost = 0;
|
|
|
|
unsigned ExtraCost2 = 0;
|
|
|
|
MachineBasicBlock *BB = nullptr;
|
|
|
|
MachineBasicBlock *TrueBB = nullptr;
|
|
|
|
MachineBasicBlock *FalseBB = nullptr;
|
2008-08-15 06:49:33 +08:00
|
|
|
SmallVector<MachineOperand, 4> BrCond;
|
|
|
|
SmallVector<MachineOperand, 4> Predicate;
|
2017-09-23 07:46:57 +08:00
|
|
|
|
2007-06-16 17:34:52 +08:00
|
|
|
BBInfo() : IsDone(false), IsBeingAnalyzed(false),
|
2007-06-12 06:26:22 +08:00
|
|
|
IsAnalyzed(false), IsEnqueued(false), IsBrAnalyzable(false),
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
IsBrReversible(false), HasFallThrough(false),
|
|
|
|
IsUnpredicable(false), CannotBeCopied(false),
|
2017-09-23 07:46:57 +08:00
|
|
|
ClobbersPred(false) {}
|
2007-06-16 17:34:52 +08:00
|
|
|
};
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Record information about pending if-conversions to attempt:
|
2007-06-16 17:34:52 +08:00
|
|
|
/// BBI - Corresponding BBInfo.
|
|
|
|
/// Kind - Type of block. See IfcvtKind.
|
2009-05-14 07:25:24 +08:00
|
|
|
/// NeedSubsumption - True if the to-be-predicated BB has already been
|
2007-06-16 17:34:52 +08:00
|
|
|
/// predicated.
|
2007-06-18 16:37:25 +08:00
|
|
|
/// NumDups - Number of instructions that would be duplicated due
|
|
|
|
/// to this if-conversion. (For diamonds, the number of
|
|
|
|
/// identical instructions at the beginnings of both
|
|
|
|
/// paths).
|
|
|
|
/// NumDups2 - For diamonds, the number of identical instructions
|
|
|
|
/// at the ends of both paths.
|
2007-06-16 17:34:52 +08:00
|
|
|
struct IfcvtToken {
|
|
|
|
BBInfo &BBI;
|
|
|
|
IfcvtKind Kind;
|
2007-06-18 16:37:25 +08:00
|
|
|
unsigned NumDups;
|
|
|
|
unsigned NumDups2;
|
2016-08-25 05:34:24 +08:00
|
|
|
bool NeedSubsumption : 1;
|
|
|
|
bool TClobbersPred : 1;
|
|
|
|
bool FClobbersPred : 1;
|
2017-09-23 07:46:57 +08:00
|
|
|
|
2016-08-25 05:34:24 +08:00
|
|
|
IfcvtToken(BBInfo &b, IfcvtKind k, bool s, unsigned d, unsigned d2 = 0,
|
|
|
|
bool tc = false, bool fc = false)
|
|
|
|
: BBI(b), Kind(k), NumDups(d), NumDups2(d2), NeedSubsumption(s),
|
|
|
|
TClobbersPred(tc), FClobbersPred(fc) {}
|
2007-05-16 10:00:57 +08:00
|
|
|
};
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Results of if-conversion feasibility analysis indexed by basic block
|
|
|
|
/// number.
|
2007-05-16 10:00:57 +08:00
|
|
|
std::vector<BBInfo> BBAnalysis;
|
2013-09-30 23:28:56 +08:00
|
|
|
TargetSchedModel SchedModel;
|
2007-05-16 10:00:57 +08:00
|
|
|
|
2013-01-12 04:05:37 +08:00
|
|
|
const TargetLoweringBase *TLI;
|
2007-05-16 10:00:57 +08:00
|
|
|
const TargetInstrInfo *TII;
|
2010-06-16 15:35:02 +08:00
|
|
|
const TargetRegisterInfo *TRI;
|
2011-08-04 06:34:43 +08:00
|
|
|
const MachineBranchProbabilityInfo *MBPI;
|
2012-06-09 05:53:50 +08:00
|
|
|
MachineRegisterInfo *MRI;
|
2011-08-04 06:34:43 +08:00
|
|
|
|
2013-12-14 14:52:56 +08:00
|
|
|
LivePhysRegs Redefs;
|
2013-10-15 04:45:17 +08:00
|
|
|
|
2012-06-09 05:53:50 +08:00
|
|
|
bool PreRegAlloc;
|
2007-05-16 10:00:57 +08:00
|
|
|
bool MadeChange;
|
2017-09-23 07:46:57 +08:00
|
|
|
int FnNum = -1;
|
2016-10-25 07:23:02 +08:00
|
|
|
std::function<bool(const MachineFunction &)> PredicateFtor;
|
2015-06-09 02:50:43 +08:00
|
|
|
|
2007-05-16 10:00:57 +08:00
|
|
|
public:
|
|
|
|
static char ID;
|
2017-09-23 07:46:57 +08:00
|
|
|
|
2016-10-25 07:23:02 +08:00
|
|
|
IfConverter(std::function<bool(const MachineFunction &)> Ftor = nullptr)
|
2017-09-23 07:46:57 +08:00
|
|
|
: MachineFunctionPass(ID), PredicateFtor(std::move(Ftor)) {
|
2010-10-20 01:21:58 +08:00
|
|
|
initializeIfConverterPass(*PassRegistry::getPassRegistry());
|
|
|
|
}
|
2011-08-04 06:34:43 +08:00
|
|
|
|
2014-03-07 17:26:03 +08:00
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const override {
|
2014-08-08 03:30:13 +08:00
|
|
|
AU.addRequired<MachineBlockFrequencyInfo>();
|
2011-08-04 06:34:43 +08:00
|
|
|
AU.addRequired<MachineBranchProbabilityInfo>();
|
2019-12-06 01:39:37 +08:00
|
|
|
AU.addRequired<ProfileSummaryInfoWrapperPass>();
|
2010-09-29 04:42:15 +08:00
|
|
|
MachineFunctionPass::getAnalysisUsage(AU);
|
|
|
|
}
|
2007-05-16 10:00:57 +08:00
|
|
|
|
2014-03-07 17:26:03 +08:00
|
|
|
bool runOnMachineFunction(MachineFunction &MF) override;
|
2007-05-16 10:00:57 +08:00
|
|
|
|
2016-04-05 01:09:25 +08:00
|
|
|
MachineFunctionProperties getRequiredProperties() const override {
|
|
|
|
return MachineFunctionProperties().set(
|
2016-08-25 09:27:13 +08:00
|
|
|
MachineFunctionProperties::Property::NoVRegs);
|
2016-04-05 01:09:25 +08:00
|
|
|
}
|
|
|
|
|
2007-05-16 10:00:57 +08:00
|
|
|
private:
|
2016-09-15 04:43:16 +08:00
|
|
|
bool reverseBranchCondition(BBInfo &BBI) const;
|
2010-10-02 06:45:50 +08:00
|
|
|
bool ValidSimple(BBInfo &TrueBBI, unsigned &Dups,
|
2015-09-11 07:10:42 +08:00
|
|
|
BranchProbability Prediction) const;
|
2007-06-08 17:36:04 +08:00
|
|
|
bool ValidTriangle(BBInfo &TrueBBI, BBInfo &FalseBBI,
|
2010-10-02 06:45:50 +08:00
|
|
|
bool FalseBranch, unsigned &Dups,
|
2015-09-11 07:10:42 +08:00
|
|
|
BranchProbability Prediction) const;
|
2016-08-25 05:34:24 +08:00
|
|
|
bool CountDuplicatedInstructions(
|
|
|
|
MachineBasicBlock::iterator &TIB, MachineBasicBlock::iterator &FIB,
|
|
|
|
MachineBasicBlock::iterator &TIE, MachineBasicBlock::iterator &FIE,
|
|
|
|
unsigned &Dups1, unsigned &Dups2,
|
|
|
|
MachineBasicBlock &TBB, MachineBasicBlock &FBB,
|
|
|
|
bool SkipUnconditionalBranches) const;
|
2007-06-18 16:37:25 +08:00
|
|
|
bool ValidDiamond(BBInfo &TrueBBI, BBInfo &FalseBBI,
|
2016-08-25 05:34:24 +08:00
|
|
|
unsigned &Dups1, unsigned &Dups2,
|
|
|
|
BBInfo &TrueBBICalc, BBInfo &FalseBBICalc) const;
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
bool ValidForkedDiamond(BBInfo &TrueBBI, BBInfo &FalseBBI,
|
|
|
|
unsigned &Dups1, unsigned &Dups2,
|
|
|
|
BBInfo &TrueBBICalc, BBInfo &FalseBBICalc) const;
|
2016-08-06 09:52:34 +08:00
|
|
|
void AnalyzeBranches(BBInfo &BBI);
|
|
|
|
void ScanInstructions(BBInfo &BBI,
|
|
|
|
MachineBasicBlock::iterator &Begin,
|
2016-08-25 05:34:24 +08:00
|
|
|
MachineBasicBlock::iterator &End,
|
|
|
|
bool BranchUnpredicable = false) const;
|
|
|
|
bool RescanInstructions(
|
|
|
|
MachineBasicBlock::iterator &TIB, MachineBasicBlock::iterator &FIB,
|
|
|
|
MachineBasicBlock::iterator &TIE, MachineBasicBlock::iterator &FIE,
|
|
|
|
BBInfo &TrueBBI, BBInfo &FalseBBI) const;
|
2016-08-17 10:52:01 +08:00
|
|
|
void AnalyzeBlock(MachineBasicBlock &MBB,
|
2016-02-23 01:51:28 +08:00
|
|
|
std::vector<std::unique_ptr<IfcvtToken>> &Tokens);
|
2018-07-17 02:51:40 +08:00
|
|
|
bool FeasibilityAnalysis(BBInfo &BBI, SmallVectorImpl<MachineOperand> &Pred,
|
2016-08-25 05:34:24 +08:00
|
|
|
bool isTriangle = false, bool RevBranch = false,
|
|
|
|
bool hasCommonTail = false);
|
2016-02-23 01:51:28 +08:00
|
|
|
void AnalyzeBlocks(MachineFunction &MF,
|
|
|
|
std::vector<std::unique_ptr<IfcvtToken>> &Tokens);
|
2016-08-17 10:52:01 +08:00
|
|
|
void InvalidatePreds(MachineBasicBlock &MBB);
|
2007-06-16 17:34:52 +08:00
|
|
|
bool IfConvertSimple(BBInfo &BBI, IfcvtKind Kind);
|
|
|
|
bool IfConvertTriangle(BBInfo &BBI, IfcvtKind Kind);
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
bool IfConvertDiamondCommon(BBInfo &BBI, BBInfo &TrueBBI, BBInfo &FalseBBI,
|
|
|
|
unsigned NumDups1, unsigned NumDups2,
|
|
|
|
bool TClobbersPred, bool FClobbersPred,
|
2016-08-30 02:27:12 +08:00
|
|
|
bool RemoveBranch, bool MergeAddEdges);
|
2007-06-18 16:37:25 +08:00
|
|
|
bool IfConvertDiamond(BBInfo &BBI, IfcvtKind Kind,
|
2016-08-25 05:34:24 +08:00
|
|
|
unsigned NumDups1, unsigned NumDups2,
|
|
|
|
bool TClobbers, bool FClobbers);
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
bool IfConvertForkedDiamond(BBInfo &BBI, IfcvtKind Kind,
|
|
|
|
unsigned NumDups1, unsigned NumDups2,
|
|
|
|
bool TClobbers, bool FClobbers);
|
2007-05-23 15:23:16 +08:00
|
|
|
void PredicateBlock(BBInfo &BBI,
|
2007-06-18 16:37:25 +08:00
|
|
|
MachineBasicBlock::iterator E,
|
2010-06-16 15:35:02 +08:00
|
|
|
SmallVectorImpl<MachineOperand> &Cond,
|
2018-11-07 03:00:11 +08:00
|
|
|
SmallSet<MCPhysReg, 4> *LaterRedefs = nullptr);
|
2007-06-15 15:36:12 +08:00
|
|
|
void CopyAndPredicateBlock(BBInfo &ToBBI, BBInfo &FromBBI,
|
2008-08-15 06:49:33 +08:00
|
|
|
SmallVectorImpl<MachineOperand> &Cond,
|
2007-06-15 15:36:12 +08:00
|
|
|
bool IgnoreBr = false);
|
Reapply my if-conversion cleanup from svn r106939 with fixes.
There are 2 changes relative to the previous version of the patch:
1) For the "simple" if-conversion case, there's no need to worry about
RemoveExtraEdges not handling an unanalyzable branch. Predicated terminators
are ignored in this context, so RemoveExtraEdges does the right thing.
This might break someday if we ever treat indirect branches (BRIND) as
predicable, but for now, I just removed this part of the patch, because
in the case where we do not add an unconditional branch, we rely on keeping
the fall-through edge to CvtBBI (which is empty after this transformation).
The change relative to the previous patch is:
@@ -1036,10 +1036,6 @@
IterIfcvt = false;
}
- // RemoveExtraEdges won't work if the block has an unanalyzable branch,
- // which is typically the case for IfConvertSimple, so explicitly remove
- // CvtBBI as a successor.
- BBI.BB->removeSuccessor(CvtBBI->BB);
RemoveExtraEdges(BBI);
// Update block info. BB can be iteratively if-converted.
2) My patch exposed a bug in the code for merging the tail of a "diamond",
which had previously never been exercised. The code was simply checking that
the tail had a single predecessor, but there was a case in
MultiSource/Benchmarks/VersaBench/dbms where that single predecessor was
neither edge of the diamond. I added the following change to check for
that:
@@ -1276,7 +1276,18 @@
// tail, add a unconditional branch to it.
if (TailBB) {
BBInfo TailBBI = BBAnalysis[TailBB->getNumber()];
- if (TailBB->pred_size() == 1 && !TailBBI.HasFallThrough) {
+ bool CanMergeTail = !TailBBI.HasFallThrough;
+ // There may still be a fall-through edge from BBI1 or BBI2 to TailBB;
+ // check if there are any other predecessors besides those.
+ unsigned NumPreds = TailBB->pred_size();
+ if (NumPreds > 1)
+ CanMergeTail = false;
+ else if (NumPreds == 1 && CanMergeTail) {
+ MachineBasicBlock::pred_iterator PI = TailBB->pred_begin();
+ if (*PI != BBI1->BB && *PI != BBI2->BB)
+ CanMergeTail = false;
+ }
+ if (CanMergeTail) {
MergeBlocks(BBI, TailBBI);
TailBBI.IsDone = true;
} else {
With these fixes, I was able to run all the SingleSource and MultiSource
tests successfully.
llvm-svn: 107110
2010-06-29 08:55:23 +08:00
|
|
|
void MergeBlocks(BBInfo &ToBBI, BBInfo &FromBBI, bool AddEdges = true);
|
2007-06-01 08:12:12 +08:00
|
|
|
|
2010-11-03 08:45:17 +08:00
|
|
|
bool MeetIfcvtSizeLimit(MachineBasicBlock &BB,
|
|
|
|
unsigned Cycle, unsigned Extra,
|
2015-09-11 07:10:42 +08:00
|
|
|
BranchProbability Prediction) const {
|
2010-11-03 08:45:17 +08:00
|
|
|
return Cycle > 0 && TII->isProfitableToIfCvt(BB, Cycle, Extra,
|
2011-07-10 10:58:07 +08:00
|
|
|
Prediction);
|
2010-06-26 06:42:03 +08:00
|
|
|
}
|
|
|
|
|
2019-10-10 17:58:28 +08:00
|
|
|
bool MeetIfcvtSizeLimit(BBInfo &TBBInfo, BBInfo &FBBInfo,
|
|
|
|
MachineBasicBlock &CommBB, unsigned Dups,
|
|
|
|
BranchProbability Prediction, bool Forked) const {
|
|
|
|
const MachineFunction &MF = *TBBInfo.BB->getParent();
|
|
|
|
if (MF.getFunction().hasMinSize()) {
|
|
|
|
MachineBasicBlock::iterator TIB = TBBInfo.BB->begin();
|
|
|
|
MachineBasicBlock::iterator FIB = FBBInfo.BB->begin();
|
|
|
|
MachineBasicBlock::iterator TIE = TBBInfo.BB->end();
|
|
|
|
MachineBasicBlock::iterator FIE = FBBInfo.BB->end();
|
|
|
|
|
|
|
|
unsigned Dups1, Dups2;
|
|
|
|
if (!CountDuplicatedInstructions(TIB, FIB, TIE, FIE, Dups1, Dups2,
|
|
|
|
*TBBInfo.BB, *FBBInfo.BB,
|
|
|
|
/*SkipUnconditionalBranches*/ true))
|
|
|
|
llvm_unreachable("should already have been checked by ValidDiamond");
|
|
|
|
|
|
|
|
unsigned BranchBytes = 0;
|
|
|
|
unsigned CommonBytes = 0;
|
|
|
|
|
|
|
|
// Count common instructions at the start of the true and false blocks.
|
|
|
|
for (auto &I : make_range(TBBInfo.BB->begin(), TIB)) {
|
|
|
|
LLVM_DEBUG(dbgs() << "Common inst: " << I);
|
|
|
|
CommonBytes += TII->getInstSizeInBytes(I);
|
|
|
|
}
|
|
|
|
for (auto &I : make_range(FBBInfo.BB->begin(), FIB)) {
|
|
|
|
LLVM_DEBUG(dbgs() << "Common inst: " << I);
|
|
|
|
CommonBytes += TII->getInstSizeInBytes(I);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Count instructions at the end of the true and false blocks, after
|
|
|
|
// the ones we plan to predicate. Analyzable branches will be removed
|
|
|
|
// (unless this is a forked diamond), and all other instructions are
|
|
|
|
// common between the two blocks.
|
|
|
|
for (auto &I : make_range(TIE, TBBInfo.BB->end())) {
|
|
|
|
if (I.isBranch() && TBBInfo.IsBrAnalyzable && !Forked) {
|
|
|
|
LLVM_DEBUG(dbgs() << "Saving branch: " << I);
|
|
|
|
BranchBytes += TII->predictBranchSizeForIfCvt(I);
|
|
|
|
} else {
|
|
|
|
LLVM_DEBUG(dbgs() << "Common inst: " << I);
|
|
|
|
CommonBytes += TII->getInstSizeInBytes(I);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (auto &I : make_range(FIE, FBBInfo.BB->end())) {
|
|
|
|
if (I.isBranch() && FBBInfo.IsBrAnalyzable && !Forked) {
|
|
|
|
LLVM_DEBUG(dbgs() << "Saving branch: " << I);
|
|
|
|
BranchBytes += TII->predictBranchSizeForIfCvt(I);
|
|
|
|
} else {
|
|
|
|
LLVM_DEBUG(dbgs() << "Common inst: " << I);
|
|
|
|
CommonBytes += TII->getInstSizeInBytes(I);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (auto &I : CommBB.terminators()) {
|
|
|
|
if (I.isBranch()) {
|
|
|
|
LLVM_DEBUG(dbgs() << "Saving branch: " << I);
|
|
|
|
BranchBytes += TII->predictBranchSizeForIfCvt(I);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// The common instructions in one branch will be eliminated, halving
|
|
|
|
// their code size.
|
|
|
|
CommonBytes /= 2;
|
|
|
|
|
|
|
|
// Count the instructions which we need to predicate.
|
|
|
|
unsigned NumPredicatedInstructions = 0;
|
|
|
|
for (auto &I : make_range(TIB, TIE)) {
|
|
|
|
if (!I.isDebugInstr()) {
|
|
|
|
LLVM_DEBUG(dbgs() << "Predicating: " << I);
|
|
|
|
NumPredicatedInstructions++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (auto &I : make_range(FIB, FIE)) {
|
|
|
|
if (!I.isDebugInstr()) {
|
|
|
|
LLVM_DEBUG(dbgs() << "Predicating: " << I);
|
|
|
|
NumPredicatedInstructions++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Even though we're optimising for size at the expense of performance,
|
|
|
|
// avoid creating really long predicated blocks.
|
|
|
|
if (NumPredicatedInstructions > 15)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Some targets (e.g. Thumb2) need to insert extra instructions to
|
|
|
|
// start predicated blocks.
|
|
|
|
unsigned ExtraPredicateBytes = TII->extraSizeToPredicateInstructions(
|
|
|
|
MF, NumPredicatedInstructions);
|
|
|
|
|
|
|
|
LLVM_DEBUG(dbgs() << "MeetIfcvtSizeLimit(BranchBytes=" << BranchBytes
|
|
|
|
<< ", CommonBytes=" << CommonBytes
|
|
|
|
<< ", NumPredicatedInstructions="
|
|
|
|
<< NumPredicatedInstructions
|
|
|
|
<< ", ExtraPredicateBytes=" << ExtraPredicateBytes
|
|
|
|
<< ")\n");
|
|
|
|
return (BranchBytes + CommonBytes) > ExtraPredicateBytes;
|
|
|
|
} else {
|
|
|
|
unsigned TCycle = TBBInfo.NonPredSize + TBBInfo.ExtraCost - Dups;
|
|
|
|
unsigned FCycle = FBBInfo.NonPredSize + FBBInfo.ExtraCost - Dups;
|
|
|
|
bool Res = TCycle > 0 && FCycle > 0 &&
|
|
|
|
TII->isProfitableToIfCvt(
|
|
|
|
*TBBInfo.BB, TCycle, TBBInfo.ExtraCost2, *FBBInfo.BB,
|
|
|
|
FCycle, FBBInfo.ExtraCost2, Prediction);
|
|
|
|
LLVM_DEBUG(dbgs() << "MeetIfcvtSizeLimit(TCycle=" << TCycle
|
|
|
|
<< ", FCycle=" << FCycle
|
|
|
|
<< ", TExtra=" << TBBInfo.ExtraCost2 << ", FExtra="
|
|
|
|
<< FBBInfo.ExtraCost2 << ") = " << Res << "\n");
|
|
|
|
return Res;
|
|
|
|
}
|
2007-06-18 16:37:25 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Returns true if Block ends without a terminator.
|
2007-06-07 10:12:15 +08:00
|
|
|
bool blockAlwaysFallThrough(BBInfo &BBI) const {
|
2014-04-14 08:51:57 +08:00
|
|
|
return BBI.IsBrAnalyzable && BBI.TrueBB == nullptr;
|
2007-06-06 18:16:17 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Used to sort if-conversion candidates.
|
2016-02-23 01:51:28 +08:00
|
|
|
static bool IfcvtTokenCmp(const std::unique_ptr<IfcvtToken> &C1,
|
|
|
|
const std::unique_ptr<IfcvtToken> &C2) {
|
2007-06-18 16:37:25 +08:00
|
|
|
int Incr1 = (C1->Kind == ICDiamond)
|
|
|
|
? -(int)(C1->NumDups + C1->NumDups2) : (int)C1->NumDups;
|
|
|
|
int Incr2 = (C2->Kind == ICDiamond)
|
|
|
|
? -(int)(C2->NumDups + C2->NumDups2) : (int)C2->NumDups;
|
|
|
|
if (Incr1 > Incr2)
|
2007-06-16 17:34:52 +08:00
|
|
|
return true;
|
2007-06-18 16:37:25 +08:00
|
|
|
else if (Incr1 == Incr2) {
|
2009-05-14 07:25:24 +08:00
|
|
|
// Favors subsumption.
|
2015-03-09 09:57:13 +08:00
|
|
|
if (!C1->NeedSubsumption && C2->NeedSubsumption)
|
2007-06-16 17:34:52 +08:00
|
|
|
return true;
|
2009-05-14 07:25:24 +08:00
|
|
|
else if (C1->NeedSubsumption == C2->NeedSubsumption) {
|
2007-06-16 17:34:52 +08:00
|
|
|
// Favors diamond over triangle, etc.
|
|
|
|
if ((unsigned)C1->Kind < (unsigned)C2->Kind)
|
|
|
|
return true;
|
|
|
|
else if (C1->Kind == C2->Kind)
|
|
|
|
return C1->BBI.BB->getNumber() < C2->BBI.BB->getNumber();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2007-06-01 08:12:12 +08:00
|
|
|
}
|
2007-05-16 10:00:57 +08:00
|
|
|
};
|
2007-06-16 17:34:52 +08:00
|
|
|
|
2017-09-23 07:46:57 +08:00
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
char IfConverter::ID = 0;
|
2007-05-16 10:00:57 +08:00
|
|
|
|
2012-02-09 05:23:13 +08:00
|
|
|
char &llvm::IfConverterID = IfConverter::ID;
|
|
|
|
|
2017-05-26 05:26:32 +08:00
|
|
|
INITIALIZE_PASS_BEGIN(IfConverter, DEBUG_TYPE, "If Converter", false, false)
|
2011-08-04 06:34:43 +08:00
|
|
|
INITIALIZE_PASS_DEPENDENCY(MachineBranchProbabilityInfo)
|
2019-12-06 01:39:37 +08:00
|
|
|
INITIALIZE_PASS_DEPENDENCY(ProfileSummaryInfoWrapperPass)
|
2017-05-26 05:26:32 +08:00
|
|
|
INITIALIZE_PASS_END(IfConverter, DEBUG_TYPE, "If Converter", false, false)
|
2009-10-29 04:46:46 +08:00
|
|
|
|
2007-05-16 10:00:57 +08:00
|
|
|
bool IfConverter::runOnMachineFunction(MachineFunction &MF) {
|
2017-12-16 06:22:58 +08:00
|
|
|
if (skipFunction(MF.getFunction()) || (PredicateFtor && !PredicateFtor(MF)))
|
2015-06-09 02:50:43 +08:00
|
|
|
return false;
|
|
|
|
|
2015-01-27 15:31:29 +08:00
|
|
|
const TargetSubtargetInfo &ST = MF.getSubtarget();
|
|
|
|
TLI = ST.getTargetLowering();
|
|
|
|
TII = ST.getInstrInfo();
|
|
|
|
TRI = ST.getRegisterInfo();
|
2020-01-28 02:05:54 +08:00
|
|
|
MBFIWrapper MBFI(getAnalysis<MachineBlockFrequencyInfo>());
|
2011-08-04 06:34:43 +08:00
|
|
|
MBPI = &getAnalysis<MachineBranchProbabilityInfo>();
|
2019-12-06 01:39:37 +08:00
|
|
|
ProfileSummaryInfo *PSI =
|
|
|
|
&getAnalysis<ProfileSummaryInfoWrapperPass>().getPSI();
|
2012-06-09 05:53:50 +08:00
|
|
|
MRI = &MF.getRegInfo();
|
2018-04-09 03:56:04 +08:00
|
|
|
SchedModel.init(&ST);
|
2013-09-30 23:28:56 +08:00
|
|
|
|
2007-05-16 10:00:57 +08:00
|
|
|
if (!TII) return false;
|
|
|
|
|
2012-06-09 05:53:50 +08:00
|
|
|
PreRegAlloc = MRI->isSSA();
|
|
|
|
|
|
|
|
bool BFChange = false;
|
|
|
|
if (!PreRegAlloc) {
|
|
|
|
// Tail merge tend to expose more if-conversion opportunities.
|
2019-12-06 01:39:37 +08:00
|
|
|
BranchFolder BF(true, false, MBFI, *MBPI, PSI);
|
2019-10-01 01:54:50 +08:00
|
|
|
auto *MMIWP = getAnalysisIfAvailable<MachineModuleInfoWrapperPass>();
|
|
|
|
BFChange = BF.OptimizeFunction(
|
|
|
|
MF, TII, ST.getRegisterInfo(),
|
|
|
|
MMIWP ? &MMIWP->getMMI() : nullptr);
|
2012-06-09 05:53:50 +08:00
|
|
|
}
|
2010-06-19 07:09:54 +08:00
|
|
|
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "\nIfcvt: function (" << ++FnNum << ") \'"
|
|
|
|
<< MF.getName() << "\'");
|
2007-06-09 03:10:51 +08:00
|
|
|
|
|
|
|
if (FnNum < IfCvtFnStart || (IfCvtFnStop != -1 && FnNum > IfCvtFnStop)) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << " skipped\n");
|
2007-06-09 03:10:51 +08:00
|
|
|
return false;
|
|
|
|
}
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "\n");
|
2007-06-01 08:12:12 +08:00
|
|
|
|
2007-05-16 10:00:57 +08:00
|
|
|
MF.RenumberBlocks();
|
2007-06-01 08:12:12 +08:00
|
|
|
BBAnalysis.resize(MF.getNumBlockIDs());
|
2007-05-16 10:00:57 +08:00
|
|
|
|
2016-02-23 01:51:28 +08:00
|
|
|
std::vector<std::unique_ptr<IfcvtToken>> Tokens;
|
2007-05-18 09:55:58 +08:00
|
|
|
MadeChange = false;
|
2007-06-15 15:36:12 +08:00
|
|
|
unsigned NumIfCvts = NumSimple + NumSimpleFalse + NumTriangle +
|
|
|
|
NumTriangleRev + NumTriangleFalse + NumTriangleFRev + NumDiamonds;
|
|
|
|
while (IfCvtLimit == -1 || (int)NumIfCvts < IfCvtLimit) {
|
2009-05-14 07:25:24 +08:00
|
|
|
// Do an initial analysis for each basic block and find all the potential
|
|
|
|
// candidates to perform if-conversion.
|
2010-06-16 02:57:15 +08:00
|
|
|
bool Change = false;
|
|
|
|
AnalyzeBlocks(MF, Tokens);
|
2007-06-16 17:34:52 +08:00
|
|
|
while (!Tokens.empty()) {
|
2016-02-23 01:51:28 +08:00
|
|
|
std::unique_ptr<IfcvtToken> Token = std::move(Tokens.back());
|
2007-06-16 17:34:52 +08:00
|
|
|
Tokens.pop_back();
|
|
|
|
BBInfo &BBI = Token->BBI;
|
|
|
|
IfcvtKind Kind = Token->Kind;
|
2008-11-04 21:02:59 +08:00
|
|
|
unsigned NumDups = Token->NumDups;
|
2008-11-05 02:05:30 +08:00
|
|
|
unsigned NumDups2 = Token->NumDups2;
|
2008-11-04 21:02:59 +08:00
|
|
|
|
2007-06-12 06:26:22 +08:00
|
|
|
// If the block has been evicted out of the queue or it has already been
|
|
|
|
// marked dead (due to it being predicated), then skip it.
|
2007-06-16 17:34:52 +08:00
|
|
|
if (BBI.IsDone)
|
|
|
|
BBI.IsEnqueued = false;
|
|
|
|
if (!BBI.IsEnqueued)
|
2007-06-12 06:26:22 +08:00
|
|
|
continue;
|
2007-06-16 17:34:52 +08:00
|
|
|
|
2007-06-15 04:28:52 +08:00
|
|
|
BBI.IsEnqueued = false;
|
2007-06-12 06:26:22 +08:00
|
|
|
|
2007-06-04 14:47:22 +08:00
|
|
|
bool RetVal = false;
|
2007-06-16 17:34:52 +08:00
|
|
|
switch (Kind) {
|
2012-02-05 16:31:47 +08:00
|
|
|
default: llvm_unreachable("Unexpected!");
|
2007-06-04 14:47:22 +08:00
|
|
|
case ICSimple:
|
2007-06-06 09:12:44 +08:00
|
|
|
case ICSimpleFalse: {
|
2007-06-16 17:34:52 +08:00
|
|
|
bool isFalse = Kind == ICSimpleFalse;
|
2007-06-09 09:03:43 +08:00
|
|
|
if ((isFalse && DisableSimpleF) || (!isFalse && DisableSimple)) break;
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "Ifcvt (Simple"
|
|
|
|
<< (Kind == ICSimpleFalse ? " false" : "")
|
|
|
|
<< "): " << printMBBReference(*BBI.BB) << " ("
|
|
|
|
<< ((Kind == ICSimpleFalse) ? BBI.FalseBB->getNumber()
|
|
|
|
: BBI.TrueBB->getNumber())
|
|
|
|
<< ") ");
|
2007-06-16 17:34:52 +08:00
|
|
|
RetVal = IfConvertSimple(BBI, Kind);
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << (RetVal ? "succeeded!" : "failed!") << "\n");
|
2008-02-20 19:10:28 +08:00
|
|
|
if (RetVal) {
|
2010-06-22 23:08:57 +08:00
|
|
|
if (isFalse) ++NumSimpleFalse;
|
|
|
|
else ++NumSimple;
|
2008-02-20 19:10:28 +08:00
|
|
|
}
|
2007-06-04 14:47:22 +08:00
|
|
|
break;
|
2007-06-06 09:12:44 +08:00
|
|
|
}
|
2007-05-23 15:23:16 +08:00
|
|
|
case ICTriangle:
|
2007-06-13 07:54:05 +08:00
|
|
|
case ICTriangleRev:
|
2007-06-09 09:03:43 +08:00
|
|
|
case ICTriangleFalse:
|
2007-06-10 08:19:17 +08:00
|
|
|
case ICTriangleFRev: {
|
2007-06-16 17:34:52 +08:00
|
|
|
bool isFalse = Kind == ICTriangleFalse;
|
|
|
|
bool isRev = (Kind == ICTriangleRev || Kind == ICTriangleFRev);
|
2007-06-13 07:54:05 +08:00
|
|
|
if (DisableTriangle && !isFalse && !isRev) break;
|
|
|
|
if (DisableTriangleR && !isFalse && isRev) break;
|
|
|
|
if (DisableTriangleF && isFalse && !isRev) break;
|
|
|
|
if (DisableTriangleFR && isFalse && isRev) break;
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "Ifcvt (Triangle");
|
2007-06-09 09:03:43 +08:00
|
|
|
if (isFalse)
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << " false");
|
2007-06-13 07:54:05 +08:00
|
|
|
if (isRev)
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << " rev");
|
|
|
|
LLVM_DEBUG(dbgs() << "): " << printMBBReference(*BBI.BB)
|
|
|
|
<< " (T:" << BBI.TrueBB->getNumber()
|
|
|
|
<< ",F:" << BBI.FalseBB->getNumber() << ") ");
|
2007-06-16 17:34:52 +08:00
|
|
|
RetVal = IfConvertTriangle(BBI, Kind);
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << (RetVal ? "succeeded!" : "failed!") << "\n");
|
2007-06-09 09:03:43 +08:00
|
|
|
if (RetVal) {
|
2007-06-13 07:54:05 +08:00
|
|
|
if (isFalse) {
|
2010-06-22 23:08:57 +08:00
|
|
|
if (isRev) ++NumTriangleFRev;
|
|
|
|
else ++NumTriangleFalse;
|
2007-06-13 07:54:05 +08:00
|
|
|
} else {
|
2010-06-22 23:08:57 +08:00
|
|
|
if (isRev) ++NumTriangleRev;
|
|
|
|
else ++NumTriangle;
|
2007-06-13 07:54:05 +08:00
|
|
|
}
|
2007-06-09 09:03:43 +08:00
|
|
|
}
|
2007-05-23 15:23:16 +08:00
|
|
|
break;
|
2007-06-10 08:19:17 +08:00
|
|
|
}
|
2017-09-23 07:46:57 +08:00
|
|
|
case ICDiamond:
|
2007-06-09 03:10:51 +08:00
|
|
|
if (DisableDiamond) break;
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "Ifcvt (Diamond): " << printMBBReference(*BBI.BB)
|
|
|
|
<< " (T:" << BBI.TrueBB->getNumber()
|
|
|
|
<< ",F:" << BBI.FalseBB->getNumber() << ") ");
|
2016-08-25 05:34:24 +08:00
|
|
|
RetVal = IfConvertDiamond(BBI, Kind, NumDups, NumDups2,
|
|
|
|
Token->TClobbersPred,
|
|
|
|
Token->FClobbersPred);
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << (RetVal ? "succeeded!" : "failed!") << "\n");
|
2010-06-22 23:08:57 +08:00
|
|
|
if (RetVal) ++NumDiamonds;
|
2007-05-23 15:23:16 +08:00
|
|
|
break;
|
2017-09-23 07:46:57 +08:00
|
|
|
case ICForkedDiamond:
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
if (DisableForkedDiamond) break;
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "Ifcvt (Forked Diamond): "
|
|
|
|
<< printMBBReference(*BBI.BB)
|
|
|
|
<< " (T:" << BBI.TrueBB->getNumber()
|
|
|
|
<< ",F:" << BBI.FalseBB->getNumber() << ") ");
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
RetVal = IfConvertForkedDiamond(BBI, Kind, NumDups, NumDups2,
|
|
|
|
Token->TClobbersPred,
|
|
|
|
Token->FClobbersPred);
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << (RetVal ? "succeeded!" : "failed!") << "\n");
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
if (RetVal) ++NumForkedDiamonds;
|
|
|
|
break;
|
|
|
|
}
|
2007-06-16 17:34:52 +08:00
|
|
|
|
2017-09-14 23:53:11 +08:00
|
|
|
if (RetVal && MRI->tracksLiveness())
|
|
|
|
recomputeLivenessFlags(*BBI.BB);
|
|
|
|
|
2007-06-04 14:47:22 +08:00
|
|
|
Change |= RetVal;
|
2007-06-09 03:10:51 +08:00
|
|
|
|
2007-06-15 15:36:12 +08:00
|
|
|
NumIfCvts = NumSimple + NumSimpleFalse + NumTriangle + NumTriangleRev +
|
|
|
|
NumTriangleFalse + NumTriangleFRev + NumDiamonds;
|
|
|
|
if (IfCvtLimit != -1 && (int)NumIfCvts >= IfCvtLimit)
|
2007-06-09 03:10:51 +08:00
|
|
|
break;
|
2007-05-16 10:00:57 +08:00
|
|
|
}
|
2007-05-23 15:23:16 +08:00
|
|
|
|
|
|
|
if (!Change)
|
|
|
|
break;
|
2007-06-04 14:47:22 +08:00
|
|
|
MadeChange |= Change;
|
2007-05-16 10:00:57 +08:00
|
|
|
}
|
2007-05-18 09:55:58 +08:00
|
|
|
|
2007-06-16 17:34:52 +08:00
|
|
|
Tokens.clear();
|
2007-05-18 09:55:58 +08:00
|
|
|
BBAnalysis.clear();
|
|
|
|
|
2010-06-19 06:17:13 +08:00
|
|
|
if (MadeChange && IfCvtBranchFold) {
|
2019-12-06 01:39:37 +08:00
|
|
|
BranchFolder BF(false, false, MBFI, *MBPI, PSI);
|
2019-10-01 01:54:50 +08:00
|
|
|
auto *MMIWP = getAnalysisIfAvailable<MachineModuleInfoWrapperPass>();
|
|
|
|
BF.OptimizeFunction(
|
|
|
|
MF, TII, MF.getSubtarget().getRegisterInfo(),
|
|
|
|
MMIWP ? &MMIWP->getMMI() : nullptr);
|
2009-09-04 15:47:40 +08:00
|
|
|
}
|
|
|
|
|
2010-06-19 07:09:54 +08:00
|
|
|
MadeChange |= BFChange;
|
2007-05-16 10:00:57 +08:00
|
|
|
return MadeChange;
|
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// BB has a fallthrough. Find its 'false' successor given its 'true' successor.
|
2007-05-16 10:00:57 +08:00
|
|
|
static MachineBasicBlock *findFalseBlock(MachineBasicBlock *BB,
|
2007-05-18 08:20:58 +08:00
|
|
|
MachineBasicBlock *TrueBB) {
|
2016-08-17 10:51:59 +08:00
|
|
|
for (MachineBasicBlock *SuccBB : BB->successors()) {
|
2007-05-18 08:20:58 +08:00
|
|
|
if (SuccBB != TrueBB)
|
2007-05-16 10:00:57 +08:00
|
|
|
return SuccBB;
|
|
|
|
}
|
2014-04-14 08:51:57 +08:00
|
|
|
return nullptr;
|
2007-05-16 10:00:57 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Reverse the condition of the end of the block branch. Swap block's 'true'
|
|
|
|
/// and 'false' successors.
|
2016-09-15 04:43:16 +08:00
|
|
|
bool IfConverter::reverseBranchCondition(BBInfo &BBI) const {
|
2010-06-18 06:43:56 +08:00
|
|
|
DebugLoc dl; // FIXME: this is nowhere
|
2016-09-15 04:43:16 +08:00
|
|
|
if (!TII->reverseBranchCondition(BBI.BrCond)) {
|
|
|
|
TII->removeBranch(*BBI.BB);
|
2016-09-15 01:24:15 +08:00
|
|
|
TII->insertBranch(*BBI.BB, BBI.FalseBB, BBI.TrueBB, BBI.BrCond, dl);
|
2007-06-04 14:47:22 +08:00
|
|
|
std::swap(BBI.TrueBB, BBI.FalseBB);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Returns the next block in the function blocks ordering. If it is the end,
|
|
|
|
/// returns NULL.
|
2016-08-17 10:52:01 +08:00
|
|
|
static inline MachineBasicBlock *getNextBlock(MachineBasicBlock &MBB) {
|
|
|
|
MachineFunction::iterator I = MBB.getIterator();
|
|
|
|
MachineFunction::iterator E = MBB.getParent()->end();
|
2007-06-09 09:03:43 +08:00
|
|
|
if (++I == E)
|
2014-04-14 08:51:57 +08:00
|
|
|
return nullptr;
|
2015-10-10 03:13:58 +08:00
|
|
|
return &*I;
|
2007-06-09 09:03:43 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Returns true if the 'true' block (along with its predecessor) forms a valid
|
|
|
|
/// simple shape for ifcvt. It also returns the number of instructions that the
|
|
|
|
/// ifcvt would need to duplicate if performed in Dups.
|
2010-09-29 04:42:15 +08:00
|
|
|
bool IfConverter::ValidSimple(BBInfo &TrueBBI, unsigned &Dups,
|
2015-09-11 07:10:42 +08:00
|
|
|
BranchProbability Prediction) const {
|
2007-06-16 17:34:52 +08:00
|
|
|
Dups = 0;
|
2007-06-18 16:37:25 +08:00
|
|
|
if (TrueBBI.IsBeingAnalyzed || TrueBBI.IsDone)
|
2007-06-12 06:26:22 +08:00
|
|
|
return false;
|
|
|
|
|
2007-06-20 05:45:13 +08:00
|
|
|
if (TrueBBI.IsBrAnalyzable)
|
|
|
|
return false;
|
|
|
|
|
2007-06-16 05:18:05 +08:00
|
|
|
if (TrueBBI.BB->pred_size() > 1) {
|
|
|
|
if (TrueBBI.CannotBeCopied ||
|
2010-09-29 04:42:15 +08:00
|
|
|
!TII->isProfitableToDupForIfCvt(*TrueBBI.BB, TrueBBI.NonPredSize,
|
2011-07-10 10:58:07 +08:00
|
|
|
Prediction))
|
2007-06-15 15:36:12 +08:00
|
|
|
return false;
|
2007-06-16 17:34:52 +08:00
|
|
|
Dups = TrueBBI.NonPredSize;
|
2007-06-15 15:36:12 +08:00
|
|
|
}
|
|
|
|
|
2007-06-20 05:45:13 +08:00
|
|
|
return true;
|
2007-06-06 18:16:17 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Returns true if the 'true' and 'false' blocks (along with their common
|
|
|
|
/// predecessor) forms a valid triangle shape for ifcvt. If 'FalseBranch' is
|
|
|
|
/// true, it checks if 'true' block's false branch branches to the 'false' block
|
|
|
|
/// rather than the other way around. It also returns the number of instructions
|
|
|
|
/// that the ifcvt would need to duplicate if performed in 'Dups'.
|
2007-06-08 17:36:04 +08:00
|
|
|
bool IfConverter::ValidTriangle(BBInfo &TrueBBI, BBInfo &FalseBBI,
|
2010-10-02 06:45:50 +08:00
|
|
|
bool FalseBranch, unsigned &Dups,
|
2015-09-11 07:10:42 +08:00
|
|
|
BranchProbability Prediction) const {
|
2007-06-16 17:34:52 +08:00
|
|
|
Dups = 0;
|
2019-09-26 14:35:55 +08:00
|
|
|
if (TrueBBI.BB == FalseBBI.BB)
|
|
|
|
return false;
|
|
|
|
|
2007-06-18 16:37:25 +08:00
|
|
|
if (TrueBBI.IsBeingAnalyzed || TrueBBI.IsDone)
|
2007-06-12 06:26:22 +08:00
|
|
|
return false;
|
|
|
|
|
2007-06-16 05:18:05 +08:00
|
|
|
if (TrueBBI.BB->pred_size() > 1) {
|
|
|
|
if (TrueBBI.CannotBeCopied)
|
|
|
|
return false;
|
|
|
|
|
2007-06-15 15:36:12 +08:00
|
|
|
unsigned Size = TrueBBI.NonPredSize;
|
2007-06-16 17:34:52 +08:00
|
|
|
if (TrueBBI.IsBrAnalyzable) {
|
2008-01-29 21:02:09 +08:00
|
|
|
if (TrueBBI.TrueBB && TrueBBI.BrCond.empty())
|
2009-05-14 07:25:24 +08:00
|
|
|
// Ends with an unconditional branch. It will be removed.
|
2007-06-16 17:34:52 +08:00
|
|
|
--Size;
|
|
|
|
else {
|
|
|
|
MachineBasicBlock *FExit = FalseBranch
|
|
|
|
? TrueBBI.TrueBB : TrueBBI.FalseBB;
|
|
|
|
if (FExit)
|
|
|
|
// Require a conditional branch
|
|
|
|
++Size;
|
|
|
|
}
|
|
|
|
}
|
2011-07-10 10:58:07 +08:00
|
|
|
if (!TII->isProfitableToDupForIfCvt(*TrueBBI.BB, Size, Prediction))
|
2007-06-15 15:36:12 +08:00
|
|
|
return false;
|
2007-06-16 17:34:52 +08:00
|
|
|
Dups = Size;
|
2007-06-15 15:36:12 +08:00
|
|
|
}
|
2007-06-07 10:12:15 +08:00
|
|
|
|
2007-06-08 17:36:04 +08:00
|
|
|
MachineBasicBlock *TExit = FalseBranch ? TrueBBI.FalseBB : TrueBBI.TrueBB;
|
|
|
|
if (!TExit && blockAlwaysFallThrough(TrueBBI)) {
|
2015-10-10 03:13:58 +08:00
|
|
|
MachineFunction::iterator I = TrueBBI.BB->getIterator();
|
2007-06-07 10:12:15 +08:00
|
|
|
if (++I == TrueBBI.BB->getParent()->end())
|
|
|
|
return false;
|
2015-10-10 03:13:58 +08:00
|
|
|
TExit = &*I;
|
2007-06-07 10:12:15 +08:00
|
|
|
}
|
2007-06-08 17:36:04 +08:00
|
|
|
return TExit && TExit == FalseBBI.BB;
|
2007-06-07 10:12:15 +08:00
|
|
|
}
|
|
|
|
|
2016-08-06 09:52:33 +08:00
|
|
|
/// Count duplicated instructions and move the iterators to show where they
|
|
|
|
/// are.
|
|
|
|
/// @param TIB True Iterator Begin
|
|
|
|
/// @param FIB False Iterator Begin
|
|
|
|
/// These two iterators initially point to the first instruction of the two
|
|
|
|
/// blocks, and finally point to the first non-shared instruction.
|
|
|
|
/// @param TIE True Iterator End
|
|
|
|
/// @param FIE False Iterator End
|
|
|
|
/// These two iterators initially point to End() for the two blocks() and
|
|
|
|
/// finally point to the first shared instruction in the tail.
|
|
|
|
/// Upon return [TIB, TIE), and [FIB, FIE) mark the un-duplicated portions of
|
|
|
|
/// two blocks.
|
2016-08-25 05:34:24 +08:00
|
|
|
/// @param Dups1 count of duplicated instructions at the beginning of the 2
|
|
|
|
/// blocks.
|
|
|
|
/// @param Dups2 count of duplicated instructions at the end of the 2 blocks.
|
|
|
|
/// @param SkipUnconditionalBranches if true, Don't make sure that
|
|
|
|
/// unconditional branches at the end of the blocks are the same. True is
|
|
|
|
/// passed when the blocks are analyzable to allow for fallthrough to be
|
|
|
|
/// handled.
|
|
|
|
/// @return false if the shared portion prevents if conversion.
|
|
|
|
bool IfConverter::CountDuplicatedInstructions(
|
2016-07-28 04:19:33 +08:00
|
|
|
MachineBasicBlock::iterator &TIB,
|
|
|
|
MachineBasicBlock::iterator &FIB,
|
|
|
|
MachineBasicBlock::iterator &TIE,
|
|
|
|
MachineBasicBlock::iterator &FIE,
|
|
|
|
unsigned &Dups1, unsigned &Dups2,
|
|
|
|
MachineBasicBlock &TBB, MachineBasicBlock &FBB,
|
2016-08-25 05:34:24 +08:00
|
|
|
bool SkipUnconditionalBranches) const {
|
2010-10-26 08:02:24 +08:00
|
|
|
while (TIB != TIE && FIB != FIE) {
|
2010-06-19 05:52:57 +08:00
|
|
|
// Skip dbg_value instructions. These do not count.
|
2016-12-16 19:10:26 +08:00
|
|
|
TIB = skipDebugInstructionsForward(TIB, TIE);
|
|
|
|
FIB = skipDebugInstructionsForward(FIB, FIE);
|
2017-01-27 04:02:47 +08:00
|
|
|
if (TIB == TIE || FIB == FIE)
|
2016-08-06 09:52:31 +08:00
|
|
|
break;
|
2016-02-28 04:01:33 +08:00
|
|
|
if (!TIB->isIdenticalTo(*FIB))
|
2007-06-18 16:37:25 +08:00
|
|
|
break;
|
2016-08-25 05:34:24 +08:00
|
|
|
// A pred-clobbering instruction in the shared portion prevents
|
|
|
|
// if-conversion.
|
|
|
|
std::vector<MachineOperand> PredDefs;
|
|
|
|
if (TII->DefinesPredicate(*TIB, PredDefs))
|
|
|
|
return false;
|
2016-09-02 09:20:06 +08:00
|
|
|
// If we get all the way to the branch instructions, don't count them.
|
|
|
|
if (!TIB->isBranch())
|
|
|
|
++Dups1;
|
2010-10-26 08:02:24 +08:00
|
|
|
++TIB;
|
|
|
|
++FIB;
|
2007-06-18 16:37:25 +08:00
|
|
|
}
|
|
|
|
|
2016-01-20 21:14:52 +08:00
|
|
|
// Check for already containing all of the block.
|
|
|
|
if (TIB == TIE || FIB == FIE)
|
2016-08-25 05:34:24 +08:00
|
|
|
return true;
|
2016-08-19 06:09:23 +08:00
|
|
|
// Now, in preparation for counting duplicate instructions at the ends of the
|
2017-01-27 04:02:47 +08:00
|
|
|
// blocks, switch to reverse_iterators. Note that getReverse() returns an
|
|
|
|
// iterator that points to the same instruction, unlike std::reverse_iterator.
|
|
|
|
// We have to do our own shifting so that we get the same range.
|
|
|
|
MachineBasicBlock::reverse_iterator RTIE = std::next(TIE.getReverse());
|
|
|
|
MachineBasicBlock::reverse_iterator RFIE = std::next(FIE.getReverse());
|
|
|
|
const MachineBasicBlock::reverse_iterator RTIB = std::next(TIB.getReverse());
|
|
|
|
const MachineBasicBlock::reverse_iterator RFIB = std::next(FIB.getReverse());
|
2016-08-19 06:09:23 +08:00
|
|
|
|
2016-07-28 04:19:33 +08:00
|
|
|
if (!TBB.succ_empty() || !FBB.succ_empty()) {
|
2016-08-25 05:34:24 +08:00
|
|
|
if (SkipUnconditionalBranches) {
|
2017-01-27 04:02:47 +08:00
|
|
|
while (RTIE != RTIB && RTIE->isUnconditionalBranch())
|
|
|
|
++RTIE;
|
|
|
|
while (RFIE != RFIB && RFIE->isUnconditionalBranch())
|
|
|
|
++RFIE;
|
2016-07-28 04:19:33 +08:00
|
|
|
}
|
2010-10-26 08:02:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Count duplicate instructions at the ends of the blocks.
|
2017-01-27 04:02:47 +08:00
|
|
|
while (RTIE != RTIB && RFIE != RFIB) {
|
2010-06-19 05:52:57 +08:00
|
|
|
// Skip dbg_value instructions. These do not count.
|
2017-01-27 04:02:47 +08:00
|
|
|
// Note that these are reverse iterators going forward.
|
|
|
|
RTIE = skipDebugInstructionsForward(RTIE, RTIB);
|
|
|
|
RFIE = skipDebugInstructionsForward(RFIE, RFIB);
|
|
|
|
if (RTIE == RTIB || RFIE == RFIB)
|
2016-08-06 09:52:31 +08:00
|
|
|
break;
|
2017-01-27 04:02:47 +08:00
|
|
|
if (!RTIE->isIdenticalTo(*RFIE))
|
2007-06-18 16:37:25 +08:00
|
|
|
break;
|
2016-08-25 05:34:24 +08:00
|
|
|
// We have to verify that any branch instructions are the same, and then we
|
|
|
|
// don't count them toward the # of duplicate instructions.
|
2017-01-27 04:02:47 +08:00
|
|
|
if (!RTIE->isBranch())
|
2016-07-28 04:19:33 +08:00
|
|
|
++Dups2;
|
2017-01-27 04:02:47 +08:00
|
|
|
++RTIE;
|
|
|
|
++RFIE;
|
2007-06-18 16:37:25 +08:00
|
|
|
}
|
2017-01-27 04:02:47 +08:00
|
|
|
TIE = std::next(RTIE.getReverse());
|
|
|
|
FIE = std::next(RFIE.getReverse());
|
2016-08-25 05:34:24 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// RescanInstructions - Run ScanInstructions on a pair of blocks.
|
|
|
|
/// @param TIB - True Iterator Begin, points to first non-shared instruction
|
|
|
|
/// @param FIB - False Iterator Begin, points to first non-shared instruction
|
|
|
|
/// @param TIE - True Iterator End, points past last non-shared instruction
|
|
|
|
/// @param FIE - False Iterator End, points past last non-shared instruction
|
|
|
|
/// @param TrueBBI - BBInfo to update for the true block.
|
|
|
|
/// @param FalseBBI - BBInfo to update for the false block.
|
|
|
|
/// @returns - false if either block cannot be predicated or if both blocks end
|
|
|
|
/// with a predicate-clobbering instruction.
|
|
|
|
bool IfConverter::RescanInstructions(
|
|
|
|
MachineBasicBlock::iterator &TIB, MachineBasicBlock::iterator &FIB,
|
|
|
|
MachineBasicBlock::iterator &TIE, MachineBasicBlock::iterator &FIE,
|
|
|
|
BBInfo &TrueBBI, BBInfo &FalseBBI) const {
|
|
|
|
bool BranchUnpredicable = true;
|
|
|
|
TrueBBI.IsUnpredicable = FalseBBI.IsUnpredicable = false;
|
|
|
|
ScanInstructions(TrueBBI, TIB, TIE, BranchUnpredicable);
|
|
|
|
if (TrueBBI.IsUnpredicable)
|
|
|
|
return false;
|
|
|
|
ScanInstructions(FalseBBI, FIB, FIE, BranchUnpredicable);
|
|
|
|
if (FalseBBI.IsUnpredicable)
|
|
|
|
return false;
|
|
|
|
if (TrueBBI.ClobbersPred && FalseBBI.ClobbersPred)
|
|
|
|
return false;
|
|
|
|
return true;
|
2016-07-28 04:19:33 +08:00
|
|
|
}
|
|
|
|
|
2016-08-30 02:27:12 +08:00
|
|
|
#ifndef NDEBUG
|
|
|
|
static void verifySameBranchInstructions(
|
|
|
|
MachineBasicBlock *MBB1,
|
|
|
|
MachineBasicBlock *MBB2) {
|
2017-01-27 04:02:47 +08:00
|
|
|
const MachineBasicBlock::reverse_iterator B1 = MBB1->rend();
|
|
|
|
const MachineBasicBlock::reverse_iterator B2 = MBB2->rend();
|
|
|
|
MachineBasicBlock::reverse_iterator E1 = MBB1->rbegin();
|
|
|
|
MachineBasicBlock::reverse_iterator E2 = MBB2->rbegin();
|
|
|
|
while (E1 != B1 && E2 != B2) {
|
|
|
|
skipDebugInstructionsForward(E1, B1);
|
|
|
|
skipDebugInstructionsForward(E2, B2);
|
|
|
|
if (E1 == B1 && E2 == B2)
|
2016-08-30 02:27:12 +08:00
|
|
|
break;
|
|
|
|
|
2017-01-27 04:02:47 +08:00
|
|
|
if (E1 == B1) {
|
2016-08-30 02:27:12 +08:00
|
|
|
assert(!E2->isBranch() && "Branch mis-match, one block is empty.");
|
|
|
|
break;
|
|
|
|
}
|
2017-01-27 04:02:47 +08:00
|
|
|
if (E2 == B2) {
|
2016-08-30 02:27:12 +08:00
|
|
|
assert(!E1->isBranch() && "Branch mis-match, one block is empty.");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (E1->isBranch() || E2->isBranch())
|
|
|
|
assert(E1->isIdenticalTo(*E2) &&
|
|
|
|
"Branch mis-match, branch instructions don't match.");
|
|
|
|
else
|
|
|
|
break;
|
2017-01-27 04:02:47 +08:00
|
|
|
++E1;
|
|
|
|
++E2;
|
2016-08-30 02:27:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
/// ValidForkedDiamond - Returns true if the 'true' and 'false' blocks (along
|
|
|
|
/// with their common predecessor) form a diamond if a common tail block is
|
|
|
|
/// extracted.
|
|
|
|
/// While not strictly a diamond, this pattern would form a diamond if
|
|
|
|
/// tail-merging had merged the shared tails.
|
|
|
|
/// EBB
|
|
|
|
/// _/ \_
|
|
|
|
/// | |
|
|
|
|
/// TBB FBB
|
|
|
|
/// / \ / \
|
|
|
|
/// FalseBB TrueBB FalseBB
|
|
|
|
/// Currently only handles analyzable branches.
|
|
|
|
/// Specifically excludes actual diamonds to avoid overlap.
|
|
|
|
bool IfConverter::ValidForkedDiamond(
|
|
|
|
BBInfo &TrueBBI, BBInfo &FalseBBI,
|
|
|
|
unsigned &Dups1, unsigned &Dups2,
|
|
|
|
BBInfo &TrueBBICalc, BBInfo &FalseBBICalc) const {
|
|
|
|
Dups1 = Dups2 = 0;
|
|
|
|
if (TrueBBI.IsBeingAnalyzed || TrueBBI.IsDone ||
|
|
|
|
FalseBBI.IsBeingAnalyzed || FalseBBI.IsDone)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!TrueBBI.IsBrAnalyzable || !FalseBBI.IsBrAnalyzable)
|
|
|
|
return false;
|
|
|
|
// Don't IfConvert blocks that can't be folded into their predecessor.
|
|
|
|
if (TrueBBI.BB->pred_size() > 1 || FalseBBI.BB->pred_size() > 1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// This function is specifically looking for conditional tails, as
|
|
|
|
// unconditional tails are already handled by the standard diamond case.
|
|
|
|
if (TrueBBI.BrCond.size() == 0 ||
|
|
|
|
FalseBBI.BrCond.size() == 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
MachineBasicBlock *TT = TrueBBI.TrueBB;
|
|
|
|
MachineBasicBlock *TF = TrueBBI.FalseBB;
|
|
|
|
MachineBasicBlock *FT = FalseBBI.TrueBB;
|
|
|
|
MachineBasicBlock *FF = FalseBBI.FalseBB;
|
|
|
|
|
|
|
|
if (!TT)
|
|
|
|
TT = getNextBlock(*TrueBBI.BB);
|
|
|
|
if (!TF)
|
|
|
|
TF = getNextBlock(*TrueBBI.BB);
|
|
|
|
if (!FT)
|
|
|
|
FT = getNextBlock(*FalseBBI.BB);
|
|
|
|
if (!FF)
|
|
|
|
FF = getNextBlock(*FalseBBI.BB);
|
|
|
|
|
|
|
|
if (!TT || !TF)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check successors. If they don't match, bail.
|
|
|
|
if (!((TT == FT && TF == FF) || (TF == FT && TT == FF)))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
bool FalseReversed = false;
|
|
|
|
if (TF == FT && TT == FF) {
|
|
|
|
// If the branches are opposing, but we can't reverse, don't do it.
|
|
|
|
if (!FalseBBI.IsBrReversible)
|
|
|
|
return false;
|
|
|
|
FalseReversed = true;
|
2016-09-15 04:43:16 +08:00
|
|
|
reverseBranchCondition(FalseBBI);
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
}
|
|
|
|
auto UnReverseOnExit = make_scope_exit([&]() {
|
|
|
|
if (FalseReversed)
|
2016-09-15 04:43:16 +08:00
|
|
|
reverseBranchCondition(FalseBBI);
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
// Count duplicate instructions at the beginning of the true and false blocks.
|
|
|
|
MachineBasicBlock::iterator TIB = TrueBBI.BB->begin();
|
|
|
|
MachineBasicBlock::iterator FIB = FalseBBI.BB->begin();
|
|
|
|
MachineBasicBlock::iterator TIE = TrueBBI.BB->end();
|
|
|
|
MachineBasicBlock::iterator FIE = FalseBBI.BB->end();
|
|
|
|
if(!CountDuplicatedInstructions(TIB, FIB, TIE, FIE, Dups1, Dups2,
|
|
|
|
*TrueBBI.BB, *FalseBBI.BB,
|
|
|
|
/* SkipUnconditionalBranches */ true))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
TrueBBICalc.BB = TrueBBI.BB;
|
|
|
|
FalseBBICalc.BB = FalseBBI.BB;
|
2019-10-10 17:58:28 +08:00
|
|
|
TrueBBICalc.IsBrAnalyzable = TrueBBI.IsBrAnalyzable;
|
|
|
|
FalseBBICalc.IsBrAnalyzable = FalseBBI.IsBrAnalyzable;
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
if (!RescanInstructions(TIB, FIB, TIE, FIE, TrueBBICalc, FalseBBICalc))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// The size is used to decide whether to if-convert, and the shared portions
|
|
|
|
// are subtracted off. Because of the subtraction, we just use the size that
|
|
|
|
// was calculated by the original ScanInstructions, as it is correct.
|
|
|
|
TrueBBICalc.NonPredSize = TrueBBI.NonPredSize;
|
|
|
|
FalseBBICalc.NonPredSize = FalseBBI.NonPredSize;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-07-28 04:19:33 +08:00
|
|
|
/// ValidDiamond - Returns true if the 'true' and 'false' blocks (along
|
|
|
|
/// with their common predecessor) forms a valid diamond shape for ifcvt.
|
2016-08-25 05:34:24 +08:00
|
|
|
bool IfConverter::ValidDiamond(
|
|
|
|
BBInfo &TrueBBI, BBInfo &FalseBBI,
|
|
|
|
unsigned &Dups1, unsigned &Dups2,
|
|
|
|
BBInfo &TrueBBICalc, BBInfo &FalseBBICalc) const {
|
2016-07-28 04:19:33 +08:00
|
|
|
Dups1 = Dups2 = 0;
|
|
|
|
if (TrueBBI.IsBeingAnalyzed || TrueBBI.IsDone ||
|
|
|
|
FalseBBI.IsBeingAnalyzed || FalseBBI.IsDone)
|
|
|
|
return false;
|
2007-06-18 16:37:25 +08:00
|
|
|
|
2016-07-28 04:19:33 +08:00
|
|
|
MachineBasicBlock *TT = TrueBBI.TrueBB;
|
|
|
|
MachineBasicBlock *FT = FalseBBI.TrueBB;
|
|
|
|
|
|
|
|
if (!TT && blockAlwaysFallThrough(TrueBBI))
|
2016-08-17 10:52:01 +08:00
|
|
|
TT = getNextBlock(*TrueBBI.BB);
|
2016-07-28 04:19:33 +08:00
|
|
|
if (!FT && blockAlwaysFallThrough(FalseBBI))
|
2016-08-17 10:52:01 +08:00
|
|
|
FT = getNextBlock(*FalseBBI.BB);
|
2016-07-28 04:19:33 +08:00
|
|
|
if (TT != FT)
|
|
|
|
return false;
|
|
|
|
if (!TT && (TrueBBI.IsBrAnalyzable || FalseBBI.IsBrAnalyzable))
|
|
|
|
return false;
|
|
|
|
if (TrueBBI.BB->pred_size() > 1 || FalseBBI.BB->pred_size() > 1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// FIXME: Allow true block to have an early exit?
|
2016-08-25 05:34:24 +08:00
|
|
|
if (TrueBBI.FalseBB || FalseBBI.FalseBB)
|
2016-07-28 04:19:33 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// Count duplicate instructions at the beginning and end of the true and
|
|
|
|
// false blocks.
|
2016-08-25 05:34:24 +08:00
|
|
|
// Skip unconditional branches only if we are considering an analyzable
|
|
|
|
// diamond. Otherwise the branches must be the same.
|
|
|
|
bool SkipUnconditionalBranches =
|
|
|
|
TrueBBI.IsBrAnalyzable && FalseBBI.IsBrAnalyzable;
|
2016-07-28 04:19:33 +08:00
|
|
|
MachineBasicBlock::iterator TIB = TrueBBI.BB->begin();
|
|
|
|
MachineBasicBlock::iterator FIB = FalseBBI.BB->begin();
|
|
|
|
MachineBasicBlock::iterator TIE = TrueBBI.BB->end();
|
|
|
|
MachineBasicBlock::iterator FIE = FalseBBI.BB->end();
|
2016-08-25 05:34:24 +08:00
|
|
|
if(!CountDuplicatedInstructions(TIB, FIB, TIE, FIE, Dups1, Dups2,
|
|
|
|
*TrueBBI.BB, *FalseBBI.BB,
|
|
|
|
SkipUnconditionalBranches))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
TrueBBICalc.BB = TrueBBI.BB;
|
|
|
|
FalseBBICalc.BB = FalseBBI.BB;
|
2019-10-10 17:58:28 +08:00
|
|
|
TrueBBICalc.IsBrAnalyzable = TrueBBI.IsBrAnalyzable;
|
|
|
|
FalseBBICalc.IsBrAnalyzable = FalseBBI.IsBrAnalyzable;
|
2016-08-25 05:34:24 +08:00
|
|
|
if (!RescanInstructions(TIB, FIB, TIE, FIE, TrueBBICalc, FalseBBICalc))
|
|
|
|
return false;
|
|
|
|
// The size is used to decide whether to if-convert, and the shared portions
|
|
|
|
// are subtracted off. Because of the subtraction, we just use the size that
|
|
|
|
// was calculated by the original ScanInstructions, as it is correct.
|
|
|
|
TrueBBICalc.NonPredSize = TrueBBI.NonPredSize;
|
|
|
|
FalseBBICalc.NonPredSize = FalseBBI.NonPredSize;
|
2007-06-18 16:37:25 +08:00
|
|
|
return true;
|
2007-06-07 10:12:15 +08:00
|
|
|
}
|
|
|
|
|
2016-08-06 09:52:34 +08:00
|
|
|
/// AnalyzeBranches - Look at the branches at the end of a block to determine if
|
|
|
|
/// the block is predicable.
|
|
|
|
void IfConverter::AnalyzeBranches(BBInfo &BBI) {
|
2007-06-12 06:26:22 +08:00
|
|
|
if (BBI.IsDone)
|
|
|
|
return;
|
|
|
|
|
2014-04-14 08:51:57 +08:00
|
|
|
BBI.TrueBB = BBI.FalseBB = nullptr;
|
2007-06-12 06:26:22 +08:00
|
|
|
BBI.BrCond.clear();
|
|
|
|
BBI.IsBrAnalyzable =
|
2016-07-15 22:41:04 +08:00
|
|
|
!TII->analyzeBranch(*BBI.BB, BBI.TrueBB, BBI.FalseBB, BBI.BrCond);
|
2019-09-10 02:29:27 +08:00
|
|
|
if (!BBI.IsBrAnalyzable) {
|
|
|
|
BBI.TrueBB = nullptr;
|
|
|
|
BBI.FalseBB = nullptr;
|
|
|
|
BBI.BrCond.clear();
|
|
|
|
}
|
|
|
|
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
SmallVector<MachineOperand, 4> RevCond(BBI.BrCond.begin(), BBI.BrCond.end());
|
|
|
|
BBI.IsBrReversible = (RevCond.size() == 0) ||
|
2016-09-15 04:43:16 +08:00
|
|
|
!TII->reverseBranchCondition(RevCond);
|
2014-04-14 08:51:57 +08:00
|
|
|
BBI.HasFallThrough = BBI.IsBrAnalyzable && BBI.FalseBB == nullptr;
|
2007-06-12 06:26:22 +08:00
|
|
|
|
|
|
|
if (BBI.BrCond.size()) {
|
|
|
|
// No false branch. This BB must end with a conditional branch and a
|
|
|
|
// fallthrough.
|
|
|
|
if (!BBI.FalseBB)
|
2010-06-16 06:18:54 +08:00
|
|
|
BBI.FalseBB = findFalseBlock(BBI.BB, BBI.TrueBB);
|
2009-06-16 05:24:34 +08:00
|
|
|
if (!BBI.FalseBB) {
|
|
|
|
// Malformed bcc? True and false blocks are the same?
|
|
|
|
BBI.IsUnpredicable = true;
|
|
|
|
}
|
2007-06-12 06:26:22 +08:00
|
|
|
}
|
2016-08-06 09:52:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// ScanInstructions - Scan all the instructions in the block to determine if
|
|
|
|
/// the block is predicable. In most cases, that means all the instructions
|
|
|
|
/// in the block are isPredicable(). Also checks if the block contains any
|
|
|
|
/// instruction which can clobber a predicate (e.g. condition code register).
|
|
|
|
/// If so, the block is not predicable unless it's the last instruction.
|
|
|
|
void IfConverter::ScanInstructions(BBInfo &BBI,
|
|
|
|
MachineBasicBlock::iterator &Begin,
|
2016-08-25 05:34:24 +08:00
|
|
|
MachineBasicBlock::iterator &End,
|
|
|
|
bool BranchUnpredicable) const {
|
2016-08-06 09:52:34 +08:00
|
|
|
if (BBI.IsDone || BBI.IsUnpredicable)
|
|
|
|
return;
|
|
|
|
|
|
|
|
bool AlreadyPredicated = !BBI.Predicate.empty();
|
2007-06-12 06:26:22 +08:00
|
|
|
|
|
|
|
BBI.NonPredSize = 0;
|
2010-10-26 08:02:21 +08:00
|
|
|
BBI.ExtraCost = 0;
|
2010-11-03 08:45:17 +08:00
|
|
|
BBI.ExtraCost2 = 0;
|
2007-06-12 06:26:22 +08:00
|
|
|
BBI.ClobbersPred = false;
|
2016-08-17 10:51:59 +08:00
|
|
|
for (MachineInstr &MI : make_range(Begin, End)) {
|
2018-05-09 10:42:00 +08:00
|
|
|
if (MI.isDebugInstr())
|
2010-06-05 07:01:26 +08:00
|
|
|
continue;
|
|
|
|
|
2016-04-15 09:38:41 +08:00
|
|
|
// It's unsafe to duplicate convergent instructions in this context, so set
|
|
|
|
// BBI.CannotBeCopied to true if MI is convergent. To see why, consider the
|
|
|
|
// following CFG, which is subject to our "simple" transformation.
|
|
|
|
//
|
|
|
|
// BB0 // if (c1) goto BB1; else goto BB2;
|
|
|
|
// / \
|
|
|
|
// BB1 |
|
|
|
|
// | BB2 // if (c2) goto TBB; else goto FBB;
|
|
|
|
// | / |
|
|
|
|
// | / |
|
|
|
|
// TBB |
|
|
|
|
// | |
|
|
|
|
// | FBB
|
|
|
|
// |
|
|
|
|
// exit
|
|
|
|
//
|
|
|
|
// Suppose we want to move TBB's contents up into BB1 and BB2 (in BB1 they'd
|
|
|
|
// be unconditional, and in BB2, they'd be predicated upon c2), and suppose
|
|
|
|
// TBB contains a convergent instruction. This is safe iff doing so does
|
|
|
|
// not add a control-flow dependency to the convergent instruction -- i.e.,
|
|
|
|
// it's safe iff the set of control flows that leads us to the convergent
|
|
|
|
// instruction does not get smaller after the transformation.
|
|
|
|
//
|
|
|
|
// Originally we executed TBB if c1 || c2. After the transformation, there
|
|
|
|
// are two copies of TBB's instructions. We get to the first if c1, and we
|
|
|
|
// get to the second if !c1 && c2.
|
|
|
|
//
|
|
|
|
// There are clearly fewer ways to satisfy the condition "c1" than
|
|
|
|
// "c1 || c2". Since we've shrunk the set of control flows which lead to
|
|
|
|
// our convergent instruction, the transformation is unsafe.
|
|
|
|
if (MI.isNotDuplicable() || MI.isConvergent())
|
2007-06-16 05:18:05 +08:00
|
|
|
BBI.CannotBeCopied = true;
|
|
|
|
|
2016-02-23 10:46:52 +08:00
|
|
|
bool isPredicated = TII->isPredicated(MI);
|
|
|
|
bool isCondBr = BBI.IsBrAnalyzable && MI.isConditionalBranch();
|
2007-06-12 06:26:22 +08:00
|
|
|
|
2016-08-25 05:34:24 +08:00
|
|
|
if (BranchUnpredicable && MI.isBranch()) {
|
|
|
|
BBI.IsUnpredicable = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-09-09 22:21:49 +08:00
|
|
|
// A conditional branch is not predicable, but it may be eliminated.
|
|
|
|
if (isCondBr)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!isPredicated) {
|
|
|
|
BBI.NonPredSize++;
|
2016-02-23 10:46:52 +08:00
|
|
|
unsigned ExtraPredCost = TII->getPredicationCost(MI);
|
|
|
|
unsigned NumCycles = SchedModel.computeInstrLatency(&MI, false);
|
2013-09-09 22:21:49 +08:00
|
|
|
if (NumCycles > 1)
|
|
|
|
BBI.ExtraCost += NumCycles-1;
|
|
|
|
BBI.ExtraCost2 += ExtraPredCost;
|
|
|
|
} else if (!AlreadyPredicated) {
|
|
|
|
// FIXME: This instruction is already predicated before the
|
|
|
|
// if-conversion pass. It's probably something like a conditional move.
|
|
|
|
// Mark this block unpredicable for now.
|
|
|
|
BBI.IsUnpredicable = true;
|
|
|
|
return;
|
2007-07-07 07:24:39 +08:00
|
|
|
}
|
2007-06-12 06:26:22 +08:00
|
|
|
|
|
|
|
if (BBI.ClobbersPred && !isPredicated) {
|
|
|
|
// Predicate modification instruction should end the block (except for
|
|
|
|
// already predicated instructions and end of block branches).
|
|
|
|
// Predicate may have been modified, the subsequent (currently)
|
2007-07-07 07:24:39 +08:00
|
|
|
// unpredicated instructions cannot be correctly predicated.
|
2007-06-12 06:26:22 +08:00
|
|
|
BBI.IsUnpredicable = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-07-11 01:50:43 +08:00
|
|
|
// FIXME: Make use of PredDefs? e.g. ADDC, SUBC sets predicates but are
|
|
|
|
// still potentially predicable.
|
|
|
|
std::vector<MachineOperand> PredDefs;
|
2016-02-23 10:46:52 +08:00
|
|
|
if (TII->DefinesPredicate(MI, PredDefs))
|
2007-06-12 06:26:22 +08:00
|
|
|
BBI.ClobbersPred = true;
|
|
|
|
|
2016-02-23 10:46:52 +08:00
|
|
|
if (!TII->isPredicable(MI)) {
|
2007-06-12 06:26:22 +08:00
|
|
|
BBI.IsUnpredicable = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Determine if the block is a suitable candidate to be predicated by the
|
|
|
|
/// specified predicate.
|
2016-08-25 05:34:24 +08:00
|
|
|
/// @param BBI BBInfo for the block to check
|
|
|
|
/// @param Pred Predicate array for the branch that leads to BBI
|
|
|
|
/// @param isTriangle true if the Analysis is for a triangle
|
|
|
|
/// @param RevBranch true if Reverse(Pred) leads to BBI (e.g. BBI is the false
|
|
|
|
/// case
|
|
|
|
/// @param hasCommonTail true if BBI shares a tail with a sibling block that
|
|
|
|
/// contains any instruction that would make the block unpredicable.
|
2007-06-12 06:26:22 +08:00
|
|
|
bool IfConverter::FeasibilityAnalysis(BBInfo &BBI,
|
2008-08-15 06:49:33 +08:00
|
|
|
SmallVectorImpl<MachineOperand> &Pred,
|
2016-08-25 05:34:24 +08:00
|
|
|
bool isTriangle, bool RevBranch,
|
|
|
|
bool hasCommonTail) {
|
2007-06-16 17:34:52 +08:00
|
|
|
// If the block is dead or unpredicable, then it cannot be predicated.
|
2016-08-25 05:34:24 +08:00
|
|
|
// Two blocks may share a common unpredicable tail, but this doesn't prevent
|
|
|
|
// them from being if-converted. The non-shared portion is assumed to have
|
|
|
|
// been checked
|
|
|
|
if (BBI.IsDone || (BBI.IsUnpredicable && !hasCommonTail))
|
2007-06-12 06:26:22 +08:00
|
|
|
return false;
|
|
|
|
|
[CodeGen][IfCvt] Don't re-ifcvt blocks with unanalyzable terminators.
If we couldn't analyze its terminator (i.e., it's an indirectbr, or some
other weirdness), we can't safely re-if-convert a predicated block,
because we can't tell whether the predicated terminator can
fallthrough (it does).
Currently, we would completely ignore the fallthrough successor. In
the added testcase, this means we used to generate:
...
@ %entry:
cmp r5, #21
ittt ne
@ %cc1f:
cmpne r7, #42
@ %cc2t:
strne.w r5, [r8]
movne pc, r10
@ %cc1t:
...
Whereas the successor of %cc1f was originally %bb1.
With the fix, we get the correct:
...
@ %entry:
cmp r5, #21
itt eq
@ %cc1t:
streq.w r5, [r11]
moveq pc, r0
@ %cc1f:
cmp r7, #42
itt ne
@ %cc2t:
strne.w r5, [r8]
movne pc, r10
@ %bb1:
...
rdar://20192768
Differential Revision: http://reviews.llvm.org/D8509
llvm-svn: 232872
2015-03-21 09:23:15 +08:00
|
|
|
// If it is already predicated but we couldn't analyze its terminator, the
|
|
|
|
// latter might fallthrough, but we can't determine where to.
|
|
|
|
// Conservatively avoid if-converting again.
|
|
|
|
if (BBI.Predicate.size() && !BBI.IsBrAnalyzable)
|
|
|
|
return false;
|
|
|
|
|
2013-07-25 04:20:37 +08:00
|
|
|
// If it is already predicated, check if the new predicate subsumes
|
|
|
|
// its predicate.
|
|
|
|
if (BBI.Predicate.size() && !TII->SubsumesPredicate(Pred, BBI.Predicate))
|
2007-06-12 06:26:22 +08:00
|
|
|
return false;
|
|
|
|
|
2016-08-25 05:34:24 +08:00
|
|
|
if (!hasCommonTail && BBI.BrCond.size()) {
|
2007-06-12 06:26:22 +08:00
|
|
|
if (!isTriangle)
|
|
|
|
return false;
|
|
|
|
|
2009-05-14 07:25:24 +08:00
|
|
|
// Test predicate subsumption.
|
2008-08-15 06:49:33 +08:00
|
|
|
SmallVector<MachineOperand, 4> RevPred(Pred.begin(), Pred.end());
|
|
|
|
SmallVector<MachineOperand, 4> Cond(BBI.BrCond.begin(), BBI.BrCond.end());
|
2007-06-12 06:26:22 +08:00
|
|
|
if (RevBranch) {
|
2016-09-15 04:43:16 +08:00
|
|
|
if (TII->reverseBranchCondition(Cond))
|
2007-06-12 06:26:22 +08:00
|
|
|
return false;
|
|
|
|
}
|
2016-09-15 04:43:16 +08:00
|
|
|
if (TII->reverseBranchCondition(RevPred) ||
|
2007-06-12 06:26:22 +08:00
|
|
|
!TII->SubsumesPredicate(Cond, RevPred))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Analyze the structure of the sub-CFG starting from the specified block.
|
|
|
|
/// Record its successors and whether it looks like an if-conversion candidate.
|
2016-02-23 01:51:28 +08:00
|
|
|
void IfConverter::AnalyzeBlock(
|
2016-08-17 10:52:01 +08:00
|
|
|
MachineBasicBlock &MBB, std::vector<std::unique_ptr<IfcvtToken>> &Tokens) {
|
2015-06-25 04:34:35 +08:00
|
|
|
struct BBState {
|
2016-08-17 10:52:01 +08:00
|
|
|
BBState(MachineBasicBlock &MBB) : MBB(&MBB), SuccsAnalyzed(false) {}
|
2015-06-25 04:34:35 +08:00
|
|
|
MachineBasicBlock *MBB;
|
|
|
|
|
|
|
|
/// This flag is true if MBB's successors have been analyzed.
|
|
|
|
bool SuccsAnalyzed;
|
|
|
|
};
|
2007-05-16 10:00:57 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
// Push MBB to the stack.
|
|
|
|
SmallVector<BBState, 16> BBStack(1, MBB);
|
2007-05-18 08:20:58 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
while (!BBStack.empty()) {
|
|
|
|
BBState &State = BBStack.back();
|
|
|
|
MachineBasicBlock *BB = State.MBB;
|
|
|
|
BBInfo &BBI = BBAnalysis[BB->getNumber()];
|
2007-05-25 08:59:01 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
if (!State.SuccsAnalyzed) {
|
|
|
|
if (BBI.IsAnalyzed || BBI.IsBeingAnalyzed) {
|
|
|
|
BBStack.pop_back();
|
|
|
|
continue;
|
|
|
|
}
|
2007-06-12 06:26:22 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
BBI.BB = BB;
|
|
|
|
BBI.IsBeingAnalyzed = true;
|
2007-05-17 05:54:37 +08:00
|
|
|
|
2016-08-06 09:52:34 +08:00
|
|
|
AnalyzeBranches(BBI);
|
|
|
|
MachineBasicBlock::iterator Begin = BBI.BB->begin();
|
|
|
|
MachineBasicBlock::iterator End = BBI.BB->end();
|
|
|
|
ScanInstructions(BBI, Begin, End);
|
2007-05-18 08:20:58 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
// Unanalyzable or ends with fallthrough or unconditional branch, or if is
|
|
|
|
// not considered for ifcvt anymore.
|
|
|
|
if (!BBI.IsBrAnalyzable || BBI.BrCond.empty() || BBI.IsDone) {
|
|
|
|
BBI.IsBeingAnalyzed = false;
|
|
|
|
BBI.IsAnalyzed = true;
|
|
|
|
BBStack.pop_back();
|
|
|
|
continue;
|
|
|
|
}
|
2009-06-16 05:24:34 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
// Do not ifcvt if either path is a back edge to the entry block.
|
|
|
|
if (BBI.TrueBB == BB || BBI.FalseBB == BB) {
|
|
|
|
BBI.IsBeingAnalyzed = false;
|
|
|
|
BBI.IsAnalyzed = true;
|
|
|
|
BBStack.pop_back();
|
|
|
|
continue;
|
|
|
|
}
|
2007-06-06 18:16:17 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
// Do not ifcvt if true and false fallthrough blocks are the same.
|
|
|
|
if (!BBI.FalseBB) {
|
|
|
|
BBI.IsBeingAnalyzed = false;
|
|
|
|
BBI.IsAnalyzed = true;
|
|
|
|
BBStack.pop_back();
|
|
|
|
continue;
|
|
|
|
}
|
2007-06-06 18:16:17 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
// Push the False and True blocks to the stack.
|
|
|
|
State.SuccsAnalyzed = true;
|
2016-08-17 10:52:01 +08:00
|
|
|
BBStack.push_back(*BBI.FalseBB);
|
|
|
|
BBStack.push_back(*BBI.TrueBB);
|
2015-06-25 04:34:35 +08:00
|
|
|
continue;
|
|
|
|
}
|
2007-06-08 17:36:04 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
BBInfo &TrueBBI = BBAnalysis[BBI.TrueBB->getNumber()];
|
|
|
|
BBInfo &FalseBBI = BBAnalysis[BBI.FalseBB->getNumber()];
|
2011-08-04 06:34:43 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
if (TrueBBI.IsDone && FalseBBI.IsDone) {
|
|
|
|
BBI.IsBeingAnalyzed = false;
|
|
|
|
BBI.IsAnalyzed = true;
|
|
|
|
BBStack.pop_back();
|
|
|
|
continue;
|
|
|
|
}
|
2011-08-04 06:34:43 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
SmallVector<MachineOperand, 4>
|
|
|
|
RevCond(BBI.BrCond.begin(), BBI.BrCond.end());
|
2016-09-15 04:43:16 +08:00
|
|
|
bool CanRevCond = !TII->reverseBranchCondition(RevCond);
|
2007-06-16 17:34:52 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
unsigned Dups = 0;
|
|
|
|
unsigned Dups2 = 0;
|
|
|
|
bool TNeedSub = !TrueBBI.Predicate.empty();
|
|
|
|
bool FNeedSub = !FalseBBI.Predicate.empty();
|
|
|
|
bool Enqueued = false;
|
2010-06-16 06:18:54 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
BranchProbability Prediction = MBPI->getEdgeProbability(BB, TrueBBI.BB);
|
2007-06-16 17:34:52 +08:00
|
|
|
|
2016-08-25 05:34:24 +08:00
|
|
|
if (CanRevCond) {
|
|
|
|
BBInfo TrueBBICalc, FalseBBICalc;
|
2019-10-10 17:58:28 +08:00
|
|
|
auto feasibleDiamond = [&](bool Forked) {
|
|
|
|
bool MeetsSize = MeetIfcvtSizeLimit(TrueBBICalc, FalseBBICalc, *BB,
|
|
|
|
Dups + Dups2, Prediction, Forked);
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
bool TrueFeasible = FeasibilityAnalysis(TrueBBI, BBI.BrCond,
|
|
|
|
/* IsTriangle */ false, /* RevCond */ false,
|
|
|
|
/* hasCommonTail */ true);
|
|
|
|
bool FalseFeasible = FeasibilityAnalysis(FalseBBI, RevCond,
|
|
|
|
/* IsTriangle */ false, /* RevCond */ false,
|
|
|
|
/* hasCommonTail */ true);
|
|
|
|
return MeetsSize && TrueFeasible && FalseFeasible;
|
|
|
|
};
|
|
|
|
|
2016-08-25 05:34:24 +08:00
|
|
|
if (ValidDiamond(TrueBBI, FalseBBI, Dups, Dups2,
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
TrueBBICalc, FalseBBICalc)) {
|
2019-10-10 17:58:28 +08:00
|
|
|
if (feasibleDiamond(false)) {
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
// Diamond:
|
|
|
|
// EBB
|
|
|
|
// / \_
|
|
|
|
// | |
|
|
|
|
// TBB FBB
|
|
|
|
// \ /
|
|
|
|
// TailBB
|
|
|
|
// Note TailBB can be empty.
|
2019-08-15 23:54:37 +08:00
|
|
|
Tokens.push_back(std::make_unique<IfcvtToken>(
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
BBI, ICDiamond, TNeedSub | FNeedSub, Dups, Dups2,
|
|
|
|
(bool) TrueBBICalc.ClobbersPred, (bool) FalseBBICalc.ClobbersPred));
|
|
|
|
Enqueued = true;
|
|
|
|
}
|
|
|
|
} else if (ValidForkedDiamond(TrueBBI, FalseBBI, Dups, Dups2,
|
|
|
|
TrueBBICalc, FalseBBICalc)) {
|
2019-10-10 17:58:28 +08:00
|
|
|
if (feasibleDiamond(true)) {
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
// ForkedDiamond:
|
|
|
|
// if TBB and FBB have a common tail that includes their conditional
|
|
|
|
// branch instructions, then we can If Convert this pattern.
|
|
|
|
// EBB
|
|
|
|
// _/ \_
|
|
|
|
// | |
|
|
|
|
// TBB FBB
|
|
|
|
// / \ / \
|
|
|
|
// FalseBB TrueBB FalseBB
|
|
|
|
//
|
2019-08-15 23:54:37 +08:00
|
|
|
Tokens.push_back(std::make_unique<IfcvtToken>(
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
BBI, ICForkedDiamond, TNeedSub | FNeedSub, Dups, Dups2,
|
|
|
|
(bool) TrueBBICalc.ClobbersPred, (bool) FalseBBICalc.ClobbersPred));
|
|
|
|
Enqueued = true;
|
|
|
|
}
|
2016-08-25 05:34:24 +08:00
|
|
|
}
|
2015-06-25 04:34:35 +08:00
|
|
|
}
|
2007-06-16 17:34:52 +08:00
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
if (ValidTriangle(TrueBBI, FalseBBI, false, Dups, Prediction) &&
|
|
|
|
MeetIfcvtSizeLimit(*TrueBBI.BB, TrueBBI.NonPredSize + TrueBBI.ExtraCost,
|
|
|
|
TrueBBI.ExtraCost2, Prediction) &&
|
|
|
|
FeasibilityAnalysis(TrueBBI, BBI.BrCond, true)) {
|
|
|
|
// Triangle:
|
|
|
|
// EBB
|
|
|
|
// | \_
|
|
|
|
// | |
|
|
|
|
// | TBB
|
|
|
|
// | /
|
|
|
|
// FBB
|
2016-02-23 01:51:28 +08:00
|
|
|
Tokens.push_back(
|
2019-08-15 23:54:37 +08:00
|
|
|
std::make_unique<IfcvtToken>(BBI, ICTriangle, TNeedSub, Dups));
|
2007-06-16 17:34:52 +08:00
|
|
|
Enqueued = true;
|
|
|
|
}
|
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
if (ValidTriangle(TrueBBI, FalseBBI, true, Dups, Prediction) &&
|
|
|
|
MeetIfcvtSizeLimit(*TrueBBI.BB, TrueBBI.NonPredSize + TrueBBI.ExtraCost,
|
|
|
|
TrueBBI.ExtraCost2, Prediction) &&
|
|
|
|
FeasibilityAnalysis(TrueBBI, BBI.BrCond, true, true)) {
|
2016-02-23 01:51:28 +08:00
|
|
|
Tokens.push_back(
|
2019-08-15 23:54:37 +08:00
|
|
|
std::make_unique<IfcvtToken>(BBI, ICTriangleRev, TNeedSub, Dups));
|
2007-06-16 17:34:52 +08:00
|
|
|
Enqueued = true;
|
|
|
|
}
|
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
if (ValidSimple(TrueBBI, Dups, Prediction) &&
|
|
|
|
MeetIfcvtSizeLimit(*TrueBBI.BB, TrueBBI.NonPredSize + TrueBBI.ExtraCost,
|
|
|
|
TrueBBI.ExtraCost2, Prediction) &&
|
|
|
|
FeasibilityAnalysis(TrueBBI, BBI.BrCond)) {
|
|
|
|
// Simple (split, no rejoin):
|
|
|
|
// EBB
|
|
|
|
// | \_
|
|
|
|
// | |
|
|
|
|
// | TBB---> exit
|
|
|
|
// |
|
|
|
|
// FBB
|
2016-02-23 01:51:28 +08:00
|
|
|
Tokens.push_back(
|
2019-08-15 23:54:37 +08:00
|
|
|
std::make_unique<IfcvtToken>(BBI, ICSimple, TNeedSub, Dups));
|
2007-06-16 17:34:52 +08:00
|
|
|
Enqueued = true;
|
2007-06-04 14:47:22 +08:00
|
|
|
}
|
|
|
|
|
2015-06-25 04:34:35 +08:00
|
|
|
if (CanRevCond) {
|
|
|
|
// Try the other path...
|
|
|
|
if (ValidTriangle(FalseBBI, TrueBBI, false, Dups,
|
|
|
|
Prediction.getCompl()) &&
|
|
|
|
MeetIfcvtSizeLimit(*FalseBBI.BB,
|
|
|
|
FalseBBI.NonPredSize + FalseBBI.ExtraCost,
|
|
|
|
FalseBBI.ExtraCost2, Prediction.getCompl()) &&
|
|
|
|
FeasibilityAnalysis(FalseBBI, RevCond, true)) {
|
2019-08-15 23:54:37 +08:00
|
|
|
Tokens.push_back(std::make_unique<IfcvtToken>(BBI, ICTriangleFalse,
|
2016-02-23 01:51:28 +08:00
|
|
|
FNeedSub, Dups));
|
2015-06-25 04:34:35 +08:00
|
|
|
Enqueued = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ValidTriangle(FalseBBI, TrueBBI, true, Dups,
|
|
|
|
Prediction.getCompl()) &&
|
|
|
|
MeetIfcvtSizeLimit(*FalseBBI.BB,
|
|
|
|
FalseBBI.NonPredSize + FalseBBI.ExtraCost,
|
|
|
|
FalseBBI.ExtraCost2, Prediction.getCompl()) &&
|
|
|
|
FeasibilityAnalysis(FalseBBI, RevCond, true, true)) {
|
2016-02-23 01:51:28 +08:00
|
|
|
Tokens.push_back(
|
2019-08-15 23:54:37 +08:00
|
|
|
std::make_unique<IfcvtToken>(BBI, ICTriangleFRev, FNeedSub, Dups));
|
2015-06-25 04:34:35 +08:00
|
|
|
Enqueued = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ValidSimple(FalseBBI, Dups, Prediction.getCompl()) &&
|
|
|
|
MeetIfcvtSizeLimit(*FalseBBI.BB,
|
|
|
|
FalseBBI.NonPredSize + FalseBBI.ExtraCost,
|
|
|
|
FalseBBI.ExtraCost2, Prediction.getCompl()) &&
|
|
|
|
FeasibilityAnalysis(FalseBBI, RevCond)) {
|
2016-02-23 01:51:28 +08:00
|
|
|
Tokens.push_back(
|
2019-08-15 23:54:37 +08:00
|
|
|
std::make_unique<IfcvtToken>(BBI, ICSimpleFalse, FNeedSub, Dups));
|
2015-06-25 04:34:35 +08:00
|
|
|
Enqueued = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BBI.IsEnqueued = Enqueued;
|
|
|
|
BBI.IsBeingAnalyzed = false;
|
|
|
|
BBI.IsAnalyzed = true;
|
|
|
|
BBStack.pop_back();
|
|
|
|
}
|
2007-05-19 02:14:37 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Analyze all blocks and find entries for all if-conversion candidates.
|
2016-02-23 01:51:28 +08:00
|
|
|
void IfConverter::AnalyzeBlocks(
|
|
|
|
MachineFunction &MF, std::vector<std::unique_ptr<IfcvtToken>> &Tokens) {
|
2016-08-17 10:51:59 +08:00
|
|
|
for (MachineBasicBlock &MBB : MF)
|
2016-08-17 10:52:01 +08:00
|
|
|
AnalyzeBlock(MBB, Tokens);
|
2007-05-31 03:49:19 +08:00
|
|
|
|
2007-06-01 08:12:12 +08:00
|
|
|
// Sort to favor more complex ifcvt scheme.
|
2019-04-23 22:51:27 +08:00
|
|
|
llvm::stable_sort(Tokens, IfcvtTokenCmp);
|
2007-05-16 10:00:57 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Returns true either if ToMBB is the next block after MBB or that all the
|
|
|
|
/// intervening blocks are empty (given MBB can fall through to its next block).
|
2016-08-17 10:52:01 +08:00
|
|
|
static bool canFallThroughTo(MachineBasicBlock &MBB, MachineBasicBlock &ToMBB) {
|
|
|
|
MachineFunction::iterator PI = MBB.getIterator();
|
2014-03-02 20:27:27 +08:00
|
|
|
MachineFunction::iterator I = std::next(PI);
|
2016-08-17 10:52:01 +08:00
|
|
|
MachineFunction::iterator TI = ToMBB.getIterator();
|
|
|
|
MachineFunction::iterator E = MBB.getParent()->end();
|
2010-06-16 15:35:02 +08:00
|
|
|
while (I != TI) {
|
|
|
|
// Check isSuccessor to avoid case where the next block is empty, but
|
|
|
|
// it's not a successor.
|
2015-10-10 03:13:58 +08:00
|
|
|
if (I == E || !I->empty() || !PI->isSuccessor(&*I))
|
2007-06-04 14:47:22 +08:00
|
|
|
return false;
|
2010-06-16 15:35:02 +08:00
|
|
|
PI = I++;
|
|
|
|
}
|
2017-05-10 21:06:13 +08:00
|
|
|
// Finally see if the last I is indeed a successor to PI.
|
|
|
|
return PI->isSuccessor(&*I);
|
2007-05-22 06:22:58 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Invalidate predecessor BB info so it would be re-analyzed to determine if it
|
|
|
|
/// can be if-converted. If predecessor is already enqueued, dequeue it!
|
2016-08-17 10:52:01 +08:00
|
|
|
void IfConverter::InvalidatePreds(MachineBasicBlock &MBB) {
|
|
|
|
for (const MachineBasicBlock *Predecessor : MBB.predecessors()) {
|
2014-08-10 01:21:29 +08:00
|
|
|
BBInfo &PBBI = BBAnalysis[Predecessor->getNumber()];
|
2016-08-17 10:52:01 +08:00
|
|
|
if (PBBI.IsDone || PBBI.BB == &MBB)
|
2007-06-15 07:13:19 +08:00
|
|
|
continue;
|
2007-06-15 07:34:09 +08:00
|
|
|
PBBI.IsAnalyzed = false;
|
|
|
|
PBBI.IsEnqueued = false;
|
2007-05-23 15:23:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Inserts an unconditional branch from \p MBB to \p ToMBB.
|
2016-08-17 10:52:01 +08:00
|
|
|
static void InsertUncondBranch(MachineBasicBlock &MBB, MachineBasicBlock &ToMBB,
|
2007-05-30 06:31:16 +08:00
|
|
|
const TargetInstrInfo *TII) {
|
2010-06-18 06:43:56 +08:00
|
|
|
DebugLoc dl; // FIXME: this is nowhere
|
2008-08-23 00:07:55 +08:00
|
|
|
SmallVector<MachineOperand, 0> NoCond;
|
2016-09-15 01:24:15 +08:00
|
|
|
TII->insertBranch(MBB, &ToMBB, nullptr, NoCond, dl);
|
2007-05-30 06:31:16 +08:00
|
|
|
}
|
|
|
|
|
2013-10-12 03:04:37 +08:00
|
|
|
/// Behaves like LiveRegUnits::StepForward() but also adds implicit uses to all
|
2016-08-03 13:46:35 +08:00
|
|
|
/// values defined in MI which are also live/used by MI.
|
2016-02-23 10:46:52 +08:00
|
|
|
static void UpdatePredRedefs(MachineInstr &MI, LivePhysRegs &Redefs) {
|
2017-10-11 07:50:49 +08:00
|
|
|
const TargetRegisterInfo *TRI = MI.getMF()->getSubtarget().getRegisterInfo();
|
2016-08-03 13:46:35 +08:00
|
|
|
|
|
|
|
// Before stepping forward past MI, remember which regs were live
|
|
|
|
// before MI. This is needed to set the Undef flag only when reg is
|
|
|
|
// dead.
|
2018-11-07 03:00:11 +08:00
|
|
|
SparseSet<MCPhysReg, identity<MCPhysReg>> LiveBeforeMI;
|
2016-08-03 13:46:35 +08:00
|
|
|
LiveBeforeMI.setUniverse(TRI->getNumRegs());
|
2016-08-17 10:51:59 +08:00
|
|
|
for (unsigned Reg : Redefs)
|
2016-08-03 13:46:35 +08:00
|
|
|
LiveBeforeMI.insert(Reg);
|
|
|
|
|
2018-11-07 03:00:11 +08:00
|
|
|
SmallVector<std::pair<MCPhysReg, const MachineOperand*>, 4> Clobbers;
|
2016-02-23 10:46:52 +08:00
|
|
|
Redefs.stepForward(MI, Clobbers);
|
2015-05-06 04:14:22 +08:00
|
|
|
|
|
|
|
// Now add the implicit uses for each of the clobbered values.
|
2016-08-17 10:52:01 +08:00
|
|
|
for (auto Clobber : Clobbers) {
|
2015-05-06 04:14:22 +08:00
|
|
|
// FIXME: Const cast here is nasty, but better than making StepForward
|
|
|
|
// take a mutable instruction instead of const.
|
2016-08-17 10:52:01 +08:00
|
|
|
unsigned Reg = Clobber.first;
|
|
|
|
MachineOperand &Op = const_cast<MachineOperand&>(*Clobber.second);
|
2015-05-07 06:51:04 +08:00
|
|
|
MachineInstr *OpMI = Op.getParent();
|
2017-10-11 07:50:49 +08:00
|
|
|
MachineInstrBuilder MIB(*OpMI->getMF(), OpMI);
|
2015-05-06 06:09:41 +08:00
|
|
|
if (Op.isRegMask()) {
|
|
|
|
// First handle regmasks. They clobber any entries in the mask which
|
|
|
|
// means that we need a def for those registers.
|
2016-08-17 10:52:01 +08:00
|
|
|
if (LiveBeforeMI.count(Reg))
|
|
|
|
MIB.addReg(Reg, RegState::Implicit);
|
2015-05-06 06:09:41 +08:00
|
|
|
|
|
|
|
// We also need to add an implicit def of this register for the later
|
|
|
|
// use to read from.
|
|
|
|
// For the register allocator to have allocated a register clobbered
|
|
|
|
// by the call which is used later, it must be the case that
|
|
|
|
// the call doesn't return.
|
2016-08-17 10:52:01 +08:00
|
|
|
MIB.addReg(Reg, RegState::Implicit | RegState::Define);
|
2015-05-06 06:09:41 +08:00
|
|
|
continue;
|
|
|
|
}
|
2016-08-17 10:52:01 +08:00
|
|
|
if (LiveBeforeMI.count(Reg))
|
|
|
|
MIB.addReg(Reg, RegState::Implicit);
|
2016-09-29 04:07:41 +08:00
|
|
|
else {
|
|
|
|
bool HasLiveSubReg = false;
|
|
|
|
for (MCSubRegIterator S(Reg, TRI); S.isValid(); ++S) {
|
|
|
|
if (!LiveBeforeMI.count(*S))
|
|
|
|
continue;
|
|
|
|
HasLiveSubReg = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (HasLiveSubReg)
|
|
|
|
MIB.addReg(Reg, RegState::Implicit);
|
|
|
|
}
|
2010-06-16 15:35:02 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// If convert a simple (split, no rejoin) sub-CFG.
|
2007-06-16 17:34:52 +08:00
|
|
|
bool IfConverter::IfConvertSimple(BBInfo &BBI, IfcvtKind Kind) {
|
2007-05-22 06:22:58 +08:00
|
|
|
BBInfo &TrueBBI = BBAnalysis[BBI.TrueBB->getNumber()];
|
|
|
|
BBInfo &FalseBBI = BBAnalysis[BBI.FalseBB->getNumber()];
|
|
|
|
BBInfo *CvtBBI = &TrueBBI;
|
|
|
|
BBInfo *NextBBI = &FalseBBI;
|
2007-05-23 15:23:16 +08:00
|
|
|
|
2008-08-15 06:49:33 +08:00
|
|
|
SmallVector<MachineOperand, 4> Cond(BBI.BrCond.begin(), BBI.BrCond.end());
|
2007-06-16 17:34:52 +08:00
|
|
|
if (Kind == ICSimpleFalse)
|
2007-06-02 04:29:21 +08:00
|
|
|
std::swap(CvtBBI, NextBBI);
|
2007-06-16 05:18:05 +08:00
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
MachineBasicBlock &CvtMBB = *CvtBBI->BB;
|
|
|
|
MachineBasicBlock &NextMBB = *NextBBI->BB;
|
2007-06-18 16:37:25 +08:00
|
|
|
if (CvtBBI->IsDone ||
|
2016-08-17 10:52:01 +08:00
|
|
|
(CvtBBI->CannotBeCopied && CvtMBB.pred_size() > 1)) {
|
2007-06-16 05:18:05 +08:00
|
|
|
// Something has changed. It's no longer safe to predicate this block.
|
|
|
|
BBI.IsAnalyzed = false;
|
|
|
|
CvtBBI->IsAnalyzed = false;
|
|
|
|
return false;
|
2007-06-02 04:29:21 +08:00
|
|
|
}
|
2007-05-23 15:23:16 +08:00
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
if (CvtMBB.hasAddressTaken())
|
2013-05-06 02:03:49 +08:00
|
|
|
// Conservatively abort if-conversion if BB's address is taken.
|
|
|
|
return false;
|
|
|
|
|
2007-06-16 17:34:52 +08:00
|
|
|
if (Kind == ICSimpleFalse)
|
2016-09-15 04:43:16 +08:00
|
|
|
if (TII->reverseBranchCondition(Cond))
|
2012-02-05 16:31:47 +08:00
|
|
|
llvm_unreachable("Unable to reverse branch condition!");
|
2007-06-16 05:18:05 +08:00
|
|
|
|
2016-12-08 08:15:51 +08:00
|
|
|
Redefs.init(*TRI);
|
2017-01-06 04:01:19 +08:00
|
|
|
|
|
|
|
if (MRI->tracksLiveness()) {
|
2019-01-09 13:11:10 +08:00
|
|
|
// Initialize liveins to the first BB. These are potentially redefined by
|
2017-01-06 04:01:19 +08:00
|
|
|
// predicated instructions.
|
|
|
|
Redefs.addLiveIns(CvtMBB);
|
|
|
|
Redefs.addLiveIns(NextMBB);
|
|
|
|
}
|
2010-06-16 15:35:02 +08:00
|
|
|
|
2017-06-26 17:33:04 +08:00
|
|
|
// Remove the branches from the entry so we can add the contents of the true
|
|
|
|
// block to it.
|
|
|
|
BBI.NonPredSize -= TII->removeBranch(*BBI.BB);
|
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
if (CvtMBB.pred_size() > 1) {
|
2009-05-14 07:25:24 +08:00
|
|
|
// Copy instructions in the true block, predicate them, and add them to
|
2007-06-15 15:36:12 +08:00
|
|
|
// the entry block.
|
2013-10-15 04:45:17 +08:00
|
|
|
CopyAndPredicateBlock(BBI, *CvtBBI, Cond);
|
2013-04-11 06:05:25 +08:00
|
|
|
|
[IfConversion] Maintain the CFG when predicating/merging blocks in IfConvert*
Summary:
This fixes PR32721 in IfConvertTriangle and possible similar problems in
IfConvertSimple, IfConvertDiamond and IfConvertForkedDiamond.
In PR32721 we had a triangle
EBB
| \
| |
| TBB
| /
FBB
where FBB didn't have any successors at all since it ended with an
unconditional return. Then TBB and FBB were be merged into EBB, but EBB
would still keep its successors, and the use of analyzeBranch and
CorrectExtraCFGEdges wouldn't help to remove them since the return
instruction is not analyzable (at least not on ARM).
The edge updating code and branch probability updating code is now pushed
into MergeBlocks() which allows us to share the same update logic between
more callsites. This lets us remove several dependencies on analyzeBranch
and completely eliminate RemoveExtraEdges.
One thing that showed up with this patch was that IfConversion sometimes
left a successor with 0% probability even if there was no branch or
fallthrough to the successor.
One such example from the test case ifcvt_bad_zero_prob_succ.mir. The
indirect branch tBRIND can only jump to bb.1, but without the patch we
got:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000), %bb.2(0x00000000)
tBRIND %r1, 1, %cpsr
B %bb.1
bb.2:
There is no way to jump from bb.1 to bb2, but still there is a 0% edge
from bb.1 to bb.2.
With the patch applied we instead get the expected:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000)
tBRIND %r1, 1, %cpsr
B %bb.1
Since bb.2 had no predecessor at all, it was removed.
Several testcases had to be updated due to this since the removed
successor made the "Branch Probability Basic Block Placement" pass
sometimes place blocks in a different order.
Finally added a couple of new test cases:
* PR32721_ifcvt_triangle_unanalyzable.mir:
Regression test for the original problem dexcribed in PR 32721.
* ifcvt_triangleWoCvtToNextEdge.mir:
Regression test for problem that caused a revert of my first attempt
to solve PR 32721.
* ifcvt_simple_bad_zero_prob_succ.mir:
Test case showing the problem where a wrong successor with 0% probability
was previously left.
* ifcvt_[diamond|forked_diamond|simple]_unanalyzable.mir
Very simple test cases for the simple and (forked) diamond cases
involving unanalyzable branches that can be nice to have as a base if
wanting to write more complicated tests.
Reviewers: iteratee, MatzeB, grosser, kparzysz
Reviewed By: kparzysz
Subscribers: kbarton, davide, aemerson, nemanjai, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D34099
llvm-svn: 310697
2017-08-11 14:57:08 +08:00
|
|
|
// Keep the CFG updated.
|
2016-08-17 10:52:01 +08:00
|
|
|
BBI.BB->removeSuccessor(&CvtMBB, true);
|
2007-06-15 15:36:12 +08:00
|
|
|
} else {
|
2017-06-26 17:33:04 +08:00
|
|
|
// Predicate the instructions in the true block.
|
2016-08-17 10:52:01 +08:00
|
|
|
PredicateBlock(*CvtBBI, CvtMBB.end(), Cond);
|
2007-05-22 06:22:58 +08:00
|
|
|
|
[IfConversion] Maintain the CFG when predicating/merging blocks in IfConvert*
Summary:
This fixes PR32721 in IfConvertTriangle and possible similar problems in
IfConvertSimple, IfConvertDiamond and IfConvertForkedDiamond.
In PR32721 we had a triangle
EBB
| \
| |
| TBB
| /
FBB
where FBB didn't have any successors at all since it ended with an
unconditional return. Then TBB and FBB were be merged into EBB, but EBB
would still keep its successors, and the use of analyzeBranch and
CorrectExtraCFGEdges wouldn't help to remove them since the return
instruction is not analyzable (at least not on ARM).
The edge updating code and branch probability updating code is now pushed
into MergeBlocks() which allows us to share the same update logic between
more callsites. This lets us remove several dependencies on analyzeBranch
and completely eliminate RemoveExtraEdges.
One thing that showed up with this patch was that IfConversion sometimes
left a successor with 0% probability even if there was no branch or
fallthrough to the successor.
One such example from the test case ifcvt_bad_zero_prob_succ.mir. The
indirect branch tBRIND can only jump to bb.1, but without the patch we
got:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000), %bb.2(0x00000000)
tBRIND %r1, 1, %cpsr
B %bb.1
bb.2:
There is no way to jump from bb.1 to bb2, but still there is a 0% edge
from bb.1 to bb.2.
With the patch applied we instead get the expected:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000)
tBRIND %r1, 1, %cpsr
B %bb.1
Since bb.2 had no predecessor at all, it was removed.
Several testcases had to be updated due to this since the removed
successor made the "Branch Probability Basic Block Placement" pass
sometimes place blocks in a different order.
Finally added a couple of new test cases:
* PR32721_ifcvt_triangle_unanalyzable.mir:
Regression test for the original problem dexcribed in PR 32721.
* ifcvt_triangleWoCvtToNextEdge.mir:
Regression test for problem that caused a revert of my first attempt
to solve PR 32721.
* ifcvt_simple_bad_zero_prob_succ.mir:
Test case showing the problem where a wrong successor with 0% probability
was previously left.
* ifcvt_[diamond|forked_diamond|simple]_unanalyzable.mir
Very simple test cases for the simple and (forked) diamond cases
involving unanalyzable branches that can be nice to have as a base if
wanting to write more complicated tests.
Reviewers: iteratee, MatzeB, grosser, kparzysz
Reviewed By: kparzysz
Subscribers: kbarton, davide, aemerson, nemanjai, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D34099
llvm-svn: 310697
2017-08-11 14:57:08 +08:00
|
|
|
// Merge converted block into entry block. The BB to Cvt edge is removed
|
|
|
|
// by MergeBlocks.
|
2007-06-15 15:36:12 +08:00
|
|
|
MergeBlocks(BBI, *CvtBBI);
|
|
|
|
}
|
2007-06-07 10:12:15 +08:00
|
|
|
|
2007-06-06 10:08:52 +08:00
|
|
|
bool IterIfcvt = true;
|
2016-08-17 10:52:01 +08:00
|
|
|
if (!canFallThroughTo(*BBI.BB, NextMBB)) {
|
|
|
|
InsertUncondBranch(*BBI.BB, NextMBB, TII);
|
2007-06-09 09:03:43 +08:00
|
|
|
BBI.HasFallThrough = false;
|
2007-06-08 17:36:04 +08:00
|
|
|
// Now ifcvt'd block will look like this:
|
|
|
|
// BB:
|
|
|
|
// ...
|
|
|
|
// t, f = cmp
|
|
|
|
// if t op
|
|
|
|
// b BBf
|
|
|
|
//
|
|
|
|
// We cannot further ifcvt this block because the unconditional branch
|
|
|
|
// will have to be predicated on the new condition, that will not be
|
|
|
|
// available if cmp executes.
|
|
|
|
IterIfcvt = false;
|
2007-06-06 10:08:52 +08:00
|
|
|
}
|
2007-05-23 15:23:16 +08:00
|
|
|
|
|
|
|
// Update block info. BB can be iteratively if-converted.
|
2007-06-12 06:26:22 +08:00
|
|
|
if (!IterIfcvt)
|
|
|
|
BBI.IsDone = true;
|
2016-08-17 10:52:01 +08:00
|
|
|
InvalidatePreds(*BBI.BB);
|
2007-06-12 06:26:22 +08:00
|
|
|
CvtBBI->IsDone = true;
|
2007-05-22 06:22:58 +08:00
|
|
|
|
|
|
|
// FIXME: Must maintain LiveIns.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// If convert a triangle sub-CFG.
|
2007-06-16 17:34:52 +08:00
|
|
|
bool IfConverter::IfConvertTriangle(BBInfo &BBI, IfcvtKind Kind) {
|
2007-05-23 15:23:16 +08:00
|
|
|
BBInfo &TrueBBI = BBAnalysis[BBI.TrueBB->getNumber()];
|
2007-06-09 09:03:43 +08:00
|
|
|
BBInfo &FalseBBI = BBAnalysis[BBI.FalseBB->getNumber()];
|
|
|
|
BBInfo *CvtBBI = &TrueBBI;
|
|
|
|
BBInfo *NextBBI = &FalseBBI;
|
2010-06-18 06:43:56 +08:00
|
|
|
DebugLoc dl; // FIXME: this is nowhere
|
2007-06-09 09:03:43 +08:00
|
|
|
|
2008-08-15 06:49:33 +08:00
|
|
|
SmallVector<MachineOperand, 4> Cond(BBI.BrCond.begin(), BBI.BrCond.end());
|
2007-06-16 17:34:52 +08:00
|
|
|
if (Kind == ICTriangleFalse || Kind == ICTriangleFRev)
|
2007-06-09 09:03:43 +08:00
|
|
|
std::swap(CvtBBI, NextBBI);
|
2007-06-16 05:18:05 +08:00
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
MachineBasicBlock &CvtMBB = *CvtBBI->BB;
|
|
|
|
MachineBasicBlock &NextMBB = *NextBBI->BB;
|
2007-06-18 16:37:25 +08:00
|
|
|
if (CvtBBI->IsDone ||
|
2016-08-17 10:52:01 +08:00
|
|
|
(CvtBBI->CannotBeCopied && CvtMBB.pred_size() > 1)) {
|
2007-06-16 05:18:05 +08:00
|
|
|
// Something has changed. It's no longer safe to predicate this block.
|
|
|
|
BBI.IsAnalyzed = false;
|
|
|
|
CvtBBI->IsAnalyzed = false;
|
|
|
|
return false;
|
2007-06-09 09:03:43 +08:00
|
|
|
}
|
2007-06-16 05:18:05 +08:00
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
if (CvtMBB.hasAddressTaken())
|
2013-05-06 02:03:49 +08:00
|
|
|
// Conservatively abort if-conversion if BB's address is taken.
|
|
|
|
return false;
|
|
|
|
|
2007-06-16 17:34:52 +08:00
|
|
|
if (Kind == ICTriangleFalse || Kind == ICTriangleFRev)
|
2016-09-15 04:43:16 +08:00
|
|
|
if (TII->reverseBranchCondition(Cond))
|
2012-02-05 16:31:47 +08:00
|
|
|
llvm_unreachable("Unable to reverse branch condition!");
|
2007-06-16 05:18:05 +08:00
|
|
|
|
2007-06-16 17:34:52 +08:00
|
|
|
if (Kind == ICTriangleRev || Kind == ICTriangleFRev) {
|
2016-09-15 04:43:16 +08:00
|
|
|
if (reverseBranchCondition(*CvtBBI)) {
|
Optimized FCMP_OEQ and FCMP_UNE for x86.
Where previously LLVM might emit code like this:
ucomisd %xmm1, %xmm0
setne %al
setp %cl
orb %al, %cl
jne .LBB4_2
it now emits this:
ucomisd %xmm1, %xmm0
jne .LBB4_2
jp .LBB4_2
It has fewer instructions and uses fewer registers, but it does
have more branches. And in the case that this code is followed by
a non-fallthrough edge, it may be followed by a jmp instruction,
resulting in three branch instructions in sequence. Some effort
is made to avoid this situation.
To achieve this, X86ISelLowering.cpp now recognizes FCMP_OEQ and
FCMP_UNE in lowered form, and replace them with code that emits
two branches, except in the case where it would require converting
a fall-through edge to an explicit branch.
Also, X86InstrInfo.cpp's branch analysis and transform code now
knows now to handle blocks with multiple conditional branches. It
uses loops instead of having fixed checks for up to two
instructions. It can now analyze and transform code generated
from FCMP_OEQ and FCMP_UNE.
llvm-svn: 57873
2008-10-21 11:29:32 +08:00
|
|
|
// BB has been changed, modify its predecessors (except for this
|
|
|
|
// one) so they don't get ifcvt'ed based on bad intel.
|
2016-08-17 10:52:01 +08:00
|
|
|
for (MachineBasicBlock *PBB : CvtMBB.predecessors()) {
|
Optimized FCMP_OEQ and FCMP_UNE for x86.
Where previously LLVM might emit code like this:
ucomisd %xmm1, %xmm0
setne %al
setp %cl
orb %al, %cl
jne .LBB4_2
it now emits this:
ucomisd %xmm1, %xmm0
jne .LBB4_2
jp .LBB4_2
It has fewer instructions and uses fewer registers, but it does
have more branches. And in the case that this code is followed by
a non-fallthrough edge, it may be followed by a jmp instruction,
resulting in three branch instructions in sequence. Some effort
is made to avoid this situation.
To achieve this, X86ISelLowering.cpp now recognizes FCMP_OEQ and
FCMP_UNE in lowered form, and replace them with code that emits
two branches, except in the case where it would require converting
a fall-through edge to an explicit branch.
Also, X86InstrInfo.cpp's branch analysis and transform code now
knows now to handle blocks with multiple conditional branches. It
uses loops instead of having fixed checks for up to two
instructions. It can now analyze and transform code generated
from FCMP_OEQ and FCMP_UNE.
llvm-svn: 57873
2008-10-21 11:29:32 +08:00
|
|
|
if (PBB == BBI.BB)
|
|
|
|
continue;
|
|
|
|
BBInfo &PBBI = BBAnalysis[PBB->getNumber()];
|
|
|
|
if (PBBI.IsEnqueued) {
|
|
|
|
PBBI.IsAnalyzed = false;
|
|
|
|
PBBI.IsEnqueued = false;
|
|
|
|
}
|
2007-06-15 07:34:09 +08:00
|
|
|
}
|
2007-06-13 07:54:05 +08:00
|
|
|
}
|
|
|
|
}
|
2007-06-01 15:41:07 +08:00
|
|
|
|
2010-06-19 13:33:57 +08:00
|
|
|
// Initialize liveins to the first BB. These are potentially redefined by
|
2010-06-16 15:35:02 +08:00
|
|
|
// predicated instructions.
|
2016-12-08 08:15:51 +08:00
|
|
|
Redefs.init(*TRI);
|
2017-01-06 04:01:19 +08:00
|
|
|
if (MRI->tracksLiveness()) {
|
|
|
|
Redefs.addLiveIns(CvtMBB);
|
|
|
|
Redefs.addLiveIns(NextMBB);
|
|
|
|
}
|
2010-06-16 15:35:02 +08:00
|
|
|
|
2014-04-14 08:51:57 +08:00
|
|
|
bool HasEarlyExit = CvtBBI->FalseBB != nullptr;
|
2015-12-01 13:29:22 +08:00
|
|
|
BranchProbability CvtNext, CvtFalse, BBNext, BBCvt;
|
2014-08-10 01:21:29 +08:00
|
|
|
|
2014-01-30 07:18:47 +08:00
|
|
|
if (HasEarlyExit) {
|
2016-08-17 10:52:01 +08:00
|
|
|
// Get probabilities before modifying CvtMBB and BBI.BB.
|
|
|
|
CvtNext = MBPI->getEdgeProbability(&CvtMBB, &NextMBB);
|
|
|
|
CvtFalse = MBPI->getEdgeProbability(&CvtMBB, CvtBBI->FalseBB);
|
|
|
|
BBNext = MBPI->getEdgeProbability(BBI.BB, &NextMBB);
|
|
|
|
BBCvt = MBPI->getEdgeProbability(BBI.BB, &CvtMBB);
|
2014-01-30 07:18:47 +08:00
|
|
|
}
|
2014-08-10 01:21:29 +08:00
|
|
|
|
2017-06-26 17:33:04 +08:00
|
|
|
// Remove the branches from the entry so we can add the contents of the true
|
|
|
|
// block to it.
|
|
|
|
BBI.NonPredSize -= TII->removeBranch(*BBI.BB);
|
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
if (CvtMBB.pred_size() > 1) {
|
2009-05-14 07:25:24 +08:00
|
|
|
// Copy instructions in the true block, predicate them, and add them to
|
2007-06-15 15:36:12 +08:00
|
|
|
// the entry block.
|
2013-10-15 04:45:17 +08:00
|
|
|
CopyAndPredicateBlock(BBI, *CvtBBI, Cond, true);
|
2007-06-15 15:36:12 +08:00
|
|
|
} else {
|
|
|
|
// Predicate the 'true' block after removing its branch.
|
2016-09-15 04:43:16 +08:00
|
|
|
CvtBBI->NonPredSize -= TII->removeBranch(CvtMBB);
|
2016-08-17 10:52:01 +08:00
|
|
|
PredicateBlock(*CvtBBI, CvtMBB.end(), Cond);
|
2007-05-16 10:00:57 +08:00
|
|
|
|
2017-05-12 14:28:58 +08:00
|
|
|
// Now merge the entry of the triangle with the true block.
|
Reapply my if-conversion cleanup from svn r106939 with fixes.
There are 2 changes relative to the previous version of the patch:
1) For the "simple" if-conversion case, there's no need to worry about
RemoveExtraEdges not handling an unanalyzable branch. Predicated terminators
are ignored in this context, so RemoveExtraEdges does the right thing.
This might break someday if we ever treat indirect branches (BRIND) as
predicable, but for now, I just removed this part of the patch, because
in the case where we do not add an unconditional branch, we rely on keeping
the fall-through edge to CvtBBI (which is empty after this transformation).
The change relative to the previous patch is:
@@ -1036,10 +1036,6 @@
IterIfcvt = false;
}
- // RemoveExtraEdges won't work if the block has an unanalyzable branch,
- // which is typically the case for IfConvertSimple, so explicitly remove
- // CvtBBI as a successor.
- BBI.BB->removeSuccessor(CvtBBI->BB);
RemoveExtraEdges(BBI);
// Update block info. BB can be iteratively if-converted.
2) My patch exposed a bug in the code for merging the tail of a "diamond",
which had previously never been exercised. The code was simply checking that
the tail had a single predecessor, but there was a case in
MultiSource/Benchmarks/VersaBench/dbms where that single predecessor was
neither edge of the diamond. I added the following change to check for
that:
@@ -1276,7 +1276,18 @@
// tail, add a unconditional branch to it.
if (TailBB) {
BBInfo TailBBI = BBAnalysis[TailBB->getNumber()];
- if (TailBB->pred_size() == 1 && !TailBBI.HasFallThrough) {
+ bool CanMergeTail = !TailBBI.HasFallThrough;
+ // There may still be a fall-through edge from BBI1 or BBI2 to TailBB;
+ // check if there are any other predecessors besides those.
+ unsigned NumPreds = TailBB->pred_size();
+ if (NumPreds > 1)
+ CanMergeTail = false;
+ else if (NumPreds == 1 && CanMergeTail) {
+ MachineBasicBlock::pred_iterator PI = TailBB->pred_begin();
+ if (*PI != BBI1->BB && *PI != BBI2->BB)
+ CanMergeTail = false;
+ }
+ if (CanMergeTail) {
MergeBlocks(BBI, TailBBI);
TailBBI.IsDone = true;
} else {
With these fixes, I was able to run all the SingleSource and MultiSource
tests successfully.
llvm-svn: 107110
2010-06-29 08:55:23 +08:00
|
|
|
MergeBlocks(BBI, *CvtBBI, false);
|
2007-06-19 06:44:57 +08:00
|
|
|
}
|
|
|
|
|
[IfConversion] Maintain the CFG when predicating/merging blocks in IfConvert*
Summary:
This fixes PR32721 in IfConvertTriangle and possible similar problems in
IfConvertSimple, IfConvertDiamond and IfConvertForkedDiamond.
In PR32721 we had a triangle
EBB
| \
| |
| TBB
| /
FBB
where FBB didn't have any successors at all since it ended with an
unconditional return. Then TBB and FBB were be merged into EBB, but EBB
would still keep its successors, and the use of analyzeBranch and
CorrectExtraCFGEdges wouldn't help to remove them since the return
instruction is not analyzable (at least not on ARM).
The edge updating code and branch probability updating code is now pushed
into MergeBlocks() which allows us to share the same update logic between
more callsites. This lets us remove several dependencies on analyzeBranch
and completely eliminate RemoveExtraEdges.
One thing that showed up with this patch was that IfConversion sometimes
left a successor with 0% probability even if there was no branch or
fallthrough to the successor.
One such example from the test case ifcvt_bad_zero_prob_succ.mir. The
indirect branch tBRIND can only jump to bb.1, but without the patch we
got:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000), %bb.2(0x00000000)
tBRIND %r1, 1, %cpsr
B %bb.1
bb.2:
There is no way to jump from bb.1 to bb2, but still there is a 0% edge
from bb.1 to bb.2.
With the patch applied we instead get the expected:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000)
tBRIND %r1, 1, %cpsr
B %bb.1
Since bb.2 had no predecessor at all, it was removed.
Several testcases had to be updated due to this since the removed
successor made the "Branch Probability Basic Block Placement" pass
sometimes place blocks in a different order.
Finally added a couple of new test cases:
* PR32721_ifcvt_triangle_unanalyzable.mir:
Regression test for the original problem dexcribed in PR 32721.
* ifcvt_triangleWoCvtToNextEdge.mir:
Regression test for problem that caused a revert of my first attempt
to solve PR 32721.
* ifcvt_simple_bad_zero_prob_succ.mir:
Test case showing the problem where a wrong successor with 0% probability
was previously left.
* ifcvt_[diamond|forked_diamond|simple]_unanalyzable.mir
Very simple test cases for the simple and (forked) diamond cases
involving unanalyzable branches that can be nice to have as a base if
wanting to write more complicated tests.
Reviewers: iteratee, MatzeB, grosser, kparzysz
Reviewed By: kparzysz
Subscribers: kbarton, davide, aemerson, nemanjai, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D34099
llvm-svn: 310697
2017-08-11 14:57:08 +08:00
|
|
|
// Keep the CFG updated.
|
|
|
|
BBI.BB->removeSuccessor(&CvtMBB, true);
|
|
|
|
|
2007-06-04 14:47:22 +08:00
|
|
|
// If 'true' block has a 'false' successor, add an exit branch to it.
|
2007-06-05 09:31:40 +08:00
|
|
|
if (HasEarlyExit) {
|
2008-08-15 06:49:33 +08:00
|
|
|
SmallVector<MachineOperand, 4> RevCond(CvtBBI->BrCond.begin(),
|
|
|
|
CvtBBI->BrCond.end());
|
2016-09-15 04:43:16 +08:00
|
|
|
if (TII->reverseBranchCondition(RevCond))
|
2012-02-05 16:31:47 +08:00
|
|
|
llvm_unreachable("Unable to reverse branch condition!");
|
2015-12-01 13:29:22 +08:00
|
|
|
|
|
|
|
// Update the edge probability for both CvtBBI->FalseBB and NextBBI.
|
2016-08-17 10:52:01 +08:00
|
|
|
// NewNext = New_Prob(BBI.BB, NextMBB) =
|
|
|
|
// Prob(BBI.BB, NextMBB) +
|
|
|
|
// Prob(BBI.BB, CvtMBB) * Prob(CvtMBB, NextMBB)
|
2015-12-01 13:29:22 +08:00
|
|
|
// NewFalse = New_Prob(BBI.BB, CvtBBI->FalseBB) =
|
2016-08-17 10:52:01 +08:00
|
|
|
// Prob(BBI.BB, CvtMBB) * Prob(CvtMBB, CvtBBI->FalseBB)
|
|
|
|
auto NewTrueBB = getNextBlock(*BBI.BB);
|
2015-12-01 13:29:22 +08:00
|
|
|
auto NewNext = BBNext + BBCvt * CvtNext;
|
2016-08-12 11:55:06 +08:00
|
|
|
auto NewTrueBBIter = find(BBI.BB->successors(), NewTrueBB);
|
2015-12-02 05:50:20 +08:00
|
|
|
if (NewTrueBBIter != BBI.BB->succ_end())
|
|
|
|
BBI.BB->setSuccProbability(NewTrueBBIter, NewNext);
|
2015-12-01 13:29:22 +08:00
|
|
|
|
|
|
|
auto NewFalse = BBCvt * CvtFalse;
|
2016-09-15 01:24:15 +08:00
|
|
|
TII->insertBranch(*BBI.BB, CvtBBI->FalseBB, nullptr, RevCond, dl);
|
2015-12-01 13:29:22 +08:00
|
|
|
BBI.BB->addSuccessor(CvtBBI->FalseBB, NewFalse);
|
2007-06-15 15:36:12 +08:00
|
|
|
}
|
2007-06-08 17:36:04 +08:00
|
|
|
|
|
|
|
// Merge in the 'false' block if the 'false' block has no other
|
2009-05-14 07:25:24 +08:00
|
|
|
// predecessors. Otherwise, add an unconditional branch to 'false'.
|
2007-06-05 08:07:37 +08:00
|
|
|
bool FalseBBDead = false;
|
2007-06-06 10:08:52 +08:00
|
|
|
bool IterIfcvt = true;
|
2016-08-17 10:52:01 +08:00
|
|
|
bool isFallThrough = canFallThroughTo(*BBI.BB, NextMBB);
|
2007-06-07 16:13:00 +08:00
|
|
|
if (!isFallThrough) {
|
|
|
|
// Only merge them if the true block does not fallthrough to the false
|
|
|
|
// block. By not merging them, we make it possible to iteratively
|
|
|
|
// ifcvt the blocks.
|
2007-06-18 16:37:25 +08:00
|
|
|
if (!HasEarlyExit &&
|
2017-05-29 14:12:18 +08:00
|
|
|
NextMBB.pred_size() == 1 && !NextBBI->HasFallThrough &&
|
2016-08-17 10:52:01 +08:00
|
|
|
!NextMBB.hasAddressTaken()) {
|
2007-06-09 09:03:43 +08:00
|
|
|
MergeBlocks(BBI, *NextBBI);
|
2007-06-07 16:13:00 +08:00
|
|
|
FalseBBDead = true;
|
|
|
|
} else {
|
2016-08-17 10:52:01 +08:00
|
|
|
InsertUncondBranch(*BBI.BB, NextMBB, TII);
|
2007-06-09 09:03:43 +08:00
|
|
|
BBI.HasFallThrough = false;
|
2007-06-07 16:13:00 +08:00
|
|
|
}
|
2007-06-08 17:36:04 +08:00
|
|
|
// Mixed predicated and unpredicated code. This cannot be iteratively
|
|
|
|
// predicated.
|
|
|
|
IterIfcvt = false;
|
2007-06-06 10:08:52 +08:00
|
|
|
}
|
2007-05-16 10:00:57 +08:00
|
|
|
|
2007-05-23 15:23:16 +08:00
|
|
|
// Update block info. BB can be iteratively if-converted.
|
2010-06-16 06:18:54 +08:00
|
|
|
if (!IterIfcvt)
|
2007-06-12 06:26:22 +08:00
|
|
|
BBI.IsDone = true;
|
2016-08-17 10:52:01 +08:00
|
|
|
InvalidatePreds(*BBI.BB);
|
2007-06-12 06:26:22 +08:00
|
|
|
CvtBBI->IsDone = true;
|
2007-06-05 08:07:37 +08:00
|
|
|
if (FalseBBDead)
|
2007-06-12 06:26:22 +08:00
|
|
|
NextBBI->IsDone = true;
|
2007-05-16 10:00:57 +08:00
|
|
|
|
2007-05-22 06:22:58 +08:00
|
|
|
// FIXME: Must maintain LiveIns.
|
|
|
|
return true;
|
2007-05-16 10:00:57 +08:00
|
|
|
}
|
|
|
|
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
/// Common code shared between diamond conversions.
|
|
|
|
/// \p BBI, \p TrueBBI, and \p FalseBBI form the diamond shape.
|
|
|
|
/// \p NumDups1 - number of shared instructions at the beginning of \p TrueBBI
|
|
|
|
/// and FalseBBI
|
|
|
|
/// \p NumDups2 - number of shared instructions at the end of \p TrueBBI
|
|
|
|
/// and \p FalseBBI
|
2016-08-30 02:27:12 +08:00
|
|
|
/// \p RemoveBranch - Remove the common branch of the two blocks before
|
|
|
|
/// predicating. Only false for unanalyzable fallthrough
|
|
|
|
/// cases. The caller will replace the branch if necessary.
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
/// \p MergeAddEdges - Add successor edges when merging blocks. Only false for
|
|
|
|
/// unanalyzable fallthrough
|
|
|
|
bool IfConverter::IfConvertDiamondCommon(
|
|
|
|
BBInfo &BBI, BBInfo &TrueBBI, BBInfo &FalseBBI,
|
|
|
|
unsigned NumDups1, unsigned NumDups2,
|
|
|
|
bool TClobbersPred, bool FClobbersPred,
|
2016-08-30 02:27:12 +08:00
|
|
|
bool RemoveBranch, bool MergeAddEdges) {
|
2007-05-16 10:00:57 +08:00
|
|
|
|
2007-06-18 16:37:25 +08:00
|
|
|
if (TrueBBI.IsDone || FalseBBI.IsDone ||
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
TrueBBI.BB->pred_size() > 1 || FalseBBI.BB->pred_size() > 1) {
|
2007-06-18 16:37:25 +08:00
|
|
|
// Something has changed. It's no longer safe to predicate these blocks.
|
|
|
|
BBI.IsAnalyzed = false;
|
|
|
|
TrueBBI.IsAnalyzed = false;
|
|
|
|
FalseBBI.IsAnalyzed = false;
|
|
|
|
return false;
|
2007-05-22 06:22:58 +08:00
|
|
|
}
|
2007-06-18 16:37:25 +08:00
|
|
|
|
2013-05-06 02:03:49 +08:00
|
|
|
if (TrueBBI.BB->hasAddressTaken() || FalseBBI.BB->hasAddressTaken())
|
|
|
|
// Conservatively abort if-conversion if either BB has its address taken.
|
|
|
|
return false;
|
|
|
|
|
Reapply my if-conversion cleanup from svn r106939 with fixes.
There are 2 changes relative to the previous version of the patch:
1) For the "simple" if-conversion case, there's no need to worry about
RemoveExtraEdges not handling an unanalyzable branch. Predicated terminators
are ignored in this context, so RemoveExtraEdges does the right thing.
This might break someday if we ever treat indirect branches (BRIND) as
predicable, but for now, I just removed this part of the patch, because
in the case where we do not add an unconditional branch, we rely on keeping
the fall-through edge to CvtBBI (which is empty after this transformation).
The change relative to the previous patch is:
@@ -1036,10 +1036,6 @@
IterIfcvt = false;
}
- // RemoveExtraEdges won't work if the block has an unanalyzable branch,
- // which is typically the case for IfConvertSimple, so explicitly remove
- // CvtBBI as a successor.
- BBI.BB->removeSuccessor(CvtBBI->BB);
RemoveExtraEdges(BBI);
// Update block info. BB can be iteratively if-converted.
2) My patch exposed a bug in the code for merging the tail of a "diamond",
which had previously never been exercised. The code was simply checking that
the tail had a single predecessor, but there was a case in
MultiSource/Benchmarks/VersaBench/dbms where that single predecessor was
neither edge of the diamond. I added the following change to check for
that:
@@ -1276,7 +1276,18 @@
// tail, add a unconditional branch to it.
if (TailBB) {
BBInfo TailBBI = BBAnalysis[TailBB->getNumber()];
- if (TailBB->pred_size() == 1 && !TailBBI.HasFallThrough) {
+ bool CanMergeTail = !TailBBI.HasFallThrough;
+ // There may still be a fall-through edge from BBI1 or BBI2 to TailBB;
+ // check if there are any other predecessors besides those.
+ unsigned NumPreds = TailBB->pred_size();
+ if (NumPreds > 1)
+ CanMergeTail = false;
+ else if (NumPreds == 1 && CanMergeTail) {
+ MachineBasicBlock::pred_iterator PI = TailBB->pred_begin();
+ if (*PI != BBI1->BB && *PI != BBI2->BB)
+ CanMergeTail = false;
+ }
+ if (CanMergeTail) {
MergeBlocks(BBI, TailBBI);
TailBBI.IsDone = true;
} else {
With these fixes, I was able to run all the SingleSource and MultiSource
tests successfully.
llvm-svn: 107110
2010-06-29 08:55:23 +08:00
|
|
|
// Put the predicated instructions from the 'true' block before the
|
|
|
|
// instructions from the 'false' block, unless the true block would clobber
|
|
|
|
// the predicate, in which case, do the opposite.
|
2007-06-06 07:46:14 +08:00
|
|
|
BBInfo *BBI1 = &TrueBBI;
|
|
|
|
BBInfo *BBI2 = &FalseBBI;
|
2008-08-15 06:49:33 +08:00
|
|
|
SmallVector<MachineOperand, 4> RevCond(BBI.BrCond.begin(), BBI.BrCond.end());
|
2016-09-15 04:43:16 +08:00
|
|
|
if (TII->reverseBranchCondition(RevCond))
|
2012-02-05 16:31:47 +08:00
|
|
|
llvm_unreachable("Unable to reverse branch condition!");
|
2008-08-15 06:49:33 +08:00
|
|
|
SmallVector<MachineOperand, 4> *Cond1 = &BBI.BrCond;
|
|
|
|
SmallVector<MachineOperand, 4> *Cond2 = &RevCond;
|
2007-06-16 17:34:52 +08:00
|
|
|
|
|
|
|
// Figure out the more profitable ordering.
|
|
|
|
bool DoSwap = false;
|
2016-08-25 05:34:24 +08:00
|
|
|
if (TClobbersPred && !FClobbersPred)
|
2007-06-16 17:34:52 +08:00
|
|
|
DoSwap = true;
|
2016-09-03 02:29:28 +08:00
|
|
|
else if (!TClobbersPred && !FClobbersPred) {
|
2007-06-19 06:44:57 +08:00
|
|
|
if (TrueBBI.NonPredSize > FalseBBI.NonPredSize)
|
2007-06-16 17:34:52 +08:00
|
|
|
DoSwap = true;
|
2016-09-03 02:29:28 +08:00
|
|
|
} else if (TClobbersPred && FClobbersPred)
|
|
|
|
llvm_unreachable("Predicate info cannot be clobbered by both sides.");
|
2007-06-16 17:34:52 +08:00
|
|
|
if (DoSwap) {
|
2007-06-06 08:57:55 +08:00
|
|
|
std::swap(BBI1, BBI2);
|
|
|
|
std::swap(Cond1, Cond2);
|
|
|
|
}
|
2007-06-06 07:46:14 +08:00
|
|
|
|
2007-06-18 16:37:25 +08:00
|
|
|
// Remove the conditional branch from entry to the blocks.
|
2016-09-15 04:43:16 +08:00
|
|
|
BBI.NonPredSize -= TII->removeBranch(*BBI.BB);
|
2007-06-18 16:37:25 +08:00
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
MachineBasicBlock &MBB1 = *BBI1->BB;
|
|
|
|
MachineBasicBlock &MBB2 = *BBI2->BB;
|
|
|
|
|
2016-08-12 02:42:06 +08:00
|
|
|
// Initialize the Redefs:
|
|
|
|
// - BB2 live-in regs need implicit uses before being redefined by BB1
|
|
|
|
// instructions.
|
|
|
|
// - BB1 live-out regs need implicit uses before being redefined by BB2
|
|
|
|
// instructions. We start with BB1 live-ins so we have the live-out regs
|
|
|
|
// after tracking the BB1 instructions.
|
2016-12-08 08:15:51 +08:00
|
|
|
Redefs.init(*TRI);
|
2017-01-06 04:01:19 +08:00
|
|
|
if (MRI->tracksLiveness()) {
|
|
|
|
Redefs.addLiveIns(MBB1);
|
|
|
|
Redefs.addLiveIns(MBB2);
|
|
|
|
}
|
2010-06-16 15:35:02 +08:00
|
|
|
|
2007-06-18 16:37:25 +08:00
|
|
|
// Remove the duplicated instructions at the beginnings of both paths.
|
2018-04-20 01:26:46 +08:00
|
|
|
// Skip dbg_value instructions.
|
2016-08-17 10:52:01 +08:00
|
|
|
MachineBasicBlock::iterator DI1 = MBB1.getFirstNonDebugInstr();
|
|
|
|
MachineBasicBlock::iterator DI2 = MBB2.getFirstNonDebugInstr();
|
2007-06-18 16:37:25 +08:00
|
|
|
BBI1->NonPredSize -= NumDups1;
|
|
|
|
BBI2->NonPredSize -= NumDups1;
|
2010-06-29 04:26:00 +08:00
|
|
|
|
|
|
|
// Skip past the dups on each side separately since there may be
|
2018-04-20 01:26:46 +08:00
|
|
|
// differing dbg_value entries. NumDups1 can include a "return"
|
|
|
|
// instruction, if it's not marked as "branch".
|
2010-06-29 04:26:00 +08:00
|
|
|
for (unsigned i = 0; i < NumDups1; ++DI1) {
|
2018-04-20 01:26:46 +08:00
|
|
|
if (DI1 == MBB1.end())
|
|
|
|
break;
|
2018-05-09 10:42:00 +08:00
|
|
|
if (!DI1->isDebugInstr())
|
2010-06-29 04:26:00 +08:00
|
|
|
++i;
|
|
|
|
}
|
2010-06-26 07:05:46 +08:00
|
|
|
while (NumDups1 != 0) {
|
2019-10-08 23:43:12 +08:00
|
|
|
// Since this instruction is going to be deleted, update call
|
|
|
|
// site info state if the instruction is call instruction.
|
2020-02-27 18:44:53 +08:00
|
|
|
if (DI2->shouldUpdateCallSiteInfo())
|
2019-10-08 23:43:12 +08:00
|
|
|
MBB2.getParent()->eraseCallSiteInfo(&*DI2);
|
|
|
|
|
2007-06-18 16:37:25 +08:00
|
|
|
++DI2;
|
2018-04-20 01:26:46 +08:00
|
|
|
if (DI2 == MBB2.end())
|
|
|
|
break;
|
2018-05-09 10:42:00 +08:00
|
|
|
if (!DI2->isDebugInstr())
|
2010-06-29 04:26:00 +08:00
|
|
|
--NumDups1;
|
2007-06-18 16:37:25 +08:00
|
|
|
}
|
2010-06-16 15:35:02 +08:00
|
|
|
|
2017-01-06 04:01:19 +08:00
|
|
|
if (MRI->tracksLiveness()) {
|
|
|
|
for (const MachineInstr &MI : make_range(MBB1.begin(), DI1)) {
|
2018-11-07 03:00:11 +08:00
|
|
|
SmallVector<std::pair<MCPhysReg, const MachineOperand*>, 4> Dummy;
|
2017-01-06 04:01:19 +08:00
|
|
|
Redefs.stepForward(MI, Dummy);
|
|
|
|
}
|
2013-10-12 03:04:37 +08:00
|
|
|
}
|
2018-04-20 01:26:46 +08:00
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
BBI.BB->splice(BBI.BB->end(), &MBB1, MBB1.begin(), DI1);
|
|
|
|
MBB2.erase(MBB2.begin(), DI2);
|
2007-06-18 16:37:25 +08:00
|
|
|
|
2018-04-20 01:26:46 +08:00
|
|
|
// The branches have been checked to match, so it is safe to remove the
|
|
|
|
// branch in BB1 and rely on the copy in BB2. The complication is that
|
|
|
|
// the blocks may end with a return instruction, which may or may not
|
|
|
|
// be marked as "branch". If it's not, then it could be included in
|
|
|
|
// "dups1", leaving the blocks potentially empty after moving the common
|
|
|
|
// duplicates.
|
2016-08-30 02:27:12 +08:00
|
|
|
#ifndef NDEBUG
|
|
|
|
// Unanalyzable branches must match exactly. Check that now.
|
|
|
|
if (!BBI1->IsBrAnalyzable)
|
|
|
|
verifySameBranchInstructions(&MBB1, &MBB2);
|
|
|
|
#endif
|
2019-09-06 04:02:38 +08:00
|
|
|
// Remove duplicated instructions from the tail of MBB1: any branch
|
|
|
|
// instructions, and the common instructions counted by NumDups2.
|
2016-08-17 10:52:01 +08:00
|
|
|
DI1 = MBB1.end();
|
2019-09-06 04:02:38 +08:00
|
|
|
while (DI1 != MBB1.begin()) {
|
|
|
|
MachineBasicBlock::iterator Prev = std::prev(DI1);
|
|
|
|
if (!Prev->isBranch() && !Prev->isDebugInstr())
|
|
|
|
break;
|
|
|
|
DI1 = Prev;
|
|
|
|
}
|
2010-06-15 05:30:32 +08:00
|
|
|
for (unsigned i = 0; i != NumDups2; ) {
|
|
|
|
// NumDups2 only counted non-dbg_value instructions, so this won't
|
|
|
|
// run off the head of the list.
|
2016-08-17 10:52:01 +08:00
|
|
|
assert(DI1 != MBB1.begin());
|
2019-10-08 23:43:12 +08:00
|
|
|
|
2007-06-18 16:37:25 +08:00
|
|
|
--DI1;
|
2019-10-08 23:43:12 +08:00
|
|
|
|
|
|
|
// Since this instruction is going to be deleted, update call
|
|
|
|
// site info state if the instruction is call instruction.
|
2020-02-27 18:44:53 +08:00
|
|
|
if (DI1->shouldUpdateCallSiteInfo())
|
2019-10-08 23:43:12 +08:00
|
|
|
MBB1.getParent()->eraseCallSiteInfo(&*DI1);
|
|
|
|
|
2010-06-15 05:30:32 +08:00
|
|
|
// skip dbg_value instructions
|
2018-05-09 10:42:00 +08:00
|
|
|
if (!DI1->isDebugInstr())
|
2010-06-15 05:30:32 +08:00
|
|
|
++i;
|
|
|
|
}
|
2016-08-17 10:52:01 +08:00
|
|
|
MBB1.erase(DI1, MBB1.end());
|
2007-06-06 07:46:14 +08:00
|
|
|
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
DI2 = BBI2->BB->end();
|
2016-08-30 02:27:12 +08:00
|
|
|
// The branches have been checked to match. Skip over the branch in the false
|
|
|
|
// block so that we don't try to predicate it.
|
|
|
|
if (RemoveBranch)
|
2016-09-15 04:43:16 +08:00
|
|
|
BBI2->NonPredSize -= TII->removeBranch(*BBI2->BB);
|
2016-08-30 02:27:12 +08:00
|
|
|
else {
|
2018-04-20 01:26:46 +08:00
|
|
|
// Make DI2 point to the end of the range where the common "tail"
|
|
|
|
// instructions could be found.
|
|
|
|
while (DI2 != MBB2.begin()) {
|
|
|
|
MachineBasicBlock::iterator Prev = std::prev(DI2);
|
2018-05-09 10:42:00 +08:00
|
|
|
if (!Prev->isBranch() && !Prev->isDebugInstr())
|
2018-04-20 01:26:46 +08:00
|
|
|
break;
|
|
|
|
DI2 = Prev;
|
|
|
|
}
|
2016-08-30 02:27:12 +08:00
|
|
|
}
|
2007-06-18 16:37:25 +08:00
|
|
|
while (NumDups2 != 0) {
|
2010-06-15 05:30:32 +08:00
|
|
|
// NumDups2 only counted non-dbg_value instructions, so this won't
|
|
|
|
// run off the head of the list.
|
2016-08-17 10:52:01 +08:00
|
|
|
assert(DI2 != MBB2.begin());
|
2007-06-18 16:37:25 +08:00
|
|
|
--DI2;
|
2010-06-15 05:30:32 +08:00
|
|
|
// skip dbg_value instructions
|
2018-05-09 10:42:00 +08:00
|
|
|
if (!DI2->isDebugInstr())
|
2010-06-15 05:30:32 +08:00
|
|
|
--NumDups2;
|
2007-06-18 16:37:25 +08:00
|
|
|
}
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
|
|
|
|
// Remember which registers would later be defined by the false block.
|
|
|
|
// This allows us not to predicate instructions in the true block that would
|
|
|
|
// later be re-defined. That is, rather than
|
|
|
|
// subeq r0, r1, #1
|
|
|
|
// addne r0, r1, #1
|
|
|
|
// generate:
|
|
|
|
// sub r0, r1, #1
|
|
|
|
// addne r0, r1, #1
|
2018-11-07 03:00:11 +08:00
|
|
|
SmallSet<MCPhysReg, 4> RedefsByFalse;
|
|
|
|
SmallSet<MCPhysReg, 4> ExtUses;
|
2016-08-17 10:52:01 +08:00
|
|
|
if (TII->isProfitableToUnpredicate(MBB1, MBB2)) {
|
|
|
|
for (const MachineInstr &FI : make_range(MBB2.begin(), DI2)) {
|
2018-05-09 10:42:00 +08:00
|
|
|
if (FI.isDebugInstr())
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
continue;
|
2018-11-07 03:00:11 +08:00
|
|
|
SmallVector<MCPhysReg, 4> Defs;
|
2016-08-17 10:51:59 +08:00
|
|
|
for (const MachineOperand &MO : FI.operands()) {
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
if (!MO.isReg())
|
|
|
|
continue;
|
Apply llvm-prefer-register-over-unsigned from clang-tidy to LLVM
Summary:
This clang-tidy check is looking for unsigned integer variables whose initializer
starts with an implicit cast from llvm::Register and changes the type of the
variable to llvm::Register (dropping the llvm:: where possible).
Partial reverts in:
X86FrameLowering.cpp - Some functions return unsigned and arguably should be MCRegister
X86FixupLEAs.cpp - Some functions return unsigned and arguably should be MCRegister
X86FrameLowering.cpp - Some functions return unsigned and arguably should be MCRegister
HexagonBitSimplify.cpp - Function takes BitTracker::RegisterRef which appears to be unsigned&
MachineVerifier.cpp - Ambiguous operator==() given MCRegister and const Register
PPCFastISel.cpp - No Register::operator-=()
PeepholeOptimizer.cpp - TargetInstrInfo::optimizeLoadInstr() takes an unsigned&
MachineTraceMetrics.cpp - MachineTraceMetrics lacks a suitable constructor
Manual fixups in:
ARMFastISel.cpp - ARMEmitLoad() now takes a Register& instead of unsigned&
HexagonSplitDouble.cpp - Ternary operator was ambiguous between unsigned/Register
HexagonConstExtenders.cpp - Has a local class named Register, used llvm::Register instead of Register.
PPCFastISel.cpp - PPCEmitLoad() now takes a Register& instead of unsigned&
Depends on D65919
Reviewers: arsenm, bogner, craig.topper, RKSimon
Reviewed By: arsenm
Subscribers: RKSimon, craig.topper, lenary, aemerson, wuzish, jholewinski, MatzeB, qcolombet, dschuff, jyknight, dylanmckay, sdardis, nemanjai, jvesely, wdng, nhaehnle, sbc100, jgravelle-google, kristof.beyls, hiraditya, aheejin, kbarton, fedor.sergeev, javed.absar, asb, rbar, johnrusso, simoncook, apazos, sabuasal, niosHD, jrtc27, MaskRay, zzheng, edward-jones, atanasyan, rogfer01, MartinMosbeck, brucehoult, the_o, tpr, PkmX, jocewei, jsji, Petar.Avramovic, asbirlea, Jim, s.egerton, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D65962
llvm-svn: 369041
2019-08-16 03:22:08 +08:00
|
|
|
Register Reg = MO.getReg();
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
if (!Reg)
|
|
|
|
continue;
|
|
|
|
if (MO.isDef()) {
|
|
|
|
Defs.push_back(Reg);
|
|
|
|
} else if (!RedefsByFalse.count(Reg)) {
|
|
|
|
// These are defined before ctrl flow reach the 'false' instructions.
|
|
|
|
// They cannot be modified by the 'true' instructions.
|
2013-05-23 07:17:36 +08:00
|
|
|
for (MCSubRegIterator SubRegs(Reg, TRI, /*IncludeSelf=*/true);
|
|
|
|
SubRegs.isValid(); ++SubRegs)
|
2012-06-02 07:28:30 +08:00
|
|
|
ExtUses.insert(*SubRegs);
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-07 03:00:11 +08:00
|
|
|
for (MCPhysReg Reg : Defs) {
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
if (!ExtUses.count(Reg)) {
|
2013-05-23 07:17:36 +08:00
|
|
|
for (MCSubRegIterator SubRegs(Reg, TRI, /*IncludeSelf=*/true);
|
|
|
|
SubRegs.isValid(); ++SubRegs)
|
2012-06-02 07:28:30 +08:00
|
|
|
RedefsByFalse.insert(*SubRegs);
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Predicate the 'true' block.
|
2016-08-17 10:52:01 +08:00
|
|
|
PredicateBlock(*BBI1, MBB1.end(), *Cond1, &RedefsByFalse);
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
|
2016-01-20 21:14:52 +08:00
|
|
|
// After predicating BBI1, if there is a predicated terminator in BBI1 and
|
|
|
|
// a non-predicated in BBI2, then we don't want to predicate the one from
|
|
|
|
// BBI2. The reason is that if we merged these blocks, we would end up with
|
|
|
|
// two predicated terminators in the same block.
|
2018-04-20 01:26:46 +08:00
|
|
|
// Also, if the branches in MBB1 and MBB2 were non-analyzable, then don't
|
|
|
|
// predicate them either. They were checked to be identical, and so the
|
|
|
|
// same branch would happen regardless of which path was taken.
|
2016-08-17 10:52:01 +08:00
|
|
|
if (!MBB2.empty() && (DI2 == MBB2.end())) {
|
|
|
|
MachineBasicBlock::iterator BBI1T = MBB1.getFirstTerminator();
|
|
|
|
MachineBasicBlock::iterator BBI2T = MBB2.getFirstTerminator();
|
2018-04-20 01:26:46 +08:00
|
|
|
bool BB1Predicated = BBI1T != MBB1.end() && TII->isPredicated(*BBI1T);
|
|
|
|
bool BB2NonPredicated = BBI2T != MBB2.end() && !TII->isPredicated(*BBI2T);
|
|
|
|
if (BB2NonPredicated && (BB1Predicated || !BBI2->IsBrAnalyzable))
|
2016-01-20 21:14:52 +08:00
|
|
|
--DI2;
|
|
|
|
}
|
|
|
|
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
// Predicate the 'false' block.
|
2013-10-15 04:45:17 +08:00
|
|
|
PredicateBlock(*BBI2, DI2, *Cond2);
|
2007-06-06 07:46:14 +08:00
|
|
|
|
2007-06-19 06:44:57 +08:00
|
|
|
// Merge the true block into the entry of the diamond.
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
MergeBlocks(BBI, *BBI1, MergeAddEdges);
|
|
|
|
MergeBlocks(BBI, *BBI2, MergeAddEdges);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// If convert an almost-diamond sub-CFG where the true
|
|
|
|
/// and false blocks share a common tail.
|
|
|
|
bool IfConverter::IfConvertForkedDiamond(
|
|
|
|
BBInfo &BBI, IfcvtKind Kind,
|
|
|
|
unsigned NumDups1, unsigned NumDups2,
|
|
|
|
bool TClobbersPred, bool FClobbersPred) {
|
|
|
|
BBInfo &TrueBBI = BBAnalysis[BBI.TrueBB->getNumber()];
|
|
|
|
BBInfo &FalseBBI = BBAnalysis[BBI.FalseBB->getNumber()];
|
|
|
|
|
|
|
|
// Save the debug location for later.
|
|
|
|
DebugLoc dl;
|
|
|
|
MachineBasicBlock::iterator TIE = TrueBBI.BB->getFirstTerminator();
|
|
|
|
if (TIE != TrueBBI.BB->end())
|
|
|
|
dl = TIE->getDebugLoc();
|
|
|
|
// Removing branches from both blocks is safe, because we have already
|
|
|
|
// determined that both blocks have the same branch instructions. The branch
|
|
|
|
// will be added back at the end, unpredicated.
|
|
|
|
if (!IfConvertDiamondCommon(
|
|
|
|
BBI, TrueBBI, FalseBBI,
|
|
|
|
NumDups1, NumDups2,
|
|
|
|
TClobbersPred, FClobbersPred,
|
2016-08-30 02:27:12 +08:00
|
|
|
/* RemoveBranch */ true, /* MergeAddEdges */ true))
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// Add back the branch.
|
|
|
|
// Debug location saved above when removing the branch from BBI2
|
2016-09-15 01:24:15 +08:00
|
|
|
TII->insertBranch(*BBI.BB, TrueBBI.TrueBB, TrueBBI.FalseBB,
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
TrueBBI.BrCond, dl);
|
|
|
|
|
|
|
|
// Update block info.
|
|
|
|
BBI.IsDone = TrueBBI.IsDone = FalseBBI.IsDone = true;
|
|
|
|
InvalidatePreds(*BBI.BB);
|
|
|
|
|
|
|
|
// FIXME: Must maintain LiveIns.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// If convert a diamond sub-CFG.
|
|
|
|
bool IfConverter::IfConvertDiamond(BBInfo &BBI, IfcvtKind Kind,
|
|
|
|
unsigned NumDups1, unsigned NumDups2,
|
|
|
|
bool TClobbersPred, bool FClobbersPred) {
|
|
|
|
BBInfo &TrueBBI = BBAnalysis[BBI.TrueBB->getNumber()];
|
|
|
|
BBInfo &FalseBBI = BBAnalysis[BBI.FalseBB->getNumber()];
|
|
|
|
MachineBasicBlock *TailBB = TrueBBI.TrueBB;
|
|
|
|
|
|
|
|
// True block must fall through or end with an unanalyzable terminator.
|
|
|
|
if (!TailBB) {
|
|
|
|
if (blockAlwaysFallThrough(TrueBBI))
|
|
|
|
TailBB = FalseBBI.TrueBB;
|
|
|
|
assert((TailBB || !TrueBBI.IsBrAnalyzable) && "Unexpected!");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IfConvertDiamondCommon(
|
|
|
|
BBI, TrueBBI, FalseBBI,
|
|
|
|
NumDups1, NumDups2,
|
2016-09-03 02:29:26 +08:00
|
|
|
TClobbersPred, FClobbersPred,
|
2016-08-30 02:27:12 +08:00
|
|
|
/* RemoveBranch */ TrueBBI.IsBrAnalyzable,
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
/* MergeAddEdges */ TailBB == nullptr))
|
|
|
|
return false;
|
2007-06-06 08:57:55 +08:00
|
|
|
|
2009-05-14 07:25:24 +08:00
|
|
|
// If the if-converted block falls through or unconditionally branches into
|
|
|
|
// the tail block, and the tail block does not have other predecessors, then
|
2007-06-18 16:37:25 +08:00
|
|
|
// fold the tail block in as well. Otherwise, unless it falls through to the
|
|
|
|
// tail, add a unconditional branch to it.
|
|
|
|
if (TailBB) {
|
[IfConversion] Maintain the CFG when predicating/merging blocks in IfConvert*
Summary:
This fixes PR32721 in IfConvertTriangle and possible similar problems in
IfConvertSimple, IfConvertDiamond and IfConvertForkedDiamond.
In PR32721 we had a triangle
EBB
| \
| |
| TBB
| /
FBB
where FBB didn't have any successors at all since it ended with an
unconditional return. Then TBB and FBB were be merged into EBB, but EBB
would still keep its successors, and the use of analyzeBranch and
CorrectExtraCFGEdges wouldn't help to remove them since the return
instruction is not analyzable (at least not on ARM).
The edge updating code and branch probability updating code is now pushed
into MergeBlocks() which allows us to share the same update logic between
more callsites. This lets us remove several dependencies on analyzeBranch
and completely eliminate RemoveExtraEdges.
One thing that showed up with this patch was that IfConversion sometimes
left a successor with 0% probability even if there was no branch or
fallthrough to the successor.
One such example from the test case ifcvt_bad_zero_prob_succ.mir. The
indirect branch tBRIND can only jump to bb.1, but without the patch we
got:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000), %bb.2(0x00000000)
tBRIND %r1, 1, %cpsr
B %bb.1
bb.2:
There is no way to jump from bb.1 to bb2, but still there is a 0% edge
from bb.1 to bb.2.
With the patch applied we instead get the expected:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000)
tBRIND %r1, 1, %cpsr
B %bb.1
Since bb.2 had no predecessor at all, it was removed.
Several testcases had to be updated due to this since the removed
successor made the "Branch Probability Basic Block Placement" pass
sometimes place blocks in a different order.
Finally added a couple of new test cases:
* PR32721_ifcvt_triangle_unanalyzable.mir:
Regression test for the original problem dexcribed in PR 32721.
* ifcvt_triangleWoCvtToNextEdge.mir:
Regression test for problem that caused a revert of my first attempt
to solve PR 32721.
* ifcvt_simple_bad_zero_prob_succ.mir:
Test case showing the problem where a wrong successor with 0% probability
was previously left.
* ifcvt_[diamond|forked_diamond|simple]_unanalyzable.mir
Very simple test cases for the simple and (forked) diamond cases
involving unanalyzable branches that can be nice to have as a base if
wanting to write more complicated tests.
Reviewers: iteratee, MatzeB, grosser, kparzysz
Reviewed By: kparzysz
Subscribers: kbarton, davide, aemerson, nemanjai, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D34099
llvm-svn: 310697
2017-08-11 14:57:08 +08:00
|
|
|
// We need to remove the edges to the true and false blocks manually since
|
|
|
|
// we didn't let IfConvertDiamondCommon update the CFG.
|
|
|
|
BBI.BB->removeSuccessor(TrueBBI.BB);
|
|
|
|
BBI.BB->removeSuccessor(FalseBBI.BB, true);
|
|
|
|
|
2011-11-05 07:49:14 +08:00
|
|
|
BBInfo &TailBBI = BBAnalysis[TailBB->getNumber()];
|
2013-05-06 02:03:49 +08:00
|
|
|
bool CanMergeTail = !TailBBI.HasFallThrough &&
|
|
|
|
!TailBBI.BB->hasAddressTaken();
|
2016-01-20 21:14:52 +08:00
|
|
|
// The if-converted block can still have a predicated terminator
|
|
|
|
// (e.g. a predicated return). If that is the case, we cannot merge
|
|
|
|
// it with the tail block.
|
|
|
|
MachineBasicBlock::const_iterator TI = BBI.BB->getFirstTerminator();
|
2016-02-23 10:46:52 +08:00
|
|
|
if (TI != BBI.BB->end() && TII->isPredicated(*TI))
|
2016-01-20 21:14:52 +08:00
|
|
|
CanMergeTail = false;
|
Reapply my if-conversion cleanup from svn r106939 with fixes.
There are 2 changes relative to the previous version of the patch:
1) For the "simple" if-conversion case, there's no need to worry about
RemoveExtraEdges not handling an unanalyzable branch. Predicated terminators
are ignored in this context, so RemoveExtraEdges does the right thing.
This might break someday if we ever treat indirect branches (BRIND) as
predicable, but for now, I just removed this part of the patch, because
in the case where we do not add an unconditional branch, we rely on keeping
the fall-through edge to CvtBBI (which is empty after this transformation).
The change relative to the previous patch is:
@@ -1036,10 +1036,6 @@
IterIfcvt = false;
}
- // RemoveExtraEdges won't work if the block has an unanalyzable branch,
- // which is typically the case for IfConvertSimple, so explicitly remove
- // CvtBBI as a successor.
- BBI.BB->removeSuccessor(CvtBBI->BB);
RemoveExtraEdges(BBI);
// Update block info. BB can be iteratively if-converted.
2) My patch exposed a bug in the code for merging the tail of a "diamond",
which had previously never been exercised. The code was simply checking that
the tail had a single predecessor, but there was a case in
MultiSource/Benchmarks/VersaBench/dbms where that single predecessor was
neither edge of the diamond. I added the following change to check for
that:
@@ -1276,7 +1276,18 @@
// tail, add a unconditional branch to it.
if (TailBB) {
BBInfo TailBBI = BBAnalysis[TailBB->getNumber()];
- if (TailBB->pred_size() == 1 && !TailBBI.HasFallThrough) {
+ bool CanMergeTail = !TailBBI.HasFallThrough;
+ // There may still be a fall-through edge from BBI1 or BBI2 to TailBB;
+ // check if there are any other predecessors besides those.
+ unsigned NumPreds = TailBB->pred_size();
+ if (NumPreds > 1)
+ CanMergeTail = false;
+ else if (NumPreds == 1 && CanMergeTail) {
+ MachineBasicBlock::pred_iterator PI = TailBB->pred_begin();
+ if (*PI != BBI1->BB && *PI != BBI2->BB)
+ CanMergeTail = false;
+ }
+ if (CanMergeTail) {
MergeBlocks(BBI, TailBBI);
TailBBI.IsDone = true;
} else {
With these fixes, I was able to run all the SingleSource and MultiSource
tests successfully.
llvm-svn: 107110
2010-06-29 08:55:23 +08:00
|
|
|
// There may still be a fall-through edge from BBI1 or BBI2 to TailBB;
|
|
|
|
// check if there are any other predecessors besides those.
|
|
|
|
unsigned NumPreds = TailBB->pred_size();
|
|
|
|
if (NumPreds > 1)
|
|
|
|
CanMergeTail = false;
|
|
|
|
else if (NumPreds == 1 && CanMergeTail) {
|
|
|
|
MachineBasicBlock::pred_iterator PI = TailBB->pred_begin();
|
CodeGen: If Convert blocks that would form a diamond when tail-merged.
The following function currently relies on tail-merging for if
conversion to succeed. The common tail of cond_true and cond_false is
extracted, and this then forms a diamond pattern that can be
successfully if converted.
If this block does not get extracted, either because tail-merging is
disabled or the threshold is higher, we should still recognize this
pattern and if-convert it.
Fixed a regression in the original commit. Need to un-reverse branches after
reversing them, or other conversions go awry.
define i32 @t2(i32 %a, i32 %b) nounwind {
entry:
%tmp1434 = icmp eq i32 %a, %b ; <i1> [#uses=1]
br i1 %tmp1434, label %bb17, label %bb.outer
bb.outer: ; preds = %cond_false, %entry
%b_addr.021.0.ph = phi i32 [ %b, %entry ], [ %tmp10, %cond_false ]
%a_addr.026.0.ph = phi i32 [ %a, %entry ], [ %a_addr.026.0, %cond_false ]
br label %bb
bb: ; preds = %cond_true, %bb.outer
%indvar = phi i32 [ 0, %bb.outer ], [ %indvar.next, %cond_true ]
%tmp. = sub i32 0, %b_addr.021.0.ph
%tmp.40 = mul i32 %indvar, %tmp.
%a_addr.026.0 = add i32 %tmp.40, %a_addr.026.0.ph
%tmp3 = icmp sgt i32 %a_addr.026.0, %b_addr.021.0.ph
br i1 %tmp3, label %cond_true, label %cond_false
cond_true: ; preds = %bb
%tmp7 = sub i32 %a_addr.026.0, %b_addr.021.0.ph
%tmp1437 = icmp eq i32 %tmp7, %b_addr.021.0.ph
%indvar.next = add i32 %indvar, 1
br i1 %tmp1437, label %bb17, label %bb
cond_false: ; preds = %bb
%tmp10 = sub i32 %b_addr.021.0.ph, %a_addr.026.0
%tmp14 = icmp eq i32 %a_addr.026.0, %tmp10
br i1 %tmp14, label %bb17, label %bb.outer
bb17: ; preds = %cond_false, %cond_true, %entry
%a_addr.026.1 = phi i32 [ %a, %entry ], [ %tmp7, %cond_true ], [ %a_addr.026.0, %cond_false ]
ret i32 %a_addr.026.1
}
Without tail-merging or diamond-tail if conversion:
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ble LBB1_3
@ BB#2: @ %cond_true
@ in Loop: Header=BB1_1 Depth=1
subs r0, r0, r1
cmp r1, r0
it ne
cmpne r0, r1
bgt LBB1_4
LBB1_3: @ %cond_false
@ in Loop: Header=BB1_1 Depth=1
subs r1, r1, r0
cmp r1, r0
bne LBB1_1
LBB1_4: @ %bb17
bx lr
With diamond-tail if conversion, but without tail-merging:
@ BB#0: @ %entry
cmp r0, r1
it eq
bxeq lr
LBB1_1: @ %bb
@ =>This Inner Loop Header: Depth=1
cmp r0, r1
ite le
suble r1, r1, r0
subgt r0, r0, r1
cmp r1, r0
bne LBB1_1
@ BB#2: @ %bb17
bx lr
llvm-svn: 279671
2016-08-25 05:34:27 +08:00
|
|
|
if (*PI != TrueBBI.BB && *PI != FalseBBI.BB)
|
Reapply my if-conversion cleanup from svn r106939 with fixes.
There are 2 changes relative to the previous version of the patch:
1) For the "simple" if-conversion case, there's no need to worry about
RemoveExtraEdges not handling an unanalyzable branch. Predicated terminators
are ignored in this context, so RemoveExtraEdges does the right thing.
This might break someday if we ever treat indirect branches (BRIND) as
predicable, but for now, I just removed this part of the patch, because
in the case where we do not add an unconditional branch, we rely on keeping
the fall-through edge to CvtBBI (which is empty after this transformation).
The change relative to the previous patch is:
@@ -1036,10 +1036,6 @@
IterIfcvt = false;
}
- // RemoveExtraEdges won't work if the block has an unanalyzable branch,
- // which is typically the case for IfConvertSimple, so explicitly remove
- // CvtBBI as a successor.
- BBI.BB->removeSuccessor(CvtBBI->BB);
RemoveExtraEdges(BBI);
// Update block info. BB can be iteratively if-converted.
2) My patch exposed a bug in the code for merging the tail of a "diamond",
which had previously never been exercised. The code was simply checking that
the tail had a single predecessor, but there was a case in
MultiSource/Benchmarks/VersaBench/dbms where that single predecessor was
neither edge of the diamond. I added the following change to check for
that:
@@ -1276,7 +1276,18 @@
// tail, add a unconditional branch to it.
if (TailBB) {
BBInfo TailBBI = BBAnalysis[TailBB->getNumber()];
- if (TailBB->pred_size() == 1 && !TailBBI.HasFallThrough) {
+ bool CanMergeTail = !TailBBI.HasFallThrough;
+ // There may still be a fall-through edge from BBI1 or BBI2 to TailBB;
+ // check if there are any other predecessors besides those.
+ unsigned NumPreds = TailBB->pred_size();
+ if (NumPreds > 1)
+ CanMergeTail = false;
+ else if (NumPreds == 1 && CanMergeTail) {
+ MachineBasicBlock::pred_iterator PI = TailBB->pred_begin();
+ if (*PI != BBI1->BB && *PI != BBI2->BB)
+ CanMergeTail = false;
+ }
+ if (CanMergeTail) {
MergeBlocks(BBI, TailBBI);
TailBBI.IsDone = true;
} else {
With these fixes, I was able to run all the SingleSource and MultiSource
tests successfully.
llvm-svn: 107110
2010-06-29 08:55:23 +08:00
|
|
|
CanMergeTail = false;
|
|
|
|
}
|
|
|
|
if (CanMergeTail) {
|
2007-06-19 06:44:57 +08:00
|
|
|
MergeBlocks(BBI, TailBBI);
|
2007-06-18 16:37:25 +08:00
|
|
|
TailBBI.IsDone = true;
|
|
|
|
} else {
|
2015-12-01 13:29:22 +08:00
|
|
|
BBI.BB->addSuccessor(TailBB, BranchProbability::getOne());
|
2016-08-17 10:52:01 +08:00
|
|
|
InsertUncondBranch(*BBI.BB, *TailBB, TII);
|
2007-06-19 06:44:57 +08:00
|
|
|
BBI.HasFallThrough = false;
|
2007-06-18 16:37:25 +08:00
|
|
|
}
|
2007-05-22 06:22:58 +08:00
|
|
|
}
|
2007-05-18 08:20:58 +08:00
|
|
|
|
2007-06-06 07:46:14 +08:00
|
|
|
// Update block info.
|
2007-06-12 06:26:22 +08:00
|
|
|
BBI.IsDone = TrueBBI.IsDone = FalseBBI.IsDone = true;
|
2016-08-17 10:52:01 +08:00
|
|
|
InvalidatePreds(*BBI.BB);
|
2007-05-16 10:00:57 +08:00
|
|
|
|
2007-05-22 06:22:58 +08:00
|
|
|
// FIXME: Must maintain LiveIns.
|
|
|
|
return true;
|
2007-05-16 10:00:57 +08:00
|
|
|
}
|
|
|
|
|
2016-07-01 07:04:51 +08:00
|
|
|
static bool MaySpeculate(const MachineInstr &MI,
|
2018-11-07 03:00:11 +08:00
|
|
|
SmallSet<MCPhysReg, 4> &LaterRedefs) {
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
bool SawStore = true;
|
2016-07-01 07:04:51 +08:00
|
|
|
if (!MI.isSafeToMove(nullptr, SawStore))
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
return false;
|
|
|
|
|
2016-08-17 10:51:59 +08:00
|
|
|
for (const MachineOperand &MO : MI.operands()) {
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
if (!MO.isReg())
|
|
|
|
continue;
|
Apply llvm-prefer-register-over-unsigned from clang-tidy to LLVM
Summary:
This clang-tidy check is looking for unsigned integer variables whose initializer
starts with an implicit cast from llvm::Register and changes the type of the
variable to llvm::Register (dropping the llvm:: where possible).
Partial reverts in:
X86FrameLowering.cpp - Some functions return unsigned and arguably should be MCRegister
X86FixupLEAs.cpp - Some functions return unsigned and arguably should be MCRegister
X86FrameLowering.cpp - Some functions return unsigned and arguably should be MCRegister
HexagonBitSimplify.cpp - Function takes BitTracker::RegisterRef which appears to be unsigned&
MachineVerifier.cpp - Ambiguous operator==() given MCRegister and const Register
PPCFastISel.cpp - No Register::operator-=()
PeepholeOptimizer.cpp - TargetInstrInfo::optimizeLoadInstr() takes an unsigned&
MachineTraceMetrics.cpp - MachineTraceMetrics lacks a suitable constructor
Manual fixups in:
ARMFastISel.cpp - ARMEmitLoad() now takes a Register& instead of unsigned&
HexagonSplitDouble.cpp - Ternary operator was ambiguous between unsigned/Register
HexagonConstExtenders.cpp - Has a local class named Register, used llvm::Register instead of Register.
PPCFastISel.cpp - PPCEmitLoad() now takes a Register& instead of unsigned&
Depends on D65919
Reviewers: arsenm, bogner, craig.topper, RKSimon
Reviewed By: arsenm
Subscribers: RKSimon, craig.topper, lenary, aemerson, wuzish, jholewinski, MatzeB, qcolombet, dschuff, jyknight, dylanmckay, sdardis, nemanjai, jvesely, wdng, nhaehnle, sbc100, jgravelle-google, kristof.beyls, hiraditya, aheejin, kbarton, fedor.sergeev, javed.absar, asb, rbar, johnrusso, simoncook, apazos, sabuasal, niosHD, jrtc27, MaskRay, zzheng, edward-jones, atanasyan, rogfer01, MartinMosbeck, brucehoult, the_o, tpr, PkmX, jocewei, jsji, Petar.Avramovic, asbirlea, Jim, s.egerton, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D65962
llvm-svn: 369041
2019-08-16 03:22:08 +08:00
|
|
|
Register Reg = MO.getReg();
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
if (!Reg)
|
|
|
|
continue;
|
|
|
|
if (MO.isDef() && !LaterRedefs.count(Reg))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Predicate instructions from the start of the block to the specified end with
|
|
|
|
/// the specified condition.
|
2007-05-23 15:23:16 +08:00
|
|
|
void IfConverter::PredicateBlock(BBInfo &BBI,
|
2007-06-18 16:37:25 +08:00
|
|
|
MachineBasicBlock::iterator E,
|
2010-06-16 15:35:02 +08:00
|
|
|
SmallVectorImpl<MachineOperand> &Cond,
|
2018-11-07 03:00:11 +08:00
|
|
|
SmallSet<MCPhysReg, 4> *LaterRedefs) {
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
bool AnyUnpred = false;
|
2014-04-14 08:51:57 +08:00
|
|
|
bool MaySpec = LaterRedefs != nullptr;
|
2016-08-17 10:51:59 +08:00
|
|
|
for (MachineInstr &I : make_range(BBI.BB->begin(), E)) {
|
2018-05-09 10:42:00 +08:00
|
|
|
if (I.isDebugInstr() || TII->isPredicated(I))
|
2007-05-16 10:00:57 +08:00
|
|
|
continue;
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
// It may be possible not to predicate an instruction if it's the 'true'
|
|
|
|
// side of a diamond and the 'false' side may re-define the instruction's
|
|
|
|
// defs.
|
2015-05-20 05:22:20 +08:00
|
|
|
if (MaySpec && MaySpeculate(I, *LaterRedefs)) {
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
AnyUnpred = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// If any instruction is predicated, then every instruction after it must
|
|
|
|
// be predicated.
|
|
|
|
MaySpec = false;
|
2016-07-01 07:04:51 +08:00
|
|
|
if (!TII->PredicateInstruction(I, Cond)) {
|
2009-07-13 04:07:01 +08:00
|
|
|
#ifndef NDEBUG
|
2016-07-01 07:04:51 +08:00
|
|
|
dbgs() << "Unable to predicate " << I << "!\n";
|
2009-07-13 04:07:01 +08:00
|
|
|
#endif
|
2014-04-14 08:51:57 +08:00
|
|
|
llvm_unreachable(nullptr);
|
2007-05-17 05:54:37 +08:00
|
|
|
}
|
2010-06-16 15:35:02 +08:00
|
|
|
|
2010-06-19 13:33:57 +08:00
|
|
|
// If the predicated instruction now redefines a register as the result of
|
2010-06-16 15:35:02 +08:00
|
|
|
// if-conversion, add an implicit kill.
|
2016-07-01 07:04:51 +08:00
|
|
|
UpdatePredRedefs(I, Redefs);
|
2007-05-16 10:00:57 +08:00
|
|
|
}
|
2007-05-23 15:23:16 +08:00
|
|
|
|
2015-02-28 18:11:12 +08:00
|
|
|
BBI.Predicate.append(Cond.begin(), Cond.end());
|
2007-06-15 15:36:12 +08:00
|
|
|
|
2007-06-15 04:28:52 +08:00
|
|
|
BBI.IsAnalyzed = false;
|
2007-05-23 15:23:16 +08:00
|
|
|
BBI.NonPredSize = 0;
|
2007-06-09 03:17:12 +08:00
|
|
|
|
2010-06-22 23:08:57 +08:00
|
|
|
++NumIfConvBBs;
|
Add a if-conversion optimization that allows 'true' side of a diamond to be
unpredicated. That is, turn
subeq r0, r1, #1
addne r0, r1, #1
into
sub r0, r1, #1
addne r0, r1, #1
For targets where conditional instructions are always executed, this may be
beneficial. It may remove pseudo anti-dependency in out-of-order execution
CPUs. e.g.
op r1, ...
str r1, [r10] ; end-of-life of r1 as div result
cmp r0, #65
movne r1, #44 ; raw dependency on previous r1
moveq r1, #12
If movne is unpredicated, then
op r1, ...
str r1, [r10]
cmp r0, #65
mov r1, #44 ; r1 written unconditionally
moveq r1, #12
Both mov and moveq are no longer depdendent on the first instruction. This gives
the out-of-order execution engine more freedom to reorder them.
This has passed entire LLVM test suite. But it has not been enabled for any ARM
variant pending more performance evaluation.
rdar://8951196
llvm-svn: 146914
2011-12-20 06:01:30 +08:00
|
|
|
if (AnyUnpred)
|
|
|
|
++NumUnpred;
|
2007-06-04 14:47:22 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Copy and predicate instructions from source BB to the destination block.
|
|
|
|
/// Skip end of block branches if IgnoreBr is true.
|
2007-06-15 15:36:12 +08:00
|
|
|
void IfConverter::CopyAndPredicateBlock(BBInfo &ToBBI, BBInfo &FromBBI,
|
2008-08-15 06:49:33 +08:00
|
|
|
SmallVectorImpl<MachineOperand> &Cond,
|
2007-06-15 15:36:12 +08:00
|
|
|
bool IgnoreBr) {
|
2008-07-08 07:14:23 +08:00
|
|
|
MachineFunction &MF = *ToBBI.BB->getParent();
|
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
MachineBasicBlock &FromMBB = *FromBBI.BB;
|
|
|
|
for (MachineInstr &I : FromMBB) {
|
2007-06-15 15:36:12 +08:00
|
|
|
// Do not copy the end of the block branches.
|
2016-02-23 10:46:52 +08:00
|
|
|
if (IgnoreBr && I.isBranch())
|
2007-06-15 15:36:12 +08:00
|
|
|
break;
|
|
|
|
|
2016-02-23 10:46:52 +08:00
|
|
|
MachineInstr *MI = MF.CloneMachineInstr(&I);
|
2019-10-08 23:43:12 +08:00
|
|
|
// Make a copy of the call site info.
|
2020-02-10 16:49:14 +08:00
|
|
|
if (I.isCandidateForCallSiteEntry())
|
|
|
|
MF.copyCallSiteInfo(&I, MI);
|
2019-10-08 23:43:12 +08:00
|
|
|
|
2007-06-15 15:36:12 +08:00
|
|
|
ToBBI.BB->insert(ToBBI.BB->end(), MI);
|
2010-10-26 08:02:21 +08:00
|
|
|
ToBBI.NonPredSize++;
|
2016-02-23 10:46:52 +08:00
|
|
|
unsigned ExtraPredCost = TII->getPredicationCost(I);
|
|
|
|
unsigned NumCycles = SchedModel.computeInstrLatency(&I, false);
|
2010-11-03 08:45:17 +08:00
|
|
|
if (NumCycles > 1)
|
|
|
|
ToBBI.ExtraCost += NumCycles-1;
|
|
|
|
ToBBI.ExtraCost2 += ExtraPredCost;
|
2007-06-15 15:36:12 +08:00
|
|
|
|
2018-05-09 10:42:00 +08:00
|
|
|
if (!TII->isPredicated(I) && !MI->isDebugInstr()) {
|
2016-02-23 10:46:52 +08:00
|
|
|
if (!TII->PredicateInstruction(*MI, Cond)) {
|
2009-07-13 04:07:01 +08:00
|
|
|
#ifndef NDEBUG
|
2016-02-23 10:46:52 +08:00
|
|
|
dbgs() << "Unable to predicate " << I << "!\n";
|
2009-07-13 04:07:01 +08:00
|
|
|
#endif
|
2014-04-14 08:51:57 +08:00
|
|
|
llvm_unreachable(nullptr);
|
2007-06-15 15:36:12 +08:00
|
|
|
}
|
2010-06-16 15:35:02 +08:00
|
|
|
}
|
|
|
|
|
2010-06-19 13:33:57 +08:00
|
|
|
// If the predicated instruction now redefines a register as the result of
|
2010-06-16 15:35:02 +08:00
|
|
|
// if-conversion, add an implicit kill.
|
2016-02-23 10:46:52 +08:00
|
|
|
UpdatePredRedefs(*MI, Redefs);
|
2007-06-15 15:36:12 +08:00
|
|
|
}
|
|
|
|
|
Reapply my if-conversion cleanup from svn r106939 with fixes.
There are 2 changes relative to the previous version of the patch:
1) For the "simple" if-conversion case, there's no need to worry about
RemoveExtraEdges not handling an unanalyzable branch. Predicated terminators
are ignored in this context, so RemoveExtraEdges does the right thing.
This might break someday if we ever treat indirect branches (BRIND) as
predicable, but for now, I just removed this part of the patch, because
in the case where we do not add an unconditional branch, we rely on keeping
the fall-through edge to CvtBBI (which is empty after this transformation).
The change relative to the previous patch is:
@@ -1036,10 +1036,6 @@
IterIfcvt = false;
}
- // RemoveExtraEdges won't work if the block has an unanalyzable branch,
- // which is typically the case for IfConvertSimple, so explicitly remove
- // CvtBBI as a successor.
- BBI.BB->removeSuccessor(CvtBBI->BB);
RemoveExtraEdges(BBI);
// Update block info. BB can be iteratively if-converted.
2) My patch exposed a bug in the code for merging the tail of a "diamond",
which had previously never been exercised. The code was simply checking that
the tail had a single predecessor, but there was a case in
MultiSource/Benchmarks/VersaBench/dbms where that single predecessor was
neither edge of the diamond. I added the following change to check for
that:
@@ -1276,7 +1276,18 @@
// tail, add a unconditional branch to it.
if (TailBB) {
BBInfo TailBBI = BBAnalysis[TailBB->getNumber()];
- if (TailBB->pred_size() == 1 && !TailBBI.HasFallThrough) {
+ bool CanMergeTail = !TailBBI.HasFallThrough;
+ // There may still be a fall-through edge from BBI1 or BBI2 to TailBB;
+ // check if there are any other predecessors besides those.
+ unsigned NumPreds = TailBB->pred_size();
+ if (NumPreds > 1)
+ CanMergeTail = false;
+ else if (NumPreds == 1 && CanMergeTail) {
+ MachineBasicBlock::pred_iterator PI = TailBB->pred_begin();
+ if (*PI != BBI1->BB && *PI != BBI2->BB)
+ CanMergeTail = false;
+ }
+ if (CanMergeTail) {
MergeBlocks(BBI, TailBBI);
TailBBI.IsDone = true;
} else {
With these fixes, I was able to run all the SingleSource and MultiSource
tests successfully.
llvm-svn: 107110
2010-06-29 08:55:23 +08:00
|
|
|
if (!IgnoreBr) {
|
2016-08-17 10:52:01 +08:00
|
|
|
std::vector<MachineBasicBlock *> Succs(FromMBB.succ_begin(),
|
|
|
|
FromMBB.succ_end());
|
|
|
|
MachineBasicBlock *NBB = getNextBlock(FromMBB);
|
2014-04-14 08:51:57 +08:00
|
|
|
MachineBasicBlock *FallThrough = FromBBI.HasFallThrough ? NBB : nullptr;
|
2007-06-19 06:44:57 +08:00
|
|
|
|
2016-08-17 10:51:59 +08:00
|
|
|
for (MachineBasicBlock *Succ : Succs) {
|
Reapply my if-conversion cleanup from svn r106939 with fixes.
There are 2 changes relative to the previous version of the patch:
1) For the "simple" if-conversion case, there's no need to worry about
RemoveExtraEdges not handling an unanalyzable branch. Predicated terminators
are ignored in this context, so RemoveExtraEdges does the right thing.
This might break someday if we ever treat indirect branches (BRIND) as
predicable, but for now, I just removed this part of the patch, because
in the case where we do not add an unconditional branch, we rely on keeping
the fall-through edge to CvtBBI (which is empty after this transformation).
The change relative to the previous patch is:
@@ -1036,10 +1036,6 @@
IterIfcvt = false;
}
- // RemoveExtraEdges won't work if the block has an unanalyzable branch,
- // which is typically the case for IfConvertSimple, so explicitly remove
- // CvtBBI as a successor.
- BBI.BB->removeSuccessor(CvtBBI->BB);
RemoveExtraEdges(BBI);
// Update block info. BB can be iteratively if-converted.
2) My patch exposed a bug in the code for merging the tail of a "diamond",
which had previously never been exercised. The code was simply checking that
the tail had a single predecessor, but there was a case in
MultiSource/Benchmarks/VersaBench/dbms where that single predecessor was
neither edge of the diamond. I added the following change to check for
that:
@@ -1276,7 +1276,18 @@
// tail, add a unconditional branch to it.
if (TailBB) {
BBInfo TailBBI = BBAnalysis[TailBB->getNumber()];
- if (TailBB->pred_size() == 1 && !TailBBI.HasFallThrough) {
+ bool CanMergeTail = !TailBBI.HasFallThrough;
+ // There may still be a fall-through edge from BBI1 or BBI2 to TailBB;
+ // check if there are any other predecessors besides those.
+ unsigned NumPreds = TailBB->pred_size();
+ if (NumPreds > 1)
+ CanMergeTail = false;
+ else if (NumPreds == 1 && CanMergeTail) {
+ MachineBasicBlock::pred_iterator PI = TailBB->pred_begin();
+ if (*PI != BBI1->BB && *PI != BBI2->BB)
+ CanMergeTail = false;
+ }
+ if (CanMergeTail) {
MergeBlocks(BBI, TailBBI);
TailBBI.IsDone = true;
} else {
With these fixes, I was able to run all the SingleSource and MultiSource
tests successfully.
llvm-svn: 107110
2010-06-29 08:55:23 +08:00
|
|
|
// Fallthrough edge can't be transferred.
|
|
|
|
if (Succ == FallThrough)
|
|
|
|
continue;
|
|
|
|
ToBBI.BB->addSuccessor(Succ);
|
|
|
|
}
|
2007-06-19 06:44:57 +08:00
|
|
|
}
|
|
|
|
|
2015-02-28 18:11:12 +08:00
|
|
|
ToBBI.Predicate.append(FromBBI.Predicate.begin(), FromBBI.Predicate.end());
|
|
|
|
ToBBI.Predicate.append(Cond.begin(), Cond.end());
|
2007-06-15 15:36:12 +08:00
|
|
|
|
|
|
|
ToBBI.ClobbersPred |= FromBBI.ClobbersPred;
|
|
|
|
ToBBI.IsAnalyzed = false;
|
|
|
|
|
2010-06-22 23:08:57 +08:00
|
|
|
++NumDupBBs;
|
2007-06-15 15:36:12 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:57 +08:00
|
|
|
/// Move all instructions from FromBB to the end of ToBB. This will leave
|
|
|
|
/// FromBB as an empty block, so remove all of its successor edges except for
|
|
|
|
/// the fall-through edge. If AddEdges is true, i.e., when FromBBI's branch is
|
[IfConversion] Maintain the CFG when predicating/merging blocks in IfConvert*
Summary:
This fixes PR32721 in IfConvertTriangle and possible similar problems in
IfConvertSimple, IfConvertDiamond and IfConvertForkedDiamond.
In PR32721 we had a triangle
EBB
| \
| |
| TBB
| /
FBB
where FBB didn't have any successors at all since it ended with an
unconditional return. Then TBB and FBB were be merged into EBB, but EBB
would still keep its successors, and the use of analyzeBranch and
CorrectExtraCFGEdges wouldn't help to remove them since the return
instruction is not analyzable (at least not on ARM).
The edge updating code and branch probability updating code is now pushed
into MergeBlocks() which allows us to share the same update logic between
more callsites. This lets us remove several dependencies on analyzeBranch
and completely eliminate RemoveExtraEdges.
One thing that showed up with this patch was that IfConversion sometimes
left a successor with 0% probability even if there was no branch or
fallthrough to the successor.
One such example from the test case ifcvt_bad_zero_prob_succ.mir. The
indirect branch tBRIND can only jump to bb.1, but without the patch we
got:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000), %bb.2(0x00000000)
tBRIND %r1, 1, %cpsr
B %bb.1
bb.2:
There is no way to jump from bb.1 to bb2, but still there is a 0% edge
from bb.1 to bb.2.
With the patch applied we instead get the expected:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000)
tBRIND %r1, 1, %cpsr
B %bb.1
Since bb.2 had no predecessor at all, it was removed.
Several testcases had to be updated due to this since the removed
successor made the "Branch Probability Basic Block Placement" pass
sometimes place blocks in a different order.
Finally added a couple of new test cases:
* PR32721_ifcvt_triangle_unanalyzable.mir:
Regression test for the original problem dexcribed in PR 32721.
* ifcvt_triangleWoCvtToNextEdge.mir:
Regression test for problem that caused a revert of my first attempt
to solve PR 32721.
* ifcvt_simple_bad_zero_prob_succ.mir:
Test case showing the problem where a wrong successor with 0% probability
was previously left.
* ifcvt_[diamond|forked_diamond|simple]_unanalyzable.mir
Very simple test cases for the simple and (forked) diamond cases
involving unanalyzable branches that can be nice to have as a base if
wanting to write more complicated tests.
Reviewers: iteratee, MatzeB, grosser, kparzysz
Reviewed By: kparzysz
Subscribers: kbarton, davide, aemerson, nemanjai, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D34099
llvm-svn: 310697
2017-08-11 14:57:08 +08:00
|
|
|
/// being moved, add those successor edges to ToBBI and remove the old edge
|
|
|
|
/// from ToBBI to FromBBI.
|
Reapply my if-conversion cleanup from svn r106939 with fixes.
There are 2 changes relative to the previous version of the patch:
1) For the "simple" if-conversion case, there's no need to worry about
RemoveExtraEdges not handling an unanalyzable branch. Predicated terminators
are ignored in this context, so RemoveExtraEdges does the right thing.
This might break someday if we ever treat indirect branches (BRIND) as
predicable, but for now, I just removed this part of the patch, because
in the case where we do not add an unconditional branch, we rely on keeping
the fall-through edge to CvtBBI (which is empty after this transformation).
The change relative to the previous patch is:
@@ -1036,10 +1036,6 @@
IterIfcvt = false;
}
- // RemoveExtraEdges won't work if the block has an unanalyzable branch,
- // which is typically the case for IfConvertSimple, so explicitly remove
- // CvtBBI as a successor.
- BBI.BB->removeSuccessor(CvtBBI->BB);
RemoveExtraEdges(BBI);
// Update block info. BB can be iteratively if-converted.
2) My patch exposed a bug in the code for merging the tail of a "diamond",
which had previously never been exercised. The code was simply checking that
the tail had a single predecessor, but there was a case in
MultiSource/Benchmarks/VersaBench/dbms where that single predecessor was
neither edge of the diamond. I added the following change to check for
that:
@@ -1276,7 +1276,18 @@
// tail, add a unconditional branch to it.
if (TailBB) {
BBInfo TailBBI = BBAnalysis[TailBB->getNumber()];
- if (TailBB->pred_size() == 1 && !TailBBI.HasFallThrough) {
+ bool CanMergeTail = !TailBBI.HasFallThrough;
+ // There may still be a fall-through edge from BBI1 or BBI2 to TailBB;
+ // check if there are any other predecessors besides those.
+ unsigned NumPreds = TailBB->pred_size();
+ if (NumPreds > 1)
+ CanMergeTail = false;
+ else if (NumPreds == 1 && CanMergeTail) {
+ MachineBasicBlock::pred_iterator PI = TailBB->pred_begin();
+ if (*PI != BBI1->BB && *PI != BBI2->BB)
+ CanMergeTail = false;
+ }
+ if (CanMergeTail) {
MergeBlocks(BBI, TailBBI);
TailBBI.IsDone = true;
} else {
With these fixes, I was able to run all the SingleSource and MultiSource
tests successfully.
llvm-svn: 107110
2010-06-29 08:55:23 +08:00
|
|
|
void IfConverter::MergeBlocks(BBInfo &ToBBI, BBInfo &FromBBI, bool AddEdges) {
|
2016-08-17 10:52:01 +08:00
|
|
|
MachineBasicBlock &FromMBB = *FromBBI.BB;
|
|
|
|
assert(!FromMBB.hasAddressTaken() &&
|
2013-05-06 02:03:49 +08:00
|
|
|
"Removing a BB whose address is taken!");
|
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
// In case FromMBB contains terminators (e.g. return instruction),
|
2016-01-20 21:14:52 +08:00
|
|
|
// first move the non-terminator instructions, then the terminators.
|
2016-08-17 10:52:01 +08:00
|
|
|
MachineBasicBlock::iterator FromTI = FromMBB.getFirstTerminator();
|
2016-01-20 21:14:52 +08:00
|
|
|
MachineBasicBlock::iterator ToTI = ToBBI.BB->getFirstTerminator();
|
2016-08-17 10:52:01 +08:00
|
|
|
ToBBI.BB->splice(ToTI, &FromMBB, FromMBB.begin(), FromTI);
|
2016-01-20 21:14:52 +08:00
|
|
|
|
|
|
|
// If FromBB has non-predicated terminator we should copy it at the end.
|
2016-08-17 10:52:01 +08:00
|
|
|
if (FromTI != FromMBB.end() && !TII->isPredicated(*FromTI))
|
2016-01-20 21:14:52 +08:00
|
|
|
ToTI = ToBBI.BB->end();
|
2016-08-17 10:52:01 +08:00
|
|
|
ToBBI.BB->splice(ToTI, &FromMBB, FromTI, FromMBB.end());
|
2007-05-23 15:23:16 +08:00
|
|
|
|
2015-12-17 09:29:08 +08:00
|
|
|
// Force normalizing the successors' probabilities of ToBBI.BB to convert all
|
|
|
|
// unknown probabilities into known ones.
|
|
|
|
// FIXME: This usage is too tricky and in the future we would like to
|
|
|
|
// eliminate all unknown probabilities in MBB.
|
2017-03-07 03:12:42 +08:00
|
|
|
if (ToBBI.IsBrAnalyzable)
|
|
|
|
ToBBI.BB->normalizeSuccProbs();
|
2015-12-17 09:29:08 +08:00
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
SmallVector<MachineBasicBlock *, 4> FromSuccs(FromMBB.succ_begin(),
|
|
|
|
FromMBB.succ_end());
|
|
|
|
MachineBasicBlock *NBB = getNextBlock(FromMBB);
|
2014-04-14 08:51:57 +08:00
|
|
|
MachineBasicBlock *FallThrough = FromBBI.HasFallThrough ? NBB : nullptr;
|
2016-08-17 10:52:01 +08:00
|
|
|
// The edge probability from ToBBI.BB to FromMBB, which is only needed when
|
|
|
|
// AddEdges is true and FromMBB is a successor of ToBBI.BB.
|
2015-12-01 13:29:22 +08:00
|
|
|
auto To2FromProb = BranchProbability::getZero();
|
2016-08-17 10:52:01 +08:00
|
|
|
if (AddEdges && ToBBI.BB->isSuccessor(&FromMBB)) {
|
[IfConversion] Maintain the CFG when predicating/merging blocks in IfConvert*
Summary:
This fixes PR32721 in IfConvertTriangle and possible similar problems in
IfConvertSimple, IfConvertDiamond and IfConvertForkedDiamond.
In PR32721 we had a triangle
EBB
| \
| |
| TBB
| /
FBB
where FBB didn't have any successors at all since it ended with an
unconditional return. Then TBB and FBB were be merged into EBB, but EBB
would still keep its successors, and the use of analyzeBranch and
CorrectExtraCFGEdges wouldn't help to remove them since the return
instruction is not analyzable (at least not on ARM).
The edge updating code and branch probability updating code is now pushed
into MergeBlocks() which allows us to share the same update logic between
more callsites. This lets us remove several dependencies on analyzeBranch
and completely eliminate RemoveExtraEdges.
One thing that showed up with this patch was that IfConversion sometimes
left a successor with 0% probability even if there was no branch or
fallthrough to the successor.
One such example from the test case ifcvt_bad_zero_prob_succ.mir. The
indirect branch tBRIND can only jump to bb.1, but without the patch we
got:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000), %bb.2(0x00000000)
tBRIND %r1, 1, %cpsr
B %bb.1
bb.2:
There is no way to jump from bb.1 to bb2, but still there is a 0% edge
from bb.1 to bb.2.
With the patch applied we instead get the expected:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000)
tBRIND %r1, 1, %cpsr
B %bb.1
Since bb.2 had no predecessor at all, it was removed.
Several testcases had to be updated due to this since the removed
successor made the "Branch Probability Basic Block Placement" pass
sometimes place blocks in a different order.
Finally added a couple of new test cases:
* PR32721_ifcvt_triangle_unanalyzable.mir:
Regression test for the original problem dexcribed in PR 32721.
* ifcvt_triangleWoCvtToNextEdge.mir:
Regression test for problem that caused a revert of my first attempt
to solve PR 32721.
* ifcvt_simple_bad_zero_prob_succ.mir:
Test case showing the problem where a wrong successor with 0% probability
was previously left.
* ifcvt_[diamond|forked_diamond|simple]_unanalyzable.mir
Very simple test cases for the simple and (forked) diamond cases
involving unanalyzable branches that can be nice to have as a base if
wanting to write more complicated tests.
Reviewers: iteratee, MatzeB, grosser, kparzysz
Reviewed By: kparzysz
Subscribers: kbarton, davide, aemerson, nemanjai, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D34099
llvm-svn: 310697
2017-08-11 14:57:08 +08:00
|
|
|
// Remove the old edge but remember the edge probability so we can calculate
|
|
|
|
// the correct weights on the new edges being added further down.
|
2016-08-17 10:52:01 +08:00
|
|
|
To2FromProb = MBPI->getEdgeProbability(ToBBI.BB, &FromMBB);
|
[IfConversion] Maintain the CFG when predicating/merging blocks in IfConvert*
Summary:
This fixes PR32721 in IfConvertTriangle and possible similar problems in
IfConvertSimple, IfConvertDiamond and IfConvertForkedDiamond.
In PR32721 we had a triangle
EBB
| \
| |
| TBB
| /
FBB
where FBB didn't have any successors at all since it ended with an
unconditional return. Then TBB and FBB were be merged into EBB, but EBB
would still keep its successors, and the use of analyzeBranch and
CorrectExtraCFGEdges wouldn't help to remove them since the return
instruction is not analyzable (at least not on ARM).
The edge updating code and branch probability updating code is now pushed
into MergeBlocks() which allows us to share the same update logic between
more callsites. This lets us remove several dependencies on analyzeBranch
and completely eliminate RemoveExtraEdges.
One thing that showed up with this patch was that IfConversion sometimes
left a successor with 0% probability even if there was no branch or
fallthrough to the successor.
One such example from the test case ifcvt_bad_zero_prob_succ.mir. The
indirect branch tBRIND can only jump to bb.1, but without the patch we
got:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000), %bb.2(0x00000000)
tBRIND %r1, 1, %cpsr
B %bb.1
bb.2:
There is no way to jump from bb.1 to bb2, but still there is a 0% edge
from bb.1 to bb.2.
With the patch applied we instead get the expected:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000)
tBRIND %r1, 1, %cpsr
B %bb.1
Since bb.2 had no predecessor at all, it was removed.
Several testcases had to be updated due to this since the removed
successor made the "Branch Probability Basic Block Placement" pass
sometimes place blocks in a different order.
Finally added a couple of new test cases:
* PR32721_ifcvt_triangle_unanalyzable.mir:
Regression test for the original problem dexcribed in PR 32721.
* ifcvt_triangleWoCvtToNextEdge.mir:
Regression test for problem that caused a revert of my first attempt
to solve PR 32721.
* ifcvt_simple_bad_zero_prob_succ.mir:
Test case showing the problem where a wrong successor with 0% probability
was previously left.
* ifcvt_[diamond|forked_diamond|simple]_unanalyzable.mir
Very simple test cases for the simple and (forked) diamond cases
involving unanalyzable branches that can be nice to have as a base if
wanting to write more complicated tests.
Reviewers: iteratee, MatzeB, grosser, kparzysz
Reviewed By: kparzysz
Subscribers: kbarton, davide, aemerson, nemanjai, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D34099
llvm-svn: 310697
2017-08-11 14:57:08 +08:00
|
|
|
ToBBI.BB->removeSuccessor(&FromMBB);
|
2015-12-01 13:29:22 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:51:59 +08:00
|
|
|
for (MachineBasicBlock *Succ : FromSuccs) {
|
2007-06-09 06:01:07 +08:00
|
|
|
// Fallthrough edge can't be transferred.
|
2007-06-07 10:12:15 +08:00
|
|
|
if (Succ == FallThrough)
|
|
|
|
continue;
|
2015-09-19 04:22:41 +08:00
|
|
|
|
2015-12-01 13:29:22 +08:00
|
|
|
auto NewProb = BranchProbability::getZero();
|
2015-09-19 04:22:41 +08:00
|
|
|
if (AddEdges) {
|
2015-12-01 13:29:22 +08:00
|
|
|
// Calculate the edge probability for the edge from ToBBI.BB to Succ,
|
2016-08-17 10:52:01 +08:00
|
|
|
// which is a portion of the edge probability from FromMBB to Succ. The
|
|
|
|
// portion ratio is the edge probability from ToBBI.BB to FromMBB (if
|
2019-01-09 13:11:10 +08:00
|
|
|
// FromBBI is a successor of ToBBI.BB. See comment below for exception).
|
2016-08-17 10:52:01 +08:00
|
|
|
NewProb = MBPI->getEdgeProbability(&FromMBB, Succ);
|
2015-09-19 04:22:41 +08:00
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
// To2FromProb is 0 when FromMBB is not a successor of ToBBI.BB. This
|
|
|
|
// only happens when if-converting a diamond CFG and FromMBB is the
|
|
|
|
// tail BB. In this case FromMBB post-dominates ToBBI.BB and hence we
|
|
|
|
// could just use the probabilities on FromMBB's out-edges when adding
|
2015-12-01 13:29:22 +08:00
|
|
|
// new successors.
|
|
|
|
if (!To2FromProb.isZero())
|
|
|
|
NewProb *= To2FromProb;
|
2015-09-19 04:22:41 +08:00
|
|
|
}
|
|
|
|
|
2016-08-17 10:52:01 +08:00
|
|
|
FromMBB.removeSuccessor(Succ);
|
2015-09-19 04:22:41 +08:00
|
|
|
|
|
|
|
if (AddEdges) {
|
2015-12-01 13:29:22 +08:00
|
|
|
// If the edge from ToBBI.BB to Succ already exists, update the
|
2015-12-13 17:26:17 +08:00
|
|
|
// probability of this edge by adding NewProb to it. An example is shown
|
2016-08-17 10:52:01 +08:00
|
|
|
// below, in which A is ToBBI.BB and B is FromMBB. In this case we
|
2015-12-01 13:29:22 +08:00
|
|
|
// don't have to set C as A's successor as it already is. We only need to
|
|
|
|
// update the edge probability on A->C. Note that B will not be
|
|
|
|
// immediately removed from A's successors. It is possible that B->D is
|
|
|
|
// not removed either if D is a fallthrough of B. Later the edge A->D
|
|
|
|
// (generated here) and B->D will be combined into one edge. To maintain
|
|
|
|
// correct edge probability of this combined edge, we need to set the edge
|
|
|
|
// probability of A->B to zero, which is already done above. The edge
|
|
|
|
// probability on A->D is calculated by scaling the original probability
|
|
|
|
// on A->B by the probability of B->D.
|
2015-09-19 04:22:41 +08:00
|
|
|
//
|
|
|
|
// Before ifcvt: After ifcvt (assume B->D is kept):
|
|
|
|
//
|
|
|
|
// A A
|
|
|
|
// /| /|\
|
|
|
|
// / B / B|
|
|
|
|
// | /| | ||
|
|
|
|
// |/ | | |/
|
|
|
|
// C D C D
|
|
|
|
//
|
|
|
|
if (ToBBI.BB->isSuccessor(Succ))
|
2015-12-01 13:29:22 +08:00
|
|
|
ToBBI.BB->setSuccProbability(
|
2016-08-12 11:55:06 +08:00
|
|
|
find(ToBBI.BB->successors(), Succ),
|
2015-12-01 13:29:22 +08:00
|
|
|
MBPI->getEdgeProbability(ToBBI.BB, Succ) + NewProb);
|
2015-09-19 04:22:41 +08:00
|
|
|
else
|
2015-12-01 13:29:22 +08:00
|
|
|
ToBBI.BB->addSuccessor(Succ, NewProb);
|
2015-09-19 04:22:41 +08:00
|
|
|
}
|
2007-06-07 10:12:15 +08:00
|
|
|
}
|
2007-06-04 14:47:22 +08:00
|
|
|
|
[IfConversion] Maintain the CFG when predicating/merging blocks in IfConvert*
Summary:
This fixes PR32721 in IfConvertTriangle and possible similar problems in
IfConvertSimple, IfConvertDiamond and IfConvertForkedDiamond.
In PR32721 we had a triangle
EBB
| \
| |
| TBB
| /
FBB
where FBB didn't have any successors at all since it ended with an
unconditional return. Then TBB and FBB were be merged into EBB, but EBB
would still keep its successors, and the use of analyzeBranch and
CorrectExtraCFGEdges wouldn't help to remove them since the return
instruction is not analyzable (at least not on ARM).
The edge updating code and branch probability updating code is now pushed
into MergeBlocks() which allows us to share the same update logic between
more callsites. This lets us remove several dependencies on analyzeBranch
and completely eliminate RemoveExtraEdges.
One thing that showed up with this patch was that IfConversion sometimes
left a successor with 0% probability even if there was no branch or
fallthrough to the successor.
One such example from the test case ifcvt_bad_zero_prob_succ.mir. The
indirect branch tBRIND can only jump to bb.1, but without the patch we
got:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000), %bb.2(0x00000000)
tBRIND %r1, 1, %cpsr
B %bb.1
bb.2:
There is no way to jump from bb.1 to bb2, but still there is a 0% edge
from bb.1 to bb.2.
With the patch applied we instead get the expected:
bb.0:
successors: %bb.1(0x80000000)
bb.1:
successors: %bb.1(0x80000000)
tBRIND %r1, 1, %cpsr
B %bb.1
Since bb.2 had no predecessor at all, it was removed.
Several testcases had to be updated due to this since the removed
successor made the "Branch Probability Basic Block Placement" pass
sometimes place blocks in a different order.
Finally added a couple of new test cases:
* PR32721_ifcvt_triangle_unanalyzable.mir:
Regression test for the original problem dexcribed in PR 32721.
* ifcvt_triangleWoCvtToNextEdge.mir:
Regression test for problem that caused a revert of my first attempt
to solve PR 32721.
* ifcvt_simple_bad_zero_prob_succ.mir:
Test case showing the problem where a wrong successor with 0% probability
was previously left.
* ifcvt_[diamond|forked_diamond|simple]_unanalyzable.mir
Very simple test cases for the simple and (forked) diamond cases
involving unanalyzable branches that can be nice to have as a base if
wanting to write more complicated tests.
Reviewers: iteratee, MatzeB, grosser, kparzysz
Reviewed By: kparzysz
Subscribers: kbarton, davide, aemerson, nemanjai, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D34099
llvm-svn: 310697
2017-08-11 14:57:08 +08:00
|
|
|
// Move the now empty FromMBB out of the way to the end of the function so
|
|
|
|
// it doesn't interfere with fallthrough checks done by canFallThroughTo().
|
|
|
|
MachineBasicBlock *Last = &*FromMBB.getParent()->rbegin();
|
|
|
|
if (Last != &FromMBB)
|
|
|
|
FromMBB.moveAfter(Last);
|
2007-06-09 06:01:07 +08:00
|
|
|
|
2015-12-13 17:26:17 +08:00
|
|
|
// Normalize the probabilities of ToBBI.BB's successors with all adjustment
|
|
|
|
// we've done above.
|
2017-03-07 03:12:42 +08:00
|
|
|
if (ToBBI.IsBrAnalyzable && FromBBI.IsBrAnalyzable)
|
|
|
|
ToBBI.BB->normalizeSuccProbs();
|
2015-12-13 17:26:17 +08:00
|
|
|
|
2015-02-28 18:11:12 +08:00
|
|
|
ToBBI.Predicate.append(FromBBI.Predicate.begin(), FromBBI.Predicate.end());
|
2007-06-15 15:36:12 +08:00
|
|
|
FromBBI.Predicate.clear();
|
|
|
|
|
2007-05-23 15:23:16 +08:00
|
|
|
ToBBI.NonPredSize += FromBBI.NonPredSize;
|
2010-10-26 08:02:21 +08:00
|
|
|
ToBBI.ExtraCost += FromBBI.ExtraCost;
|
2010-11-03 08:45:17 +08:00
|
|
|
ToBBI.ExtraCost2 += FromBBI.ExtraCost2;
|
2007-05-23 15:23:16 +08:00
|
|
|
FromBBI.NonPredSize = 0;
|
2010-10-26 08:02:21 +08:00
|
|
|
FromBBI.ExtraCost = 0;
|
2010-11-03 08:45:17 +08:00
|
|
|
FromBBI.ExtraCost2 = 0;
|
2007-06-06 18:16:17 +08:00
|
|
|
|
2007-06-12 06:26:22 +08:00
|
|
|
ToBBI.ClobbersPred |= FromBBI.ClobbersPred;
|
2007-06-09 09:03:43 +08:00
|
|
|
ToBBI.HasFallThrough = FromBBI.HasFallThrough;
|
2007-06-15 04:28:52 +08:00
|
|
|
ToBBI.IsAnalyzed = false;
|
|
|
|
FromBBI.IsAnalyzed = false;
|
2007-05-16 10:00:57 +08:00
|
|
|
}
|
2015-06-09 02:50:43 +08:00
|
|
|
|
|
|
|
FunctionPass *
|
2016-10-25 07:23:02 +08:00
|
|
|
llvm::createIfConverter(std::function<bool(const MachineFunction &)> Ftor) {
|
2016-06-13 00:13:55 +08:00
|
|
|
return new IfConverter(std::move(Ftor));
|
2015-06-09 02:50:43 +08:00
|
|
|
}
|