Skip to content

Commit 08ed216

Browse files
ibooksteinMaskRay
authored andcommitted
[IR] Refactor GlobalIFunc to inherit from GlobalObject, Remove GlobalIndirectSymbol
As discussed in: * https://reviews.llvm.org/D94166 * https://lists.llvm.org/pipermail/llvm-dev/2020-September/145031.html The GlobalIndirectSymbol class lost most of its meaning in https://reviews.llvm.org/D109792, which disambiguated getBaseObject (now getAliaseeObject) between GlobalIFunc and everything else. In addition, as long as GlobalIFunc is not a GlobalObject and getAliaseeObject returns GlobalObjects, a GlobalAlias whose aliasee is a GlobalIFunc cannot currently be modeled properly. Creating aliases for GlobalIFuncs does happen in the wild (e.g. glibc). In addition, calling getAliaseeObject on a GlobalIFunc will currently return nullptr, which is undesirable because it should return the object itself for non-aliases. This patch refactors the GlobalIFunc class to inherit directly from GlobalObject, and removes GlobalIndirectSymbol (while inlining the relevant parts into GlobalAlias and GlobalIFunc). This allows for calling getAliaseeObject() on a GlobalIFunc to return the GlobalIFunc itself, making getAliaseeObject() more consistent and enabling alias-to-ifunc to be properly modeled in the IR. I exercised some judgement in the API clients of GlobalIndirectSymbol: some were 'monomorphized' for GlobalAlias and GlobalIFunc, and some remained shared (with the type adapted to become GlobalValue). Reviewed By: MaskRay Differential Revision: https://reviews.llvm.org/D108872
1 parent 7562f3d commit 08ed216

File tree

22 files changed

+419
-384
lines changed

22 files changed

+419
-384
lines changed

clang/lib/CodeGen/CodeGenModule.cpp

