2011-10-17 06:15:07 +08:00
|
|
|
//===- LowerExpectIntrinsic.cpp - Lower expect intrinsic ------------------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2011-10-17 06:15:07 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This pass lowers the 'expect' intrinsic to LLVM metadata.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2015-01-24 19:13:02 +08:00
|
|
|
#include "llvm/Transforms/Scalar/LowerExpectIntrinsic.h"
|
2015-01-24 18:47:13 +08:00
|
|
|
#include "llvm/ADT/SmallVector.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
2017-06-02 10:09:31 +08:00
|
|
|
#include "llvm/ADT/iterator_range.h"
|
2013-01-02 19:36:10 +08:00
|
|
|
#include "llvm/IR/BasicBlock.h"
|
|
|
|
#include "llvm/IR/Constants.h"
|
|
|
|
#include "llvm/IR/Function.h"
|
|
|
|
#include "llvm/IR/Instructions.h"
|
|
|
|
#include "llvm/IR/Intrinsics.h"
|
|
|
|
#include "llvm/IR/LLVMContext.h"
|
|
|
|
#include "llvm/IR/MDBuilder.h"
|
|
|
|
#include "llvm/IR/Metadata.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"
|
2011-07-07 02:22:43 +08:00
|
|
|
#include "llvm/Pass.h"
|
2021-03-21 02:45:56 +08:00
|
|
|
#include "llvm/Support/BranchProbability.h"
|
2011-07-07 02:22:43 +08:00
|
|
|
#include "llvm/Support/Debug.h"
|
2015-01-24 19:13:02 +08:00
|
|
|
#include "llvm/Transforms/Scalar.h"
|
2011-07-07 02:22:43 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
2014-04-22 10:55:47 +08:00
|
|
|
#define DEBUG_TYPE "lower-expect-intrinsic"
|
|
|
|
|
2015-01-24 18:57:25 +08:00
|
|
|
STATISTIC(ExpectIntrinsicsHandled,
|
|
|
|
"Number of 'expect' intrinsic instructions handled");
|
2011-07-07 02:22:43 +08:00
|
|
|
|
2020-10-31 15:15:46 +08:00
|
|
|
static std::tuple<uint32_t, uint32_t>
|
2020-07-09 19:48:56 +08:00
|
|
|
getBranchWeight(Intrinsic::ID IntrinsicID, CallInst *CI, int BranchCount) {
|
2020-06-23 01:20:42 +08:00
|
|
|
if (IntrinsicID == Intrinsic::expect) {
|
|
|
|
// __builtin_expect
|
2020-06-23 03:32:21 +08:00
|
|
|
return std::make_tuple(LikelyBranchWeight.getValue(),
|
|
|
|
UnlikelyBranchWeight.getValue());
|
2020-06-23 01:20:42 +08:00
|
|
|
} else {
|
|
|
|
// __builtin_expect_with_probability
|
|
|
|
assert(CI->getNumOperands() >= 3 &&
|
|
|
|
"expect with probability must have 3 arguments");
|
|
|
|
ConstantFP *Confidence = dyn_cast<ConstantFP>(CI->getArgOperand(2));
|
|
|
|
double TrueProb = Confidence->getValueAPF().convertToDouble();
|
|
|
|
assert((TrueProb >= 0.0 && TrueProb <= 1.0) &&
|
|
|
|
"probability value must be in the range [0.0, 1.0]");
|
|
|
|
double FalseProb = (1.0 - TrueProb) / (BranchCount - 1);
|
2020-10-31 15:15:46 +08:00
|
|
|
uint32_t LikelyBW = ceil((TrueProb * (double)(INT32_MAX - 1)) + 1.0);
|
|
|
|
uint32_t UnlikelyBW = ceil((FalseProb * (double)(INT32_MAX - 1)) + 1.0);
|
2020-06-23 03:32:21 +08:00
|
|
|
return std::make_tuple(LikelyBW, UnlikelyBW);
|
2020-06-23 01:20:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-24 18:39:24 +08:00
|
|
|
static bool handleSwitchExpect(SwitchInst &SI) {
|
|
|
|
CallInst *CI = dyn_cast<CallInst>(SI.getCondition());
|
2011-07-07 02:22:43 +08:00
|
|
|
if (!CI)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
Function *Fn = CI->getCalledFunction();
|
2020-06-23 01:20:42 +08:00
|
|
|
if (!Fn || (Fn->getIntrinsicID() != Intrinsic::expect &&
|
|
|
|
Fn->getIntrinsicID() != Intrinsic::expect_with_probability))
|
2011-07-07 02:22:43 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
Value *ArgValue = CI->getArgOperand(0);
|
|
|
|
ConstantInt *ExpectedValue = dyn_cast<ConstantInt>(CI->getArgOperand(1));
|
|
|
|
if (!ExpectedValue)
|
|
|
|
return false;
|
|
|
|
|
2017-04-12 15:27:28 +08:00
|
|
|
SwitchInst::CaseHandle Case = *SI.findCaseValue(ExpectedValue);
|
2015-01-24 18:39:24 +08:00
|
|
|
unsigned n = SI.getNumCases(); // +1 for default case.
|
2020-10-31 15:15:46 +08:00
|
|
|
uint32_t LikelyBranchWeightVal, UnlikelyBranchWeightVal;
|
2020-06-23 01:20:42 +08:00
|
|
|
std::tie(LikelyBranchWeightVal, UnlikelyBranchWeightVal) =
|
|
|
|
getBranchWeight(Fn->getIntrinsicID(), CI, n + 1);
|
|
|
|
|
2020-10-31 15:15:46 +08:00
|
|
|
SmallVector<uint32_t, 16> Weights(n + 1, UnlikelyBranchWeightVal);
|
2011-07-07 02:22:43 +08:00
|
|
|
|
2019-09-12 00:19:50 +08:00
|
|
|
uint64_t Index = (Case == *SI.case_default()) ? 0 : Case.getCaseIndex() + 1;
|
2020-06-23 01:20:42 +08:00
|
|
|
Weights[Index] = LikelyBranchWeightVal;
|
2019-09-12 00:19:50 +08:00
|
|
|
|
|
|
|
SI.setCondition(ArgValue);
|
2011-07-07 02:22:43 +08:00
|
|
|
|
2015-01-24 18:39:24 +08:00
|
|
|
SI.setMetadata(LLVMContext::MD_prof,
|
|
|
|
MDBuilder(CI->getContext()).createBranchWeights(Weights));
|
2011-07-07 02:22:43 +08:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-06-02 10:09:31 +08:00
|
|
|
/// Handler for PHINodes that define the value argument to an
|
|
|
|
/// @llvm.expect call.
|
|
|
|
///
|
|
|
|
/// If the operand of the phi has a constant value and it 'contradicts'
|
|
|
|
/// with the expected value of phi def, then the corresponding incoming
|
|
|
|
/// edge of the phi is unlikely to be taken. Using that information,
|
|
|
|
/// the branch probability info for the originating branch can be inferred.
|
|
|
|
static void handlePhiDef(CallInst *Expect) {
|
|
|
|
Value &Arg = *Expect->getArgOperand(0);
|
2017-06-08 02:32:24 +08:00
|
|
|
ConstantInt *ExpectedValue = dyn_cast<ConstantInt>(Expect->getArgOperand(1));
|
|
|
|
if (!ExpectedValue)
|
|
|
|
return;
|
2017-06-02 10:09:31 +08:00
|
|
|
const APInt &ExpectedPhiValue = ExpectedValue->getValue();
|
|
|
|
|
|
|
|
// Walk up in backward a list of instructions that
|
|
|
|
// have 'copy' semantics by 'stripping' the copies
|
|
|
|
// until a PHI node or an instruction of unknown kind
|
|
|
|
// is reached. Negation via xor is also handled.
|
|
|
|
//
|
|
|
|
// C = PHI(...);
|
|
|
|
// B = C;
|
|
|
|
// A = B;
|
|
|
|
// D = __builtin_expect(A, 0);
|
|
|
|
//
|
|
|
|
Value *V = &Arg;
|
|
|
|
SmallVector<Instruction *, 4> Operations;
|
|
|
|
while (!isa<PHINode>(V)) {
|
|
|
|
if (ZExtInst *ZExt = dyn_cast<ZExtInst>(V)) {
|
|
|
|
V = ZExt->getOperand(0);
|
|
|
|
Operations.push_back(ZExt);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (SExtInst *SExt = dyn_cast<SExtInst>(V)) {
|
|
|
|
V = SExt->getOperand(0);
|
|
|
|
Operations.push_back(SExt);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
BinaryOperator *BinOp = dyn_cast<BinaryOperator>(V);
|
|
|
|
if (!BinOp || BinOp->getOpcode() != Instruction::Xor)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ConstantInt *CInt = dyn_cast<ConstantInt>(BinOp->getOperand(1));
|
|
|
|
if (!CInt)
|
|
|
|
return;
|
|
|
|
|
|
|
|
V = BinOp->getOperand(0);
|
|
|
|
Operations.push_back(BinOp);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Executes the recorded operations on input 'Value'.
|
|
|
|
auto ApplyOperations = [&](const APInt &Value) {
|
|
|
|
APInt Result = Value;
|
|
|
|
for (auto Op : llvm::reverse(Operations)) {
|
|
|
|
switch (Op->getOpcode()) {
|
|
|
|
case Instruction::Xor:
|
|
|
|
Result ^= cast<ConstantInt>(Op->getOperand(1))->getValue();
|
|
|
|
break;
|
|
|
|
case Instruction::ZExt:
|
|
|
|
Result = Result.zext(Op->getType()->getIntegerBitWidth());
|
|
|
|
break;
|
|
|
|
case Instruction::SExt:
|
|
|
|
Result = Result.sext(Op->getType()->getIntegerBitWidth());
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
llvm_unreachable("Unexpected operation");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Result;
|
|
|
|
};
|
|
|
|
|
2019-10-03 00:03:45 +08:00
|
|
|
auto *PhiDef = cast<PHINode>(V);
|
2017-06-02 10:09:31 +08:00
|
|
|
|
|
|
|
// Get the first dominating conditional branch of the operand
|
|
|
|
// i's incoming block.
|
|
|
|
auto GetDomConditional = [&](unsigned i) -> BranchInst * {
|
|
|
|
BasicBlock *BB = PhiDef->getIncomingBlock(i);
|
|
|
|
BranchInst *BI = dyn_cast<BranchInst>(BB->getTerminator());
|
|
|
|
if (BI && BI->isConditional())
|
|
|
|
return BI;
|
|
|
|
BB = BB->getSinglePredecessor();
|
|
|
|
if (!BB)
|
|
|
|
return nullptr;
|
|
|
|
BI = dyn_cast<BranchInst>(BB->getTerminator());
|
|
|
|
if (!BI || BI->isUnconditional())
|
|
|
|
return nullptr;
|
|
|
|
return BI;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Now walk through all Phi operands to find phi oprerands with values
|
|
|
|
// conflicting with the expected phi output value. Any such operand
|
|
|
|
// indicates the incoming edge to that operand is unlikely.
|
|
|
|
for (unsigned i = 0, e = PhiDef->getNumIncomingValues(); i != e; ++i) {
|
|
|
|
|
|
|
|
Value *PhiOpnd = PhiDef->getIncomingValue(i);
|
|
|
|
ConstantInt *CI = dyn_cast<ConstantInt>(PhiOpnd);
|
|
|
|
if (!CI)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Not an interesting case when IsUnlikely is false -- we can not infer
|
|
|
|
// anything useful when the operand value matches the expected phi
|
|
|
|
// output.
|
|
|
|
if (ExpectedPhiValue == ApplyOperations(CI->getValue()))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
BranchInst *BI = GetDomConditional(i);
|
|
|
|
if (!BI)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
MDBuilder MDB(PhiDef->getContext());
|
|
|
|
|
|
|
|
// There are two situations in which an operand of the PhiDef comes
|
|
|
|
// from a given successor of a branch instruction BI.
|
|
|
|
// 1) When the incoming block of the operand is the successor block;
|
|
|
|
// 2) When the incoming block is BI's enclosing block and the
|
|
|
|
// successor is the PhiDef's enclosing block.
|
|
|
|
//
|
|
|
|
// Returns true if the operand which comes from OpndIncomingBB
|
|
|
|
// comes from outgoing edge of BI that leads to Succ block.
|
|
|
|
auto *OpndIncomingBB = PhiDef->getIncomingBlock(i);
|
|
|
|
auto IsOpndComingFromSuccessor = [&](BasicBlock *Succ) {
|
|
|
|
if (OpndIncomingBB == Succ)
|
|
|
|
// If this successor is the incoming block for this
|
|
|
|
// Phi operand, then this successor does lead to the Phi.
|
|
|
|
return true;
|
|
|
|
if (OpndIncomingBB == BI->getParent() && Succ == PhiDef->getParent())
|
|
|
|
// Otherwise, if the edge is directly from the branch
|
|
|
|
// to the Phi, this successor is the one feeding this
|
|
|
|
// Phi operand.
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
};
|
2020-10-31 15:15:46 +08:00
|
|
|
uint32_t LikelyBranchWeightVal, UnlikelyBranchWeightVal;
|
2020-06-23 01:20:42 +08:00
|
|
|
std::tie(LikelyBranchWeightVal, UnlikelyBranchWeightVal) = getBranchWeight(
|
|
|
|
Expect->getCalledFunction()->getIntrinsicID(), Expect, 2);
|
2017-06-02 10:09:31 +08:00
|
|
|
|
|
|
|
if (IsOpndComingFromSuccessor(BI->getSuccessor(1)))
|
2020-06-23 01:20:42 +08:00
|
|
|
BI->setMetadata(LLVMContext::MD_prof,
|
|
|
|
MDB.createBranchWeights(LikelyBranchWeightVal,
|
|
|
|
UnlikelyBranchWeightVal));
|
2017-06-02 10:09:31 +08:00
|
|
|
else if (IsOpndComingFromSuccessor(BI->getSuccessor(0)))
|
2020-06-23 01:20:42 +08:00
|
|
|
BI->setMetadata(LLVMContext::MD_prof,
|
|
|
|
MDB.createBranchWeights(UnlikelyBranchWeightVal,
|
|
|
|
LikelyBranchWeightVal));
|
2017-06-02 10:09:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-03 06:03:40 +08:00
|
|
|
// Handle both BranchInst and SelectInst.
|
|
|
|
template <class BrSelInst> static bool handleBrSelExpect(BrSelInst &BSI) {
|
2011-07-07 02:22:43 +08:00
|
|
|
|
|
|
|
// Handle non-optimized IR code like:
|
2014-02-03 06:43:55 +08:00
|
|
|
// %expval = call i64 @llvm.expect.i64(i64 %conv1, i64 1)
|
2011-07-07 02:22:43 +08:00
|
|
|
// %tobool = icmp ne i64 %expval, 0
|
|
|
|
// br i1 %tobool, label %if.then, label %if.end
|
2014-02-03 06:43:55 +08:00
|
|
|
//
|
|
|
|
// Or the following simpler case:
|
|
|
|
// %expval = call i1 @llvm.expect.i1(i1 %cmp, i1 1)
|
|
|
|
// br i1 %expval, label %if.then, label %if.end
|
|
|
|
|
|
|
|
CallInst *CI;
|
2011-07-07 02:22:43 +08:00
|
|
|
|
2016-09-03 06:03:40 +08:00
|
|
|
ICmpInst *CmpI = dyn_cast<ICmpInst>(BSI.getCondition());
|
2017-06-02 03:05:55 +08:00
|
|
|
CmpInst::Predicate Predicate;
|
2017-06-02 07:05:11 +08:00
|
|
|
ConstantInt *CmpConstOperand = nullptr;
|
2014-02-03 06:43:55 +08:00
|
|
|
if (!CmpI) {
|
2016-09-03 06:03:40 +08:00
|
|
|
CI = dyn_cast<CallInst>(BSI.getCondition());
|
2017-06-02 03:05:55 +08:00
|
|
|
Predicate = CmpInst::ICMP_NE;
|
2014-02-03 06:43:55 +08:00
|
|
|
} else {
|
2017-06-02 03:05:55 +08:00
|
|
|
Predicate = CmpI->getPredicate();
|
|
|
|
if (Predicate != CmpInst::ICMP_NE && Predicate != CmpInst::ICMP_EQ)
|
2014-02-03 06:43:55 +08:00
|
|
|
return false;
|
2017-06-02 07:05:11 +08:00
|
|
|
|
|
|
|
CmpConstOperand = dyn_cast<ConstantInt>(CmpI->getOperand(1));
|
2017-06-02 03:05:55 +08:00
|
|
|
if (!CmpConstOperand)
|
|
|
|
return false;
|
2014-02-03 06:43:55 +08:00
|
|
|
CI = dyn_cast<CallInst>(CmpI->getOperand(0));
|
|
|
|
}
|
2011-07-07 02:22:43 +08:00
|
|
|
|
|
|
|
if (!CI)
|
|
|
|
return false;
|
|
|
|
|
2017-06-02 07:05:11 +08:00
|
|
|
uint64_t ValueComparedTo = 0;
|
|
|
|
if (CmpConstOperand) {
|
|
|
|
if (CmpConstOperand->getBitWidth() > 64)
|
|
|
|
return false;
|
|
|
|
ValueComparedTo = CmpConstOperand->getZExtValue();
|
|
|
|
}
|
|
|
|
|
2011-07-07 02:22:43 +08:00
|
|
|
Function *Fn = CI->getCalledFunction();
|
2020-06-23 01:20:42 +08:00
|
|
|
if (!Fn || (Fn->getIntrinsicID() != Intrinsic::expect &&
|
|
|
|
Fn->getIntrinsicID() != Intrinsic::expect_with_probability))
|
2011-07-07 02:22:43 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
Value *ArgValue = CI->getArgOperand(0);
|
|
|
|
ConstantInt *ExpectedValue = dyn_cast<ConstantInt>(CI->getArgOperand(1));
|
|
|
|
if (!ExpectedValue)
|
|
|
|
return false;
|
|
|
|
|
2012-05-26 21:59:43 +08:00
|
|
|
MDBuilder MDB(CI->getContext());
|
|
|
|
MDNode *Node;
|
2011-07-07 02:22:43 +08:00
|
|
|
|
2020-10-31 15:15:46 +08:00
|
|
|
uint32_t LikelyBranchWeightVal, UnlikelyBranchWeightVal;
|
2020-06-23 01:20:42 +08:00
|
|
|
std::tie(LikelyBranchWeightVal, UnlikelyBranchWeightVal) =
|
|
|
|
getBranchWeight(Fn->getIntrinsicID(), CI, 2);
|
|
|
|
|
2017-06-02 03:05:55 +08:00
|
|
|
if ((ExpectedValue->getZExtValue() == ValueComparedTo) ==
|
2019-09-12 00:19:50 +08:00
|
|
|
(Predicate == CmpInst::ICMP_EQ)) {
|
2020-06-23 01:20:42 +08:00
|
|
|
Node =
|
|
|
|
MDB.createBranchWeights(LikelyBranchWeightVal, UnlikelyBranchWeightVal);
|
2019-09-12 00:19:50 +08:00
|
|
|
} else {
|
2020-06-23 01:20:42 +08:00
|
|
|
Node =
|
|
|
|
MDB.createBranchWeights(UnlikelyBranchWeightVal, LikelyBranchWeightVal);
|
2019-09-12 00:19:50 +08:00
|
|
|
}
|
2011-07-07 02:22:43 +08:00
|
|
|
|
2014-02-03 06:43:55 +08:00
|
|
|
if (CmpI)
|
|
|
|
CmpI->setOperand(0, ArgValue);
|
|
|
|
else
|
2016-09-03 06:03:40 +08:00
|
|
|
BSI.setCondition(ArgValue);
|
2019-09-12 00:19:50 +08:00
|
|
|
|
|
|
|
BSI.setMetadata(LLVMContext::MD_prof, Node);
|
|
|
|
|
2011-07-07 02:22:43 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-09-03 06:03:40 +08:00
|
|
|
static bool handleBranchExpect(BranchInst &BI) {
|
|
|
|
if (BI.isUnconditional())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return handleBrSelExpect<BranchInst>(BI);
|
|
|
|
}
|
|
|
|
|
2015-01-24 19:13:02 +08:00
|
|
|
static bool lowerExpectIntrinsic(Function &F) {
|
2015-01-24 19:12:57 +08:00
|
|
|
bool Changed = false;
|
|
|
|
|
2015-01-24 18:57:19 +08:00
|
|
|
for (BasicBlock &BB : F) {
|
2011-07-07 02:22:43 +08:00
|
|
|
// Create "block_weights" metadata.
|
2015-01-24 18:57:19 +08:00
|
|
|
if (BranchInst *BI = dyn_cast<BranchInst>(BB.getTerminator())) {
|
2015-01-24 18:39:24 +08:00
|
|
|
if (handleBranchExpect(*BI))
|
2015-01-24 18:57:25 +08:00
|
|
|
ExpectIntrinsicsHandled++;
|
2015-01-24 18:57:19 +08:00
|
|
|
} else if (SwitchInst *SI = dyn_cast<SwitchInst>(BB.getTerminator())) {
|
2015-01-24 18:39:24 +08:00
|
|
|
if (handleSwitchExpect(*SI))
|
2015-01-24 18:57:25 +08:00
|
|
|
ExpectIntrinsicsHandled++;
|
2011-07-07 02:22:43 +08:00
|
|
|
}
|
|
|
|
|
2016-09-03 06:03:40 +08:00
|
|
|
// Remove llvm.expect intrinsics. Iterate backwards in order
|
|
|
|
// to process select instructions before the intrinsic gets
|
|
|
|
// removed.
|
|
|
|
for (auto BI = BB.rbegin(), BE = BB.rend(); BI != BE;) {
|
|
|
|
Instruction *Inst = &*BI++;
|
|
|
|
CallInst *CI = dyn_cast<CallInst>(Inst);
|
|
|
|
if (!CI) {
|
|
|
|
if (SelectInst *SI = dyn_cast<SelectInst>(Inst)) {
|
|
|
|
if (handleBrSelExpect(*SI))
|
|
|
|
ExpectIntrinsicsHandled++;
|
|
|
|
}
|
2011-07-07 02:22:43 +08:00
|
|
|
continue;
|
2016-09-03 06:03:40 +08:00
|
|
|
}
|
2011-07-07 02:22:43 +08:00
|
|
|
|
|
|
|
Function *Fn = CI->getCalledFunction();
|
2020-06-23 01:20:42 +08:00
|
|
|
if (Fn && (Fn->getIntrinsicID() == Intrinsic::expect ||
|
|
|
|
Fn->getIntrinsicID() == Intrinsic::expect_with_probability)) {
|
2017-06-02 10:09:31 +08:00
|
|
|
// Before erasing the llvm.expect, walk backward to find
|
|
|
|
// phi that define llvm.expect's first arg, and
|
|
|
|
// infer branch probability:
|
|
|
|
handlePhiDef(CI);
|
2011-07-07 07:50:16 +08:00
|
|
|
Value *Exp = CI->getArgOperand(0);
|
|
|
|
CI->replaceAllUsesWith(Exp);
|
2011-07-07 02:22:43 +08:00
|
|
|
CI->eraseFromParent();
|
2015-01-24 19:12:57 +08:00
|
|
|
Changed = true;
|
2011-07-07 07:50:16 +08:00
|
|
|
}
|
2011-07-07 02:22:43 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-24 19:12:57 +08:00
|
|
|
return Changed;
|
2011-07-07 02:22:43 +08:00
|
|
|
}
|
|
|
|
|
2016-06-17 08:11:01 +08:00
|
|
|
PreservedAnalyses LowerExpectIntrinsicPass::run(Function &F,
|
|
|
|
FunctionAnalysisManager &) {
|
2015-01-24 19:13:02 +08:00
|
|
|
if (lowerExpectIntrinsic(F))
|
|
|
|
return PreservedAnalyses::none();
|
|
|
|
|
|
|
|
return PreservedAnalyses::all();
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
2018-05-01 23:54:18 +08:00
|
|
|
/// Legacy pass for lowering expect intrinsics out of the IR.
|
2015-01-24 19:13:02 +08:00
|
|
|
///
|
|
|
|
/// When this pass is run over a function it uses expect intrinsics which feed
|
|
|
|
/// branches and switches to provide branch weight metadata for those
|
|
|
|
/// terminators. It then removes the expect intrinsics from the IR so the rest
|
|
|
|
/// of the optimizer can ignore them.
|
|
|
|
class LowerExpectIntrinsic : public FunctionPass {
|
|
|
|
public:
|
|
|
|
static char ID;
|
|
|
|
LowerExpectIntrinsic() : FunctionPass(ID) {
|
|
|
|
initializeLowerExpectIntrinsicPass(*PassRegistry::getPassRegistry());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool runOnFunction(Function &F) override { return lowerExpectIntrinsic(F); }
|
|
|
|
};
|
2015-06-23 17:49:53 +08:00
|
|
|
}
|
2015-01-24 19:13:02 +08:00
|
|
|
|
2011-07-07 02:22:43 +08:00
|
|
|
char LowerExpectIntrinsic::ID = 0;
|
2015-01-24 18:30:14 +08:00
|
|
|
INITIALIZE_PASS(LowerExpectIntrinsic, "lower-expect",
|
|
|
|
"Lower 'expect' Intrinsics", false, false)
|
2011-07-07 02:22:43 +08:00
|
|
|
|
|
|
|
FunctionPass *llvm::createLowerExpectIntrinsicPass() {
|
|
|
|
return new LowerExpectIntrinsic();
|
|
|
|
}
|