blob: 34dc1cccdd5b567adcbfe2bfa227f981863f6ba4 [file] [log] [blame]
Mehdi Amini27d23792016-09-16 16:56:30 +00001//===- NameAnonGlobals.cpp - ThinLTO Support: Name Unnamed Globals --------===//
Mehdi Aminid5faa262016-04-12 21:35:28 +00002//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
Mehdi Amini27d23792016-09-16 16:56:30 +000010// This file implements naming anonymous globals to make sure they can be
11// referred to by ThinLTO.
Mehdi Aminid5faa262016-04-12 21:35:28 +000012//
13//===----------------------------------------------------------------------===//
14
Mehdi Amini27d23792016-09-16 16:56:30 +000015#include "llvm/Transforms/Utils/NameAnonGlobals.h"
Teresa Johnson4223dd82016-08-12 14:03:36 +000016
Mehdi Aminid5faa262016-04-12 21:35:28 +000017#include "llvm/ADT/SmallString.h"
18#include "llvm/IR/Module.h"
19#include "llvm/Support/MD5.h"
20#include "llvm/Transforms/Utils/ModuleUtils.h"
21
22using namespace llvm;
23
Benjamin Kramerb7d33112016-08-06 11:13:10 +000024namespace {
Mehdi Aminid5faa262016-04-12 21:35:28 +000025// Compute a "unique" hash for the module based on the name of the public
Mehdi Amini27d23792016-09-16 16:56:30 +000026// globals.
Mehdi Aminid5faa262016-04-12 21:35:28 +000027class ModuleHasher {
28 Module &TheModule;
29 std::string TheHash;
30
31public:
32 ModuleHasher(Module &M) : TheModule(M) {}
33
34 /// Return the lazily computed hash.
35 std::string &get() {
36 if (!TheHash.empty())
37 // Cache hit :)
38 return TheHash;
39
40 MD5 Hasher;
41 for (auto &F : TheModule) {
42 if (F.isDeclaration() || F.hasLocalLinkage() || !F.hasName())
43 continue;
44 auto Name = F.getName();
45 Hasher.update(Name);
46 }
47 for (auto &GV : TheModule.globals()) {
48 if (GV.isDeclaration() || GV.hasLocalLinkage() || !GV.hasName())
49 continue;
50 auto Name = GV.getName();
51 Hasher.update(Name);
52 }
53
54 // Now return the result.
55 MD5::MD5Result Hash;
56 Hasher.final(Hash);
57 SmallString<32> Result;
58 MD5::stringifyResult(Hash, Result);
59 TheHash = Result.str();
60 return TheHash;
61 }
62};
Benjamin Kramerb7d33112016-08-06 11:13:10 +000063} // end anonymous namespace
Mehdi Aminid5faa262016-04-12 21:35:28 +000064
Mehdi Amini27d23792016-09-16 16:56:30 +000065// Rename all the anon globals in the module
66bool llvm::nameUnamedGlobals(Module &M) {
Mehdi Aminid5faa262016-04-12 21:35:28 +000067 bool Changed = false;
68 ModuleHasher ModuleHash(M);
69 int count = 0;
Mehdi Amini27d23792016-09-16 16:56:30 +000070 auto RenameIfNeed = [&](GlobalValue &GV) {
Mehdi Amini2cac7872016-09-16 16:56:25 +000071 if (GV.hasName())
72 return;
73 GV.setName(Twine("anon.") + ModuleHash.get() + "." + Twine(count++));
Mehdi Aminid5faa262016-04-12 21:35:28 +000074 Changed = true;
Mehdi Amini2cac7872016-09-16 16:56:25 +000075 };
76 for (auto &GO : M.global_objects())
77 RenameIfNeed(GO);
78 for (auto &GA : M.aliases())
79 RenameIfNeed(GA);
80
Mehdi Aminid5faa262016-04-12 21:35:28 +000081 return Changed;
82}
83
84namespace {
85
Mehdi Amini27d23792016-09-16 16:56:30 +000086// Legacy pass that provides a name to every anon globals.
87class NameAnonGlobalLegacyPass : public ModulePass {
Mehdi Aminid5faa262016-04-12 21:35:28 +000088
89public:
90 /// Pass identification, replacement for typeid
91 static char ID;
92
93 /// Specify pass name for debug output
Mehdi Amini117296c2016-10-01 02:56:57 +000094 StringRef getPassName() const override { return "Name Anon Globals"; }
Mehdi Aminid5faa262016-04-12 21:35:28 +000095
Mehdi Amini27d23792016-09-16 16:56:30 +000096 explicit NameAnonGlobalLegacyPass() : ModulePass(ID) {}
Mehdi Aminid5faa262016-04-12 21:35:28 +000097
Mehdi Amini27d23792016-09-16 16:56:30 +000098 bool runOnModule(Module &M) override { return nameUnamedGlobals(M); }
Mehdi Aminid5faa262016-04-12 21:35:28 +000099};
Mehdi Amini27d23792016-09-16 16:56:30 +0000100char NameAnonGlobalLegacyPass::ID = 0;
Mehdi Aminid5faa262016-04-12 21:35:28 +0000101
102} // anonymous namespace
103
Mehdi Amini27d23792016-09-16 16:56:30 +0000104PreservedAnalyses NameAnonGlobalPass::run(Module &M,
105 ModuleAnalysisManager &AM) {
106 if (!nameUnamedGlobals(M))
Teresa Johnson4223dd82016-08-12 14:03:36 +0000107 return PreservedAnalyses::all();
108
109 return PreservedAnalyses::none();
110}
111
Mehdi Amini27d23792016-09-16 16:56:30 +0000112INITIALIZE_PASS_BEGIN(NameAnonGlobalLegacyPass, "name-anon-globals",
113 "Provide a name to nameless globals", false, false)
114INITIALIZE_PASS_END(NameAnonGlobalLegacyPass, "name-anon-globals",
115 "Provide a name to nameless globals", false, false)
Mehdi Aminid5faa262016-04-12 21:35:28 +0000116
117namespace llvm {
Mehdi Amini27d23792016-09-16 16:56:30 +0000118ModulePass *createNameAnonGlobalPass() {
119 return new NameAnonGlobalLegacyPass();
Teresa Johnson4223dd82016-08-12 14:03:36 +0000120}
Mehdi Aminid5faa262016-04-12 21:35:28 +0000121}