2008-02-27 04:26:43 +08:00
|
|
|
//===-LTOCodeGenerator.cpp - LLVM Link Time Optimizer ---------------------===//
|
|
|
|
//
|
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
|
2012-03-31 18:50:14 +08:00
|
|
|
//
|
2008-02-27 04:26:43 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2012-03-31 18:50:14 +08:00
|
|
|
// This file implements the Link Time Optimization library. This library is
|
2008-02-27 04:26:43 +08:00
|
|
|
// intended to be used by linker to optimize code at link time.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2016-07-15 05:21:16 +08:00
|
|
|
#include "llvm/LTO/legacy/LTOCodeGenerator.h"
|
2016-04-12 14:34:10 +08:00
|
|
|
|
2016-02-17 05:41:51 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
2012-12-04 18:44:52 +08:00
|
|
|
#include "llvm/ADT/StringExtras.h"
|
2009-06-17 14:52:10 +08:00
|
|
|
#include "llvm/Analysis/Passes.h"
|
2015-01-15 10:16:27 +08:00
|
|
|
#include "llvm/Analysis/TargetLibraryInfo.h"
|
2015-01-31 19:17:59 +08:00
|
|
|
#include "llvm/Analysis/TargetTransformInfo.h"
|
2016-11-11 13:34:58 +08:00
|
|
|
#include "llvm/Bitcode/BitcodeWriter.h"
|
2015-08-28 07:37:36 +08:00
|
|
|
#include "llvm/CodeGen/ParallelCG.h"
|
2017-11-17 09:07:10 +08:00
|
|
|
#include "llvm/CodeGen/TargetSubtargetInfo.h"
|
2012-03-31 19:15:43 +08:00
|
|
|
#include "llvm/Config/config.h"
|
2013-01-02 19:36:10 +08:00
|
|
|
#include "llvm/IR/Constants.h"
|
|
|
|
#include "llvm/IR/DataLayout.h"
|
2016-05-10 03:57:15 +08:00
|
|
|
#include "llvm/IR/DebugInfo.h"
|
2013-01-02 19:36:10 +08:00
|
|
|
#include "llvm/IR/DerivedTypes.h"
|
2014-01-16 06:04:35 +08:00
|
|
|
#include "llvm/IR/DiagnosticInfo.h"
|
|
|
|
#include "llvm/IR/DiagnosticPrinter.h"
|
2013-01-02 19:36:10 +08:00
|
|
|
#include "llvm/IR/LLVMContext.h"
|
2015-02-13 18:01:29 +08:00
|
|
|
#include "llvm/IR/LegacyPassManager.h"
|
2014-01-08 05:19:40 +08:00
|
|
|
#include "llvm/IR/Mangler.h"
|
2013-01-02 19:36:10 +08:00
|
|
|
#include "llvm/IR/Module.h"
|
2018-09-26 21:01:43 +08:00
|
|
|
#include "llvm/IR/PassTimingInfo.h"
|
2014-01-13 17:26:24 +08:00
|
|
|
#include "llvm/IR/Verifier.h"
|
2013-07-23 02:40:34 +08:00
|
|
|
#include "llvm/InitializePasses.h"
|
2017-02-11 07:49:38 +08:00
|
|
|
#include "llvm/LTO/LTO.h"
|
2016-07-15 05:21:16 +08:00
|
|
|
#include "llvm/LTO/legacy/LTOModule.h"
|
|
|
|
#include "llvm/LTO/legacy/UpdateCompilerUsed.h"
|
2014-03-06 11:42:23 +08:00
|
|
|
#include "llvm/Linker/Linker.h"
|
2010-03-13 02:44:54 +08:00
|
|
|
#include "llvm/MC/MCAsmInfo.h"
|
|
|
|
#include "llvm/MC/MCContext.h"
|
2011-06-29 09:14:12 +08:00
|
|
|
#include "llvm/MC/SubtargetFeature.h"
|
2010-03-13 02:44:54 +08:00
|
|
|
#include "llvm/Support/CommandLine.h"
|
2013-06-18 02:05:35 +08:00
|
|
|
#include "llvm/Support/FileSystem.h"
|
2010-11-30 02:47:54 +08:00
|
|
|
#include "llvm/Support/Host.h"
|
2012-12-04 18:44:52 +08:00
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
2010-11-30 02:47:54 +08:00
|
|
|
#include "llvm/Support/Signals.h"
|
2011-08-25 02:08:43 +08:00
|
|
|
#include "llvm/Support/TargetRegistry.h"
|
|
|
|
#include "llvm/Support/TargetSelect.h"
|
2012-12-04 18:44:52 +08:00
|
|
|
#include "llvm/Support/ToolOutputFile.h"
|
2016-11-19 02:06:28 +08:00
|
|
|
#include "llvm/Support/YAMLTraits.h"
|
2014-03-04 18:07:28 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2013-10-01 00:39:19 +08:00
|
|
|
#include "llvm/Target/TargetOptions.h"
|
2012-12-04 18:44:52 +08:00
|
|
|
#include "llvm/Transforms/IPO.h"
|
2016-04-13 14:32:46 +08:00
|
|
|
#include "llvm/Transforms/IPO/Internalize.h"
|
2012-12-04 18:44:52 +08:00
|
|
|
#include "llvm/Transforms/IPO/PassManagerBuilder.h"
|
2013-03-30 07:28:55 +08:00
|
|
|
#include "llvm/Transforms/ObjCARC.h"
|
2016-10-26 07:53:31 +08:00
|
|
|
#include "llvm/Transforms/Utils/ModuleUtils.h"
|
2014-06-13 01:38:55 +08:00
|
|
|
#include <system_error>
|
2008-02-27 04:26:43 +08:00
|
|
|
using namespace llvm;
|
|
|
|
|
2012-03-31 18:49:43 +08:00
|
|
|
const char* LTOCodeGenerator::getVersionString() {
|
2008-02-27 04:26:43 +08:00
|
|
|
#ifdef LLVM_VERSION_INFO
|
2012-03-31 18:49:43 +08:00
|
|
|
return PACKAGE_NAME " version " PACKAGE_VERSION ", " LLVM_VERSION_INFO;
|
2008-02-27 04:26:43 +08:00
|
|
|
#else
|
2012-03-31 18:49:43 +08:00
|
|
|
return PACKAGE_NAME " version " PACKAGE_VERSION;
|
2008-02-27 04:26:43 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
Add a flag to the LLVMContext to disable name for Value other than GlobalValue
Summary:
This is intended to be a performance flag, on the same level as clang
cc1 option "--disable-free". LLVM will never initialize it by default,
it will be up to the client creating the LLVMContext to request this
behavior. Clang will do it by default in Release build (just like
--disable-free).
"opt" and "llc" can opt-in using -disable-named-value command line
option.
When performing LTO on llvm-tblgen, the initial merging of IR peaks
at 92MB without this patch, and 86MB after this patch,setNameImpl()
drops from 6.5MB to 0.5MB.
The total link time goes from ~29.5s to ~27.8s.
Compared to a compile-time flag (like the IRBuilder one), it performs
very close. I profiled on SROA and obtain these results:
420ms with IRBuilder that preserve name
372ms with IRBuilder that strip name
375ms with IRBuilder that preserve name, and a runtime flag to strip
Reviewers: chandlerc, dexonsmith, bogner
Subscribers: joker.eph, llvm-commits
Differential Revision: http://reviews.llvm.org/D17946
From: Mehdi Amini <mehdi.amini@apple.com>
llvm-svn: 263086
2016-03-10 09:28:54 +08:00
|
|
|
namespace llvm {
|
|
|
|
cl::opt<bool> LTODiscardValueNames(
|
2016-03-11 01:06:52 +08:00
|
|
|
"lto-discard-value-names",
|
|
|
|
cl::desc("Strip names from Value during LTO (other than GlobalValue)."),
|
Add a flag to the LLVMContext to disable name for Value other than GlobalValue
Summary:
This is intended to be a performance flag, on the same level as clang
cc1 option "--disable-free". LLVM will never initialize it by default,
it will be up to the client creating the LLVMContext to request this
behavior. Clang will do it by default in Release build (just like
--disable-free).
"opt" and "llc" can opt-in using -disable-named-value command line
option.
When performing LTO on llvm-tblgen, the initial merging of IR peaks
at 92MB without this patch, and 86MB after this patch,setNameImpl()
drops from 6.5MB to 0.5MB.
The total link time goes from ~29.5s to ~27.8s.
Compared to a compile-time flag (like the IRBuilder one), it performs
very close. I profiled on SROA and obtain these results:
420ms with IRBuilder that preserve name
372ms with IRBuilder that strip name
375ms with IRBuilder that preserve name, and a runtime flag to strip
Reviewers: chandlerc, dexonsmith, bogner
Subscribers: joker.eph, llvm-commits
Differential Revision: http://reviews.llvm.org/D17946
From: Mehdi Amini <mehdi.amini@apple.com>
llvm-svn: 263086
2016-03-10 09:28:54 +08:00
|
|
|
#ifdef NDEBUG
|
|
|
|
cl::init(true),
|
|
|
|
#else
|
|
|
|
cl::init(false),
|
|
|
|
#endif
|
|
|
|
cl::Hidden);
|
2016-05-10 03:57:15 +08:00
|
|
|
|
2016-11-20 02:20:05 +08:00
|
|
|
cl::opt<std::string>
|
2016-11-22 15:35:14 +08:00
|
|
|
LTORemarksFilename("lto-pass-remarks-output",
|
2016-11-20 02:20:05 +08:00
|
|
|
cl::desc("Output filename for pass remarks"),
|
|
|
|
cl::value_desc("filename"));
|
2016-12-03 01:53:56 +08:00
|
|
|
|
|
|
|
cl::opt<bool> LTOPassRemarksWithHotness(
|
|
|
|
"lto-pass-remarks-with-hotness",
|
|
|
|
cl::desc("With PGO, include profile count in optimization remarks"),
|
|
|
|
cl::Hidden);
|
2016-11-20 02:20:05 +08:00
|
|
|
}
|
2016-11-19 02:06:28 +08:00
|
|
|
|
2015-12-04 10:42:28 +08:00
|
|
|
LTOCodeGenerator::LTOCodeGenerator(LLVMContext &Context)
|
2015-12-05 06:08:53 +08:00
|
|
|
: Context(Context), MergedModule(new Module("ld-temp.o", Context)),
|
2015-12-19 03:28:59 +08:00
|
|
|
TheLinker(new Linker(*MergedModule)) {
|
Add a flag to the LLVMContext to disable name for Value other than GlobalValue
Summary:
This is intended to be a performance flag, on the same level as clang
cc1 option "--disable-free". LLVM will never initialize it by default,
it will be up to the client creating the LLVMContext to request this
behavior. Clang will do it by default in Release build (just like
--disable-free).
"opt" and "llc" can opt-in using -disable-named-value command line
option.
When performing LTO on llvm-tblgen, the initial merging of IR peaks
at 92MB without this patch, and 86MB after this patch,setNameImpl()
drops from 6.5MB to 0.5MB.
The total link time goes from ~29.5s to ~27.8s.
Compared to a compile-time flag (like the IRBuilder one), it performs
very close. I profiled on SROA and obtain these results:
420ms with IRBuilder that preserve name
372ms with IRBuilder that strip name
375ms with IRBuilder that preserve name, and a runtime flag to strip
Reviewers: chandlerc, dexonsmith, bogner
Subscribers: joker.eph, llvm-commits
Differential Revision: http://reviews.llvm.org/D17946
From: Mehdi Amini <mehdi.amini@apple.com>
llvm-svn: 263086
2016-03-10 09:28:54 +08:00
|
|
|
Context.setDiscardValueNames(LTODiscardValueNames);
|
2016-04-19 12:55:25 +08:00
|
|
|
Context.enableDebugTypeODRUniquing();
|
2015-04-28 07:19:26 +08:00
|
|
|
initializeLTOPasses();
|
2014-11-12 07:03:29 +08:00
|
|
|
}
|
|
|
|
|
2015-08-25 06:22:53 +08:00
|
|
|
LTOCodeGenerator::~LTOCodeGenerator() {}
|
2008-02-27 04:26:43 +08:00
|
|
|
|
2015-09-01 18:13:49 +08:00
|
|
|
// Initialize LTO passes. Please keep this function in sync with
|
2013-07-23 14:44:34 +08:00
|
|
|
// PassManagerBuilder::populateLTOPassManager(), and make sure all LTO
|
2014-01-11 04:24:35 +08:00
|
|
|
// passes are initialized.
|
2013-07-23 02:40:34 +08:00
|
|
|
void LTOCodeGenerator::initializeLTOPasses() {
|
|
|
|
PassRegistry &R = *PassRegistry::getPassRegistry();
|
|
|
|
|
2016-04-27 04:15:52 +08:00
|
|
|
initializeInternalizeLegacyPassPass(R);
|
2016-05-06 05:05:36 +08:00
|
|
|
initializeIPSCCPLegacyPassPass(R);
|
2016-04-26 08:28:01 +08:00
|
|
|
initializeGlobalOptLegacyPassPass(R);
|
2016-05-05 08:51:09 +08:00
|
|
|
initializeConstantMergeLegacyPassPass(R);
|
2013-07-23 02:40:34 +08:00
|
|
|
initializeDAHPass(R);
|
2015-01-21 06:44:35 +08:00
|
|
|
initializeInstructionCombiningPassPass(R);
|
2013-07-23 02:40:34 +08:00
|
|
|
initializeSimpleInlinerPass(R);
|
|
|
|
initializePruneEHPass(R);
|
2016-05-04 03:39:15 +08:00
|
|
|
initializeGlobalDCELegacyPassPass(R);
|
2013-07-23 02:40:34 +08:00
|
|
|
initializeArgPromotionPass(R);
|
|
|
|
initializeJumpThreadingPass(R);
|
2015-09-12 17:09:14 +08:00
|
|
|
initializeSROALegacyPassPass(R);
|
2016-02-18 19:03:11 +08:00
|
|
|
initializePostOrderFunctionAttrsLegacyPassPass(R);
|
[PM] Port ReversePostOrderFunctionAttrs to the new PM
Below are my super rough notes when porting. They can probably serve as
a basic guide for porting other passes to the new PM. As I port more
passes I'll expand and generalize this and make a proper
docs/HowToPortToNewPassManager.rst document. There is also missing
documentation for general concepts and API's in the new PM which will
require some documentation.
Once there is proper documentation in place we can put up a list of
passes that have to be ported and game-ify/crowdsource the rest of the
porting (at least of the middle end; the backend is still unclear).
I will however be taking personal responsibility for ensuring that the
LLD/ELF LTO pipeline is ported in a timely fashion. The remaining passes
to be ported are (do something like
`git grep "<the string in the bullet point below>"` to find the pass):
General Scalar:
[ ] Simplify the CFG
[ ] Jump Threading
[ ] MemCpy Optimization
[ ] Promote Memory to Register
[ ] MergedLoadStoreMotion
[ ] Lazy Value Information Analysis
General IPO:
[ ] Dead Argument Elimination
[ ] Deduce function attributes in RPO
Loop stuff / vectorization stuff:
[ ] Alignment from assumptions
[ ] Canonicalize natural loops
[ ] Delete dead loops
[ ] Loop Access Analysis
[ ] Loop Invariant Code Motion
[ ] Loop Vectorization
[ ] SLP Vectorizer
[ ] Unroll loops
Devirtualization / CFI:
[ ] Cross-DSO CFI
[ ] Whole program devirtualization
[ ] Lower bitset metadata
CGSCC passes:
[ ] Function Integration/Inlining
[ ] Remove unused exception handling info
[ ] Promote 'by reference' arguments to scalars
Please let me know if you are interested in working on any of the passes
in the above list (e.g. reply to the post-commit thread for this patch).
I'll probably be tackling "General Scalar" and "General IPO" first FWIW.
Steps as I port "Deduce function attributes in RPO"
---------------------------------------------------
(note: if you are doing any work based on these notes, please leave a
note in the post-commit review thread for this commit with any
improvements / suggestions / incompleteness you ran into!)
Note: "Deduce function attributes in RPO" is a module pass.
1. Do preparatory refactoring.
Do preparatory factoring. In this case all I had to do was to pull out a static helper (r272503).
(TODO: give more advice here e.g. if pass holds state or something)
2. Rename the old pass class.
llvm/lib/Transforms/IPO/FunctionAttrs.cpp
Rename class ReversePostOrderFunctionAttrs -> ReversePostOrderFunctionAttrsLegacyPass
in preparation for adding a class ReversePostOrderFunctionAttrs as the pass in the new PM.
(edit: actually wait what? The new class name will be
ReversePostOrderFunctionAttrsPass, so it doesn't conflict. So this step is
sort of useless churn).
llvm/include/llvm/InitializePasses.h
llvm/lib/LTO/LTOCodeGenerator.cpp
llvm/lib/Transforms/IPO/IPO.cpp
llvm/lib/Transforms/IPO/FunctionAttrs.cpp
Rename initializeReversePostOrderFunctionAttrsPass -> initializeReversePostOrderFunctionAttrsLegacyPassPass
(note that the "PassPass" thing falls out of `s/ReversePostOrderFunctionAttrs/ReversePostOrderFunctionAttrsLegacyPass/`)
Note that the INITIALIZE_PASS macro is what creates this identifier name, so renaming the class requires this renaming too.
Note that createReversePostOrderFunctionAttrsPass does not need to be
renamed since its name is not generated from the class name.
3. Add the new PM pass class.
In the new PM all passes need to have their
declaration in a header somewhere, so you will often need to add a header.
In this case
llvm/include/llvm/Transforms/IPO/FunctionAttrs.h is already there because
PostOrderFunctionAttrsPass was already ported.
The file-level comment from the .cpp file can be used as the file-level
comment for the new header. You may want to tweak the wording slightly
from "this file implements" to "this file provides" or similar.
Add declaration for the new PM pass in this header:
class ReversePostOrderFunctionAttrsPass
: public PassInfoMixin<ReversePostOrderFunctionAttrsPass> {
public:
PreservedAnalyses run(Module &M, AnalysisManager<Module> &AM);
};
Its name should end with `Pass` for consistency (note that this doesn't
collide with the names of most old PM passes). E.g. call it
`<name of the old PM pass>Pass`.
Also, move the doxygen comment from the old PM pass to the declaration of
this class in the header.
Also, include the declaration for the new PM class
`llvm/Transforms/IPO/FunctionAttrs.h` at the top of the file (in this case,
it was already done when the other pass in this file was ported).
Now define the `run` method for the new class.
The main things here are:
a) Use AM.getResult<...>(M) to get results instead of `getAnalysis<...>()`
b) If the old PM pass would have returned "false" (i.e. `Changed ==
false`), then you should return PreservedAnalyses::all();
c) In the old PM getAnalysisUsage method, observe the calls
`AU.addPreserved<...>();`.
In the case `Changed == true`, for each preserved analysis you should do
call `PA.preserve<...>()` on a PreservedAnalyses object and return it.
E.g.:
PreservedAnalyses PA;
PA.preserve<CallGraphAnalysis>();
return PA;
Note that calls to skipModule/skipFunction are not supported in the new PM
currently, so optnone and optimization bisect support do not work. You can
just drop those calls for now.
4. Add the pass to the new PM pass registry to make it available in opt.
In llvm/lib/Passes/PassBuilder.cpp add a #include for your header.
`#include "llvm/Transforms/IPO/FunctionAttrs.h"`
In this case there is already an include (from when
PostOrderFunctionAttrsPass was ported).
Add your pass to llvm/lib/Passes/PassRegistry.def
In this case, I added
`MODULE_PASS("rpo-functionattrs", ReversePostOrderFunctionAttrsPass())`
The string is from the `INITIALIZE_PASS*` macros used in the old pass
manager.
Then choose a test that uses the pass and use the new PM `-passes=...` to
run it.
E.g. in this case there is a test that does:
; RUN: opt < %s -basicaa -functionattrs -rpo-functionattrs -S | FileCheck %s
I have added the line:
; RUN: opt < %s -aa-pipeline=basic-aa -passes='require<targetlibinfo>,cgscc(function-attrs),rpo-functionattrs' -S | FileCheck %s
The `-aa-pipeline=basic-aa` and
`require<targetlibinfo>,cgscc(function-attrs)` are what is needed to run
functionattrs in the new PM (note that in the new PM "functionattrs"
becomes "function-attrs" for some reason). This is just pulled from
`readattrs.ll` which contains the change from when functionattrs was ported
to the new PM.
Adding rpo-functionattrs causes the pass that was just ported to run.
llvm-svn: 272505
2016-06-12 15:48:51 +08:00
|
|
|
initializeReversePostOrderFunctionAttrsLegacyPassPass(R);
|
[PM/AA] Rebuild LLVM's alias analysis infrastructure in a way compatible
with the new pass manager, and no longer relying on analysis groups.
This builds essentially a ground-up new AA infrastructure stack for
LLVM. The core ideas are the same that are used throughout the new pass
manager: type erased polymorphism and direct composition. The design is
as follows:
- FunctionAAResults is a type-erasing alias analysis results aggregation
interface to walk a single query across a range of results from
different alias analyses. Currently this is function-specific as we
always assume that aliasing queries are *within* a function.
- AAResultBase is a CRTP utility providing stub implementations of
various parts of the alias analysis result concept, notably in several
cases in terms of other more general parts of the interface. This can
be used to implement only a narrow part of the interface rather than
the entire interface. This isn't really ideal, this logic should be
hoisted into FunctionAAResults as currently it will cause
a significant amount of redundant work, but it faithfully models the
behavior of the prior infrastructure.
- All the alias analysis passes are ported to be wrapper passes for the
legacy PM and new-style analysis passes for the new PM with a shared
result object. In some cases (most notably CFL), this is an extremely
naive approach that we should revisit when we can specialize for the
new pass manager.
- BasicAA has been restructured to reflect that it is much more
fundamentally a function analysis because it uses dominator trees and
loop info that need to be constructed for each function.
All of the references to getting alias analysis results have been
updated to use the new aggregation interface. All the preservation and
other pass management code has been updated accordingly.
The way the FunctionAAResultsWrapperPass works is to detect the
available alias analyses when run, and add them to the results object.
This means that we should be able to continue to respect when various
passes are added to the pipeline, for example adding CFL or adding TBAA
passes should just cause their results to be available and to get folded
into this. The exception to this rule is BasicAA which really needs to
be a function pass due to using dominator trees and loop info. As
a consequence, the FunctionAAResultsWrapperPass directly depends on
BasicAA and always includes it in the aggregation.
This has significant implications for preserving analyses. Generally,
most passes shouldn't bother preserving FunctionAAResultsWrapperPass
because rebuilding the results just updates the set of known AA passes.
The exception to this rule are LoopPass instances which need to preserve
all the function analyses that the loop pass manager will end up
needing. This means preserving both BasicAAWrapperPass and the
aggregating FunctionAAResultsWrapperPass.
Now, when preserving an alias analysis, you do so by directly preserving
that analysis. This is only necessary for non-immutable-pass-provided
alias analyses though, and there are only three of interest: BasicAA,
GlobalsAA (formerly GlobalsModRef), and SCEVAA. Usually BasicAA is
preserved when needed because it (like DominatorTree and LoopInfo) is
marked as a CFG-only pass. I've expanded GlobalsAA into the preserved
set everywhere we previously were preserving all of AliasAnalysis, and
I've added SCEVAA in the intersection of that with where we preserve
SCEV itself.
One significant challenge to all of this is that the CGSCC passes were
actually using the alias analysis implementations by taking advantage of
a pretty amazing set of loop holes in the old pass manager's analysis
management code which allowed analysis groups to slide through in many
cases. Moving away from analysis groups makes this problem much more
obvious. To fix it, I've leveraged the flexibility the design of the new
PM components provides to just directly construct the relevant alias
analyses for the relevant functions in the IPO passes that need them.
This is a bit hacky, but should go away with the new pass manager, and
is already in many ways cleaner than the prior state.
Another significant challenge is that various facilities of the old
alias analysis infrastructure just don't fit any more. The most
significant of these is the alias analysis 'counter' pass. That pass
relied on the ability to snoop on AA queries at different points in the
analysis group chain. Instead, I'm planning to build printing
functionality directly into the aggregation layer. I've not included
that in this patch merely to keep it smaller.
Note that all of this needs a nearly complete rewrite of the AA
documentation. I'm planning to do that, but I'd like to make sure the
new design settles, and to flesh out a bit more of what it looks like in
the new pass manager first.
Differential Revision: http://reviews.llvm.org/D12080
llvm-svn: 247167
2015-09-10 01:55:00 +08:00
|
|
|
initializeGlobalsAAWrapperPassPass(R);
|
2016-07-13 06:37:48 +08:00
|
|
|
initializeLegacyLICMPassPass(R);
|
2016-06-18 03:10:09 +08:00
|
|
|
initializeMergedLoadStoreMotionLegacyPassPass(R);
|
2016-03-11 16:50:55 +08:00
|
|
|
initializeGVNLegacyPassPass(R);
|
2016-06-14 10:44:55 +08:00
|
|
|
initializeMemCpyOptLegacyPassPass(R);
|
2016-04-23 03:40:41 +08:00
|
|
|
initializeDCELegacyPassPass(R);
|
2013-08-06 10:43:45 +08:00
|
|
|
initializeCFGSimplifyPassPass(R);
|
2013-07-23 02:40:34 +08:00
|
|
|
}
|
|
|
|
|
2016-08-23 20:32:57 +08:00
|
|
|
void LTOCodeGenerator::setAsmUndefinedRefs(LTOModule *Mod) {
|
2016-10-08 03:05:14 +08:00
|
|
|
const std::vector<StringRef> &undefs = Mod->getAsmUndefinedRefs();
|
2016-08-23 20:32:57 +08:00
|
|
|
for (int i = 0, e = undefs.size(); i != e; ++i)
|
|
|
|
AsmUndefinedRefs[undefs[i]] = 1;
|
|
|
|
}
|
|
|
|
|
2015-09-01 07:44:06 +08:00
|
|
|
bool LTOCodeGenerator::addModule(LTOModule *Mod) {
|
|
|
|
assert(&Mod->getModule().getContext() == &Context &&
|
2014-11-12 07:13:10 +08:00
|
|
|
"Expected module in same context");
|
|
|
|
|
2015-12-19 03:28:59 +08:00
|
|
|
bool ret = TheLinker->linkInModule(Mod->takeModule());
|
2016-08-23 20:32:57 +08:00
|
|
|
setAsmUndefinedRefs(Mod);
|
2011-03-02 12:14:42 +08:00
|
|
|
|
2016-04-21 01:48:22 +08:00
|
|
|
// We've just changed the input, so let's make sure we verify it.
|
|
|
|
HasVerifiedInput = false;
|
|
|
|
|
2013-08-07 13:19:23 +08:00
|
|
|
return !ret;
|
2008-02-27 04:26:43 +08:00
|
|
|
}
|
2012-03-31 18:50:14 +08:00
|
|
|
|
2015-08-25 06:22:53 +08:00
|
|
|
void LTOCodeGenerator::setModule(std::unique_ptr<LTOModule> Mod) {
|
2015-02-24 08:45:56 +08:00
|
|
|
assert(&Mod->getModule().getContext() == &Context &&
|
|
|
|
"Expected module in same context");
|
|
|
|
|
|
|
|
AsmUndefinedRefs.clear();
|
|
|
|
|
2015-08-25 06:22:53 +08:00
|
|
|
MergedModule = Mod->takeModule();
|
2015-12-19 03:28:59 +08:00
|
|
|
TheLinker = make_unique<Linker>(*MergedModule);
|
2016-08-23 20:32:57 +08:00
|
|
|
setAsmUndefinedRefs(&*Mod);
|
2016-04-21 01:48:22 +08:00
|
|
|
|
|
|
|
// We've just changed the input, so let's make sure we verify it.
|
|
|
|
HasVerifiedInput = false;
|
2015-02-24 08:45:56 +08:00
|
|
|
}
|
|
|
|
|
2016-06-09 03:09:22 +08:00
|
|
|
void LTOCodeGenerator::setTargetOptions(const TargetOptions &Options) {
|
2015-09-01 07:44:06 +08:00
|
|
|
this->Options = Options;
|
2013-10-01 00:39:19 +08:00
|
|
|
}
|
|
|
|
|
2015-09-01 07:44:06 +08:00
|
|
|
void LTOCodeGenerator::setDebugInfo(lto_debug_model Debug) {
|
|
|
|
switch (Debug) {
|
2012-03-31 19:15:43 +08:00
|
|
|
case LTO_DEBUG_MODEL_NONE:
|
2013-09-05 01:44:24 +08:00
|
|
|
EmitDwarfDebugInfo = false;
|
2013-08-07 13:19:23 +08:00
|
|
|
return;
|
2008-02-27 04:26:43 +08:00
|
|
|
|
2012-03-31 19:15:43 +08:00
|
|
|
case LTO_DEBUG_MODEL_DWARF:
|
2013-09-05 01:44:24 +08:00
|
|
|
EmitDwarfDebugInfo = true;
|
2013-08-07 13:19:23 +08:00
|
|
|
return;
|
2012-03-31 19:15:43 +08:00
|
|
|
}
|
|
|
|
llvm_unreachable("Unknown debug format!");
|
2008-02-27 04:26:43 +08:00
|
|
|
}
|
|
|
|
|
2015-09-01 07:44:06 +08:00
|
|
|
void LTOCodeGenerator::setOptLevel(unsigned Level) {
|
|
|
|
OptLevel = Level;
|
2015-08-22 10:25:53 +08:00
|
|
|
switch (OptLevel) {
|
|
|
|
case 0:
|
|
|
|
CGOptLevel = CodeGenOpt::None;
|
2016-08-26 23:22:59 +08:00
|
|
|
return;
|
2015-08-22 10:25:53 +08:00
|
|
|
case 1:
|
|
|
|
CGOptLevel = CodeGenOpt::Less;
|
2016-08-26 23:22:59 +08:00
|
|
|
return;
|
2015-08-22 10:25:53 +08:00
|
|
|
case 2:
|
|
|
|
CGOptLevel = CodeGenOpt::Default;
|
2016-08-26 23:22:59 +08:00
|
|
|
return;
|
2015-08-22 10:25:53 +08:00
|
|
|
case 3:
|
|
|
|
CGOptLevel = CodeGenOpt::Aggressive;
|
2016-08-26 23:22:59 +08:00
|
|
|
return;
|
2015-08-22 10:25:53 +08:00
|
|
|
}
|
2016-08-26 23:22:59 +08:00
|
|
|
llvm_unreachable("Unknown optimization level!");
|
2015-08-22 10:25:53 +08:00
|
|
|
}
|
|
|
|
|
2016-10-01 09:18:23 +08:00
|
|
|
bool LTOCodeGenerator::writeMergedModules(StringRef Path) {
|
2015-11-18 03:48:12 +08:00
|
|
|
if (!determineTarget())
|
2013-08-07 13:19:23 +08:00
|
|
|
return false;
|
2009-08-23 15:49:08 +08:00
|
|
|
|
2016-04-21 01:48:22 +08:00
|
|
|
// We always run the verifier once on the merged module.
|
|
|
|
verifyMergedModuleOnce();
|
|
|
|
|
2013-08-09 07:51:04 +08:00
|
|
|
// mark which symbols can not be internalized
|
|
|
|
applyScopeRestrictions();
|
2009-08-23 15:49:08 +08:00
|
|
|
|
|
|
|
// create output file
|
2014-08-26 02:16:47 +08:00
|
|
|
std::error_code EC;
|
2017-09-23 09:03:17 +08:00
|
|
|
ToolOutputFile Out(Path, EC, sys::fs::F_None);
|
2014-08-26 02:16:47 +08:00
|
|
|
if (EC) {
|
2015-11-18 03:48:12 +08:00
|
|
|
std::string ErrMsg = "could not open bitcode file for writing: ";
|
2017-10-24 09:26:22 +08:00
|
|
|
ErrMsg += Path.str() + ": " + EC.message();
|
2015-11-18 03:48:12 +08:00
|
|
|
emitError(ErrMsg);
|
2013-08-07 13:19:23 +08:00
|
|
|
return false;
|
2009-08-23 15:49:08 +08:00
|
|
|
}
|
2012-03-31 18:50:14 +08:00
|
|
|
|
2009-08-23 15:49:08 +08:00
|
|
|
// write bitcode to it
|
2018-02-15 03:11:32 +08:00
|
|
|
WriteBitcodeToFile(*MergedModule, Out.os(), ShouldEmbedUselists);
|
2010-09-01 22:20:41 +08:00
|
|
|
Out.os().close();
|
2010-05-28 04:19:47 +08:00
|
|
|
|
2010-09-01 22:20:41 +08:00
|
|
|
if (Out.os().has_error()) {
|
2015-11-18 03:48:12 +08:00
|
|
|
std::string ErrMsg = "could not write bitcode file: ";
|
2017-10-24 09:26:22 +08:00
|
|
|
ErrMsg += Path.str() + ": " + Out.os().error().message();
|
2015-11-18 03:48:12 +08:00
|
|
|
emitError(ErrMsg);
|
2010-09-01 22:20:41 +08:00
|
|
|
Out.os().clear_error();
|
2013-08-07 13:19:23 +08:00
|
|
|
return false;
|
2009-08-23 15:49:08 +08:00
|
|
|
}
|
2012-03-31 18:50:14 +08:00
|
|
|
|
2010-08-21 00:59:15 +08:00
|
|
|
Out.keep();
|
2013-08-07 13:19:23 +08:00
|
|
|
return true;
|
2008-02-27 04:26:43 +08:00
|
|
|
}
|
|
|
|
|
2015-11-18 03:48:12 +08:00
|
|
|
bool LTOCodeGenerator::compileOptimizedToFile(const char **Name) {
|
2015-11-20 07:59:24 +08:00
|
|
|
// make unique temp output file to put generated code
|
2013-08-13 05:07:31 +08:00
|
|
|
SmallString<128> Filename;
|
|
|
|
int FD;
|
2015-11-20 07:59:24 +08:00
|
|
|
|
2016-10-01 09:18:23 +08:00
|
|
|
StringRef Extension
|
2015-11-20 07:59:24 +08:00
|
|
|
(FileType == TargetMachine::CGFT_AssemblyFile ? "s" : "o");
|
|
|
|
|
2014-06-13 10:24:39 +08:00
|
|
|
std::error_code EC =
|
2015-11-20 07:59:24 +08:00
|
|
|
sys::fs::createTemporaryFile("lto-llvm", Extension, FD, Filename);
|
2013-08-13 05:07:31 +08:00
|
|
|
if (EC) {
|
2015-11-18 03:48:12 +08:00
|
|
|
emitError(EC.message());
|
2013-08-07 13:19:23 +08:00
|
|
|
return false;
|
2013-08-13 05:07:31 +08:00
|
|
|
}
|
2011-03-23 04:57:13 +08:00
|
|
|
|
2013-08-13 05:07:31 +08:00
|
|
|
// generate object file
|
2017-09-23 09:03:17 +08:00
|
|
|
ToolOutputFile objFile(Filename, FD);
|
2012-03-31 19:15:43 +08:00
|
|
|
|
2015-11-18 03:48:12 +08:00
|
|
|
bool genResult = compileOptimized(&objFile.os());
|
2013-08-13 05:07:31 +08:00
|
|
|
objFile.os().close();
|
|
|
|
if (objFile.os().has_error()) {
|
2017-10-24 09:26:22 +08:00
|
|
|
emitError((Twine("could not write object file: ") + Filename + ": " +
|
|
|
|
objFile.os().error().message())
|
|
|
|
.str());
|
2013-08-13 05:07:31 +08:00
|
|
|
objFile.os().clear_error();
|
|
|
|
sys::fs::remove(Twine(Filename));
|
2013-08-07 13:19:23 +08:00
|
|
|
return false;
|
2013-08-13 05:07:31 +08:00
|
|
|
}
|
2012-03-31 19:15:43 +08:00
|
|
|
|
2013-08-13 05:07:31 +08:00
|
|
|
objFile.keep();
|
|
|
|
if (!genResult) {
|
|
|
|
sys::fs::remove(Twine(Filename));
|
2013-08-07 13:19:23 +08:00
|
|
|
return false;
|
2013-08-13 05:07:31 +08:00
|
|
|
}
|
2011-02-25 05:04:06 +08:00
|
|
|
|
2013-09-05 01:44:24 +08:00
|
|
|
NativeObjectPath = Filename.c_str();
|
2015-09-01 07:44:06 +08:00
|
|
|
*Name = NativeObjectPath.c_str();
|
2013-08-07 13:19:23 +08:00
|
|
|
return true;
|
2011-03-23 04:57:13 +08:00
|
|
|
}
|
2011-02-25 05:04:06 +08:00
|
|
|
|
2015-06-02 04:08:30 +08:00
|
|
|
std::unique_ptr<MemoryBuffer>
|
2015-11-18 03:48:12 +08:00
|
|
|
LTOCodeGenerator::compileOptimized() {
|
2011-03-23 04:57:13 +08:00
|
|
|
const char *name;
|
2015-11-18 03:48:12 +08:00
|
|
|
if (!compileOptimizedToFile(&name))
|
2014-04-15 14:32:26 +08:00
|
|
|
return nullptr;
|
2011-03-23 04:57:13 +08:00
|
|
|
|
|
|
|
// read .o file into memory buffer
|
2014-07-07 01:43:13 +08:00
|
|
|
ErrorOr<std::unique_ptr<MemoryBuffer>> BufferOrErr =
|
|
|
|
MemoryBuffer::getFile(name, -1, false);
|
|
|
|
if (std::error_code EC = BufferOrErr.getError()) {
|
2015-11-18 03:48:12 +08:00
|
|
|
emitError(EC.message());
|
2013-09-05 01:44:24 +08:00
|
|
|
sys::fs::remove(NativeObjectPath);
|
2014-04-15 14:32:26 +08:00
|
|
|
return nullptr;
|
2011-03-23 04:57:13 +08:00
|
|
|
}
|
2008-02-28 06:25:36 +08:00
|
|
|
|
2013-08-13 05:07:31 +08:00
|
|
|
// remove temp files
|
2013-09-05 01:44:24 +08:00
|
|
|
sys::fs::remove(NativeObjectPath);
|
2013-08-13 02:29:43 +08:00
|
|
|
|
2015-06-02 04:08:30 +08:00
|
|
|
return std::move(*BufferOrErr);
|
2008-02-27 04:26:43 +08:00
|
|
|
}
|
|
|
|
|
2015-09-16 07:05:59 +08:00
|
|
|
bool LTOCodeGenerator::compile_to_file(const char **Name, bool DisableVerify,
|
|
|
|
bool DisableInline,
|
2015-09-01 07:44:06 +08:00
|
|
|
bool DisableGVNLoadPRE,
|
2015-11-18 03:48:12 +08:00
|
|
|
bool DisableVectorization) {
|
2015-09-16 07:05:59 +08:00
|
|
|
if (!optimize(DisableVerify, DisableInline, DisableGVNLoadPRE,
|
2015-11-18 03:48:12 +08:00
|
|
|
DisableVectorization))
|
2015-02-04 02:39:15 +08:00
|
|
|
return false;
|
|
|
|
|
2015-11-18 03:48:12 +08:00
|
|
|
return compileOptimizedToFile(Name);
|
2015-02-04 02:39:15 +08:00
|
|
|
}
|
|
|
|
|
2015-06-02 04:08:30 +08:00
|
|
|
std::unique_ptr<MemoryBuffer>
|
2015-09-16 07:05:59 +08:00
|
|
|
LTOCodeGenerator::compile(bool DisableVerify, bool DisableInline,
|
2015-11-18 03:48:12 +08:00
|
|
|
bool DisableGVNLoadPRE, bool DisableVectorization) {
|
2015-09-16 07:05:59 +08:00
|
|
|
if (!optimize(DisableVerify, DisableInline, DisableGVNLoadPRE,
|
2015-11-18 03:48:12 +08:00
|
|
|
DisableVectorization))
|
2015-02-04 02:39:15 +08:00
|
|
|
return nullptr;
|
|
|
|
|
2015-11-18 03:48:12 +08:00
|
|
|
return compileOptimized();
|
2015-02-04 02:39:15 +08:00
|
|
|
}
|
|
|
|
|
2015-11-18 03:48:12 +08:00
|
|
|
bool LTOCodeGenerator::determineTarget() {
|
2014-04-15 14:32:26 +08:00
|
|
|
if (TargetMach)
|
2013-08-07 05:51:21 +08:00
|
|
|
return true;
|
2012-08-07 06:52:45 +08:00
|
|
|
|
2016-04-18 02:42:27 +08:00
|
|
|
TripleStr = MergedModule->getTargetTriple();
|
2015-08-22 10:25:53 +08:00
|
|
|
if (TripleStr.empty()) {
|
2012-10-13 01:39:25 +08:00
|
|
|
TripleStr = sys::getDefaultTargetTriple();
|
2015-08-25 06:22:53 +08:00
|
|
|
MergedModule->setTargetTriple(TripleStr);
|
2015-08-22 10:25:53 +08:00
|
|
|
}
|
2012-10-13 01:39:25 +08:00
|
|
|
llvm::Triple Triple(TripleStr);
|
2012-08-07 06:52:45 +08:00
|
|
|
|
|
|
|
// create target machine from info for merged modules
|
2015-11-18 03:48:12 +08:00
|
|
|
std::string ErrMsg;
|
2016-04-18 02:42:27 +08:00
|
|
|
MArch = TargetRegistry::lookupTarget(TripleStr, ErrMsg);
|
|
|
|
if (!MArch) {
|
2015-11-18 03:48:12 +08:00
|
|
|
emitError(ErrMsg);
|
2013-08-07 05:51:21 +08:00
|
|
|
return false;
|
2015-11-18 03:48:12 +08:00
|
|
|
}
|
2012-08-07 06:52:45 +08:00
|
|
|
|
2014-04-26 05:46:51 +08:00
|
|
|
// Construct LTOModule, hand over ownership of module and target. Use MAttr as
|
|
|
|
// the default set of features.
|
|
|
|
SubtargetFeatures Features(MAttr);
|
2012-10-13 01:39:25 +08:00
|
|
|
Features.getDefaultSubtargetFeatures(Triple);
|
2015-08-22 10:25:53 +08:00
|
|
|
FeatureStr = Features.getString();
|
2012-10-13 01:39:25 +08:00
|
|
|
// Set a default CPU for Darwin triples.
|
2013-09-05 01:44:24 +08:00
|
|
|
if (MCpu.empty() && Triple.isOSDarwin()) {
|
2012-10-13 01:39:25 +08:00
|
|
|
if (Triple.getArch() == llvm::Triple::x86_64)
|
2013-09-05 01:44:24 +08:00
|
|
|
MCpu = "core2";
|
2012-10-13 01:39:25 +08:00
|
|
|
else if (Triple.getArch() == llvm::Triple::x86)
|
2013-09-05 01:44:24 +08:00
|
|
|
MCpu = "yonah";
|
2014-07-23 20:32:47 +08:00
|
|
|
else if (Triple.getArch() == llvm::Triple::aarch64)
|
2014-03-29 18:18:08 +08:00
|
|
|
MCpu = "cyclone";
|
2012-10-13 01:39:25 +08:00
|
|
|
}
|
2013-10-01 00:39:19 +08:00
|
|
|
|
2016-04-18 02:42:27 +08:00
|
|
|
TargetMach = createTargetMachine();
|
2013-08-07 05:51:21 +08:00
|
|
|
return true;
|
2008-02-27 04:26:43 +08:00
|
|
|
}
|
|
|
|
|
2016-04-18 02:42:27 +08:00
|
|
|
std::unique_ptr<TargetMachine> LTOCodeGenerator::createTargetMachine() {
|
2017-08-03 10:16:21 +08:00
|
|
|
return std::unique_ptr<TargetMachine>(MArch->createTargetMachine(
|
|
|
|
TripleStr, MCpu, FeatureStr, Options, RelocModel, None, CGOptLevel));
|
2016-04-18 02:42:27 +08:00
|
|
|
}
|
|
|
|
|
2016-05-05 13:14:24 +08:00
|
|
|
// If a linkonce global is present in the MustPreserveSymbols, we need to make
|
|
|
|
// sure we honor this. To force the compiler to not drop it, we add it to the
|
|
|
|
// "llvm.compiler.used" global.
|
2016-05-06 04:05:33 +08:00
|
|
|
void LTOCodeGenerator::preserveDiscardableGVs(
|
2016-05-05 13:14:24 +08:00
|
|
|
Module &TheModule,
|
|
|
|
llvm::function_ref<bool(const GlobalValue &)> mustPreserveGV) {
|
2016-10-26 07:53:31 +08:00
|
|
|
std::vector<GlobalValue *> Used;
|
2016-05-05 13:14:24 +08:00
|
|
|
auto mayPreserveGlobal = [&](GlobalValue &GV) {
|
2016-09-03 00:37:31 +08:00
|
|
|
if (!GV.isDiscardableIfUnused() || GV.isDeclaration() ||
|
2016-10-26 07:53:31 +08:00
|
|
|
!mustPreserveGV(GV))
|
2016-05-05 13:14:24 +08:00
|
|
|
return;
|
2016-09-03 00:37:31 +08:00
|
|
|
if (GV.hasAvailableExternallyLinkage())
|
|
|
|
return emitWarning(
|
2016-05-06 04:05:33 +08:00
|
|
|
(Twine("Linker asked to preserve available_externally global: '") +
|
|
|
|
GV.getName() + "'").str());
|
2016-09-03 00:37:31 +08:00
|
|
|
if (GV.hasInternalLinkage())
|
|
|
|
return emitWarning((Twine("Linker asked to preserve internal global: '") +
|
2016-05-06 04:05:33 +08:00
|
|
|
GV.getName() + "'").str());
|
2016-10-26 07:53:31 +08:00
|
|
|
Used.push_back(&GV);
|
2016-05-05 13:14:24 +08:00
|
|
|
};
|
|
|
|
for (auto &GV : TheModule)
|
|
|
|
mayPreserveGlobal(GV);
|
|
|
|
for (auto &GV : TheModule.globals())
|
|
|
|
mayPreserveGlobal(GV);
|
|
|
|
for (auto &GV : TheModule.aliases())
|
|
|
|
mayPreserveGlobal(GV);
|
|
|
|
|
2016-10-26 07:53:31 +08:00
|
|
|
if (Used.empty())
|
2016-05-05 13:14:24 +08:00
|
|
|
return;
|
|
|
|
|
2016-10-26 07:53:31 +08:00
|
|
|
appendToCompilerUsed(TheModule, Used);
|
2016-05-05 13:14:24 +08:00
|
|
|
}
|
|
|
|
|
2010-03-13 02:44:54 +08:00
|
|
|
void LTOCodeGenerator::applyScopeRestrictions() {
|
2016-05-05 13:14:24 +08:00
|
|
|
if (ScopeRestrictionsDone)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Declare a callback for the internalize pass that will ask for every
|
|
|
|
// candidate GlobalValue if it can be internalized or not.
|
2016-10-13 04:12:19 +08:00
|
|
|
Mangler Mang;
|
2016-05-05 13:14:24 +08:00
|
|
|
SmallString<64> MangledName;
|
|
|
|
auto mustPreserveGV = [&](const GlobalValue &GV) -> bool {
|
|
|
|
// Unnamed globals can't be mangled, but they can't be preserved either.
|
|
|
|
if (!GV.hasName())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Need to mangle the GV as the "MustPreserveSymbols" StringSet is filled
|
|
|
|
// with the linker supplied name, which on Darwin includes a leading
|
|
|
|
// underscore.
|
|
|
|
MangledName.clear();
|
|
|
|
MangledName.reserve(GV.getName().size() + 1);
|
2016-10-13 04:12:19 +08:00
|
|
|
Mang.getNameWithPrefix(MangledName, &GV, /*CannotUsePrivateLabel=*/false);
|
2016-05-05 13:14:24 +08:00
|
|
|
return MustPreserveSymbols.count(MangledName);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Preserve linkonce value on linker request
|
|
|
|
preserveDiscardableGVs(*MergedModule, mustPreserveGV);
|
|
|
|
|
|
|
|
if (!ShouldInternalize)
|
2013-09-05 01:44:24 +08:00
|
|
|
return;
|
2010-03-13 02:44:54 +08:00
|
|
|
|
2016-04-13 13:36:06 +08:00
|
|
|
if (ShouldRestoreGlobalsLinkage) {
|
|
|
|
// Record the linkage type of non-local symbols so they can be restored
|
2016-05-05 13:14:20 +08:00
|
|
|
// prior
|
|
|
|
// to module splitting.
|
2016-04-13 13:36:06 +08:00
|
|
|
auto RecordLinkage = [&](const GlobalValue &GV) {
|
|
|
|
if (!GV.hasAvailableExternallyLinkage() && !GV.hasLocalLinkage() &&
|
|
|
|
GV.hasName())
|
|
|
|
ExternalSymbols.insert(std::make_pair(GV.getName(), GV.getLinkage()));
|
|
|
|
};
|
|
|
|
for (auto &GV : *MergedModule)
|
|
|
|
RecordLinkage(GV);
|
|
|
|
for (auto &GV : MergedModule->globals())
|
|
|
|
RecordLinkage(GV);
|
|
|
|
for (auto &GV : MergedModule->aliases())
|
|
|
|
RecordLinkage(GV);
|
|
|
|
}
|
|
|
|
|
2016-04-13 14:32:46 +08:00
|
|
|
// Update the llvm.compiler_used globals to force preserving libcalls and
|
|
|
|
// symbols referenced from asm
|
2016-06-23 03:50:42 +08:00
|
|
|
updateCompilerUsed(*MergedModule, *TargetMach, AsmUndefinedRefs);
|
2016-04-13 14:32:46 +08:00
|
|
|
|
2016-05-05 13:14:24 +08:00
|
|
|
internalizeModule(*MergedModule, mustPreserveGV);
|
2012-03-31 18:50:14 +08:00
|
|
|
|
2013-09-05 01:44:24 +08:00
|
|
|
ScopeRestrictionsDone = true;
|
2008-02-27 04:26:43 +08:00
|
|
|
}
|
|
|
|
|
2016-01-19 07:24:54 +08:00
|
|
|
/// Restore original linkage for symbols that may have been internalized
|
|
|
|
void LTOCodeGenerator::restoreLinkageForExternals() {
|
|
|
|
if (!ShouldInternalize || !ShouldRestoreGlobalsLinkage)
|
|
|
|
return;
|
|
|
|
|
|
|
|
assert(ScopeRestrictionsDone &&
|
|
|
|
"Cannot externalize without internalization!");
|
|
|
|
|
|
|
|
if (ExternalSymbols.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto externalize = [this](GlobalValue &GV) {
|
|
|
|
if (!GV.hasLocalLinkage() || !GV.hasName())
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto I = ExternalSymbols.find(GV.getName());
|
|
|
|
if (I == ExternalSymbols.end())
|
|
|
|
return;
|
|
|
|
|
2017-12-19 03:46:56 +08:00
|
|
|
GV.setLinkage(I->second);
|
|
|
|
};
|
|
|
|
|
|
|
|
llvm::for_each(MergedModule->functions(), externalize);
|
|
|
|
llvm::for_each(MergedModule->globals(), externalize);
|
|
|
|
llvm::for_each(MergedModule->aliases(), externalize);
|
|
|
|
}
|
|
|
|
|
|
|
|
void LTOCodeGenerator::verifyMergedModuleOnce() {
|
2016-04-21 01:48:22 +08:00
|
|
|
// Only run on the first call.
|
|
|
|
if (HasVerifiedInput)
|
|
|
|
return;
|
|
|
|
HasVerifiedInput = true;
|
|
|
|
|
2017-05-13 05:38:32 +08:00
|
|
|
bool BrokenDebugInfo = false;
|
2017-10-03 02:31:29 +08:00
|
|
|
if (verifyModule(*MergedModule, &dbgs(), &BrokenDebugInfo))
|
2016-04-21 01:48:22 +08:00
|
|
|
report_fatal_error("Broken module found, compilation aborted!");
|
2017-05-13 05:38:32 +08:00
|
|
|
if (BrokenDebugInfo) {
|
|
|
|
emitWarning("Invalid debug info found, debug info will be stripped");
|
|
|
|
StripDebugInfo(*MergedModule);
|
|
|
|
}
|
2016-04-21 01:48:22 +08:00
|
|
|
}
|
|
|
|
|
2016-11-19 02:06:28 +08:00
|
|
|
void LTOCodeGenerator::finishOptimizationRemarks() {
|
|
|
|
if (DiagnosticOutputFile) {
|
|
|
|
DiagnosticOutputFile->keep();
|
|
|
|
// FIXME: LTOCodeGenerator dtor is not invoked on Darwin
|
|
|
|
DiagnosticOutputFile->os().flush();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-02-27 04:26:43 +08:00
|
|
|
/// Optimize merged modules using various IPO passes
|
2015-09-16 07:05:59 +08:00
|
|
|
bool LTOCodeGenerator::optimize(bool DisableVerify, bool DisableInline,
|
|
|
|
bool DisableGVNLoadPRE,
|
2015-11-18 03:48:12 +08:00
|
|
|
bool DisableVectorization) {
|
|
|
|
if (!this->determineTarget())
|
2013-08-07 05:51:21 +08:00
|
|
|
return false;
|
2008-02-27 04:26:43 +08:00
|
|
|
|
2017-02-11 07:49:38 +08:00
|
|
|
auto DiagFileOrErr = lto::setupOptimizationRemarks(
|
2018-03-08 09:13:10 +08:00
|
|
|
Context, LTORemarksFilename, LTOPassRemarksWithHotness);
|
2017-02-11 06:16:17 +08:00
|
|
|
if (!DiagFileOrErr) {
|
|
|
|
errs() << "Error: " << toString(DiagFileOrErr.takeError()) << "\n";
|
|
|
|
report_fatal_error("Can't get an output file for the remarks");
|
|
|
|
}
|
|
|
|
DiagnosticOutputFile = std::move(*DiagFileOrErr);
|
2016-11-19 02:06:28 +08:00
|
|
|
|
2016-04-12 14:34:10 +08:00
|
|
|
// We always run the verifier once on the merged module, the `DisableVerify`
|
|
|
|
// parameter only applies to subsequent verify.
|
2016-04-21 01:48:22 +08:00
|
|
|
verifyMergedModuleOnce();
|
2016-04-12 14:34:10 +08:00
|
|
|
|
2013-05-24 05:21:50 +08:00
|
|
|
// Mark which symbols can not be internalized
|
2012-04-10 06:18:01 +08:00
|
|
|
this->applyScopeRestrictions();
|
|
|
|
|
2012-03-31 19:15:43 +08:00
|
|
|
// Instantiate the pass manager to organize the passes.
|
2015-02-13 18:01:29 +08:00
|
|
|
legacy::PassManager passes;
|
2008-02-27 04:26:43 +08:00
|
|
|
|
2012-10-09 00:39:34 +08:00
|
|
|
// Add an appropriate DataLayout instance for this module...
|
2015-08-25 05:15:35 +08:00
|
|
|
MergedModule->setDataLayout(TargetMach->createDataLayout());
|
2012-03-31 18:50:14 +08:00
|
|
|
|
2015-02-01 20:26:09 +08:00
|
|
|
passes.add(
|
|
|
|
createTargetTransformInfoWrapperPass(TargetMach->getTargetIRAnalysis()));
|
2015-01-30 21:33:42 +08:00
|
|
|
|
2014-08-22 02:49:52 +08:00
|
|
|
Triple TargetTriple(TargetMach->getTargetTriple());
|
2014-08-22 04:03:44 +08:00
|
|
|
PassManagerBuilder PMB;
|
|
|
|
PMB.DisableGVNLoadPRE = DisableGVNLoadPRE;
|
2014-10-27 05:50:58 +08:00
|
|
|
PMB.LoopVectorize = !DisableVectorization;
|
|
|
|
PMB.SLPVectorize = !DisableVectorization;
|
2014-08-22 04:03:44 +08:00
|
|
|
if (!DisableInline)
|
|
|
|
PMB.Inliner = createFunctionInliningPass();
|
[PM] Rework how the TargetLibraryInfo pass integrates with the new pass
manager to support the actual uses of it. =]
When I ported instcombine to the new pass manager I discover that it
didn't work because TLI wasn't available in the right places. This is
a somewhat surprising and/or subtle aspect of the new pass manager
design that came up before but I think is useful to be reminded of:
While the new pass manager *allows* a function pass to query a module
analysis, it requires that the module analysis is already run and cached
prior to the function pass manager starting up, possibly with
a 'require<foo>' style utility in the pass pipeline. This is an
intentional hurdle because using a module analysis from a function pass
*requires* that the module analysis is run prior to entering the
function pass manager. Otherwise the other functions in the module could
be in who-knows-what state, etc.
A somewhat surprising consequence of this design decision (at least to
me) is that you have to design a function pass that leverages
a module analysis to do so as an optional feature. Even if that means
your function pass does no work in the absence of the module analysis,
you have to handle that possibility and remain conservatively correct.
This is a natural consequence of things being able to invalidate the
module analysis and us being unable to re-run it. And it's a generally
good thing because it lets us reorder passes arbitrarily without
breaking correctness, etc.
This ends up causing problems in one case. What if we have a module
analysis that is *definitionally* impossible to invalidate. In the
places this might come up, the analysis is usually also definitionally
trivial to run even while other transformation passes run on the module,
regardless of the state of anything. And so, it follows that it is
natural to have a hard requirement on such analyses from a function
pass.
It turns out, that TargetLibraryInfo is just such an analysis, and
InstCombine has a hard requirement on it.
The approach I've taken here is to produce an analysis that models this
flexibility by making it both a module and a function analysis. This
exposes the fact that it is in fact safe to compute at any point. We can
even make it a valid CGSCC analysis at some point if that is useful.
However, we don't want to have a copy of the actual target library info
state for each function! This state is specific to the triple. The
somewhat direct and blunt approach here is to turn TLI into a pimpl,
with the state and mutators in the implementation class and the query
routines primarily in the wrapper. Then the analysis can lazily
construct and cache the implementations, keyed on the triple, and
on-demand produce wrappers of them for each function.
One minor annoyance is that we will end up with a wrapper for each
function in the module. While this is a bit wasteful (one pointer per
function) it seems tolerable. And it has the advantage of ensuring that
we pay the absolute minimum synchronization cost to access this
information should we end up with a nice parallel function pass manager
in the future. We could look into trying to mark when analysis results
are especially cheap to recompute and more eagerly GC-ing the cached
results, or we could look at supporting a variant of analyses whose
results are specifically *not* cached and expected to just be used and
discarded by the consumer. Either way, these seem like incremental
enhancements that should happen when we start profiling the memory and
CPU usage of the new pass manager and not before.
The other minor annoyance is that if we end up using the TLI in both
a module pass and a function pass, those will be produced by two
separate analyses, and thus will point to separate copies of the
implementation state. While a minor issue, I dislike this and would like
to find a way to cleanly allow a single analysis instance to be used
across multiple IR unit managers. But I don't have a good solution to
this today, and I don't want to hold up all of the work waiting to come
up with one. This too seems like a reasonable thing to incrementally
improve later.
llvm-svn: 226981
2015-01-24 10:06:09 +08:00
|
|
|
PMB.LibraryInfo = new TargetLibraryInfoImpl(TargetTriple);
|
2017-03-29 02:55:44 +08:00
|
|
|
if (Freestanding)
|
|
|
|
PMB.LibraryInfo->disableAllFunctions();
|
2015-03-20 06:01:00 +08:00
|
|
|
PMB.OptLevel = OptLevel;
|
2015-09-16 07:05:59 +08:00
|
|
|
PMB.VerifyInput = !DisableVerify;
|
|
|
|
PMB.VerifyOutput = !DisableVerify;
|
2014-08-22 04:03:44 +08:00
|
|
|
|
2015-01-30 21:33:42 +08:00
|
|
|
PMB.populateLTOPassManager(passes);
|
2008-02-27 04:26:43 +08:00
|
|
|
|
2015-02-04 02:39:15 +08:00
|
|
|
// Run our queue of passes all at once now, efficiently.
|
2015-08-25 05:15:35 +08:00
|
|
|
passes.run(*MergedModule);
|
2015-02-04 02:39:15 +08:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-11-18 03:48:12 +08:00
|
|
|
bool LTOCodeGenerator::compileOptimized(ArrayRef<raw_pwrite_stream *> Out) {
|
|
|
|
if (!this->determineTarget())
|
2015-02-04 02:39:15 +08:00
|
|
|
return false;
|
|
|
|
|
2016-04-21 01:48:22 +08:00
|
|
|
// We always run the verifier once on the merged module. If it has already
|
|
|
|
// been called in optimize(), this call will return early.
|
|
|
|
verifyMergedModuleOnce();
|
|
|
|
|
2015-08-28 07:37:36 +08:00
|
|
|
legacy::PassManager preCodeGenPasses;
|
2008-02-27 04:26:43 +08:00
|
|
|
|
2013-03-30 07:28:55 +08:00
|
|
|
// If the bitcode files contain ARC code and were compiled with optimization,
|
|
|
|
// the ObjCARCContractPass must be run, so do it unconditionally here.
|
2015-08-28 07:37:36 +08:00
|
|
|
preCodeGenPasses.add(createObjCARCContractPass());
|
|
|
|
preCodeGenPasses.run(*MergedModule);
|
|
|
|
|
2016-01-19 07:24:54 +08:00
|
|
|
// Re-externalize globals that may have been internalized to increase scope
|
|
|
|
// for splitting
|
|
|
|
restoreLinkageForExternals();
|
|
|
|
|
2015-08-28 07:37:36 +08:00
|
|
|
// Do code generation. We need to preserve the module in case the client calls
|
|
|
|
// writeMergedModules() after compilation, but we only need to allow this at
|
|
|
|
// parallelism level 1. This is achieved by having splitCodeGen return the
|
|
|
|
// original module at parallelism level 1 which we then assign back to
|
|
|
|
// MergedModule.
|
2016-04-18 02:42:27 +08:00
|
|
|
MergedModule = splitCodeGen(std::move(MergedModule), Out, {},
|
|
|
|
[&]() { return createTargetMachine(); }, FileType,
|
|
|
|
ShouldRestoreGlobalsLinkage);
|
2009-07-27 06:16:39 +08:00
|
|
|
|
2016-02-17 05:41:51 +08:00
|
|
|
// If statistics were requested, print them out after codegen.
|
|
|
|
if (llvm::AreStatisticsEnabled())
|
|
|
|
llvm::PrintStatistics();
|
2017-05-16 17:43:21 +08:00
|
|
|
reportAndResetTimings();
|
2016-02-17 05:41:51 +08:00
|
|
|
|
2016-11-29 00:51:49 +08:00
|
|
|
finishOptimizationRemarks();
|
|
|
|
|
2013-08-07 05:51:21 +08:00
|
|
|
return true;
|
2008-02-27 04:26:43 +08:00
|
|
|
}
|
|
|
|
|
2012-03-31 18:49:43 +08:00
|
|
|
/// setCodeGenDebugOptions - Set codegen debugging options to aid in debugging
|
|
|
|
/// LTO problems.
|
2016-10-01 09:18:23 +08:00
|
|
|
void LTOCodeGenerator::setCodeGenDebugOptions(StringRef Options) {
|
2015-09-01 07:44:06 +08:00
|
|
|
for (std::pair<StringRef, StringRef> o = getToken(Options); !o.first.empty();
|
|
|
|
o = getToken(o.second))
|
2015-08-21 12:45:55 +08:00
|
|
|
CodegenOptions.push_back(o.first);
|
2008-07-09 05:14:10 +08:00
|
|
|
}
|
2013-10-02 22:36:23 +08:00
|
|
|
|
|
|
|
void LTOCodeGenerator::parseCodeGenDebugOptions() {
|
|
|
|
// if options were requested, set them
|
2015-08-21 12:45:55 +08:00
|
|
|
if (!CodegenOptions.empty()) {
|
|
|
|
// ParseCommandLineOptions() expects argv[0] to be program name.
|
|
|
|
std::vector<const char *> CodegenArgv(1, "libLLVMLTO");
|
|
|
|
for (std::string &Arg : CodegenOptions)
|
|
|
|
CodegenArgv.push_back(Arg.c_str());
|
|
|
|
cl::ParseCommandLineOptions(CodegenArgv.size(), CodegenArgv.data());
|
|
|
|
}
|
2013-10-02 22:36:23 +08:00
|
|
|
}
|
2014-01-16 06:04:35 +08:00
|
|
|
|
|
|
|
|
2017-09-16 04:10:09 +08:00
|
|
|
void LTOCodeGenerator::DiagnosticHandler(const DiagnosticInfo &DI) {
|
2014-01-16 06:04:35 +08:00
|
|
|
// Map the LLVM internal diagnostic severity to the LTO diagnostic severity.
|
|
|
|
lto_codegen_diagnostic_severity_t Severity;
|
|
|
|
switch (DI.getSeverity()) {
|
|
|
|
case DS_Error:
|
|
|
|
Severity = LTO_DS_ERROR;
|
|
|
|
break;
|
|
|
|
case DS_Warning:
|
|
|
|
Severity = LTO_DS_WARNING;
|
|
|
|
break;
|
2014-02-28 17:08:45 +08:00
|
|
|
case DS_Remark:
|
|
|
|
Severity = LTO_DS_REMARK;
|
|
|
|
break;
|
2014-01-16 06:04:35 +08:00
|
|
|
case DS_Note:
|
|
|
|
Severity = LTO_DS_NOTE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Create the string that will be reported to the external diagnostic handler.
|
2014-06-27 06:52:05 +08:00
|
|
|
std::string MsgStorage;
|
|
|
|
raw_string_ostream Stream(MsgStorage);
|
|
|
|
DiagnosticPrinterRawOStream DP(Stream);
|
2014-01-16 06:04:35 +08:00
|
|
|
DI.print(DP);
|
2014-06-27 06:52:05 +08:00
|
|
|
Stream.flush();
|
2014-01-16 06:04:35 +08:00
|
|
|
|
|
|
|
// If this method has been called it means someone has set up an external
|
|
|
|
// diagnostic handler. Assert on that.
|
|
|
|
assert(DiagHandler && "Invalid diagnostic handler");
|
2014-06-27 06:52:05 +08:00
|
|
|
(*DiagHandler)(Severity, MsgStorage.c_str(), DiagContext);
|
2014-01-16 06:04:35 +08:00
|
|
|
}
|
|
|
|
|
2017-09-16 04:10:09 +08:00
|
|
|
namespace {
|
|
|
|
struct LTODiagnosticHandler : public DiagnosticHandler {
|
|
|
|
LTOCodeGenerator *CodeGenerator;
|
|
|
|
LTODiagnosticHandler(LTOCodeGenerator *CodeGenPtr)
|
|
|
|
: CodeGenerator(CodeGenPtr) {}
|
|
|
|
bool handleDiagnostics(const DiagnosticInfo &DI) override {
|
|
|
|
CodeGenerator->DiagnosticHandler(DI);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2014-01-16 06:04:35 +08:00
|
|
|
void
|
|
|
|
LTOCodeGenerator::setDiagnosticHandler(lto_diagnostic_handler_t DiagHandler,
|
|
|
|
void *Ctxt) {
|
|
|
|
this->DiagHandler = DiagHandler;
|
|
|
|
this->DiagContext = Ctxt;
|
|
|
|
if (!DiagHandler)
|
2017-09-16 04:10:09 +08:00
|
|
|
return Context.setDiagnosticHandler(nullptr);
|
2014-01-16 06:04:35 +08:00
|
|
|
// Register the LTOCodeGenerator stub in the LLVMContext to forward the
|
|
|
|
// diagnostic to the external DiagHandler.
|
2017-09-16 04:10:09 +08:00
|
|
|
Context.setDiagnosticHandler(llvm::make_unique<LTODiagnosticHandler>(this),
|
|
|
|
true);
|
2014-01-16 06:04:35 +08:00
|
|
|
}
|
2015-11-18 03:48:12 +08:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
class LTODiagnosticInfo : public DiagnosticInfo {
|
|
|
|
const Twine &Msg;
|
|
|
|
public:
|
|
|
|
LTODiagnosticInfo(const Twine &DiagMsg, DiagnosticSeverity Severity=DS_Error)
|
|
|
|
: DiagnosticInfo(DK_Linker, Severity), Msg(DiagMsg) {}
|
|
|
|
void print(DiagnosticPrinter &DP) const override { DP << Msg; }
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
void LTOCodeGenerator::emitError(const std::string &ErrMsg) {
|
|
|
|
if (DiagHandler)
|
|
|
|
(*DiagHandler)(LTO_DS_ERROR, ErrMsg.c_str(), DiagContext);
|
|
|
|
else
|
|
|
|
Context.diagnose(LTODiagnosticInfo(ErrMsg));
|
|
|
|
}
|
2016-05-06 04:05:33 +08:00
|
|
|
|
|
|
|
void LTOCodeGenerator::emitWarning(const std::string &ErrMsg) {
|
|
|
|
if (DiagHandler)
|
|
|
|
(*DiagHandler)(LTO_DS_WARNING, ErrMsg.c_str(), DiagContext);
|
|
|
|
else
|
|
|
|
Context.diagnose(LTODiagnosticInfo(ErrMsg, DS_Warning));
|
|
|
|
}
|