blob: af313a6b001d751405f0e727ff2db65816ad382f [file] [log] [blame]
Teresa Johnsonf72278f2015-11-02 18:02:11 +00001//===-- ElimAvailExtern.cpp - DCE unreachable internal functions
2//----------------===//
Teresa Johnsond3a33a12015-07-06 16:22:42 +00003//
4// The LLVM Compiler Infrastructure
5//
6// This file is distributed under the University of Illinois Open Source
7// License. See LICENSE.TXT for details.
8//
9//===----------------------------------------------------------------------===//
10//
11// This transform is designed to eliminate available external global
12// definitions from the program, turning them into declarations.
13//
14//===----------------------------------------------------------------------===//
15
16#include "llvm/Transforms/IPO.h"
17#include "llvm/ADT/Statistic.h"
18#include "llvm/IR/Constants.h"
Teresa Johnsond3a33a12015-07-06 16:22:42 +000019#include "llvm/IR/Module.h"
Teresa Johnsond3a33a12015-07-06 16:22:42 +000020#include "llvm/Transforms/Utils/GlobalStatus.h"
21#include "llvm/Pass.h"
22using namespace llvm;
23
24#define DEBUG_TYPE "elim-avail-extern"
25
Teresa Johnsond3a33a12015-07-06 16:22:42 +000026STATISTIC(NumFunctions, "Number of functions removed");
27STATISTIC(NumVariables, "Number of global variables removed");
28
29namespace {
Teresa Johnsonf72278f2015-11-02 18:02:11 +000030struct EliminateAvailableExternally : public ModulePass {
31 static char ID; // Pass identification, replacement for typeid
32 EliminateAvailableExternally() : ModulePass(ID) {
33 initializeEliminateAvailableExternallyPass(
34 *PassRegistry::getPassRegistry());
35 }
Teresa Johnsond3a33a12015-07-06 16:22:42 +000036
Teresa Johnsonf72278f2015-11-02 18:02:11 +000037 // run - Do the EliminateAvailableExternally pass on the specified module,
38 // optionally updating the specified callgraph to reflect the changes.
39 //
40 bool runOnModule(Module &M) override;
41};
Teresa Johnsond3a33a12015-07-06 16:22:42 +000042}
43
44char EliminateAvailableExternally::ID = 0;
45INITIALIZE_PASS(EliminateAvailableExternally, "elim-avail-extern",
46 "Eliminate Available Externally Globals", false, false)
47
48ModulePass *llvm::createEliminateAvailableExternallyPass() {
49 return new EliminateAvailableExternally();
50}
51
52bool EliminateAvailableExternally::runOnModule(Module &M) {
53 bool Changed = false;
54
55 // Drop initializers of available externally global variables.
Teresa Johnsonf72278f2015-11-02 18:02:11 +000056 for (GlobalVariable &GV : M.globals()) {
Yaron Keren771e3192015-09-04 20:24:24 +000057 if (!GV.hasAvailableExternallyLinkage())
Teresa Johnsond3a33a12015-07-06 16:22:42 +000058 continue;
Yaron Keren771e3192015-09-04 20:24:24 +000059 if (GV.hasInitializer()) {
60 Constant *Init = GV.getInitializer();
61 GV.setInitializer(nullptr);
Teresa Johnsond3a33a12015-07-06 16:22:42 +000062 if (isSafeToDestroyConstant(Init))
63 Init->destroyConstant();
64 }
Yaron Keren771e3192015-09-04 20:24:24 +000065 GV.removeDeadConstantUsers();
66 GV.setLinkage(GlobalValue::ExternalLinkage);
Teresa Johnsond3a33a12015-07-06 16:22:42 +000067 NumVariables++;
Teresa Johnsonc7ed52f2015-11-03 00:14:15 +000068 Changed = true;
Teresa Johnsond3a33a12015-07-06 16:22:42 +000069 }
70
71 // Drop the bodies of available externally functions.
Yaron Keren771e3192015-09-04 20:24:24 +000072 for (Function &F : M) {
73 if (!F.hasAvailableExternallyLinkage())
Teresa Johnsond3a33a12015-07-06 16:22:42 +000074 continue;
Yaron Keren771e3192015-09-04 20:24:24 +000075 if (!F.isDeclaration())
Teresa Johnsond3a33a12015-07-06 16:22:42 +000076 // This will set the linkage to external
Yaron Keren771e3192015-09-04 20:24:24 +000077 F.deleteBody();
78 F.removeDeadConstantUsers();
Teresa Johnsond3a33a12015-07-06 16:22:42 +000079 NumFunctions++;
Teresa Johnsonc7ed52f2015-11-03 00:14:15 +000080 Changed = true;
Teresa Johnsond3a33a12015-07-06 16:22:42 +000081 }
82
Teresa Johnsond3a33a12015-07-06 16:22:42 +000083 return Changed;
84}