mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-13 04:30:23 +00:00
If the global variable is removed by the linker, then don't constant merge it
with other symbols. An object in the __cfstring section is suppoed to be filled with CFString objects, which have a pointer to ___CFConstantStringClassReference followed by a pointer to a __cstring. If we allow the object in the __cstring section to be merged with another global, then it could end up in any section. Because the linker is going to remove these symbols in the final executable, we shouldn't bother to merge them. <rdar://problem/10564621> git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@147899 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
dcc296d420
commit
37b94c6b4e
@ -192,6 +192,14 @@ public:
|
|||||||
Linkage == LinkerPrivateWeakDefAutoLinkage;
|
Linkage == LinkerPrivateWeakDefAutoLinkage;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// mayBeRemovedByLinker - Whether the definition of this global may be
|
||||||
|
/// removed at link time.
|
||||||
|
static bool mayBeRemovedByLinker(LinkageTypes Linkage) {
|
||||||
|
return isLinkerPrivateLinkage(Linkage) ||
|
||||||
|
isLinkerPrivateWeakLinkage(Linkage) ||
|
||||||
|
isLinkerPrivateWeakDefAutoLinkage(Linkage);
|
||||||
|
}
|
||||||
|
|
||||||
bool hasExternalLinkage() const { return isExternalLinkage(Linkage); }
|
bool hasExternalLinkage() const { return isExternalLinkage(Linkage); }
|
||||||
bool hasAvailableExternallyLinkage() const {
|
bool hasAvailableExternallyLinkage() const {
|
||||||
return isAvailableExternallyLinkage(Linkage);
|
return isAvailableExternallyLinkage(Linkage);
|
||||||
@ -225,6 +233,8 @@ public:
|
|||||||
|
|
||||||
bool isWeakForLinker() const { return isWeakForLinker(Linkage); }
|
bool isWeakForLinker() const { return isWeakForLinker(Linkage); }
|
||||||
|
|
||||||
|
bool mayBeRemovedByLinker() const { return mayBeRemovedByLinker(Linkage); }
|
||||||
|
|
||||||
/// copyAttributesFrom - copy all additional attributes (those not needed to
|
/// copyAttributesFrom - copy all additional attributes (those not needed to
|
||||||
/// create a GlobalValue) from the GlobalValue Src to this one.
|
/// create a GlobalValue) from the GlobalValue Src to this one.
|
||||||
virtual void copyAttributesFrom(const GlobalValue *Src);
|
virtual void copyAttributesFrom(const GlobalValue *Src);
|
||||||
|
@ -140,18 +140,21 @@ bool ConstantMerge::runOnModule(Module &M) {
|
|||||||
UsedGlobals.count(GV))
|
UsedGlobals.count(GV))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
// Ignore any constants which may be removed by the linker.
|
||||||
|
if (GV->mayBeRemovedByLinker())
|
||||||
|
continue;
|
||||||
|
|
||||||
Constant *Init = GV->getInitializer();
|
Constant *Init = GV->getInitializer();
|
||||||
|
|
||||||
// Check to see if the initializer is already known.
|
// Check to see if the initializer is already known.
|
||||||
PointerIntPair<Constant*, 1, bool> Pair(Init, hasKnownAlignment(GV));
|
PointerIntPair<Constant*, 1, bool> Pair(Init, hasKnownAlignment(GV));
|
||||||
GlobalVariable *&Slot = CMap[Pair];
|
GlobalVariable *&Slot = CMap[Pair];
|
||||||
|
|
||||||
// If this is the first constant we find or if the old on is local,
|
// If this is the first constant we find or if the old one is local,
|
||||||
// replace with the current one. It the current is externally visible
|
// replace with the current one. If the current is externally visible
|
||||||
// it cannot be replace, but can be the canonical constant we merge with.
|
// it cannot be replace, but can be the canonical constant we merge with.
|
||||||
if (Slot == 0 || IsBetterCannonical(*GV, *Slot)) {
|
if (Slot == 0 || IsBetterCannonical(*GV, *Slot))
|
||||||
Slot = GV;
|
Slot = GV;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Second: identify all globals that can be merged together, filling in
|
// Second: identify all globals that can be merged together, filling in
|
||||||
@ -169,8 +172,9 @@ bool ConstantMerge::runOnModule(Module &M) {
|
|||||||
UsedGlobals.count(GV))
|
UsedGlobals.count(GV))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
// We can only replace constant with local linkage.
|
// We can only replace constants with local linkage and which aren't
|
||||||
if (!GV->hasLocalLinkage())
|
// removed by the linker.
|
||||||
|
if (!GV->hasLocalLinkage() || GV->mayBeRemovedByLinker())
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
Constant *Init = GV->getInitializer();
|
Constant *Init = GV->getInitializer();
|
||||||
|
23
test/Transforms/ConstantMerge/linker-private.ll
Normal file
23
test/Transforms/ConstantMerge/linker-private.ll
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
; RUN: opt < %s -constmerge -S | FileCheck %s
|
||||||
|
; <rdar://problem/10564621>
|
||||||
|
|
||||||
|
%0 = type opaque
|
||||||
|
%struct.NSConstantString = type { i32*, i32, i8*, i32 }
|
||||||
|
|
||||||
|
; CHECK: @.str3 = linker_private unnamed_addr constant [1 x i8] zeroinitializer, align 1
|
||||||
|
|
||||||
|
@isLogVisible = global i8 0, align 1
|
||||||
|
@__CFConstantStringClassReference = external global [0 x i32]
|
||||||
|
@.str3 = linker_private unnamed_addr constant [1 x i8] zeroinitializer, align 1
|
||||||
|
@_unnamed_cfstring_4 = private constant %struct.NSConstantString { i32* getelementptr inbounds ([0 x i32]* @__CFConstantStringClassReference, i32 0, i32 0), i32 1992, i8* getelementptr inbounds ([1 x i8]* @.str3, i32 0, i32 0), i32 0 }, section "__DATA,__cfstring"
|
||||||
|
@null.array = weak_odr constant [1 x i8] zeroinitializer, align 1
|
||||||
|
|
||||||
|
define linkonce_odr void @bar() nounwind ssp align 2 {
|
||||||
|
entry:
|
||||||
|
%stack = alloca i8*, align 4
|
||||||
|
%call = call %0* bitcast (i8* (i8*, i8*, ...)* @objc_msgSend to %0* (i8*, i8*, %0*)*)(i8* null, i8* null, %0* bitcast (%struct.NSConstantString* @_unnamed_cfstring_4 to %0*))
|
||||||
|
store i8* getelementptr inbounds ([1 x i8]* @null.array, i32 0, i32 0), i8** %stack, align 4
|
||||||
|
ret void
|
||||||
|
}
|
||||||
|
|
||||||
|
declare i8* @objc_msgSend(i8*, i8*, ...) nonlazybind
|
Loading…
Reference in New Issue
Block a user