+27-22
Original file line numberDiff line numberDiff line change
@@ -317,21 +317,21 @@ void CodeGenModule::applyGlobalValReplacements() {
317317

318318
// This is only used in aliases that we created and we know they have a
319319
// linear structure.
320-
static const llvm::GlobalObject *getAliasedGlobal(
321-
const llvm::GlobalIndirectSymbol &GIS) {
322-
llvm::SmallPtrSet<const llvm::GlobalIndirectSymbol*, 4> Visited;
323-
const llvm::Constant *C = &GIS;
320+
static const llvm::GlobalValue *getAliasedGlobal(const llvm::GlobalValue *GV) {
321+
llvm::SmallPtrSet<const llvm::GlobalValue *, 4> Visited;
324322
for (;;) {
325-
C = C->stripPointerCasts();
326-
if (auto *GO = dyn_cast<llvm::GlobalObject>(C))
327-
return GO;
328-
// stripPointerCasts will not walk over weak aliases.
329-
auto *GIS2 = dyn_cast<llvm::GlobalIndirectSymbol>(C);
330-
if (!GIS2)
323+
if (!GV || !Visited.insert(GV).second)
331324
return nullptr;
332-
if (!Visited.insert(GIS2).second)
333-
return nullptr;
334-
C = GIS2->getIndirectSymbol();
325+
326+
const llvm::Constant *C;
327+
if (auto *GA = dyn_cast<llvm::GlobalAlias>(GV))
328+
C = GA->getAliasee();
329+
else if (auto *GI = dyn_cast<llvm::GlobalIFunc>(GV))
330+
C = GI->getResolver();
331+
else
332+
return GV;
333+
334+
GV = dyn_cast<llvm::GlobalValue>(C->stripPointerCasts());
335335
}
336336
}
337337

@@ -350,9 +350,8 @@ void CodeGenModule::checkAliases() {
350350
else
351351
llvm_unreachable("Not an alias or ifunc?");
352352
StringRef MangledName = getMangledName(GD);
353-
llvm::GlobalValue *Entry = GetGlobalValue(MangledName);
354-
auto *Alias = cast<llvm::GlobalIndirectSymbol>(Entry);
355-
const llvm::GlobalValue *GV = getAliasedGlobal(*Alias);
353+
llvm::GlobalValue *Alias = GetGlobalValue(MangledName);
354+
const llvm::GlobalValue *GV = getAliasedGlobal(Alias);
356355
if (!GV) {
357356
Error = true;
358357
Diags.Report(Location, diag::err_cyclic_alias) << IsIFunc;
@@ -369,7 +368,10 @@ void CodeGenModule::checkAliases() {
369368
Diags.Report(Location, diag::err_ifunc_resolver_return);
370369
}
371370

372-
llvm::Constant *Aliasee = Alias->getIndirectSymbol();
371+
llvm::Constant *Aliasee =
372+
IsIFunc ? cast<llvm::GlobalIFunc>(Alias)->getResolver()
373+
: cast<llvm::GlobalAlias>(Alias)->getAliasee();
374+
373375
llvm::GlobalValue *AliaseeGV;
374376
if (auto CE = dyn_cast<llvm::ConstantExpr>(Aliasee))
375377
AliaseeGV = cast<llvm::GlobalValue>(CE->getOperand(0));
@@ -388,13 +390,17 @@ void CodeGenModule::checkAliases() {
388390
// compatibility with gcc we implement it by just pointing the alias
389391
// to its aliasee's aliasee. We also warn, since the user is probably
390392
// expecting the link to be weak.
391-
if (auto GA = dyn_cast<llvm::GlobalIndirectSymbol>(AliaseeGV)) {
393+
if (auto *GA = dyn_cast<llvm::GlobalAlias>(AliaseeGV)) {
392394
if (GA->isInterposable()) {
393395
Diags.Report(Location, diag::warn_alias_to_weak_alias)
394396
<< GV->getName() << GA->getName() << IsIFunc;
395397
Aliasee = llvm::ConstantExpr::getPointerBitCastOrAddrSpaceCast(
396-
GA->getIndirectSymbol(), Alias->getType());
397-
Alias->setIndirectSymbol(Aliasee);
398+
GA->getAliasee(), Alias->getType());
399+
400+
if (IsIFunc)
401+
cast<llvm::GlobalIFunc>(Alias)->setResolver(Aliasee);
402+
else
403+
cast<llvm::GlobalAlias>(Alias)->setAliasee(Aliasee);
398404
}
399405
}
400406
}
@@ -403,8 +409,7 @@ void CodeGenModule::checkAliases() {
403409

404410
for (const GlobalDecl &GD : Aliases) {
405411
StringRef MangledName = getMangledName(GD);
406-
llvm::GlobalValue *Entry = GetGlobalValue(MangledName);
407-
auto *Alias = cast<llvm::GlobalIndirectSymbol>(Entry);
412+
llvm::GlobalValue *Alias = GetGlobalValue(MangledName);
408413
Alias->replaceAllUsesWith(llvm::UndefValue::get(Alias->getType()));
409414
Alias->eraseFromParent();
410415
}

llvm/include/llvm-c/Core.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -1580,10 +1580,10 @@ LLVMTypeRef LLVMX86AMXType(void);
15801580
macro(ConstantVector) \
15811581
macro(GlobalValue) \
15821582
macro(GlobalAlias) \
1583-
macro(GlobalIFunc) \
15841583
macro(GlobalObject) \
15851584
macro(Function) \
15861585
macro(GlobalVariable) \
1586+
macro(GlobalIFunc) \
15871587
macro(UndefValue) \
15881588
macro(PoisonValue) \
15891589
macro(Instruction) \

llvm/include/llvm/AsmParser/LLParser.h

+4-5
Original file line numberDiff line numberDiff line change
@@ -304,11 +304,10 @@ namespace llvm {
304304
unsigned DLLStorageClass, bool DSOLocal,
305305
GlobalVariable::ThreadLocalMode TLM,
306306
GlobalVariable::UnnamedAddr UnnamedAddr);
307-
bool parseIndirectSymbol(const std::string &Name, LocTy NameLoc,
308-
unsigned L, unsigned Visibility,
309-
unsigned DLLStorageClass, bool DSOLocal,
310-
GlobalVariable::ThreadLocalMode TLM,
311-
GlobalVariable::UnnamedAddr UnnamedAddr);
307+
bool parseAliasOrIFunc(const std::string &Name, LocTy NameLoc, unsigned L,
308+
unsigned Visibility, unsigned DLLStorageClass,
309+
bool DSOLocal, GlobalVariable::ThreadLocalMode TLM,
310+
GlobalVariable::UnnamedAddr UnnamedAddr);
312311
bool parseComdat();
313312
bool parseStandaloneMetadata();
314313
bool parseNamedMetadata();

llvm/include/llvm/CodeGen/AsmPrinter.h

+2-3
Original file line numberDiff line numberDiff line change
@@ -41,7 +41,6 @@ class DIEAbbrev;
4141
class DwarfDebug;
4242
class GCMetadataPrinter;
4343
class GCStrategy;
44-
class GlobalIndirectSymbol;
4544
class GlobalObject;
4645
class GlobalValue;
4746
class GlobalVariable;
@@ -795,8 +794,8 @@ class AsmPrinter : public MachineFunctionPass {
795794
void emitModuleCommandLines(Module &M);
796795

797796
GCMetadataPrinter *GetOrCreateGCPrinter(GCStrategy &S);
798-
/// Emit GlobalAlias or GlobalIFunc.
799-
void emitGlobalIndirectSymbol(Module &M, const GlobalIndirectSymbol &GIS);
797+
void emitGlobalAlias(Module &M, const GlobalAlias &GA);
798+
void emitGlobalIFunc(Module &M, const GlobalIFunc &GI);
800799

801800
/// This method decides whether the specified basic block requires a label.
802801
bool shouldEmitLabelForBasicBlock(const MachineBasicBlock &MBB) const;

llvm/include/llvm/IR/GlobalAlias.h

+22-7
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,8 @@
1515
#define LLVM_IR_GLOBALALIAS_H
1616

1717
#include "llvm/ADT/ilist_node.h"
18-
#include "llvm/IR/GlobalIndirectSymbol.h"
18+
#include "llvm/IR/GlobalValue.h"
19+
#include "llvm/IR/OperandTraits.h"
1920
#include "llvm/IR/Value.h"
2021

2122
namespace llvm {
@@ -24,8 +25,7 @@ class Twine;
2425
class Module;
2526
template <typename ValueSubClass> class SymbolTableListTraits;
2627

27-
class GlobalAlias : public GlobalIndirectSymbol,
28-
public ilist_node<GlobalAlias> {
28+
class GlobalAlias : public GlobalValue, public ilist_node<GlobalAlias> {
2929
friend class SymbolTableListTraits<GlobalAlias>;
3030

3131
GlobalAlias(Type *Ty, unsigned AddressSpace, LinkageTypes Linkage,
@@ -58,6 +58,17 @@ class GlobalAlias : public GlobalIndirectSymbol,
5858
// Linkage, Type, Parent and AddressSpace taken from the Aliasee.
5959
static GlobalAlias *create(const Twine &Name, GlobalValue *Aliasee);
6060

61+
// allocate space for exactly one operand
62+
void *operator new(size_t S) { return User::operator new(S, 1); }
63+
void operator delete(void *Ptr) { User::operator delete(Ptr); }
64+
65+
/// Provide fast operand accessors
66+
DECLARE_TRANSPARENT_OPERAND_ACCESSORS(Constant);
67+
68+
void copyAttributesFrom(const GlobalAlias *Src) {
69+
GlobalValue::copyAttributesFrom(Src);
70+
}
71+
6172
/// removeFromParent - This method unlinks 'this' from the containing module,
6273
/// but does not delete it.
6374
///
@@ -71,11 +82,9 @@ class GlobalAlias : public GlobalIndirectSymbol,
7182
/// These methods retrieve and set alias target.
7283
void setAliasee(Constant *Aliasee);
7384
const Constant *getAliasee() const {
74-
return getIndirectSymbol();
75-
}
76-
Constant *getAliasee() {
77-
return getIndirectSymbol();
85+
return static_cast<Constant *>(Op<0>().get());
7886
}
87+
Constant *getAliasee() { return static_cast<Constant *>(Op<0>().get()); }
7988

8089
const GlobalObject *getAliaseeObject() const;
8190
GlobalObject *getAliaseeObject() {
@@ -94,6 +103,12 @@ class GlobalAlias : public GlobalIndirectSymbol,
94103
}
95104
};
96105

106+
template <>
107+
struct OperandTraits<GlobalAlias>
108+
: public FixedNumOperandTraits<GlobalAlias, 1> {};
109+
110+
DEFINE_TRANSPARENT_OPERAND_ACCESSORS(GlobalAlias, Constant)
111+
97112
} // end namespace llvm
98113

99114
#endif // LLVM_IR_GLOBALALIAS_H

llvm/include/llvm/IR/GlobalIFunc.h

+25-7
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,9 @@
1818
#define LLVM_IR_GLOBALIFUNC_H
1919

2020
#include "llvm/ADT/ilist_node.h"
21-
#include "llvm/IR/GlobalIndirectSymbol.h"
21+
#include "llvm/IR/Constant.h"
22+
#include "llvm/IR/GlobalObject.h"
23+
#include "llvm/IR/OperandTraits.h"
2224
#include "llvm/IR/Value.h"
2325

2426
namespace llvm {
@@ -29,8 +31,7 @@ class Module;
2931
// Traits class for using GlobalIFunc in symbol table in Module.
3032
template <typename ValueSubClass> class SymbolTableListTraits;
3133

32-
class GlobalIFunc final : public GlobalIndirectSymbol,
33-
public ilist_node<GlobalIFunc> {
34+
class GlobalIFunc final : public GlobalObject, public ilist_node<GlobalIFunc> {
3435
friend class SymbolTableListTraits<GlobalIFunc>;
3536

3637
GlobalIFunc(Type *Ty, unsigned AddressSpace, LinkageTypes Linkage,
@@ -46,6 +47,17 @@ class GlobalIFunc final : public GlobalIndirectSymbol,
4647
LinkageTypes Linkage, const Twine &Name,
4748
Constant *Resolver, Module *Parent);
4849

50+
// allocate space for exactly one operand
51+
void *operator new(size_t S) { return User::operator new(S, 1); }
52+
void operator delete(void *Ptr) { User::operator delete(Ptr); }
53+
54+
/// Provide fast operand accessors
55+
DECLARE_TRANSPARENT_OPERAND_ACCESSORS(Constant);
56+
57+
void copyAttributesFrom(const GlobalIFunc *Src) {
58+
GlobalObject::copyAttributesFrom(Src);
59+
}
60+
4961
/// This method unlinks 'this' from the containing module, but does not
5062
/// delete it.
5163
void removeFromParent();
@@ -54,11 +66,11 @@ class GlobalIFunc final : public GlobalIndirectSymbol,
5466
void eraseFromParent();
5567

5668
/// These methods retrieve and set ifunc resolver function.
57-
void setResolver(Constant *Resolver) {
58-
setIndirectSymbol(Resolver);
69+
void setResolver(Constant *Resolver) { Op<0>().set(Resolver); }
70+
const Constant *getResolver() const {
71+
return static_cast<Constant *>(Op<0>().get());
5972
}
60-
const Constant *getResolver() const { return getIndirectSymbol(); }
61-
Constant *getResolver() { return getIndirectSymbol(); }
73+
Constant *getResolver() { return static_cast<Constant *>(Op<0>().get()); }
6274

6375
// Return the resolver function after peeling off potential ConstantExpr
6476
// indirection.
@@ -74,6 +86,12 @@ class GlobalIFunc final : public GlobalIndirectSymbol,
7486
}
7587
};
7688

89+
template <>
90+
struct OperandTraits<GlobalIFunc>
91+
: public FixedNumOperandTraits<GlobalIFunc, 1> {};
92+
93+
DEFINE_TRANSPARENT_OPERAND_ACCESSORS(GlobalIFunc, Constant)
94+
7795
} // end namespace llvm
7896

7997
#endif // LLVM_IR_GLOBALIFUNC_H

llvm/include/llvm/IR/GlobalIndirectSymbol.h

-82
This file was deleted.

llvm/include/llvm/IR/GlobalObject.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -153,7 +153,8 @@ class GlobalObject : public GlobalValue {
153153
// Methods for support type inquiry through isa, cast, and dyn_cast:
154154
static bool classof(const Value *V) {
155155
return V->getValueID() == Value::FunctionVal ||
156-
V->getValueID() == Value::GlobalVariableVal;
156+
V->getValueID() == Value::GlobalVariableVal ||
157+
V->getValueID() == Value::GlobalIFuncVal;
157158
}
158159

159160
private:

llvm/include/llvm/IR/Value.h

+3-9
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,6 @@ class DataLayout;
3737
class Function;
3838
class GlobalAlias;
3939
class GlobalIFunc;
40-
class GlobalIndirectSymbol;
4140
class GlobalObject;
4241
class GlobalValue;
4342
class GlobalVariable;
@@ -1016,21 +1015,16 @@ template <> struct isa_impl<GlobalIFunc, Value> {
10161015
}
10171016
};
10181017

1019-
template <> struct isa_impl<GlobalIndirectSymbol, Value> {
1020-
static inline bool doit(const Value &Val) {
1021-
return isa<GlobalAlias>(Val) || isa<GlobalIFunc>(Val);
1022-
}
1023-
};
1024-
10251018
template <> struct isa_impl<GlobalValue, Value> {
10261019
static inline bool doit(const Value &Val) {
1027-
return isa<GlobalObject>(Val) || isa<GlobalIndirectSymbol>(Val);
1020+
return isa<GlobalObject>(Val) || isa<GlobalAlias>(Val);
10281021
}
10291022
};
10301023

10311024
template <> struct isa_impl<GlobalObject, Value> {
10321025
static inline bool doit(const Value &Val) {
1033-
return isa<GlobalVariable>(Val) || isa<Function>(Val);
1026+
return isa<GlobalVariable>(Val) || isa<Function>(Val) ||
1027+
isa<GlobalIFunc>(Val);
10341028
}
10351029
};
10361030

0 commit comments

Comments
 (0)