summaryrefslogtreecommitdiffstats
path: root/llvm/lib/Target/AArch64
diff options
context:
space:
mode:
authorTim Northover <Tim.Northover@arm.com>2013-02-06 16:43:33 +0000
committerTim Northover <Tim.Northover@arm.com>2013-02-06 16:43:33 +0000
commit228d9d3aa25f95c3fae5b38aa58d82e17f465ab8 (patch)
tree88ffd984e91fa56e5289db0dfc3452ac617037ff /llvm/lib/Target/AArch64
parent82b836a61dc4be6a7b64e9df06dddee62ffe39bc (diff)
downloadbcm5719-llvm-228d9d3aa25f95c3fae5b38aa58d82e17f465ab8.tar.gz
bcm5719-llvm-228d9d3aa25f95c3fae5b38aa58d82e17f465ab8.zip
Implement external weak (ELF) symbols on AArch64
Weakly defined symbols should evaluate to 0 if they're undefined at link-time. This is impossible to do with the usual address generation patterns, so we should use a literal pool entry to materlialise the address. llvm-svn: 174518
Diffstat (limited to 'llvm/lib/Target/AArch64')
-rw-r--r--llvm/lib/Target/AArch64/AArch64ISelLowering.cpp15
-rw-r--r--llvm/lib/Target/AArch64/AArch64InstrInfo.td6
2 files changed, 17 insertions, 4 deletions
diff --git a/llvm/lib/Target/AArch64/AArch64ISelLowering.cpp b/llvm/lib/Target/AArch64/AArch64ISelLowering.cpp
index e2e472fa9f8..071b432998b 100644
--- a/llvm/lib/Target/AArch64/AArch64ISelLowering.cpp
+++ b/llvm/lib/Target/AArch64/AArch64ISelLowering.cpp
@@ -1863,7 +1863,7 @@ AArch64TargetLowering::LowerGlobalAddressELF(SDValue Op,
// TableGen doesn't have easy access to the CodeModel or RelocationModel, so
// we make that distinction here.
- // We support the static, small memory model for now.
+ // We support the small memory model for now.
assert(getTargetMachine().getCodeModel() == CodeModel::Small);
EVT PtrVT = getPointerTy();
@@ -1871,6 +1871,18 @@ AArch64TargetLowering::LowerGlobalAddressELF(SDValue Op,
const GlobalAddressSDNode *GN = cast<GlobalAddressSDNode>(Op);
const GlobalValue *GV = GN->getGlobal();
unsigned Alignment = GV->getAlignment();
+ Reloc::Model RelocM = getTargetMachine().getRelocationModel();
+
+ if (GV->isWeakForLinker() && RelocM == Reloc::Static) {
+ // Weak symbols can't use ADRP/ADD pair since they should evaluate to
+ // zero when undefined. In PIC mode the GOT can take care of this, but in
+ // absolute mode we use a constant pool load.
+ return DAG.getLoad(PtrVT, dl, DAG.getEntryNode(),
+ DAG.getConstantPool(GV, GN->getValueType(0)),
+ MachinePointerInfo::getConstantPool(),
+ /*isVolatile=*/ false, /*isNonTemporal=*/ true,
+ /*isInvariant=*/ true, 8);
+ }
if (Alignment == 0) {
const PointerType *GVPtrTy = cast<PointerType>(GV->getType());
@@ -1886,7 +1898,6 @@ AArch64TargetLowering::LowerGlobalAddressELF(SDValue Op,
}
unsigned char HiFixup, LoFixup;
- Reloc::Model RelocM = getTargetMachine().getRelocationModel();
bool UseGOT = Subtarget->GVIsIndirectSymbol(GV, RelocM);
if (UseGOT) {
diff --git a/llvm/lib/Target/AArch64/AArch64InstrInfo.td b/llvm/lib/Target/AArch64/AArch64InstrInfo.td
index 538d4bdb325..a9ff02ae0a1 100644
--- a/llvm/lib/Target/AArch64/AArch64InstrInfo.td
+++ b/llvm/lib/Target/AArch64/AArch64InstrInfo.td
@@ -2409,8 +2409,10 @@ class A64I_LDRlitSimple<bits<2> opc, bit v, RegisterClass OutReg,
"ldr\t$Rt, $Imm19", patterns, NoItinerary>;
let mayLoad = 1 in {
- def LDRw_lit : A64I_LDRlitSimple<0b00, 0b0, GPR32>;
- def LDRx_lit : A64I_LDRlitSimple<0b01, 0b0, GPR64>;
+ def LDRw_lit : A64I_LDRlitSimple<0b00, 0b0, GPR32,
+ [(set (i32 GPR32:$Rt), (load constpool:$Imm19))]>;
+ def LDRx_lit : A64I_LDRlitSimple<0b01, 0b0, GPR64,
+ [(set (i64 GPR64:$Rt), (load constpool:$Imm19))]>;
}
def LDRs_lit : A64I_LDRlitSimple<0b00, 0b1, FPR32,
OpenPOWER on IntegriCloud