blob: bc395b97a996b23c852f88faca16ea8abaaea770 [file] [log] [blame]
Venkatraman Govindaraju08bcf292013-12-26 00:01:52 +00001//===-- SparcMCExpr.cpp - Sparc specific MC expression classes --------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file contains the implementation of the assembly expression modifiers
11// accepted by the Sparc architecture (e.g. "%hi", "%lo", ...).
12//
13//===----------------------------------------------------------------------===//
14
15#define DEBUG_TYPE "sparcmcexpr"
16#include "SparcMCExpr.h"
17#include "llvm/MC/MCContext.h"
18#include "llvm/MC/MCAssembler.h"
19#include "llvm/MC/MCELF.h"
20#include "llvm/Object/ELF.h"
21
22
23using namespace llvm;
24
25const SparcMCExpr*
26SparcMCExpr::Create(VariantKind Kind, const MCExpr *Expr,
27 MCContext &Ctx) {
28 return new (Ctx) SparcMCExpr(Kind, Expr);
29}
30
31
32void SparcMCExpr::PrintImpl(raw_ostream &OS) const
33{
34 bool closeParen = true;
35 switch (Kind) {
36 case VK_Sparc_None: closeParen = false; break;
37 case VK_Sparc_LO: OS << "%lo("; break;
38 case VK_Sparc_HI: OS << "%hi("; break;
39 case VK_Sparc_H44: OS << "%h44("; break;
40 case VK_Sparc_M44: OS << "%m44("; break;
41 case VK_Sparc_L44: OS << "%l44("; break;
42 case VK_Sparc_HH: OS << "%hh("; break;
43 case VK_Sparc_HM: OS << "%hm("; break;
44 case VK_Sparc_TLS_GD_HI22: OS << "%tgd_hi22("; break;
45 case VK_Sparc_TLS_GD_LO10: OS << "%tgd_lo10("; break;
46 case VK_Sparc_TLS_GD_ADD: OS << "%tgd_add("; break;
47 case VK_Sparc_TLS_GD_CALL: OS << "%tgd_call("; break;
48 case VK_Sparc_TLS_LDM_HI22: OS << "%tldm_hi22("; break;
49 case VK_Sparc_TLS_LDM_LO10: OS << "%tldm_lo10("; break;
50 case VK_Sparc_TLS_LDM_ADD: OS << "%tldm_add("; break;
51 case VK_Sparc_TLS_LDM_CALL: OS << "%tldm_call("; break;
52 case VK_Sparc_TLS_LDO_HIX22: OS << "%tldo_hix22("; break;
53 case VK_Sparc_TLS_LDO_LOX10: OS << "%tldo_lox10("; break;
54 case VK_Sparc_TLS_LDO_ADD: OS << "%tldo_add("; break;
55 case VK_Sparc_TLS_IE_HI22: OS << "%tie_hi22("; break;
56 case VK_Sparc_TLS_IE_LO10: OS << "%tie_lo10("; break;
57 case VK_Sparc_TLS_IE_LD: OS << "%tie_ld("; break;
58 case VK_Sparc_TLS_IE_LDX: OS << "%tie_ldx("; break;
59 case VK_Sparc_TLS_IE_ADD: OS << "%tie_add("; break;
60 case VK_Sparc_TLS_LE_HIX22: OS << "%tle_hix22("; break;
61 case VK_Sparc_TLS_LE_LOX10: OS << "%tle_lox10("; break;
62 }
63
64 const MCExpr *Expr = getSubExpr();
65 Expr->print(OS);
66 if (closeParen)
67 OS << ')';
68}
69
70bool
71SparcMCExpr::EvaluateAsRelocatableImpl(MCValue &Res,
72 const MCAsmLayout *Layout) const {
Venkatraman Govindaraju08bcf292013-12-26 00:01:52 +000073 return getSubExpr()->EvaluateAsRelocatable(Res, *Layout);
74}
75
Venkatraman Govindarajub73aeca2014-01-06 01:22:54 +000076static void fixELFSymbolsInTLSFixupsImpl(const MCExpr *Expr, MCAssembler &Asm) {
77 assert(0 && "Implement fixELFSymbolsInTLSFixupsImpl!");
78}
Venkatraman Govindaraju08bcf292013-12-26 00:01:52 +000079
80void SparcMCExpr::fixELFSymbolsInTLSFixups(MCAssembler &Asm) const {
Venkatraman Govindarajub73aeca2014-01-06 01:22:54 +000081 switch(getKind()) {
82 default: return;
83 case VK_Sparc_TLS_GD_HI22:
84 case VK_Sparc_TLS_GD_LO10:
85 case VK_Sparc_TLS_GD_ADD:
86 case VK_Sparc_TLS_GD_CALL:
87 case VK_Sparc_TLS_LDM_HI22:
88 case VK_Sparc_TLS_LDM_LO10:
89 case VK_Sparc_TLS_LDM_ADD:
90 case VK_Sparc_TLS_LDM_CALL:
91 case VK_Sparc_TLS_LDO_HIX22:
92 case VK_Sparc_TLS_LDO_LOX10:
93 case VK_Sparc_TLS_LDO_ADD:
94 case VK_Sparc_TLS_IE_HI22:
95 case VK_Sparc_TLS_IE_LO10:
96 case VK_Sparc_TLS_IE_LD:
97 case VK_Sparc_TLS_IE_LDX:
98 case VK_Sparc_TLS_IE_ADD:
99 case VK_Sparc_TLS_LE_HIX22:
100 case VK_Sparc_TLS_LE_LOX10: break;
101 }
102 fixELFSymbolsInTLSFixupsImpl(getSubExpr(), Asm);
103}
104
105// FIXME: This basically copies MCObjectStreamer::AddValueSymbols. Perhaps
106// that method should be made public?
107// FIXME: really do above: now that at least three other backends are using it.
108static void AddValueSymbolsImpl(const MCExpr *Value, MCAssembler *Asm) {
109 switch (Value->getKind()) {
110 case MCExpr::Target:
111 llvm_unreachable("Can't handle nested target expr!");
112 break;
113
114 case MCExpr::Constant:
115 break;
116
117 case MCExpr::Binary: {
118 const MCBinaryExpr *BE = cast<MCBinaryExpr>(Value);
119 AddValueSymbolsImpl(BE->getLHS(), Asm);
120 AddValueSymbolsImpl(BE->getRHS(), Asm);
121 break;
122 }
123
124 case MCExpr::SymbolRef:
125 Asm->getOrCreateSymbolData(cast<MCSymbolRefExpr>(Value)->getSymbol());
126 break;
127
128 case MCExpr::Unary:
129 AddValueSymbolsImpl(cast<MCUnaryExpr>(Value)->getSubExpr(), Asm);
130 break;
131 }
Venkatraman Govindaraju08bcf292013-12-26 00:01:52 +0000132}
133
134void SparcMCExpr::AddValueSymbols(MCAssembler *Asm) const {
Venkatraman Govindarajub73aeca2014-01-06 01:22:54 +0000135 AddValueSymbolsImpl(getSubExpr(), Asm);
Venkatraman Govindaraju08bcf292013-12-26 00:01:52 +0000136}