[ORC] Add std::tuple support to SimplePackedSerialization.
[llvm-project.git] / llvm / lib / Transforms / Utils / NameAnonGlobals.cpp
blob7083789267d9c260af5a4eb4a697d173f48e9189
1 //===- NameAnonGlobals.cpp - ThinLTO Support: Name Unnamed Globals --------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file implements naming anonymous globals to make sure they can be
10 // referred to by ThinLTO.
12 //===----------------------------------------------------------------------===//
14 #include "llvm/Transforms/Utils/NameAnonGlobals.h"
15 #include "llvm/ADT/SmallString.h"
16 #include "llvm/IR/Module.h"
17 #include "llvm/InitializePasses.h"
18 #include "llvm/Support/MD5.h"
19 #include "llvm/Transforms/Utils/ModuleUtils.h"
21 using namespace llvm;
23 namespace {
24 // Compute a "unique" hash for the module based on the name of the public
25 // globals.
26 class ModuleHasher {
27 Module &TheModule;
28 std::string TheHash;
30 public:
31 ModuleHasher(Module &M) : TheModule(M) {}
33 /// Return the lazily computed hash.
34 std::string &get() {
35 if (!TheHash.empty())
36 // Cache hit :)
37 return TheHash;
39 MD5 Hasher;
40 for (auto &F : TheModule) {
41 if (F.isDeclaration() || F.hasLocalLinkage() || !F.hasName())
42 continue;
43 auto Name = F.getName();
44 Hasher.update(Name);
46 for (auto &GV : TheModule.globals()) {
47 if (GV.isDeclaration() || GV.hasLocalLinkage() || !GV.hasName())
48 continue;
49 auto Name = GV.getName();
50 Hasher.update(Name);
53 // Now return the result.
54 MD5::MD5Result Hash;
55 Hasher.final(Hash);
56 SmallString<32> Result;
57 MD5::stringifyResult(Hash, Result);
58 TheHash = std::string(Result.str());
59 return TheHash;
62 } // end anonymous namespace
64 // Rename all the anon globals in the module
65 bool llvm::nameUnamedGlobals(Module &M) {
66 bool Changed = false;
67 ModuleHasher ModuleHash(M);
68 int count = 0;
69 auto RenameIfNeed = [&](GlobalValue &GV) {
70 if (GV.hasName())
71 return;
72 GV.setName(Twine("anon.") + ModuleHash.get() + "." + Twine(count++));
73 Changed = true;
75 for (auto &GO : M.global_objects())
76 RenameIfNeed(GO);
77 for (auto &GA : M.aliases())
78 RenameIfNeed(GA);
80 return Changed;
83 namespace {
85 // Legacy pass that provides a name to every anon globals.
86 class NameAnonGlobalLegacyPass : public ModulePass {
88 public:
89 /// Pass identification, replacement for typeid
90 static char ID;
92 /// Specify pass name for debug output
93 StringRef getPassName() const override { return "Name Anon Globals"; }
95 explicit NameAnonGlobalLegacyPass() : ModulePass(ID) {}
97 bool runOnModule(Module &M) override { return nameUnamedGlobals(M); }
99 char NameAnonGlobalLegacyPass::ID = 0;
101 } // anonymous namespace
103 PreservedAnalyses NameAnonGlobalPass::run(Module &M,
104 ModuleAnalysisManager &AM) {
105 if (!nameUnamedGlobals(M))
106 return PreservedAnalyses::all();
108 return PreservedAnalyses::none();
111 INITIALIZE_PASS_BEGIN(NameAnonGlobalLegacyPass, "name-anon-globals",
112 "Provide a name to nameless globals", false, false)
113 INITIALIZE_PASS_END(NameAnonGlobalLegacyPass, "name-anon-globals",
114 "Provide a name to nameless globals", false, false)
116 namespace llvm {
117 ModulePass *createNameAnonGlobalPass() {
118 return new NameAnonGlobalLegacyPass();