blob: 633cd362dc3e3101c2d240621e3341558e5ae080 [file] [log] [blame]
Ulrich Weigand41789de2013-05-23 22:26:41 +00001//===-- PPCMCExpr.cpp - PPC 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#define DEBUG_TYPE "ppcmcexpr"
11#include "PPCMCExpr.h"
12#include "llvm/MC/MCAssembler.h"
13#include "llvm/MC/MCContext.h"
Ulrich Weigand96e65782013-06-20 16:23:52 +000014#include "llvm/MC/MCAsmInfo.h"
Ulrich Weigand41789de2013-05-23 22:26:41 +000015
16using namespace llvm;
17
18const PPCMCExpr*
19PPCMCExpr::Create(VariantKind Kind, const MCExpr *Expr,
Ulrich Weigand266db7f2013-07-08 20:20:51 +000020 bool isDarwin, MCContext &Ctx) {
21 return new (Ctx) PPCMCExpr(Kind, Expr, isDarwin);
Ulrich Weigand41789de2013-05-23 22:26:41 +000022}
23
24void PPCMCExpr::PrintImpl(raw_ostream &OS) const {
Ulrich Weigand96e65782013-06-20 16:23:52 +000025 if (isDarwinSyntax()) {
26 switch (Kind) {
27 default: llvm_unreachable("Invalid kind!");
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000028 case VK_PPC_LO: OS << "lo16"; break;
Ulrich Weigande67c5652013-06-21 14:42:49 +000029 case VK_PPC_HI: OS << "hi16"; break;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000030 case VK_PPC_HA: OS << "ha16"; break;
Ulrich Weigand96e65782013-06-20 16:23:52 +000031 }
Ulrich Weigand41789de2013-05-23 22:26:41 +000032
Ulrich Weigand96e65782013-06-20 16:23:52 +000033 OS << '(';
34 getSubExpr()->print(OS);
35 OS << ')';
36 } else {
37 getSubExpr()->print(OS);
38
39 switch (Kind) {
40 default: llvm_unreachable("Invalid kind!");
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000041 case VK_PPC_LO: OS << "@l"; break;
Ulrich Weigande67c5652013-06-21 14:42:49 +000042 case VK_PPC_HI: OS << "@h"; break;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000043 case VK_PPC_HA: OS << "@ha"; break;
Ulrich Weigande9126f52013-06-21 14:43:42 +000044 case VK_PPC_HIGHER: OS << "@higher"; break;
45 case VK_PPC_HIGHERA: OS << "@highera"; break;
46 case VK_PPC_HIGHEST: OS << "@highest"; break;
47 case VK_PPC_HIGHESTA: OS << "@highesta"; break;
Ulrich Weigand96e65782013-06-20 16:23:52 +000048 }
49 }
Ulrich Weigand41789de2013-05-23 22:26:41 +000050}
51
52bool
53PPCMCExpr::EvaluateAsRelocatableImpl(MCValue &Res,
54 const MCAsmLayout *Layout) const {
55 MCValue Value;
56
Joerg Sonnenbergerb822af42013-08-27 20:23:19 +000057 if (Layout && !getSubExpr()->EvaluateAsRelocatable(Value, *Layout))
Ulrich Weigand41789de2013-05-23 22:26:41 +000058 return false;
59
60 if (Value.isAbsolute()) {
61 int64_t Result = Value.getConstant();
62 switch (Kind) {
63 default:
64 llvm_unreachable("Invalid kind!");
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000065 case VK_PPC_LO:
Ulrich Weigand41789de2013-05-23 22:26:41 +000066 Result = Result & 0xffff;
67 break;
Ulrich Weigande67c5652013-06-21 14:42:49 +000068 case VK_PPC_HI:
69 Result = (Result >> 16) & 0xffff;
70 break;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000071 case VK_PPC_HA:
Ulrich Weigande9126f52013-06-21 14:43:42 +000072 Result = ((Result + 0x8000) >> 16) & 0xffff;
73 break;
74 case VK_PPC_HIGHER:
75 Result = (Result >> 32) & 0xffff;
76 break;
77 case VK_PPC_HIGHERA:
78 Result = ((Result + 0x8000) >> 32) & 0xffff;
79 break;
80 case VK_PPC_HIGHEST:
81 Result = (Result >> 48) & 0xffff;
82 break;
83 case VK_PPC_HIGHESTA:
84 Result = ((Result + 0x8000) >> 48) & 0xffff;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000085 break;
Ulrich Weigand41789de2013-05-23 22:26:41 +000086 }
87 Res = MCValue::get(Result);
Joerg Sonnenbergerb822af42013-08-27 20:23:19 +000088 } else if (Layout) {
Ulrich Weigand41789de2013-05-23 22:26:41 +000089 MCContext &Context = Layout->getAssembler().getContext();
90 const MCSymbolRefExpr *Sym = Value.getSymA();
91 MCSymbolRefExpr::VariantKind Modifier = Sym->getKind();
92 if (Modifier != MCSymbolRefExpr::VK_None)
93 return false;
94 switch (Kind) {
95 default:
96 llvm_unreachable("Invalid kind!");
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000097 case VK_PPC_LO:
98 Modifier = MCSymbolRefExpr::VK_PPC_LO;
Ulrich Weigand41789de2013-05-23 22:26:41 +000099 break;
Ulrich Weigande67c5652013-06-21 14:42:49 +0000100 case VK_PPC_HI:
101 Modifier = MCSymbolRefExpr::VK_PPC_HI;
102 break;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +0000103 case VK_PPC_HA:
104 Modifier = MCSymbolRefExpr::VK_PPC_HA;
Ulrich Weigand41789de2013-05-23 22:26:41 +0000105 break;
Ulrich Weigande9126f52013-06-21 14:43:42 +0000106 case VK_PPC_HIGHERA:
107 Modifier = MCSymbolRefExpr::VK_PPC_HIGHERA;
108 break;
109 case VK_PPC_HIGHER:
110 Modifier = MCSymbolRefExpr::VK_PPC_HIGHER;
111 break;
112 case VK_PPC_HIGHEST:
113 Modifier = MCSymbolRefExpr::VK_PPC_HIGHEST;
114 break;
115 case VK_PPC_HIGHESTA:
116 Modifier = MCSymbolRefExpr::VK_PPC_HIGHESTA;
117 break;
Ulrich Weigand41789de2013-05-23 22:26:41 +0000118 }
119 Sym = MCSymbolRefExpr::Create(&Sym->getSymbol(), Modifier, Context);
120 Res = MCValue::get(Sym, Value.getSymB(), Value.getConstant());
Joerg Sonnenbergerb822af42013-08-27 20:23:19 +0000121 } else
122 return false;
Ulrich Weigand41789de2013-05-23 22:26:41 +0000123
124 return true;
125}
126
127// FIXME: This basically copies MCObjectStreamer::AddValueSymbols. Perhaps
128// that method should be made public?
129static void AddValueSymbols_(const MCExpr *Value, MCAssembler *Asm) {
130 switch (Value->getKind()) {
131 case MCExpr::Target:
132 llvm_unreachable("Can't handle nested target expr!");
133
134 case MCExpr::Constant:
135 break;
136
137 case MCExpr::Binary: {
138 const MCBinaryExpr *BE = cast<MCBinaryExpr>(Value);
139 AddValueSymbols_(BE->getLHS(), Asm);
140 AddValueSymbols_(BE->getRHS(), Asm);
141 break;
142 }
143
144 case MCExpr::SymbolRef:
145 Asm->getOrCreateSymbolData(cast<MCSymbolRefExpr>(Value)->getSymbol());
146 break;
147
148 case MCExpr::Unary:
149 AddValueSymbols_(cast<MCUnaryExpr>(Value)->getSubExpr(), Asm);
150 break;
151 }
152}
153
154void PPCMCExpr::AddValueSymbols(MCAssembler *Asm) const {
155 AddValueSymbols_(getSubExpr(), Asm);
156}