2017-03-07 05:31:18 +08:00
|
|
|
//===---- MachineOutliner.cpp - Outline instructions -----------*- C++ -*-===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2017-03-07 05:31:18 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
///
|
|
|
|
/// \file
|
|
|
|
/// Replaces repeated sequences of instructions with function calls.
|
|
|
|
///
|
|
|
|
/// This works by placing every instruction from every basic block in a
|
|
|
|
/// suffix tree, and repeatedly querying that tree for repeated sequences of
|
|
|
|
/// instructions. If a sequence of instructions appears often, then it ought
|
|
|
|
/// to be beneficial to pull out into a function.
|
|
|
|
///
|
2017-09-28 04:47:39 +08:00
|
|
|
/// The MachineOutliner communicates with a given target using hooks defined in
|
|
|
|
/// TargetInstrInfo.h. The target supplies the outliner with information on how
|
|
|
|
/// a specific sequence of instructions should be outlined. This information
|
|
|
|
/// is used to deduce the number of instructions necessary to
|
|
|
|
///
|
|
|
|
/// * Create an outlined function
|
|
|
|
/// * Call that outlined function
|
|
|
|
///
|
|
|
|
/// Targets must implement
|
|
|
|
/// * getOutliningCandidateInfo
|
2018-06-20 05:14:48 +08:00
|
|
|
/// * buildOutlinedFrame
|
2017-09-28 04:47:39 +08:00
|
|
|
/// * insertOutlinedCall
|
|
|
|
/// * isFunctionSafeToOutlineFrom
|
|
|
|
///
|
|
|
|
/// in order to make use of the MachineOutliner.
|
|
|
|
///
|
2017-03-07 05:31:18 +08:00
|
|
|
/// This was originally presented at the 2016 LLVM Developers' Meeting in the
|
|
|
|
/// talk "Reducing Code Size Using Outlining". For a high-level overview of
|
|
|
|
/// how this pass works, the talk is available on YouTube at
|
|
|
|
///
|
|
|
|
/// https://www.youtube.com/watch?v=yorld-WSOeU
|
|
|
|
///
|
|
|
|
/// The slides for the talk are available at
|
|
|
|
///
|
|
|
|
/// http://www.llvm.org/devmtg/2016-11/Slides/Paquette-Outliner.pdf
|
|
|
|
///
|
|
|
|
/// The talk provides an overview of how the outliner finds candidates and
|
|
|
|
/// ultimately outlines them. It describes how the main data structure for this
|
|
|
|
/// pass, the suffix tree, is queried and purged for candidates. It also gives
|
|
|
|
/// a simplified suffix tree construction algorithm for suffix trees based off
|
|
|
|
/// of the algorithm actually used here, Ukkonen's algorithm.
|
|
|
|
///
|
|
|
|
/// For the original RFC for this pass, please see
|
|
|
|
///
|
|
|
|
/// http://lists.llvm.org/pipermail/llvm-dev/2016-August/104170.html
|
|
|
|
///
|
|
|
|
/// For more information on the suffix tree data structure, please see
|
|
|
|
/// https://www.cs.helsinki.fi/u/ukkonen/SuffixT1withFigs.pdf
|
|
|
|
///
|
|
|
|
//===----------------------------------------------------------------------===//
|
2018-06-05 05:14:16 +08:00
|
|
|
#include "llvm/CodeGen/MachineOutliner.h"
|
2017-03-07 05:31:18 +08:00
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2020-03-06 05:54:58 +08:00
|
|
|
#include "llvm/ADT/SmallSet.h"
|
2017-03-07 05:31:18 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
|
|
|
#include "llvm/ADT/Twine.h"
|
|
|
|
#include "llvm/CodeGen/MachineModuleInfo.h"
|
2017-09-01 05:02:45 +08:00
|
|
|
#include "llvm/CodeGen/MachineOptimizationRemarkEmitter.h"
|
2017-03-07 05:31:18 +08:00
|
|
|
#include "llvm/CodeGen/Passes.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/TargetSubtargetInfo.h"
|
2018-01-18 08:00:58 +08:00
|
|
|
#include "llvm/IR/DIBuilder.h"
|
2017-03-07 05:31:18 +08:00
|
|
|
#include "llvm/IR/IRBuilder.h"
|
2018-01-20 05:21:49 +08:00
|
|
|
#include "llvm/IR/Mangler.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"
|
2018-04-20 06:17:07 +08:00
|
|
|
#include "llvm/Support/CommandLine.h"
|
2017-03-07 05:31:18 +08:00
|
|
|
#include "llvm/Support/Debug.h"
|
2020-04-09 06:21:49 +08:00
|
|
|
#include "llvm/Support/SuffixTree.h"
|
2017-03-07 05:31:18 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
#include <functional>
|
|
|
|
#include <tuple>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#define DEBUG_TYPE "machine-outliner"
|
|
|
|
|
|
|
|
using namespace llvm;
|
2017-09-01 05:02:45 +08:00
|
|
|
using namespace ore;
|
2018-06-05 05:14:16 +08:00
|
|
|
using namespace outliner;
|
2017-03-07 05:31:18 +08:00
|
|
|
|
2022-02-18 10:12:39 +08:00
|
|
|
// Statistics for outlined functions.
|
2017-03-07 05:31:18 +08:00
|
|
|
STATISTIC(NumOutlined, "Number of candidates outlined");
|
|
|
|
STATISTIC(FunctionsCreated, "Number of functions created");
|
|
|
|
|
2022-02-18 10:12:39 +08:00
|
|
|
// Statistics for instruction mapping.
|
|
|
|
STATISTIC(NumLegalInUnsignedVec, "Number of legal instrs in unsigned vector");
|
|
|
|
STATISTIC(NumIllegalInUnsignedVec,
|
|
|
|
"Number of illegal instrs in unsigned vector");
|
|
|
|
STATISTIC(NumInvisible, "Number of invisible instrs in unsigned vector");
|
|
|
|
STATISTIC(UnsignedVecSize, "Size of unsigned vector");
|
|
|
|
|
2018-04-20 06:17:07 +08:00
|
|
|
// Set to true if the user wants the outliner to run on linkonceodr linkage
|
|
|
|
// functions. This is false by default because the linker can dedupe linkonceodr
|
|
|
|
// functions. Since the outliner is confined to a single module (modulo LTO),
|
|
|
|
// this is off by default. It should, however, be the default behaviour in
|
|
|
|
// LTO.
|
|
|
|
static cl::opt<bool> EnableLinkOnceODROutlining(
|
2019-10-29 05:57:51 +08:00
|
|
|
"enable-linkonceodr-outlining", cl::Hidden,
|
2018-04-20 06:17:07 +08:00
|
|
|
cl::desc("Enable the machine outliner on linkonceodr functions"),
|
|
|
|
cl::init(false));
|
|
|
|
|
2020-04-29 15:33:47 +08:00
|
|
|
/// Number of times to re-run the outliner. This is not the total number of runs
|
|
|
|
/// as the outliner will run at least one time. The default value is set to 0,
|
|
|
|
/// meaning the outliner will run one time and rerun zero times after that.
|
|
|
|
static cl::opt<unsigned> OutlinerReruns(
|
|
|
|
"machine-outliner-reruns", cl::init(0), cl::Hidden,
|
|
|
|
cl::desc(
|
|
|
|
"Number of times to rerun the outliner after the initial outline"));
|
2020-03-18 06:40:26 +08:00
|
|
|
|
2017-03-07 05:31:18 +08:00
|
|
|
namespace {
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// Maps \p MachineInstrs to unsigned integers and stores the mappings.
|
2017-03-07 05:31:18 +08:00
|
|
|
struct InstructionMapper {
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// The next available integer to assign to a \p MachineInstr that
|
2017-03-07 05:31:18 +08:00
|
|
|
/// cannot be outlined.
|
|
|
|
///
|
|
|
|
/// Set to -3 for compatability with \p DenseMapInfo<unsigned>.
|
|
|
|
unsigned IllegalInstrNumber = -3;
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// The next available integer to assign to a \p MachineInstr that can
|
2017-03-07 05:31:18 +08:00
|
|
|
/// be outlined.
|
|
|
|
unsigned LegalInstrNumber = 0;
|
|
|
|
|
|
|
|
/// Correspondence from \p MachineInstrs to unsigned integers.
|
|
|
|
DenseMap<MachineInstr *, unsigned, MachineInstrExpressionTrait>
|
|
|
|
InstructionIntegerMap;
|
|
|
|
|
2018-11-14 07:01:34 +08:00
|
|
|
/// Correspondence between \p MachineBasicBlocks and target-defined flags.
|
|
|
|
DenseMap<MachineBasicBlock *, unsigned> MBBFlagsMap;
|
|
|
|
|
2017-03-07 05:31:18 +08:00
|
|
|
/// The vector of unsigned integers that the module is mapped to.
|
|
|
|
std::vector<unsigned> UnsignedVec;
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// Stores the location of the instruction associated with the integer
|
2017-03-07 05:31:18 +08:00
|
|
|
/// at index i in \p UnsignedVec for each index i.
|
|
|
|
std::vector<MachineBasicBlock::iterator> InstrList;
|
|
|
|
|
2018-11-02 07:09:06 +08:00
|
|
|
// Set if we added an illegal number in the previous step.
|
|
|
|
// Since each illegal number is unique, we only need one of them between
|
|
|
|
// each range of legal numbers. This lets us make sure we don't add more
|
|
|
|
// than one illegal number per range.
|
|
|
|
bool AddedIllegalLastTime = false;
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// Maps \p *It to a legal integer.
|
2017-03-07 05:31:18 +08:00
|
|
|
///
|
2018-11-08 08:33:38 +08:00
|
|
|
/// Updates \p CanOutlineWithPrevInstr, \p HaveLegalRange, \p InstrListForMBB,
|
2018-12-06 08:01:51 +08:00
|
|
|
/// \p UnsignedVecForMBB, \p InstructionIntegerMap, and \p LegalInstrNumber.
|
2017-03-07 05:31:18 +08:00
|
|
|
///
|
|
|
|
/// \returns The integer that \p *It was mapped to.
|
2018-11-08 08:02:11 +08:00
|
|
|
unsigned mapToLegalUnsigned(
|
2018-11-08 08:33:38 +08:00
|
|
|
MachineBasicBlock::iterator &It, bool &CanOutlineWithPrevInstr,
|
|
|
|
bool &HaveLegalRange, unsigned &NumLegalInBlock,
|
2018-11-08 08:02:11 +08:00
|
|
|
std::vector<unsigned> &UnsignedVecForMBB,
|
|
|
|
std::vector<MachineBasicBlock::iterator> &InstrListForMBB) {
|
2018-11-02 07:09:06 +08:00
|
|
|
// We added something legal, so we should unset the AddedLegalLastTime
|
|
|
|
// flag.
|
|
|
|
AddedIllegalLastTime = false;
|
2017-03-07 05:31:18 +08:00
|
|
|
|
2018-11-08 08:33:38 +08:00
|
|
|
// If we have at least two adjacent legal instructions (which may have
|
|
|
|
// invisible instructions in between), remember that.
|
|
|
|
if (CanOutlineWithPrevInstr)
|
|
|
|
HaveLegalRange = true;
|
|
|
|
CanOutlineWithPrevInstr = true;
|
|
|
|
|
2018-11-08 08:02:11 +08:00
|
|
|
// Keep track of the number of legal instructions we insert.
|
|
|
|
NumLegalInBlock++;
|
|
|
|
|
2017-03-07 05:31:18 +08:00
|
|
|
// Get the integer for this instruction or give it the current
|
|
|
|
// LegalInstrNumber.
|
2018-11-08 08:02:11 +08:00
|
|
|
InstrListForMBB.push_back(It);
|
2017-03-07 05:31:18 +08:00
|
|
|
MachineInstr &MI = *It;
|
|
|
|
bool WasInserted;
|
|
|
|
DenseMap<MachineInstr *, unsigned, MachineInstrExpressionTrait>::iterator
|
2017-07-28 07:24:43 +08:00
|
|
|
ResultIt;
|
2017-03-07 05:31:18 +08:00
|
|
|
std::tie(ResultIt, WasInserted) =
|
2017-07-28 07:24:43 +08:00
|
|
|
InstructionIntegerMap.insert(std::make_pair(&MI, LegalInstrNumber));
|
2017-03-07 05:31:18 +08:00
|
|
|
unsigned MINumber = ResultIt->second;
|
|
|
|
|
|
|
|
// There was an insertion.
|
2018-12-06 08:01:51 +08:00
|
|
|
if (WasInserted)
|
2017-03-07 05:31:18 +08:00
|
|
|
LegalInstrNumber++;
|
|
|
|
|
2018-11-08 08:02:11 +08:00
|
|
|
UnsignedVecForMBB.push_back(MINumber);
|
2017-03-07 05:31:18 +08:00
|
|
|
|
|
|
|
// Make sure we don't overflow or use any integers reserved by the DenseMap.
|
|
|
|
if (LegalInstrNumber >= IllegalInstrNumber)
|
|
|
|
report_fatal_error("Instruction mapping overflow!");
|
|
|
|
|
2017-07-28 07:24:43 +08:00
|
|
|
assert(LegalInstrNumber != DenseMapInfo<unsigned>::getEmptyKey() &&
|
|
|
|
"Tried to assign DenseMap tombstone or empty key to instruction.");
|
|
|
|
assert(LegalInstrNumber != DenseMapInfo<unsigned>::getTombstoneKey() &&
|
|
|
|
"Tried to assign DenseMap tombstone or empty key to instruction.");
|
2017-03-07 05:31:18 +08:00
|
|
|
|
2022-02-18 10:12:39 +08:00
|
|
|
// Statistics.
|
|
|
|
++NumLegalInUnsignedVec;
|
2017-03-07 05:31:18 +08:00
|
|
|
return MINumber;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Maps \p *It to an illegal integer.
|
|
|
|
///
|
2018-11-08 08:02:11 +08:00
|
|
|
/// Updates \p InstrListForMBB, \p UnsignedVecForMBB, and \p
|
|
|
|
/// IllegalInstrNumber.
|
2017-03-07 05:31:18 +08:00
|
|
|
///
|
|
|
|
/// \returns The integer that \p *It was mapped to.
|
2019-10-29 05:57:51 +08:00
|
|
|
unsigned mapToIllegalUnsigned(
|
|
|
|
MachineBasicBlock::iterator &It, bool &CanOutlineWithPrevInstr,
|
|
|
|
std::vector<unsigned> &UnsignedVecForMBB,
|
|
|
|
std::vector<MachineBasicBlock::iterator> &InstrListForMBB) {
|
2018-11-08 08:33:38 +08:00
|
|
|
// Can't outline an illegal instruction. Set the flag.
|
|
|
|
CanOutlineWithPrevInstr = false;
|
|
|
|
|
2018-11-02 07:09:06 +08:00
|
|
|
// Only add one illegal number per range of legal numbers.
|
|
|
|
if (AddedIllegalLastTime)
|
|
|
|
return IllegalInstrNumber;
|
|
|
|
|
|
|
|
// Remember that we added an illegal number last time.
|
|
|
|
AddedIllegalLastTime = true;
|
2017-03-07 05:31:18 +08:00
|
|
|
unsigned MINumber = IllegalInstrNumber;
|
|
|
|
|
2018-11-08 08:02:11 +08:00
|
|
|
InstrListForMBB.push_back(It);
|
|
|
|
UnsignedVecForMBB.push_back(IllegalInstrNumber);
|
2017-03-07 05:31:18 +08:00
|
|
|
IllegalInstrNumber--;
|
2022-02-18 10:12:39 +08:00
|
|
|
// Statistics.
|
|
|
|
++NumIllegalInUnsignedVec;
|
2017-03-07 05:31:18 +08:00
|
|
|
|
|
|
|
assert(LegalInstrNumber < IllegalInstrNumber &&
|
|
|
|
"Instruction mapping overflow!");
|
|
|
|
|
2017-07-28 07:24:43 +08:00
|
|
|
assert(IllegalInstrNumber != DenseMapInfo<unsigned>::getEmptyKey() &&
|
|
|
|
"IllegalInstrNumber cannot be DenseMap tombstone or empty key!");
|
2017-03-07 05:31:18 +08:00
|
|
|
|
2017-07-28 07:24:43 +08:00
|
|
|
assert(IllegalInstrNumber != DenseMapInfo<unsigned>::getTombstoneKey() &&
|
|
|
|
"IllegalInstrNumber cannot be DenseMap tombstone or empty key!");
|
2017-03-07 05:31:18 +08:00
|
|
|
|
|
|
|
return MINumber;
|
|
|
|
}
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// Transforms a \p MachineBasicBlock into a \p vector of \p unsigneds
|
2017-03-07 05:31:18 +08:00
|
|
|
/// and appends it to \p UnsignedVec and \p InstrList.
|
|
|
|
///
|
|
|
|
/// Two instructions are assigned the same integer if they are identical.
|
|
|
|
/// If an instruction is deemed unsafe to outline, then it will be assigned an
|
|
|
|
/// unique integer. The resulting mapping is placed into a suffix tree and
|
|
|
|
/// queried for candidates.
|
|
|
|
///
|
|
|
|
/// \param MBB The \p MachineBasicBlock to be translated into integers.
|
2018-08-01 08:37:20 +08:00
|
|
|
/// \param TII \p TargetInstrInfo for the function.
|
2017-03-07 05:31:18 +08:00
|
|
|
void convertToUnsignedVec(MachineBasicBlock &MBB,
|
|
|
|
const TargetInstrInfo &TII) {
|
2018-11-14 00:41:05 +08:00
|
|
|
unsigned Flags = 0;
|
2018-11-13 07:51:32 +08:00
|
|
|
|
|
|
|
// Don't even map in this case.
|
|
|
|
if (!TII.isMBBSafeToOutlineFrom(MBB, Flags))
|
|
|
|
return;
|
|
|
|
|
2018-11-14 07:01:34 +08:00
|
|
|
// Store info for the MBB for later outlining.
|
|
|
|
MBBFlagsMap[&MBB] = Flags;
|
|
|
|
|
2018-11-02 07:09:06 +08:00
|
|
|
MachineBasicBlock::iterator It = MBB.begin();
|
2018-11-08 08:02:11 +08:00
|
|
|
|
|
|
|
// The number of instructions in this block that will be considered for
|
|
|
|
// outlining.
|
|
|
|
unsigned NumLegalInBlock = 0;
|
|
|
|
|
2018-11-08 08:33:38 +08:00
|
|
|
// True if we have at least two legal instructions which aren't separated
|
|
|
|
// by an illegal instruction.
|
|
|
|
bool HaveLegalRange = false;
|
|
|
|
|
|
|
|
// True if we can perform outlining given the last mapped (non-invisible)
|
|
|
|
// instruction. This lets us know if we have a legal range.
|
|
|
|
bool CanOutlineWithPrevInstr = false;
|
|
|
|
|
2018-11-08 08:02:11 +08:00
|
|
|
// FIXME: Should this all just be handled in the target, rather than using
|
|
|
|
// repeated calls to getOutliningType?
|
|
|
|
std::vector<unsigned> UnsignedVecForMBB;
|
|
|
|
std::vector<MachineBasicBlock::iterator> InstrListForMBB;
|
|
|
|
|
2019-11-06 00:46:10 +08:00
|
|
|
for (MachineBasicBlock::iterator Et = MBB.end(); It != Et; ++It) {
|
2017-03-07 05:31:18 +08:00
|
|
|
// Keep track of where this instruction is in the module.
|
[MachineOutliner] AArch64: Handle instrs that use SP and will never need fixups
This commit does two things. Firstly, it adds a collection of flags which can
be passed along to the target to encode information about the MBB that an
instruction lives in to the outliner.
Second, it adds some of those flags to the AArch64 outliner in order to add
more stack instructions to the list of legal instructions that are handled
by the outliner. The two flags added check if
- There are calls in the MachineBasicBlock containing the instruction
- The link register is available in the entire block
If the link register is available and there are no calls, then a stack
instruction can always be outlined without fixups, regardless of what it is,
since in this case, the outliner will never modify the stack to create a
call or outlined frame.
The motivation for doing this was checking which instructions are most often
missed by the outliner. Instructions like, say
%sp<def> = ADDXri %sp, 32, 0; flags: FrameDestroy
are very common, but cannot be outlined in the case that the outliner might
modify the stack. This commit allows us to outline instructions like this.
llvm-svn: 322048
2018-01-09 08:26:18 +08:00
|
|
|
switch (TII.getOutliningType(It, Flags)) {
|
2018-06-05 05:14:16 +08:00
|
|
|
case InstrType::Illegal:
|
2019-10-29 05:57:51 +08:00
|
|
|
mapToIllegalUnsigned(It, CanOutlineWithPrevInstr, UnsignedVecForMBB,
|
|
|
|
InstrListForMBB);
|
2017-07-28 07:24:43 +08:00
|
|
|
break;
|
2017-03-07 05:31:18 +08:00
|
|
|
|
2018-06-05 05:14:16 +08:00
|
|
|
case InstrType::Legal:
|
2018-11-08 08:33:38 +08:00
|
|
|
mapToLegalUnsigned(It, CanOutlineWithPrevInstr, HaveLegalRange,
|
|
|
|
NumLegalInBlock, UnsignedVecForMBB, InstrListForMBB);
|
2017-07-28 07:24:43 +08:00
|
|
|
break;
|
2017-03-07 05:31:18 +08:00
|
|
|
|
2018-06-05 05:14:16 +08:00
|
|
|
case InstrType::LegalTerminator:
|
2018-11-08 08:33:38 +08:00
|
|
|
mapToLegalUnsigned(It, CanOutlineWithPrevInstr, HaveLegalRange,
|
|
|
|
NumLegalInBlock, UnsignedVecForMBB, InstrListForMBB);
|
2018-11-02 07:09:06 +08:00
|
|
|
// The instruction also acts as a terminator, so we have to record that
|
|
|
|
// in the string.
|
2018-11-08 08:33:38 +08:00
|
|
|
mapToIllegalUnsigned(It, CanOutlineWithPrevInstr, UnsignedVecForMBB,
|
2019-10-29 05:57:51 +08:00
|
|
|
InstrListForMBB);
|
2018-05-23 03:11:06 +08:00
|
|
|
break;
|
|
|
|
|
2018-06-05 05:14:16 +08:00
|
|
|
case InstrType::Invisible:
|
2018-11-02 07:09:06 +08:00
|
|
|
// Normally this is set by mapTo(Blah)Unsigned, but we just want to
|
|
|
|
// skip this instruction. So, unset the flag here.
|
2022-02-18 10:12:39 +08:00
|
|
|
++NumInvisible;
|
2018-09-18 02:40:21 +08:00
|
|
|
AddedIllegalLastTime = false;
|
2017-07-28 07:24:43 +08:00
|
|
|
break;
|
2017-03-07 05:31:18 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-08 08:02:11 +08:00
|
|
|
// Are there enough legal instructions in the block for outlining to be
|
|
|
|
// possible?
|
2018-11-08 08:33:38 +08:00
|
|
|
if (HaveLegalRange) {
|
2018-11-08 08:02:11 +08:00
|
|
|
// After we're done every insertion, uniquely terminate this part of the
|
|
|
|
// "string". This makes sure we won't match across basic block or function
|
|
|
|
// boundaries since the "end" is encoded uniquely and thus appears in no
|
|
|
|
// repeated substring.
|
2018-11-08 08:33:38 +08:00
|
|
|
mapToIllegalUnsigned(It, CanOutlineWithPrevInstr, UnsignedVecForMBB,
|
2019-10-29 05:57:51 +08:00
|
|
|
InstrListForMBB);
|
2020-12-29 11:55:16 +08:00
|
|
|
llvm::append_range(InstrList, InstrListForMBB);
|
|
|
|
llvm::append_range(UnsignedVec, UnsignedVecForMBB);
|
2018-11-08 08:02:11 +08:00
|
|
|
}
|
2017-03-07 05:31:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
InstructionMapper() {
|
|
|
|
// Make sure that the implementation of DenseMapInfo<unsigned> hasn't
|
|
|
|
// changed.
|
|
|
|
assert(DenseMapInfo<unsigned>::getEmptyKey() == (unsigned)-1 &&
|
2017-07-28 07:24:43 +08:00
|
|
|
"DenseMapInfo<unsigned>'s empty key isn't -1!");
|
2017-03-07 05:31:18 +08:00
|
|
|
assert(DenseMapInfo<unsigned>::getTombstoneKey() == (unsigned)-2 &&
|
2017-07-28 07:24:43 +08:00
|
|
|
"DenseMapInfo<unsigned>'s tombstone key isn't -2!");
|
2017-03-07 05:31:18 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// An interprocedural pass which finds repeated sequences of
|
2017-03-07 05:31:18 +08:00
|
|
|
/// instructions and replaces them with calls to functions.
|
|
|
|
///
|
|
|
|
/// Each instruction is mapped to an unsigned integer and placed in a string.
|
|
|
|
/// The resulting mapping is then placed in a \p SuffixTree. The \p SuffixTree
|
|
|
|
/// is then repeatedly queried for repeated sequences of instructions. Each
|
|
|
|
/// non-overlapping repeated sequence is then placed in its own
|
|
|
|
/// \p MachineFunction and each instance is then replaced with a call to that
|
|
|
|
/// function.
|
|
|
|
struct MachineOutliner : public ModulePass {
|
|
|
|
|
|
|
|
static char ID;
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// Set to true if the outliner should consider functions with
|
[MachineOutliner] Disable outlining from LinkOnceODRs by default
Say you have two identical linkonceodr functions, one in M1 and one in M2.
Say that the outliner outlines A,B,C from one function, and D,E,F from another
function (where letters are instructions). Now those functions are not
identical, and cannot be deduped. Locally to M1 and M2, these outlining
choices would be good-- to the whole program, however, this might not be true!
To mitigate this, this commit makes it so that the outliner sees linkonceodr
functions as unsafe to outline from. It also adds a flag,
-enable-linkonceodr-outlining, which allows the user to specify that they
want to outline from such functions when they know what they're doing.
Changing this handles most code size regressions in the test suite caused by
competing with linker dedupe. It also doesn't have a huge impact on the code
size improvements from the outliner. There are 6 tests that regress > 5% from
outlining WITH linkonceodrs to outlining WITHOUT linkonceodrs. Overall, most
tests either improve or are not impacted.
Not outlined vs outlined without linkonceodrs:
https://hastebin.com/raw/qeguxavuda
Not outlined vs outlined with linkonceodrs:
https://hastebin.com/raw/edepoqoqic
Outlined with linkonceodrs vs outlined without linkonceodrs:
https://hastebin.com/raw/awiqifiheb
Numbers generated using compare.py with -m size.__text. Tests run for AArch64
with -Oz -mllvm -enable-machine-outliner -mno-red-zone.
llvm-svn: 315136
2017-10-07 08:16:34 +08:00
|
|
|
/// linkonceodr linkage.
|
|
|
|
bool OutlineFromLinkOnceODRs = false;
|
|
|
|
|
2020-03-18 06:40:26 +08:00
|
|
|
/// The current repeat number of machine outlining.
|
|
|
|
unsigned OutlineRepeatedNum = 0;
|
|
|
|
|
2018-06-30 11:56:03 +08:00
|
|
|
/// Set to true if the outliner should run on all functions in the module
|
|
|
|
/// considered safe for outlining.
|
|
|
|
/// Set to true by default for compatibility with llc's -run-pass option.
|
|
|
|
/// Set when the pass is constructed in TargetPassConfig.
|
|
|
|
bool RunOnAllFunctions = true;
|
|
|
|
|
2017-03-07 05:31:18 +08:00
|
|
|
StringRef getPassName() const override { return "Machine Outliner"; }
|
|
|
|
|
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const override {
|
2019-10-01 01:54:50 +08:00
|
|
|
AU.addRequired<MachineModuleInfoWrapperPass>();
|
|
|
|
AU.addPreserved<MachineModuleInfoWrapperPass>();
|
2017-03-07 05:31:18 +08:00
|
|
|
AU.setPreservesAll();
|
|
|
|
ModulePass::getAnalysisUsage(AU);
|
|
|
|
}
|
|
|
|
|
2018-04-20 06:17:07 +08:00
|
|
|
MachineOutliner() : ModulePass(ID) {
|
2017-03-07 05:31:18 +08:00
|
|
|
initializeMachineOutlinerPass(*PassRegistry::getPassRegistry());
|
|
|
|
}
|
|
|
|
|
2018-07-25 01:37:28 +08:00
|
|
|
/// Remark output explaining that not outlining a set of candidates would be
|
|
|
|
/// better than outlining that set.
|
|
|
|
void emitNotOutliningCheaperRemark(
|
|
|
|
unsigned StringLen, std::vector<Candidate> &CandidatesForRepeatedSeq,
|
|
|
|
OutlinedFunction &OF);
|
|
|
|
|
2018-07-25 04:20:45 +08:00
|
|
|
/// Remark output explaining that a function was outlined.
|
|
|
|
void emitOutlinedFunctionRemark(OutlinedFunction &OF);
|
|
|
|
|
2018-12-06 07:39:07 +08:00
|
|
|
/// Find all repeated substrings that satisfy the outlining cost model by
|
|
|
|
/// constructing a suffix tree.
|
2017-07-28 07:24:43 +08:00
|
|
|
///
|
|
|
|
/// If a substring appears at least twice, then it must be represented by
|
2018-07-25 01:37:28 +08:00
|
|
|
/// an internal node which appears in at least two suffixes. Each suffix
|
|
|
|
/// is represented by a leaf node. To do this, we visit each internal node
|
|
|
|
/// in the tree, using the leaf children of each internal node. If an
|
|
|
|
/// internal node represents a beneficial substring, then we use each of
|
|
|
|
/// its leaf children to find the locations of its substring.
|
2017-07-28 07:24:43 +08:00
|
|
|
///
|
|
|
|
/// \param Mapper Contains outlining mapping information.
|
2018-07-25 01:37:28 +08:00
|
|
|
/// \param[out] FunctionList Filled with a list of \p OutlinedFunctions
|
|
|
|
/// each type of candidate.
|
2018-12-06 07:39:07 +08:00
|
|
|
void findCandidates(InstructionMapper &Mapper,
|
|
|
|
std::vector<OutlinedFunction> &FunctionList);
|
2017-07-28 07:24:43 +08:00
|
|
|
|
2018-12-06 06:50:26 +08:00
|
|
|
/// Replace the sequences of instructions represented by \p OutlinedFunctions
|
|
|
|
/// with calls to functions.
|
2017-03-07 05:31:18 +08:00
|
|
|
///
|
|
|
|
/// \param M The module we are outlining from.
|
|
|
|
/// \param FunctionList A list of functions to be inserted into the module.
|
|
|
|
/// \param Mapper Contains the instruction mappings for the module.
|
2018-12-06 06:50:26 +08:00
|
|
|
bool outline(Module &M, std::vector<OutlinedFunction> &FunctionList,
|
2019-10-29 05:57:51 +08:00
|
|
|
InstructionMapper &Mapper, unsigned &OutlinedFunctionNum);
|
2017-03-07 05:31:18 +08:00
|
|
|
|
|
|
|
/// Creates a function for \p OF and inserts it into the module.
|
2018-12-06 07:24:22 +08:00
|
|
|
MachineFunction *createOutlinedFunction(Module &M, OutlinedFunction &OF,
|
2018-11-08 02:36:43 +08:00
|
|
|
InstructionMapper &Mapper,
|
|
|
|
unsigned Name);
|
2017-03-07 05:31:18 +08:00
|
|
|
|
2020-04-29 15:33:47 +08:00
|
|
|
/// Calls 'doOutline()' 1 + OutlinerReruns times.
|
2020-03-18 09:33:55 +08:00
|
|
|
bool runOnModule(Module &M) override;
|
2020-03-18 06:40:26 +08:00
|
|
|
|
2017-03-07 05:31:18 +08:00
|
|
|
/// Construct a suffix tree on the instructions in \p M and outline repeated
|
|
|
|
/// strings from that tree.
|
2019-10-29 03:10:21 +08:00
|
|
|
bool doOutline(Module &M, unsigned &OutlinedFunctionNum);
|
2018-06-05 05:14:16 +08:00
|
|
|
|
|
|
|
/// Return a DISubprogram for OF if one exists, and null otherwise. Helper
|
|
|
|
/// function for remark emission.
|
|
|
|
DISubprogram *getSubprogramOrNull(const OutlinedFunction &OF) {
|
2018-12-06 07:24:22 +08:00
|
|
|
for (const Candidate &C : OF.Candidates)
|
2019-11-05 23:58:04 +08:00
|
|
|
if (MachineFunction *MF = C.getMF())
|
|
|
|
if (DISubprogram *SP = MF->getFunction().getSubprogram())
|
|
|
|
return SP;
|
2018-06-05 05:14:16 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
2018-09-12 00:33:46 +08:00
|
|
|
|
|
|
|
/// Populate and \p InstructionMapper with instruction-to-integer mappings.
|
|
|
|
/// These are used to construct a suffix tree.
|
|
|
|
void populateMapper(InstructionMapper &Mapper, Module &M,
|
|
|
|
MachineModuleInfo &MMI);
|
2017-03-07 05:31:18 +08:00
|
|
|
|
2018-09-12 07:05:34 +08:00
|
|
|
/// Initialize information necessary to output a size remark.
|
|
|
|
/// FIXME: This should be handled by the pass manager, not the outliner.
|
|
|
|
/// FIXME: This is nearly identical to the initSizeRemarkInfo in the legacy
|
|
|
|
/// pass manager.
|
2019-10-29 05:57:51 +08:00
|
|
|
void initSizeRemarkInfo(const Module &M, const MachineModuleInfo &MMI,
|
|
|
|
StringMap<unsigned> &FunctionToInstrCount);
|
2018-09-12 07:05:34 +08:00
|
|
|
|
|
|
|
/// Emit the remark.
|
|
|
|
// FIXME: This should be handled by the pass manager, not the outliner.
|
2019-10-29 05:57:51 +08:00
|
|
|
void
|
|
|
|
emitInstrCountChangedRemark(const Module &M, const MachineModuleInfo &MMI,
|
|
|
|
const StringMap<unsigned> &FunctionToInstrCount);
|
2018-09-12 07:05:34 +08:00
|
|
|
};
|
2017-03-07 05:31:18 +08:00
|
|
|
} // Anonymous namespace.
|
|
|
|
|
|
|
|
char MachineOutliner::ID = 0;
|
|
|
|
|
|
|
|
namespace llvm {
|
2018-06-30 11:56:03 +08:00
|
|
|
ModulePass *createMachineOutlinerPass(bool RunOnAllFunctions) {
|
|
|
|
MachineOutliner *OL = new MachineOutliner();
|
|
|
|
OL->RunOnAllFunctions = RunOnAllFunctions;
|
|
|
|
return OL;
|
[MachineOutliner] Disable outlining from LinkOnceODRs by default
Say you have two identical linkonceodr functions, one in M1 and one in M2.
Say that the outliner outlines A,B,C from one function, and D,E,F from another
function (where letters are instructions). Now those functions are not
identical, and cannot be deduped. Locally to M1 and M2, these outlining
choices would be good-- to the whole program, however, this might not be true!
To mitigate this, this commit makes it so that the outliner sees linkonceodr
functions as unsafe to outline from. It also adds a flag,
-enable-linkonceodr-outlining, which allows the user to specify that they
want to outline from such functions when they know what they're doing.
Changing this handles most code size regressions in the test suite caused by
competing with linker dedupe. It also doesn't have a huge impact on the code
size improvements from the outliner. There are 6 tests that regress > 5% from
outlining WITH linkonceodrs to outlining WITHOUT linkonceodrs. Overall, most
tests either improve or are not impacted.
Not outlined vs outlined without linkonceodrs:
https://hastebin.com/raw/qeguxavuda
Not outlined vs outlined with linkonceodrs:
https://hastebin.com/raw/edepoqoqic
Outlined with linkonceodrs vs outlined without linkonceodrs:
https://hastebin.com/raw/awiqifiheb
Numbers generated using compare.py with -m size.__text. Tests run for AArch64
with -Oz -mllvm -enable-machine-outliner -mno-red-zone.
llvm-svn: 315136
2017-10-07 08:16:34 +08:00
|
|
|
}
|
|
|
|
|
2017-07-28 07:24:43 +08:00
|
|
|
} // namespace llvm
|
|
|
|
|
|
|
|
INITIALIZE_PASS(MachineOutliner, DEBUG_TYPE, "Machine Function Outliner", false,
|
|
|
|
false)
|
|
|
|
|
2018-07-25 01:37:28 +08:00
|
|
|
void MachineOutliner::emitNotOutliningCheaperRemark(
|
|
|
|
unsigned StringLen, std::vector<Candidate> &CandidatesForRepeatedSeq,
|
|
|
|
OutlinedFunction &OF) {
|
2018-11-02 07:09:06 +08:00
|
|
|
// FIXME: Right now, we arbitrarily choose some Candidate from the
|
|
|
|
// OutlinedFunction. This isn't necessarily fixed, nor does it have to be.
|
|
|
|
// We should probably sort these by function name or something to make sure
|
|
|
|
// the remarks are stable.
|
2018-07-25 01:37:28 +08:00
|
|
|
Candidate &C = CandidatesForRepeatedSeq.front();
|
|
|
|
MachineOptimizationRemarkEmitter MORE(*(C.getMF()), nullptr);
|
|
|
|
MORE.emit([&]() {
|
|
|
|
MachineOptimizationRemarkMissed R(DEBUG_TYPE, "NotOutliningCheaper",
|
|
|
|
C.front()->getDebugLoc(), C.getMBB());
|
|
|
|
R << "Did not outline " << NV("Length", StringLen) << " instructions"
|
|
|
|
<< " from " << NV("NumOccurrences", CandidatesForRepeatedSeq.size())
|
|
|
|
<< " locations."
|
|
|
|
<< " Bytes from outlining all occurrences ("
|
|
|
|
<< NV("OutliningCost", OF.getOutliningCost()) << ")"
|
|
|
|
<< " >= Unoutlined instruction bytes ("
|
|
|
|
<< NV("NotOutliningCost", OF.getNotOutlinedCost()) << ")"
|
|
|
|
<< " (Also found at: ";
|
|
|
|
|
|
|
|
// Tell the user the other places the candidate was found.
|
|
|
|
for (unsigned i = 1, e = CandidatesForRepeatedSeq.size(); i < e; i++) {
|
|
|
|
R << NV((Twine("OtherStartLoc") + Twine(i)).str(),
|
|
|
|
CandidatesForRepeatedSeq[i].front()->getDebugLoc());
|
|
|
|
if (i != e - 1)
|
|
|
|
R << ", ";
|
|
|
|
}
|
|
|
|
|
|
|
|
R << ")";
|
|
|
|
return R;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-07-25 04:20:45 +08:00
|
|
|
void MachineOutliner::emitOutlinedFunctionRemark(OutlinedFunction &OF) {
|
|
|
|
MachineBasicBlock *MBB = &*OF.MF->begin();
|
|
|
|
MachineOptimizationRemarkEmitter MORE(*OF.MF, nullptr);
|
|
|
|
MachineOptimizationRemark R(DEBUG_TYPE, "OutlinedFunction",
|
|
|
|
MBB->findDebugLoc(MBB->begin()), MBB);
|
|
|
|
R << "Saved " << NV("OutliningBenefit", OF.getBenefit()) << " bytes by "
|
2018-12-06 01:57:33 +08:00
|
|
|
<< "outlining " << NV("Length", OF.getNumInstrs()) << " instructions "
|
2018-07-25 04:20:45 +08:00
|
|
|
<< "from " << NV("NumOccurrences", OF.getOccurrenceCount())
|
|
|
|
<< " locations. "
|
|
|
|
<< "(Found at: ";
|
|
|
|
|
|
|
|
// Tell the user the other places the candidate was found.
|
|
|
|
for (size_t i = 0, e = OF.Candidates.size(); i < e; i++) {
|
|
|
|
|
|
|
|
R << NV((Twine("StartLoc") + Twine(i)).str(),
|
2018-12-06 07:24:22 +08:00
|
|
|
OF.Candidates[i].front()->getDebugLoc());
|
2018-07-25 04:20:45 +08:00
|
|
|
if (i != e - 1)
|
|
|
|
R << ", ";
|
|
|
|
}
|
|
|
|
|
|
|
|
R << ")";
|
|
|
|
|
|
|
|
MORE.emit(R);
|
|
|
|
}
|
|
|
|
|
2019-10-29 05:57:51 +08:00
|
|
|
void MachineOutliner::findCandidates(
|
|
|
|
InstructionMapper &Mapper, std::vector<OutlinedFunction> &FunctionList) {
|
2017-07-28 07:24:43 +08:00
|
|
|
FunctionList.clear();
|
2018-12-06 07:39:07 +08:00
|
|
|
SuffixTree ST(Mapper.UnsignedVec);
|
2017-07-28 07:24:43 +08:00
|
|
|
|
2019-10-31 00:28:11 +08:00
|
|
|
// First, find all of the repeated substrings in the tree of minimum length
|
2018-11-07 05:46:41 +08:00
|
|
|
// 2.
|
2018-12-06 08:04:03 +08:00
|
|
|
std::vector<Candidate> CandidatesForRepeatedSeq;
|
2021-02-16 06:46:10 +08:00
|
|
|
for (const SuffixTree::RepeatedSubstring &RS : ST) {
|
2018-12-06 08:04:03 +08:00
|
|
|
CandidatesForRepeatedSeq.clear();
|
2018-11-07 05:46:41 +08:00
|
|
|
unsigned StringLen = RS.Length;
|
|
|
|
for (const unsigned &StartIdx : RS.StartIndices) {
|
|
|
|
unsigned EndIdx = StartIdx + StringLen - 1;
|
|
|
|
// Trick: Discard some candidates that would be incompatible with the
|
|
|
|
// ones we've already found for this sequence. This will save us some
|
|
|
|
// work in candidate selection.
|
|
|
|
//
|
|
|
|
// If two candidates overlap, then we can't outline them both. This
|
|
|
|
// happens when we have candidates that look like, say
|
|
|
|
//
|
|
|
|
// AA (where each "A" is an instruction).
|
|
|
|
//
|
|
|
|
// We might have some portion of the module that looks like this:
|
|
|
|
// AAAAAA (6 A's)
|
|
|
|
//
|
|
|
|
// In this case, there are 5 different copies of "AA" in this range, but
|
|
|
|
// at most 3 can be outlined. If only outlining 3 of these is going to
|
|
|
|
// be unbeneficial, then we ought to not bother.
|
|
|
|
//
|
|
|
|
// Note that two things DON'T overlap when they look like this:
|
|
|
|
// start1...end1 .... start2...end2
|
|
|
|
// That is, one must either
|
|
|
|
// * End before the other starts
|
|
|
|
// * Start after the other ends
|
2021-01-07 10:27:36 +08:00
|
|
|
if (llvm::all_of(CandidatesForRepeatedSeq, [&StartIdx,
|
|
|
|
&EndIdx](const Candidate &C) {
|
|
|
|
return (EndIdx < C.getStartIdx() || StartIdx > C.getEndIdx());
|
|
|
|
})) {
|
2018-11-07 05:46:41 +08:00
|
|
|
// It doesn't overlap with anything, so we can outline it.
|
|
|
|
// Each sequence is over [StartIt, EndIt].
|
|
|
|
// Save the candidate and its location.
|
|
|
|
|
|
|
|
MachineBasicBlock::iterator StartIt = Mapper.InstrList[StartIdx];
|
|
|
|
MachineBasicBlock::iterator EndIt = Mapper.InstrList[EndIdx];
|
2018-11-14 07:01:34 +08:00
|
|
|
MachineBasicBlock *MBB = StartIt->getParent();
|
2018-11-07 05:46:41 +08:00
|
|
|
|
|
|
|
CandidatesForRepeatedSeq.emplace_back(StartIdx, StringLen, StartIt,
|
2018-11-14 07:01:34 +08:00
|
|
|
EndIt, MBB, FunctionList.size(),
|
|
|
|
Mapper.MBBFlagsMap[MBB]);
|
2017-07-28 11:21:58 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-04 04:32:55 +08:00
|
|
|
// We've found something we might want to outline.
|
|
|
|
// Create an OutlinedFunction to store it and check if it'd be beneficial
|
|
|
|
// to outline.
|
2018-11-15 08:02:24 +08:00
|
|
|
if (CandidatesForRepeatedSeq.size() < 2)
|
2018-08-01 08:37:20 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// Arbitrarily choose a TII from the first candidate.
|
|
|
|
// FIXME: Should getOutliningCandidateInfo move to TargetMachine?
|
|
|
|
const TargetInstrInfo *TII =
|
|
|
|
CandidatesForRepeatedSeq[0].getMF()->getSubtarget().getInstrInfo();
|
|
|
|
|
2018-07-28 02:21:57 +08:00
|
|
|
OutlinedFunction OF =
|
2018-08-01 08:37:20 +08:00
|
|
|
TII->getOutliningCandidateInfo(CandidatesForRepeatedSeq);
|
2018-07-28 02:21:57 +08:00
|
|
|
|
2018-11-14 06:16:27 +08:00
|
|
|
// If we deleted too many candidates, then there's nothing worth outlining.
|
|
|
|
// FIXME: This should take target-specified instruction sizes into account.
|
|
|
|
if (OF.Candidates.size() < 2)
|
2018-07-28 02:21:57 +08:00
|
|
|
continue;
|
|
|
|
|
2017-09-01 05:02:45 +08:00
|
|
|
// Is it better to outline this candidate than not?
|
2018-07-25 01:36:13 +08:00
|
|
|
if (OF.getBenefit() < 1) {
|
2018-07-25 01:37:28 +08:00
|
|
|
emitNotOutliningCheaperRemark(StringLen, CandidatesForRepeatedSeq, OF);
|
2017-07-28 07:24:43 +08:00
|
|
|
continue;
|
2017-09-01 05:02:45 +08:00
|
|
|
}
|
2017-07-28 07:24:43 +08:00
|
|
|
|
2017-10-04 04:32:55 +08:00
|
|
|
FunctionList.push_back(OF);
|
2017-07-28 07:24:43 +08:00
|
|
|
}
|
2017-03-07 05:31:18 +08:00
|
|
|
}
|
|
|
|
|
2019-10-29 05:57:51 +08:00
|
|
|
MachineFunction *MachineOutliner::createOutlinedFunction(
|
|
|
|
Module &M, OutlinedFunction &OF, InstructionMapper &Mapper, unsigned Name) {
|
2017-03-07 05:31:18 +08:00
|
|
|
|
2019-04-10 22:52:37 +08:00
|
|
|
// Create the function name. This should be unique.
|
2018-11-08 02:36:43 +08:00
|
|
|
// FIXME: We should have a better naming scheme. This should be stable,
|
|
|
|
// regardless of changes to the outliner's cost model/traversal order.
|
2020-05-06 11:25:13 +08:00
|
|
|
std::string FunctionName = "OUTLINED_FUNCTION_";
|
2020-03-18 06:40:26 +08:00
|
|
|
if (OutlineRepeatedNum > 0)
|
2020-05-06 11:25:13 +08:00
|
|
|
FunctionName += std::to_string(OutlineRepeatedNum + 1) + "_";
|
|
|
|
FunctionName += std::to_string(Name);
|
2017-03-07 05:31:18 +08:00
|
|
|
|
|
|
|
// Create the function using an IR-level function.
|
|
|
|
LLVMContext &C = M.getContext();
|
2019-04-10 22:52:37 +08:00
|
|
|
Function *F = Function::Create(FunctionType::get(Type::getVoidTy(C), false),
|
|
|
|
Function::ExternalLinkage, FunctionName, M);
|
2017-03-07 05:31:18 +08:00
|
|
|
|
|
|
|
// NOTE: If this is linkonceodr, then we can take advantage of linker deduping
|
|
|
|
// which gives us better results when we outline from linkonceodr functions.
|
2018-04-04 05:36:00 +08:00
|
|
|
F->setLinkage(GlobalValue::InternalLinkage);
|
2017-03-07 05:31:18 +08:00
|
|
|
F->setUnnamedAddr(GlobalValue::UnnamedAddr::Global);
|
|
|
|
|
2018-05-16 07:36:46 +08:00
|
|
|
// Set optsize/minsize, so we don't insert padding between outlined
|
|
|
|
// functions.
|
|
|
|
F->addFnAttr(Attribute::OptimizeForSize);
|
|
|
|
F->addFnAttr(Attribute::MinSize);
|
|
|
|
|
2018-12-06 07:24:22 +08:00
|
|
|
Candidate &FirstCand = OF.Candidates.front();
|
2021-12-01 20:44:09 +08:00
|
|
|
const TargetInstrInfo &TII =
|
|
|
|
*FirstCand.getMF()->getSubtarget().getInstrInfo();
|
2018-10-30 04:27:07 +08:00
|
|
|
|
2021-12-01 20:44:09 +08:00
|
|
|
TII.mergeOutliningCandidateAttributes(*F, OF.Candidates);
|
2020-07-01 22:28:44 +08:00
|
|
|
|
Extend the `uwtable` attribute with unwind table kind
We have the `clang -cc1` command-line option `-funwind-tables=1|2` and
the codegen option `VALUE_CODEGENOPT(UnwindTables, 2, 0) ///< Unwind
tables (1) or asynchronous unwind tables (2)`. However, this is
encoded in LLVM IR by the presence or the absence of the `uwtable`
attribute, i.e. we lose the information whether to generate want just
some unwind tables or asynchronous unwind tables.
Asynchronous unwind tables take more space in the runtime image, I'd
estimate something like 80-90% more, as the difference is adding
roughly the same number of CFI directives as for prologues, only a bit
simpler (e.g. `.cfi_offset reg, off` vs. `.cfi_restore reg`). Or even
more, if you consider tail duplication of epilogue blocks.
Asynchronous unwind tables could also restrict code generation to
having only a finite number of frame pointer adjustments (an example
of *not* having a finite number of `SP` adjustments is on AArch64 when
untagging the stack (MTE) in some cases the compiler can modify `SP`
in a loop).
Having the CFI precise up to an instruction generally also means one
cannot bundle together CFI instructions once the prologue is done,
they need to be interspersed with ordinary instructions, which means
extra `DW_CFA_advance_loc` commands, further increasing the unwind
tables size.
That is to say, async unwind tables impose a non-negligible overhead,
yet for the most common use cases (like C++ exceptions), they are not
even needed.
This patch extends the `uwtable` attribute with an optional
value:
- `uwtable` (default to `async`)
- `uwtable(sync)`, synchronous unwind tables
- `uwtable(async)`, asynchronous (instruction precise) unwind tables
Reviewed By: MaskRay
Differential Revision: https://reviews.llvm.org/D114543
2022-02-14 21:41:34 +08:00
|
|
|
// Set uwtable, so we generate eh_frame.
|
|
|
|
UWTableKind UW = std::accumulate(
|
|
|
|
OF.Candidates.cbegin(), OF.Candidates.cend(), UWTableKind::None,
|
|
|
|
[](UWTableKind K, const outliner::Candidate &C) {
|
|
|
|
return std::max(K, C.getMF()->getFunction().getUWTableKind());
|
|
|
|
});
|
|
|
|
if (UW != UWTableKind::None)
|
|
|
|
F->setUWTableKind(UW);
|
|
|
|
|
2017-03-07 05:31:18 +08:00
|
|
|
BasicBlock *EntryBB = BasicBlock::Create(C, "entry", F);
|
|
|
|
IRBuilder<> Builder(EntryBB);
|
|
|
|
Builder.CreateRetVoid();
|
|
|
|
|
2019-10-01 01:54:50 +08:00
|
|
|
MachineModuleInfo &MMI = getAnalysis<MachineModuleInfoWrapperPass>().getMMI();
|
2017-06-06 08:44:35 +08:00
|
|
|
MachineFunction &MF = MMI.getOrCreateMachineFunction(*F);
|
2017-03-07 05:31:18 +08:00
|
|
|
MachineBasicBlock &MBB = *MF.CreateMachineBasicBlock();
|
|
|
|
|
|
|
|
// Insert the new function into the module.
|
|
|
|
MF.insert(MF.begin(), &MBB);
|
|
|
|
|
2020-04-10 09:06:38 +08:00
|
|
|
MachineFunction *OriginalMF = FirstCand.front()->getMF();
|
|
|
|
const std::vector<MCCFIInstruction> &Instrs =
|
|
|
|
OriginalMF->getFrameInstructions();
|
2018-12-06 01:57:33 +08:00
|
|
|
for (auto I = FirstCand.front(), E = std::next(FirstCand.back()); I != E;
|
|
|
|
++I) {
|
2020-11-06 03:11:43 +08:00
|
|
|
if (I->isDebugInstr())
|
|
|
|
continue;
|
2018-12-06 01:57:33 +08:00
|
|
|
MachineInstr *NewMI = MF.CloneMachineInstr(&*I);
|
2020-04-10 09:06:38 +08:00
|
|
|
if (I->isCFIInstruction()) {
|
|
|
|
unsigned CFIIndex = NewMI->getOperand(0).getCFIIndex();
|
|
|
|
MCCFIInstruction CFI = Instrs[CFIIndex];
|
|
|
|
(void)MF.addFrameInst(CFI);
|
|
|
|
}
|
2018-08-17 05:30:05 +08:00
|
|
|
NewMI->dropMemRefs(MF);
|
2017-03-07 05:31:18 +08:00
|
|
|
|
|
|
|
// Don't keep debug information for outlined instructions.
|
|
|
|
NewMI->setDebugLoc(DebugLoc());
|
|
|
|
MBB.insert(MBB.end(), NewMI);
|
|
|
|
}
|
|
|
|
|
2020-04-22 08:40:41 +08:00
|
|
|
// Set normal properties for a late MachineFunction.
|
|
|
|
MF.getProperties().reset(MachineFunctionProperties::Property::IsSSA);
|
|
|
|
MF.getProperties().set(MachineFunctionProperties::Property::NoPHIs);
|
|
|
|
MF.getProperties().set(MachineFunctionProperties::Property::NoVRegs);
|
|
|
|
MF.getProperties().set(MachineFunctionProperties::Property::TracksLiveness);
|
2018-09-21 02:53:53 +08:00
|
|
|
MF.getRegInfo().freezeReservedRegs(MF);
|
|
|
|
|
2020-04-22 08:40:41 +08:00
|
|
|
// Compute live-in set for outlined fn
|
|
|
|
const MachineRegisterInfo &MRI = MF.getRegInfo();
|
|
|
|
const TargetRegisterInfo &TRI = *MRI.getTargetRegisterInfo();
|
|
|
|
LivePhysRegs LiveIns(TRI);
|
|
|
|
for (auto &Cand : OF.Candidates) {
|
|
|
|
// Figure out live-ins at the first instruction.
|
|
|
|
MachineBasicBlock &OutlineBB = *Cand.front()->getParent();
|
|
|
|
LivePhysRegs CandLiveIns(TRI);
|
|
|
|
CandLiveIns.addLiveOuts(OutlineBB);
|
|
|
|
for (const MachineInstr &MI :
|
|
|
|
reverse(make_range(Cand.front(), OutlineBB.end())))
|
|
|
|
CandLiveIns.stepBackward(MI);
|
|
|
|
|
|
|
|
// The live-in set for the outlined function is the union of the live-ins
|
|
|
|
// from all the outlining points.
|
2021-01-10 01:24:59 +08:00
|
|
|
for (MCPhysReg Reg : CandLiveIns)
|
2020-04-22 08:40:41 +08:00
|
|
|
LiveIns.addReg(Reg);
|
|
|
|
}
|
|
|
|
addLiveIns(MBB, LiveIns);
|
|
|
|
|
|
|
|
TII.buildOutlinedFrame(MBB, MF, OF);
|
|
|
|
|
2018-01-20 05:21:49 +08:00
|
|
|
// If there's a DISubprogram associated with this outlined function, then
|
|
|
|
// emit debug info for the outlined function.
|
2018-06-05 05:14:16 +08:00
|
|
|
if (DISubprogram *SP = getSubprogramOrNull(OF)) {
|
2018-01-20 05:21:49 +08:00
|
|
|
// We have a DISubprogram. Get its DICompileUnit.
|
|
|
|
DICompileUnit *CU = SP->getUnit();
|
|
|
|
DIBuilder DB(M, true, CU);
|
|
|
|
DIFile *Unit = SP->getFile();
|
|
|
|
Mangler Mg;
|
2018-09-21 02:53:53 +08:00
|
|
|
// Get the mangled name of the function for the linkage name.
|
|
|
|
std::string Dummy;
|
|
|
|
llvm::raw_string_ostream MangledNameStream(Dummy);
|
|
|
|
Mg.getNameWithPrefix(MangledNameStream, F, false);
|
|
|
|
|
|
|
|
DISubprogram *OutlinedSP = DB.createFunction(
|
|
|
|
Unit /* Context */, F->getName(), StringRef(MangledNameStream.str()),
|
|
|
|
Unit /* File */,
|
|
|
|
0 /* Line 0 is reserved for compiler-generated code. */,
|
|
|
|
DB.createSubroutineType(DB.getOrCreateTypeArray(None)), /* void type */
|
2018-11-20 02:29:28 +08:00
|
|
|
0, /* Line 0 is reserved for compiler-generated code. */
|
2018-09-21 02:53:53 +08:00
|
|
|
DINode::DIFlags::FlagArtificial /* Compiler-generated code. */,
|
2018-11-20 02:29:28 +08:00
|
|
|
/* Outlined code is optimized code by definition. */
|
|
|
|
DISubprogram::SPFlagDefinition | DISubprogram::SPFlagOptimized);
|
2018-09-21 02:53:53 +08:00
|
|
|
|
|
|
|
// Don't add any new variables to the subprogram.
|
|
|
|
DB.finalizeSubprogram(OutlinedSP);
|
|
|
|
|
|
|
|
// Attach subprogram to the function.
|
|
|
|
F->setSubprogram(OutlinedSP);
|
2018-01-20 05:21:49 +08:00
|
|
|
// We're done with the DIBuilder.
|
|
|
|
DB.finalize();
|
|
|
|
}
|
|
|
|
|
2017-03-07 05:31:18 +08:00
|
|
|
return &MF;
|
|
|
|
}
|
|
|
|
|
2018-12-06 06:50:26 +08:00
|
|
|
bool MachineOutliner::outline(Module &M,
|
|
|
|
std::vector<OutlinedFunction> &FunctionList,
|
2019-10-29 03:10:21 +08:00
|
|
|
InstructionMapper &Mapper,
|
|
|
|
unsigned &OutlinedFunctionNum) {
|
2017-03-07 05:31:18 +08:00
|
|
|
|
|
|
|
bool OutlinedSomething = false;
|
2018-11-08 02:36:43 +08:00
|
|
|
|
2018-12-06 05:36:04 +08:00
|
|
|
// Sort by benefit. The most beneficial functions should be outlined first.
|
2019-04-23 22:51:27 +08:00
|
|
|
llvm::stable_sort(FunctionList, [](const OutlinedFunction &LHS,
|
|
|
|
const OutlinedFunction &RHS) {
|
|
|
|
return LHS.getBenefit() > RHS.getBenefit();
|
|
|
|
});
|
2018-12-06 05:36:04 +08:00
|
|
|
|
|
|
|
// Walk over each function, outlining them as we go along. Functions are
|
|
|
|
// outlined greedily, based off the sort above.
|
|
|
|
for (OutlinedFunction &OF : FunctionList) {
|
|
|
|
// If we outlined something that overlapped with a candidate in a previous
|
|
|
|
// step, then we can't outline from it.
|
2018-12-06 07:24:22 +08:00
|
|
|
erase_if(OF.Candidates, [&Mapper](Candidate &C) {
|
2018-12-06 06:47:25 +08:00
|
|
|
return std::any_of(
|
2018-12-06 07:24:22 +08:00
|
|
|
Mapper.UnsignedVec.begin() + C.getStartIdx(),
|
|
|
|
Mapper.UnsignedVec.begin() + C.getEndIdx() + 1,
|
2018-12-06 06:47:25 +08:00
|
|
|
[](unsigned I) { return (I == static_cast<unsigned>(-1)); });
|
2018-12-06 06:27:38 +08:00
|
|
|
});
|
2018-12-06 05:36:04 +08:00
|
|
|
|
|
|
|
// If we made it unbeneficial to outline this function, skip it.
|
2017-10-18 03:03:23 +08:00
|
|
|
if (OF.getBenefit() < 1)
|
2017-03-07 05:31:18 +08:00
|
|
|
continue;
|
|
|
|
|
2018-12-06 05:36:04 +08:00
|
|
|
// It's beneficial. Create the function and outline its sequence's
|
|
|
|
// occurrences.
|
|
|
|
OF.MF = createOutlinedFunction(M, OF, Mapper, OutlinedFunctionNum);
|
|
|
|
emitOutlinedFunctionRemark(OF);
|
|
|
|
FunctionsCreated++;
|
|
|
|
OutlinedFunctionNum++; // Created a function, move to the next name.
|
2017-03-07 05:31:18 +08:00
|
|
|
MachineFunction *MF = OF.MF;
|
|
|
|
const TargetSubtargetInfo &STI = MF->getSubtarget();
|
|
|
|
const TargetInstrInfo &TII = *STI.getInstrInfo();
|
|
|
|
|
2018-12-06 05:36:04 +08:00
|
|
|
// Replace occurrences of the sequence with calls to the new function.
|
2018-12-06 07:24:22 +08:00
|
|
|
for (Candidate &C : OF.Candidates) {
|
2018-12-06 05:36:04 +08:00
|
|
|
MachineBasicBlock &MBB = *C.getMBB();
|
|
|
|
MachineBasicBlock::iterator StartIt = C.front();
|
|
|
|
MachineBasicBlock::iterator EndIt = C.back();
|
|
|
|
|
|
|
|
// Insert the call.
|
|
|
|
auto CallInst = TII.insertOutlinedCall(M, MBB, StartIt, *MF, C);
|
|
|
|
|
|
|
|
// If the caller tracks liveness, then we need to make sure that
|
|
|
|
// anything we outline doesn't break liveness assumptions. The outlined
|
|
|
|
// functions themselves currently don't track liveness, but we should
|
|
|
|
// make sure that the ranges we yank things out of aren't wrong.
|
|
|
|
if (MBB.getParent()->getProperties().hasProperty(
|
|
|
|
MachineFunctionProperties::Property::TracksLiveness)) {
|
2020-03-06 05:54:58 +08:00
|
|
|
// The following code is to add implicit def operands to the call
|
2019-06-27 21:10:29 +08:00
|
|
|
// instruction. It also updates call site information for moved
|
|
|
|
// code.
|
2020-03-06 05:54:58 +08:00
|
|
|
SmallSet<Register, 2> UseRegs, DefRegs;
|
2018-12-06 05:36:04 +08:00
|
|
|
// Copy over the defs in the outlined range.
|
|
|
|
// First inst in outlined range <-- Anything that's defined in this
|
|
|
|
// ... .. range has to be added as an
|
|
|
|
// implicit Last inst in outlined range <-- def to the call
|
2019-06-27 21:10:29 +08:00
|
|
|
// instruction. Also remove call site information for outlined block
|
2020-03-06 05:54:58 +08:00
|
|
|
// of code. The exposed uses need to be copied in the outlined range.
|
2020-04-29 15:33:47 +08:00
|
|
|
for (MachineBasicBlock::reverse_iterator
|
|
|
|
Iter = EndIt.getReverse(),
|
|
|
|
Last = std::next(CallInst.getReverse());
|
2020-03-06 05:54:58 +08:00
|
|
|
Iter != Last; Iter++) {
|
|
|
|
MachineInstr *MI = &*Iter;
|
Fix the side effect of outlined function when the register is implicit use and implicit-def in the same instruction.
This is the diff associated with {D95267}, and we need to mark $x0 as live whether or not $x0 is dead.
The compiler also needs to mark register $x0 as live in for the following case.
```
$x1 = ADDXri $sp, 16, 0
BL @spam, csr_darwin_aarch64_aapcs, implicit-def dead $lr, implicit $sp, implicit $x0, implicit killed $x1, implicit-def $sp, implicit-def $x0
```
This change fixes an issue where the wrong registers were used when -machine-outliner-reruns>0.
As an example:
```
lang=c
typedef struct {
double v1;
double v2;
} D16;
typedef struct {
D16 v1;
D16 v2;
} D32;
typedef long long LL8;
typedef struct {
long long v1;
long long v2;
} LL16;
typedef struct {
LL16 v1;
LL16 v2;
} LL32;
typedef struct {
LL32 v1;
LL32 v2;
} LL64;
LL8 needx0(LL8 v0, LL8 v1);
void bar(LL64 v1, LL32 v2, LL16 v3, LL32 v4, LL8 v5, D16 v6, D16 v7, D16 v8);
LL8 foo(LL8 v0, LL64 v1, LL32 v2, LL16 v3, LL32 v4, LL8 v5, D16 v6, D16 v7, D16 v8)
{
LL8 result = needx0(v0, 0);
bar(v1, v2, v3, v4, v5, v6, v7, v8);
return result + 1;
}
```
As you can see from the `foo` function, we should not modify the value of `x0` until we call `needx0`.
This code is compiled to give the following instruction MIR code.
```
$sp = frame-setup SUBXri $sp, 256, 0
frame-setup STPDi killed $d13, killed $d12, $sp, 16
frame-setup STPDi killed $d11, killed $d10, $sp, 18
frame-setup STPDi killed $d9, killed $d8, $sp, 20
frame-setup STPXi killed $x26, killed $x25, $sp, 22
frame-setup STPXi killed $x24, killed $x23, $sp, 24
frame-setup STPXi killed $x22, killed $x21, $sp, 26
frame-setup STPXi killed $x20, killed $x19, $sp, 28
...
$x1 = MOVZXi 0, 0
BL @needx0, csr_darwin_aarch64_aapcs, implicit-def dead $lr, implicit $sp, implicit $x0, implicit $x1, implicit-def $sp, implicit-def $x0
...
```
Since there are some other instruction sequences that duplicate `foo`, after the first execution of Machine Outliner you will get:
```
$sp = frame-setup SUBXri $sp, 256, 0
frame-setup STPDi killed $d13, killed $d12, $sp, 16
frame-setup STPDi killed $d11, killed $d10, $sp, 18
frame-setup STPDi killed $d9, killed $d8, $sp, 20
$x7 = ORRXrs $xzr, $lr, 0
BL @OUTLINED_FUNCTION_0, implicit-def $lr, implicit $sp, implicit-def $lr, implicit $sp, implicit $xzr, implicit $x7, implicit $x19, implicit $x20, implicit $x21, implicit $x22, implicit $x23, implicit $x24, implicit $x25, implicit $x26
$lr = ORRXrs $xzr, $x7, 0
...
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp
...
```
For the first time we outlined the following sequence:
```
frame-setup STPXi killed $x26, killed $x25, $sp, 22
frame-setup STPXi killed $x24, killed $x23, $sp, 24
frame-setup STPXi killed $x22, killed $x21, $sp, 26
frame-setup STPXi killed $x20, killed $x19, $sp, 28
```
and
```
$x1 = MOVZXi 0, 0
BL @needx0, csr_darwin_aarch64_aapcs, implicit-def dead $lr, implicit $sp, implicit $x0, implicit $x1, implicit-def $sp, implicit-def $x0
```
When we execute the outline again, we will get:
```
$x0 = ORRXrs $xzr, $lr, 0 <---- here
BL @OUTLINED_FUNCTION_2_0, implicit-def $lr, implicit $sp, implicit-def $sp, implicit-def $lr, implicit $sp, implicit $xzr, implicit $d8, implicit $d9, implicit $d10, implicit $d11, implicit $d12, implicit $d13, implicit $x0
$lr = ORRXrs $xzr, $x0, 0
$x7 = ORRXrs $xzr, $lr, 0
BL @OUTLINED_FUNCTION_0, implicit-def $lr, implicit $sp, implicit-def $lr, implicit $sp, implicit $xzr, implicit $x7, implicit $x19, implicit $x20, implicit $x21, implicit $x22, implicit $x23, implicit $x24, implicit $x25, implicit $x26
$lr = ORRXrs $xzr, $x7, 0
...
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp
```
When calling `OUTLINED_FUNCTION_2_0`, we used `x0` to save the `lr` register.
The reason for the above error appears to be that:
```
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp
```
should be:
```
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp, implicit $x0
```
When processing the same instruction with both `implicit-def $x0` and `implicit $x0` we should keep `implicit $x0`.
A reproducible demo is available at: [https://github.com/DianQK/reproduce_outlined_function_use_live_x0](https://github.com/DianQK/reproduce_outlined_function_use_live_x0).
Reviewed By: jinlin
Differential Revision: https://reviews.llvm.org/D112911
2021-11-18 01:42:21 +08:00
|
|
|
SmallSet<Register, 2> InstrUseRegs;
|
2020-03-06 05:54:58 +08:00
|
|
|
for (MachineOperand &MOP : MI->operands()) {
|
|
|
|
// Skip over anything that isn't a register.
|
|
|
|
if (!MOP.isReg())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (MOP.isDef()) {
|
|
|
|
// Introduce DefRegs set to skip the redundant register.
|
|
|
|
DefRegs.insert(MOP.getReg());
|
Fix the side effect of outlined function when the register is implicit use and implicit-def in the same instruction.
This is the diff associated with {D95267}, and we need to mark $x0 as live whether or not $x0 is dead.
The compiler also needs to mark register $x0 as live in for the following case.
```
$x1 = ADDXri $sp, 16, 0
BL @spam, csr_darwin_aarch64_aapcs, implicit-def dead $lr, implicit $sp, implicit $x0, implicit killed $x1, implicit-def $sp, implicit-def $x0
```
This change fixes an issue where the wrong registers were used when -machine-outliner-reruns>0.
As an example:
```
lang=c
typedef struct {
double v1;
double v2;
} D16;
typedef struct {
D16 v1;
D16 v2;
} D32;
typedef long long LL8;
typedef struct {
long long v1;
long long v2;
} LL16;
typedef struct {
LL16 v1;
LL16 v2;
} LL32;
typedef struct {
LL32 v1;
LL32 v2;
} LL64;
LL8 needx0(LL8 v0, LL8 v1);
void bar(LL64 v1, LL32 v2, LL16 v3, LL32 v4, LL8 v5, D16 v6, D16 v7, D16 v8);
LL8 foo(LL8 v0, LL64 v1, LL32 v2, LL16 v3, LL32 v4, LL8 v5, D16 v6, D16 v7, D16 v8)
{
LL8 result = needx0(v0, 0);
bar(v1, v2, v3, v4, v5, v6, v7, v8);
return result + 1;
}
```
As you can see from the `foo` function, we should not modify the value of `x0` until we call `needx0`.
This code is compiled to give the following instruction MIR code.
```
$sp = frame-setup SUBXri $sp, 256, 0
frame-setup STPDi killed $d13, killed $d12, $sp, 16
frame-setup STPDi killed $d11, killed $d10, $sp, 18
frame-setup STPDi killed $d9, killed $d8, $sp, 20
frame-setup STPXi killed $x26, killed $x25, $sp, 22
frame-setup STPXi killed $x24, killed $x23, $sp, 24
frame-setup STPXi killed $x22, killed $x21, $sp, 26
frame-setup STPXi killed $x20, killed $x19, $sp, 28
...
$x1 = MOVZXi 0, 0
BL @needx0, csr_darwin_aarch64_aapcs, implicit-def dead $lr, implicit $sp, implicit $x0, implicit $x1, implicit-def $sp, implicit-def $x0
...
```
Since there are some other instruction sequences that duplicate `foo`, after the first execution of Machine Outliner you will get:
```
$sp = frame-setup SUBXri $sp, 256, 0
frame-setup STPDi killed $d13, killed $d12, $sp, 16
frame-setup STPDi killed $d11, killed $d10, $sp, 18
frame-setup STPDi killed $d9, killed $d8, $sp, 20
$x7 = ORRXrs $xzr, $lr, 0
BL @OUTLINED_FUNCTION_0, implicit-def $lr, implicit $sp, implicit-def $lr, implicit $sp, implicit $xzr, implicit $x7, implicit $x19, implicit $x20, implicit $x21, implicit $x22, implicit $x23, implicit $x24, implicit $x25, implicit $x26
$lr = ORRXrs $xzr, $x7, 0
...
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp
...
```
For the first time we outlined the following sequence:
```
frame-setup STPXi killed $x26, killed $x25, $sp, 22
frame-setup STPXi killed $x24, killed $x23, $sp, 24
frame-setup STPXi killed $x22, killed $x21, $sp, 26
frame-setup STPXi killed $x20, killed $x19, $sp, 28
```
and
```
$x1 = MOVZXi 0, 0
BL @needx0, csr_darwin_aarch64_aapcs, implicit-def dead $lr, implicit $sp, implicit $x0, implicit $x1, implicit-def $sp, implicit-def $x0
```
When we execute the outline again, we will get:
```
$x0 = ORRXrs $xzr, $lr, 0 <---- here
BL @OUTLINED_FUNCTION_2_0, implicit-def $lr, implicit $sp, implicit-def $sp, implicit-def $lr, implicit $sp, implicit $xzr, implicit $d8, implicit $d9, implicit $d10, implicit $d11, implicit $d12, implicit $d13, implicit $x0
$lr = ORRXrs $xzr, $x0, 0
$x7 = ORRXrs $xzr, $lr, 0
BL @OUTLINED_FUNCTION_0, implicit-def $lr, implicit $sp, implicit-def $lr, implicit $sp, implicit $xzr, implicit $x7, implicit $x19, implicit $x20, implicit $x21, implicit $x22, implicit $x23, implicit $x24, implicit $x25, implicit $x26
$lr = ORRXrs $xzr, $x7, 0
...
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp
```
When calling `OUTLINED_FUNCTION_2_0`, we used `x0` to save the `lr` register.
The reason for the above error appears to be that:
```
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp
```
should be:
```
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp, implicit $x0
```
When processing the same instruction with both `implicit-def $x0` and `implicit $x0` we should keep `implicit $x0`.
A reproducible demo is available at: [https://github.com/DianQK/reproduce_outlined_function_use_live_x0](https://github.com/DianQK/reproduce_outlined_function_use_live_x0).
Reviewed By: jinlin
Differential Revision: https://reviews.llvm.org/D112911
2021-11-18 01:42:21 +08:00
|
|
|
if (UseRegs.count(MOP.getReg()) &&
|
|
|
|
!InstrUseRegs.count(MOP.getReg()))
|
2020-03-06 05:54:58 +08:00
|
|
|
// Since the regiester is modeled as defined,
|
|
|
|
// it is not necessary to be put in use register set.
|
|
|
|
UseRegs.erase(MOP.getReg());
|
|
|
|
} else if (!MOP.isUndef()) {
|
|
|
|
// Any register which is not undefined should
|
|
|
|
// be put in the use register set.
|
|
|
|
UseRegs.insert(MOP.getReg());
|
Fix the side effect of outlined function when the register is implicit use and implicit-def in the same instruction.
This is the diff associated with {D95267}, and we need to mark $x0 as live whether or not $x0 is dead.
The compiler also needs to mark register $x0 as live in for the following case.
```
$x1 = ADDXri $sp, 16, 0
BL @spam, csr_darwin_aarch64_aapcs, implicit-def dead $lr, implicit $sp, implicit $x0, implicit killed $x1, implicit-def $sp, implicit-def $x0
```
This change fixes an issue where the wrong registers were used when -machine-outliner-reruns>0.
As an example:
```
lang=c
typedef struct {
double v1;
double v2;
} D16;
typedef struct {
D16 v1;
D16 v2;
} D32;
typedef long long LL8;
typedef struct {
long long v1;
long long v2;
} LL16;
typedef struct {
LL16 v1;
LL16 v2;
} LL32;
typedef struct {
LL32 v1;
LL32 v2;
} LL64;
LL8 needx0(LL8 v0, LL8 v1);
void bar(LL64 v1, LL32 v2, LL16 v3, LL32 v4, LL8 v5, D16 v6, D16 v7, D16 v8);
LL8 foo(LL8 v0, LL64 v1, LL32 v2, LL16 v3, LL32 v4, LL8 v5, D16 v6, D16 v7, D16 v8)
{
LL8 result = needx0(v0, 0);
bar(v1, v2, v3, v4, v5, v6, v7, v8);
return result + 1;
}
```
As you can see from the `foo` function, we should not modify the value of `x0` until we call `needx0`.
This code is compiled to give the following instruction MIR code.
```
$sp = frame-setup SUBXri $sp, 256, 0
frame-setup STPDi killed $d13, killed $d12, $sp, 16
frame-setup STPDi killed $d11, killed $d10, $sp, 18
frame-setup STPDi killed $d9, killed $d8, $sp, 20
frame-setup STPXi killed $x26, killed $x25, $sp, 22
frame-setup STPXi killed $x24, killed $x23, $sp, 24
frame-setup STPXi killed $x22, killed $x21, $sp, 26
frame-setup STPXi killed $x20, killed $x19, $sp, 28
...
$x1 = MOVZXi 0, 0
BL @needx0, csr_darwin_aarch64_aapcs, implicit-def dead $lr, implicit $sp, implicit $x0, implicit $x1, implicit-def $sp, implicit-def $x0
...
```
Since there are some other instruction sequences that duplicate `foo`, after the first execution of Machine Outliner you will get:
```
$sp = frame-setup SUBXri $sp, 256, 0
frame-setup STPDi killed $d13, killed $d12, $sp, 16
frame-setup STPDi killed $d11, killed $d10, $sp, 18
frame-setup STPDi killed $d9, killed $d8, $sp, 20
$x7 = ORRXrs $xzr, $lr, 0
BL @OUTLINED_FUNCTION_0, implicit-def $lr, implicit $sp, implicit-def $lr, implicit $sp, implicit $xzr, implicit $x7, implicit $x19, implicit $x20, implicit $x21, implicit $x22, implicit $x23, implicit $x24, implicit $x25, implicit $x26
$lr = ORRXrs $xzr, $x7, 0
...
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp
...
```
For the first time we outlined the following sequence:
```
frame-setup STPXi killed $x26, killed $x25, $sp, 22
frame-setup STPXi killed $x24, killed $x23, $sp, 24
frame-setup STPXi killed $x22, killed $x21, $sp, 26
frame-setup STPXi killed $x20, killed $x19, $sp, 28
```
and
```
$x1 = MOVZXi 0, 0
BL @needx0, csr_darwin_aarch64_aapcs, implicit-def dead $lr, implicit $sp, implicit $x0, implicit $x1, implicit-def $sp, implicit-def $x0
```
When we execute the outline again, we will get:
```
$x0 = ORRXrs $xzr, $lr, 0 <---- here
BL @OUTLINED_FUNCTION_2_0, implicit-def $lr, implicit $sp, implicit-def $sp, implicit-def $lr, implicit $sp, implicit $xzr, implicit $d8, implicit $d9, implicit $d10, implicit $d11, implicit $d12, implicit $d13, implicit $x0
$lr = ORRXrs $xzr, $x0, 0
$x7 = ORRXrs $xzr, $lr, 0
BL @OUTLINED_FUNCTION_0, implicit-def $lr, implicit $sp, implicit-def $lr, implicit $sp, implicit $xzr, implicit $x7, implicit $x19, implicit $x20, implicit $x21, implicit $x22, implicit $x23, implicit $x24, implicit $x25, implicit $x26
$lr = ORRXrs $xzr, $x7, 0
...
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp
```
When calling `OUTLINED_FUNCTION_2_0`, we used `x0` to save the `lr` register.
The reason for the above error appears to be that:
```
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp
```
should be:
```
BL @OUTLINED_FUNCTION_1, implicit-def $lr, implicit $sp, implicit-def $lr, implicit-def $sp, implicit-def $x0, implicit-def $x1, implicit $sp, implicit $x0
```
When processing the same instruction with both `implicit-def $x0` and `implicit $x0` we should keep `implicit $x0`.
A reproducible demo is available at: [https://github.com/DianQK/reproduce_outlined_function_use_live_x0](https://github.com/DianQK/reproduce_outlined_function_use_live_x0).
Reviewed By: jinlin
Differential Revision: https://reviews.llvm.org/D112911
2021-11-18 01:42:21 +08:00
|
|
|
InstrUseRegs.insert(MOP.getReg());
|
2020-03-06 05:54:58 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (MI->isCandidateForCallSiteEntry())
|
|
|
|
MI->getMF()->eraseCallSiteInfo(MI);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const Register &I : DefRegs)
|
2020-04-29 15:33:47 +08:00
|
|
|
// If it's a def, add it to the call instruction.
|
|
|
|
CallInst->addOperand(
|
|
|
|
MachineOperand::CreateReg(I, true, /* isDef = true */
|
|
|
|
true /* isImp = true */));
|
2020-03-06 05:54:58 +08:00
|
|
|
|
|
|
|
for (const Register &I : UseRegs)
|
|
|
|
// If it's a exposed use, add it to the call instruction.
|
|
|
|
CallInst->addOperand(
|
|
|
|
MachineOperand::CreateReg(I, false, /* isDef = false */
|
|
|
|
true /* isImp = true */));
|
2018-12-06 05:36:04 +08:00
|
|
|
}
|
2018-04-28 07:36:35 +08:00
|
|
|
|
2018-12-06 05:36:04 +08:00
|
|
|
// Erase from the point after where the call was inserted up to, and
|
|
|
|
// including, the final instruction in the sequence.
|
|
|
|
// Erase needs one past the end, so we need std::next there too.
|
|
|
|
MBB.erase(std::next(StartIt), std::next(EndIt));
|
2018-12-06 06:27:38 +08:00
|
|
|
|
2018-12-06 06:47:25 +08:00
|
|
|
// Keep track of what we removed by marking them all as -1.
|
2018-12-06 06:27:38 +08:00
|
|
|
std::for_each(Mapper.UnsignedVec.begin() + C.getStartIdx(),
|
|
|
|
Mapper.UnsignedVec.begin() + C.getEndIdx() + 1,
|
2018-12-06 06:47:25 +08:00
|
|
|
[](unsigned &I) { I = static_cast<unsigned>(-1); });
|
2018-12-06 05:36:04 +08:00
|
|
|
OutlinedSomething = true;
|
2017-03-07 05:31:18 +08:00
|
|
|
|
2018-12-06 05:36:04 +08:00
|
|
|
// Statistics.
|
|
|
|
NumOutlined++;
|
|
|
|
}
|
2017-03-07 05:31:18 +08:00
|
|
|
}
|
|
|
|
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "OutlinedSomething = " << OutlinedSomething << "\n";);
|
2017-03-07 05:31:18 +08:00
|
|
|
return OutlinedSomething;
|
|
|
|
}
|
|
|
|
|
2018-09-12 00:33:46 +08:00
|
|
|
void MachineOutliner::populateMapper(InstructionMapper &Mapper, Module &M,
|
|
|
|
MachineModuleInfo &MMI) {
|
2018-03-23 05:07:09 +08:00
|
|
|
// Build instruction mappings for each function in the module. Start by
|
|
|
|
// iterating over each Function in M.
|
2017-03-07 05:31:18 +08:00
|
|
|
for (Function &F : M) {
|
|
|
|
|
2018-03-23 05:07:09 +08:00
|
|
|
// If there's nothing in F, then there's no reason to try and outline from
|
|
|
|
// it.
|
|
|
|
if (F.empty())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// There's something in F. Check if it has a MachineFunction associated with
|
|
|
|
// it.
|
|
|
|
MachineFunction *MF = MMI.getMachineFunction(F);
|
|
|
|
|
|
|
|
// If it doesn't, then there's nothing to outline from. Move to the next
|
|
|
|
// Function.
|
|
|
|
if (!MF)
|
2017-03-07 05:31:18 +08:00
|
|
|
continue;
|
|
|
|
|
2018-08-01 08:37:20 +08:00
|
|
|
const TargetInstrInfo *TII = MF->getSubtarget().getInstrInfo();
|
|
|
|
|
2018-06-30 11:56:03 +08:00
|
|
|
if (!RunOnAllFunctions && !TII->shouldOutlineFromFunctionByDefault(*MF))
|
|
|
|
continue;
|
|
|
|
|
2018-03-23 05:07:09 +08:00
|
|
|
// We have a MachineFunction. Ask the target if it's suitable for outlining.
|
|
|
|
// If it isn't, then move on to the next Function in the module.
|
|
|
|
if (!TII->isFunctionSafeToOutlineFrom(*MF, OutlineFromLinkOnceODRs))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// We have a function suitable for outlining. Iterate over every
|
|
|
|
// MachineBasicBlock in MF and try to map its instructions to a list of
|
|
|
|
// unsigned integers.
|
|
|
|
for (MachineBasicBlock &MBB : *MF) {
|
|
|
|
// If there isn't anything in MBB, then there's no point in outlining from
|
|
|
|
// it.
|
2018-09-21 05:53:25 +08:00
|
|
|
// If there are fewer than 2 instructions in the MBB, then it can't ever
|
|
|
|
// contain something worth outlining.
|
|
|
|
// FIXME: This should be based off of the maximum size in B of an outlined
|
|
|
|
// call versus the size in B of the MBB.
|
|
|
|
if (MBB.empty() || MBB.size() < 2)
|
2018-03-23 05:07:09 +08:00
|
|
|
continue;
|
2017-03-07 05:31:18 +08:00
|
|
|
|
2018-03-23 05:07:09 +08:00
|
|
|
// Check if MBB could be the target of an indirect branch. If it is, then
|
|
|
|
// we don't want to outline from it.
|
|
|
|
if (MBB.hasAddressTaken())
|
2017-03-07 05:31:18 +08:00
|
|
|
continue;
|
|
|
|
|
2018-03-23 05:07:09 +08:00
|
|
|
// MBB is suitable for outlining. Map it to a list of unsigneds.
|
2018-08-01 08:37:20 +08:00
|
|
|
Mapper.convertToUnsignedVec(MBB, *TII);
|
2017-03-07 05:31:18 +08:00
|
|
|
}
|
2022-02-18 10:12:39 +08:00
|
|
|
|
|
|
|
// Statistics.
|
|
|
|
UnsignedVecSize = Mapper.UnsignedVec.size();
|
2017-03-07 05:31:18 +08:00
|
|
|
}
|
2018-09-12 00:33:46 +08:00
|
|
|
}
|
|
|
|
|
2018-09-12 07:05:34 +08:00
|
|
|
void MachineOutliner::initSizeRemarkInfo(
|
|
|
|
const Module &M, const MachineModuleInfo &MMI,
|
|
|
|
StringMap<unsigned> &FunctionToInstrCount) {
|
|
|
|
// Collect instruction counts for every function. We'll use this to emit
|
|
|
|
// per-function size remarks later.
|
|
|
|
for (const Function &F : M) {
|
|
|
|
MachineFunction *MF = MMI.getMachineFunction(F);
|
|
|
|
|
|
|
|
// We only care about MI counts here. If there's no MachineFunction at this
|
|
|
|
// point, then there won't be after the outliner runs, so let's move on.
|
|
|
|
if (!MF)
|
|
|
|
continue;
|
|
|
|
FunctionToInstrCount[F.getName().str()] = MF->getInstructionCount();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MachineOutliner::emitInstrCountChangedRemark(
|
|
|
|
const Module &M, const MachineModuleInfo &MMI,
|
|
|
|
const StringMap<unsigned> &FunctionToInstrCount) {
|
|
|
|
// Iterate over each function in the module and emit remarks.
|
|
|
|
// Note that we won't miss anything by doing this, because the outliner never
|
|
|
|
// deletes functions.
|
|
|
|
for (const Function &F : M) {
|
|
|
|
MachineFunction *MF = MMI.getMachineFunction(F);
|
|
|
|
|
|
|
|
// The outliner never deletes functions. If we don't have a MF here, then we
|
|
|
|
// didn't have one prior to outlining either.
|
|
|
|
if (!MF)
|
|
|
|
continue;
|
|
|
|
|
2020-01-29 03:23:46 +08:00
|
|
|
std::string Fname = std::string(F.getName());
|
2018-09-12 07:05:34 +08:00
|
|
|
unsigned FnCountAfter = MF->getInstructionCount();
|
|
|
|
unsigned FnCountBefore = 0;
|
|
|
|
|
|
|
|
// Check if the function was recorded before.
|
|
|
|
auto It = FunctionToInstrCount.find(Fname);
|
|
|
|
|
|
|
|
// Did we have a previously-recorded size? If yes, then set FnCountBefore
|
|
|
|
// to that.
|
|
|
|
if (It != FunctionToInstrCount.end())
|
|
|
|
FnCountBefore = It->second;
|
|
|
|
|
|
|
|
// Compute the delta and emit a remark if there was a change.
|
|
|
|
int64_t FnDelta = static_cast<int64_t>(FnCountAfter) -
|
|
|
|
static_cast<int64_t>(FnCountBefore);
|
|
|
|
if (FnDelta == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
MachineOptimizationRemarkEmitter MORE(*MF, nullptr);
|
|
|
|
MORE.emit([&]() {
|
|
|
|
MachineOptimizationRemarkAnalysis R("size-info", "FunctionMISizeChange",
|
2019-10-29 05:57:51 +08:00
|
|
|
DiagnosticLocation(), &MF->front());
|
2018-09-12 07:05:34 +08:00
|
|
|
R << DiagnosticInfoOptimizationBase::Argument("Pass", "Machine Outliner")
|
|
|
|
<< ": Function: "
|
|
|
|
<< DiagnosticInfoOptimizationBase::Argument("Function", F.getName())
|
|
|
|
<< ": MI instruction count changed from "
|
|
|
|
<< DiagnosticInfoOptimizationBase::Argument("MIInstrsBefore",
|
|
|
|
FnCountBefore)
|
|
|
|
<< " to "
|
|
|
|
<< DiagnosticInfoOptimizationBase::Argument("MIInstrsAfter",
|
|
|
|
FnCountAfter)
|
|
|
|
<< "; Delta: "
|
|
|
|
<< DiagnosticInfoOptimizationBase::Argument("Delta", FnDelta);
|
|
|
|
return R;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-29 15:33:47 +08:00
|
|
|
bool MachineOutliner::runOnModule(Module &M) {
|
2018-09-12 00:33:46 +08:00
|
|
|
// Check if there's anything in the module. If it's empty, then there's
|
|
|
|
// nothing to outline.
|
|
|
|
if (M.empty())
|
|
|
|
return false;
|
|
|
|
|
2019-10-29 03:10:21 +08:00
|
|
|
// Number to append to the current outlined function.
|
|
|
|
unsigned OutlinedFunctionNum = 0;
|
|
|
|
|
2020-04-29 15:33:47 +08:00
|
|
|
OutlineRepeatedNum = 0;
|
2019-10-29 03:10:21 +08:00
|
|
|
if (!doOutline(M, OutlinedFunctionNum))
|
|
|
|
return false;
|
2020-04-29 15:33:47 +08:00
|
|
|
|
|
|
|
for (unsigned I = 0; I < OutlinerReruns; ++I) {
|
|
|
|
OutlinedFunctionNum = 0;
|
|
|
|
OutlineRepeatedNum++;
|
|
|
|
if (!doOutline(M, OutlinedFunctionNum)) {
|
|
|
|
LLVM_DEBUG({
|
|
|
|
dbgs() << "Did not outline on iteration " << I + 2 << " out of "
|
|
|
|
<< OutlinerReruns + 1 << "\n";
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-29 03:10:21 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MachineOutliner::doOutline(Module &M, unsigned &OutlinedFunctionNum) {
|
2019-10-01 01:54:50 +08:00
|
|
|
MachineModuleInfo &MMI = getAnalysis<MachineModuleInfoWrapperPass>().getMMI();
|
2018-09-12 00:33:46 +08:00
|
|
|
|
|
|
|
// If the user passed -enable-machine-outliner=always or
|
|
|
|
// -enable-machine-outliner, the pass will run on all functions in the module.
|
|
|
|
// Otherwise, if the target supports default outlining, it will run on all
|
|
|
|
// functions deemed by the target to be worth outlining from by default. Tell
|
|
|
|
// the user how the outliner is running.
|
2019-10-29 05:57:51 +08:00
|
|
|
LLVM_DEBUG({
|
2018-09-12 00:33:46 +08:00
|
|
|
dbgs() << "Machine Outliner: Running on ";
|
|
|
|
if (RunOnAllFunctions)
|
|
|
|
dbgs() << "all functions";
|
|
|
|
else
|
|
|
|
dbgs() << "target-default functions";
|
2019-10-29 05:57:51 +08:00
|
|
|
dbgs() << "\n";
|
|
|
|
});
|
2018-09-12 00:33:46 +08:00
|
|
|
|
|
|
|
// If the user specifies that they want to outline from linkonceodrs, set
|
|
|
|
// it here.
|
|
|
|
OutlineFromLinkOnceODRs = EnableLinkOnceODROutlining;
|
|
|
|
InstructionMapper Mapper;
|
|
|
|
|
|
|
|
// Prepare instruction mappings for the suffix tree.
|
|
|
|
populateMapper(Mapper, M, MMI);
|
2017-03-07 05:31:18 +08:00
|
|
|
std::vector<OutlinedFunction> FunctionList;
|
|
|
|
|
2017-03-24 05:27:38 +08:00
|
|
|
// Find all of the outlining candidates.
|
2018-12-06 07:39:07 +08:00
|
|
|
findCandidates(Mapper, FunctionList);
|
2017-03-24 05:27:38 +08:00
|
|
|
|
2018-09-12 07:05:34 +08:00
|
|
|
// If we've requested size remarks, then collect the MI counts of every
|
|
|
|
// function before outlining, and the MI counts after outlining.
|
|
|
|
// FIXME: This shouldn't be in the outliner at all; it should ultimately be
|
|
|
|
// the pass manager's responsibility.
|
|
|
|
// This could pretty easily be placed in outline instead, but because we
|
|
|
|
// really ultimately *don't* want this here, it's done like this for now
|
|
|
|
// instead.
|
|
|
|
|
|
|
|
// Check if we want size remarks.
|
|
|
|
bool ShouldEmitSizeRemarks = M.shouldEmitInstrCountChangedRemark();
|
|
|
|
StringMap<unsigned> FunctionToInstrCount;
|
|
|
|
if (ShouldEmitSizeRemarks)
|
|
|
|
initSizeRemarkInfo(M, MMI, FunctionToInstrCount);
|
|
|
|
|
2017-03-24 05:27:38 +08:00
|
|
|
// Outline each of the candidates and return true if something was outlined.
|
2019-10-29 03:10:21 +08:00
|
|
|
bool OutlinedSomething =
|
|
|
|
outline(M, FunctionList, Mapper, OutlinedFunctionNum);
|
2018-01-18 08:00:58 +08:00
|
|
|
|
2018-09-12 07:05:34 +08:00
|
|
|
// If we outlined something, we definitely changed the MI count of the
|
|
|
|
// module. If we've asked for size remarks, then output them.
|
|
|
|
// FIXME: This should be in the pass manager.
|
|
|
|
if (ShouldEmitSizeRemarks && OutlinedSomething)
|
|
|
|
emitInstrCountChangedRemark(M, MMI, FunctionToInstrCount);
|
|
|
|
|
2020-04-29 15:33:47 +08:00
|
|
|
LLVM_DEBUG({
|
|
|
|
if (!OutlinedSomething)
|
|
|
|
dbgs() << "Stopped outlining at iteration " << OutlineRepeatedNum
|
|
|
|
<< " because no changes were found.\n";
|
|
|
|
});
|
2020-03-18 06:40:26 +08:00
|
|
|
|
2020-04-29 15:33:47 +08:00
|
|
|
return OutlinedSomething;
|
2020-03-18 06:40:26 +08:00
|
|
|
}
|