2017-10-11 06:49:55 +08:00
|
|
|
//===- ElimAvailExtern.cpp - DCE unreachable internal functions -----------===//
|
2015-07-07 00:22:42 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2015-07-07 00:22:42 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This transform is designed to eliminate available external global
|
|
|
|
// definitions from the program, turning them into declarations.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2016-05-05 10:37:32 +08:00
|
|
|
#include "llvm/Transforms/IPO/ElimAvailExtern.h"
|
2015-07-07 00:22:42 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
2017-10-11 06:49:55 +08:00
|
|
|
#include "llvm/IR/Constant.h"
|
|
|
|
#include "llvm/IR/Function.h"
|
|
|
|
#include "llvm/IR/GlobalValue.h"
|
|
|
|
#include "llvm/IR/GlobalVariable.h"
|
2015-07-07 00:22:42 +08:00
|
|
|
#include "llvm/IR/Module.h"
|
Sink all InitializePasses.h includes
This file lists every pass in LLVM, and is included by Pass.h, which is
very popular. Every time we add, remove, or rename a pass in LLVM, it
caused lots of recompilation.
I found this fact by looking at this table, which is sorted by the
number of times a file was changed over the last 100,000 git commits
multiplied by the number of object files that depend on it in the
current checkout:
recompiles touches affected_files header
342380 95 3604 llvm/include/llvm/ADT/STLExtras.h
314730 234 1345 llvm/include/llvm/InitializePasses.h
307036 118 2602 llvm/include/llvm/ADT/APInt.h
213049 59 3611 llvm/include/llvm/Support/MathExtras.h
170422 47 3626 llvm/include/llvm/Support/Compiler.h
162225 45 3605 llvm/include/llvm/ADT/Optional.h
158319 63 2513 llvm/include/llvm/ADT/Triple.h
140322 39 3598 llvm/include/llvm/ADT/StringRef.h
137647 59 2333 llvm/include/llvm/Support/Error.h
131619 73 1803 llvm/include/llvm/Support/FileSystem.h
Before this change, touching InitializePasses.h would cause 1345 files
to recompile. After this change, touching it only causes 550 compiles in
an incremental rebuild.
Reviewers: bkramer, asbirlea, bollu, jdoerfert
Differential Revision: https://reviews.llvm.org/D70211
2019-11-14 05:15:01 +08:00
|
|
|
#include "llvm/InitializePasses.h"
|
2017-06-06 19:49:48 +08:00
|
|
|
#include "llvm/Pass.h"
|
2016-05-05 10:37:32 +08:00
|
|
|
#include "llvm/Transforms/IPO.h"
|
2015-07-07 00:22:42 +08:00
|
|
|
#include "llvm/Transforms/Utils/GlobalStatus.h"
|
2017-10-11 06:49:55 +08:00
|
|
|
|
2015-07-07 00:22:42 +08:00
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
#define DEBUG_TYPE "elim-avail-extern"
|
|
|
|
|
|
|
|
STATISTIC(NumFunctions, "Number of functions removed");
|
|
|
|
STATISTIC(NumVariables, "Number of global variables removed");
|
|
|
|
|
2016-05-05 10:37:32 +08:00
|
|
|
static bool eliminateAvailableExternally(Module &M) {
|
2015-07-07 00:22:42 +08:00
|
|
|
bool Changed = false;
|
|
|
|
|
|
|
|
// Drop initializers of available externally global variables.
|
2015-11-03 02:02:11 +08:00
|
|
|
for (GlobalVariable &GV : M.globals()) {
|
2015-09-05 04:24:24 +08:00
|
|
|
if (!GV.hasAvailableExternallyLinkage())
|
2015-07-07 00:22:42 +08:00
|
|
|
continue;
|
2015-09-05 04:24:24 +08:00
|
|
|
if (GV.hasInitializer()) {
|
|
|
|
Constant *Init = GV.getInitializer();
|
|
|
|
GV.setInitializer(nullptr);
|
2015-07-07 00:22:42 +08:00
|
|
|
if (isSafeToDestroyConstant(Init))
|
|
|
|
Init->destroyConstant();
|
|
|
|
}
|
2015-09-05 04:24:24 +08:00
|
|
|
GV.removeDeadConstantUsers();
|
|
|
|
GV.setLinkage(GlobalValue::ExternalLinkage);
|
2015-07-07 00:22:42 +08:00
|
|
|
NumVariables++;
|
2015-11-03 08:14:15 +08:00
|
|
|
Changed = true;
|
2015-07-07 00:22:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Drop the bodies of available externally functions.
|
2015-09-05 04:24:24 +08:00
|
|
|
for (Function &F : M) {
|
|
|
|
if (!F.hasAvailableExternallyLinkage())
|
2015-07-07 00:22:42 +08:00
|
|
|
continue;
|
2015-09-05 04:24:24 +08:00
|
|
|
if (!F.isDeclaration())
|
2015-07-07 00:22:42 +08:00
|
|
|
// This will set the linkage to external
|
2015-09-05 04:24:24 +08:00
|
|
|
F.deleteBody();
|
|
|
|
F.removeDeadConstantUsers();
|
2015-07-07 00:22:42 +08:00
|
|
|
NumFunctions++;
|
2015-11-03 08:14:15 +08:00
|
|
|
Changed = true;
|
2015-07-07 00:22:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return Changed;
|
|
|
|
}
|
2016-05-05 10:37:32 +08:00
|
|
|
|
2016-06-17 08:11:01 +08:00
|
|
|
PreservedAnalyses
|
|
|
|
EliminateAvailableExternallyPass::run(Module &M, ModuleAnalysisManager &) {
|
2016-05-05 10:37:32 +08:00
|
|
|
if (!eliminateAvailableExternally(M))
|
|
|
|
return PreservedAnalyses::all();
|
|
|
|
return PreservedAnalyses::none();
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
2017-10-11 06:49:55 +08:00
|
|
|
|
2016-05-05 10:37:32 +08:00
|
|
|
struct EliminateAvailableExternallyLegacyPass : public ModulePass {
|
|
|
|
static char ID; // Pass identification, replacement for typeid
|
2017-10-11 06:49:55 +08:00
|
|
|
|
2016-05-05 10:37:32 +08:00
|
|
|
EliminateAvailableExternallyLegacyPass() : ModulePass(ID) {
|
|
|
|
initializeEliminateAvailableExternallyLegacyPassPass(
|
|
|
|
*PassRegistry::getPassRegistry());
|
|
|
|
}
|
|
|
|
|
|
|
|
// run - Do the EliminateAvailableExternally pass on the specified module,
|
|
|
|
// optionally updating the specified callgraph to reflect the changes.
|
2017-10-11 06:49:55 +08:00
|
|
|
bool runOnModule(Module &M) override {
|
2016-05-05 10:37:32 +08:00
|
|
|
if (skipModule(M))
|
|
|
|
return false;
|
|
|
|
return eliminateAvailableExternally(M);
|
|
|
|
}
|
|
|
|
};
|
2017-10-11 06:49:55 +08:00
|
|
|
|
|
|
|
} // end anonymous namespace
|
2016-05-05 10:37:32 +08:00
|
|
|
|
|
|
|
char EliminateAvailableExternallyLegacyPass::ID = 0;
|
2017-10-11 06:49:55 +08:00
|
|
|
|
2016-05-05 10:37:32 +08:00
|
|
|
INITIALIZE_PASS(EliminateAvailableExternallyLegacyPass, "elim-avail-extern",
|
|
|
|
"Eliminate Available Externally Globals", false, false)
|
|
|
|
|
|
|
|
ModulePass *llvm::createEliminateAvailableExternallyPass() {
|
|
|
|
return new EliminateAvailableExternallyLegacyPass();
|
|
|
|
}
|