summaryrefslogtreecommitdiffstats
path: root/llvm/lib/MC/MCELFStreamer.cpp
diff options
context:
space:
mode:
authorRafael Espindola <rafael.espindola@gmail.com>2010-11-24 02:19:40 +0000
committerRafael Espindola <rafael.espindola@gmail.com>2010-11-24 02:19:40 +0000
commit4e70ac7b684f8188ddafe83b8b5d02dc115d670d (patch)
tree316532e895ded10e91d26f68512d79d3466e6553 /llvm/lib/MC/MCELFStreamer.cpp
parent8e82c6372ad069056e5fa12602e0a8bcc293d383 (diff)
downloadbcm5719-llvm-4e70ac7b684f8188ddafe83b8b5d02dc115d670d.tar.gz
bcm5719-llvm-4e70ac7b684f8188ddafe83b8b5d02dc115d670d.zip
If a symbol is used as tls, mark it as tls even if not declare as so. Probably
fixes PR8659. llvm-svn: 120076
Diffstat (limited to 'llvm/lib/MC/MCELFStreamer.cpp')
-rw-r--r--llvm/lib/MC/MCELFStreamer.cpp40
1 files changed, 40 insertions, 0 deletions
diff --git a/llvm/lib/MC/MCELFStreamer.cpp b/llvm/lib/MC/MCELFStreamer.cpp
index a3c716bdcb7..1eddda9fefc 100644
--- a/llvm/lib/MC/MCELFStreamer.cpp
+++ b/llvm/lib/MC/MCELFStreamer.cpp
@@ -144,6 +144,8 @@ private:
virtual void EmitInstToFragment(const MCInst &Inst);
virtual void EmitInstToData(const MCInst &Inst);
+ void fixSymbolsInTLSFixups(const MCExpr *expr);
+
struct LocalCommon {
MCSymbolData *SD;
uint64_t Size;
@@ -450,6 +452,38 @@ void MCELFStreamer::EmitFileDirective(StringRef Filename) {
SD.setFlags(ELF_STT_File | ELF_STB_Local | ELF_STV_Default);
}
+void MCELFStreamer::fixSymbolsInTLSFixups(const MCExpr *expr) {
+ switch (expr->getKind()) {
+ case MCExpr::Target: llvm_unreachable("Can't handle target exprs yet!");
+ case MCExpr::Constant:
+ break;
+
+ case MCExpr::Binary: {
+ const MCBinaryExpr *be = cast<MCBinaryExpr>(expr);
+ fixSymbolsInTLSFixups(be->getLHS());
+ fixSymbolsInTLSFixups(be->getRHS());
+ break;
+ }
+
+ case MCExpr::SymbolRef: {
+ const MCSymbolRefExpr &symRef = *cast<MCSymbolRefExpr>(expr);
+ MCSymbolRefExpr::VariantKind kind = symRef.getKind();
+ if (kind != MCSymbolRefExpr::VK_TLSGD &&
+ kind != MCSymbolRefExpr::VK_TLSLD &&
+ kind != MCSymbolRefExpr::VK_TLSLDM &&
+ kind != MCSymbolRefExpr::VK_ARM_TLSGD)
+ return;
+ MCSymbolData &SD = getAssembler().getOrCreateSymbolData(symRef.getSymbol());
+ SetType(SD, ELF::STT_TLS);
+ break;
+ }
+
+ case MCExpr::Unary:
+ fixSymbolsInTLSFixups(cast<MCUnaryExpr>(expr)->getSubExpr());
+ break;
+ }
+}
+
void MCELFStreamer::EmitInstToFragment(const MCInst &Inst) {
MCInstFragment *IF = new MCInstFragment(Inst, getCurrentSectionData());
@@ -463,6 +497,9 @@ void MCELFStreamer::EmitInstToFragment(const MCInst &Inst) {
getAssembler().getEmitter().EncodeInstruction(Inst, VecOS, Fixups);
VecOS.flush();
+ for (unsigned i = 0, e = Fixups.size(); i != e; ++i)
+ fixSymbolsInTLSFixups(Fixups[i].getValue());
+
IF->getCode() = Code;
IF->getFixups() = Fixups;
}
@@ -476,6 +513,9 @@ void MCELFStreamer::EmitInstToData(const MCInst &Inst) {
getAssembler().getEmitter().EncodeInstruction(Inst, VecOS, Fixups);
VecOS.flush();
+ for (unsigned i = 0, e = Fixups.size(); i != e; ++i)
+ fixSymbolsInTLSFixups(Fixups[i].getValue());
+
// Add the fixups and data.
for (unsigned i = 0, e = Fixups.size(); i != e; ++i) {
Fixups[i].setOffset(Fixups[i].getOffset() + DF->getContents().size());
OpenPOWER on IntegriCloud