blob: 10d068dc49e61e760cec7d86f66794f6af52a22b [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
Ulrich Weigand41789de2013-05-23 22:26:41 +000010#include "PPCMCExpr.h"
Chandler Carruth8a8cd2b2014-01-07 11:48:04 +000011#include "llvm/MC/MCAsmInfo.h"
Ulrich Weigand41789de2013-05-23 22:26:41 +000012#include "llvm/MC/MCAssembler.h"
13#include "llvm/MC/MCContext.h"
14
15using namespace llvm;
16
Chandler Carruth84e68b22014-04-22 02:41:26 +000017#define DEBUG_TYPE "ppcmcexpr"
18
Ulrich Weigand41789de2013-05-23 22:26:41 +000019const PPCMCExpr*
20PPCMCExpr::Create(VariantKind Kind, const MCExpr *Expr,
Ulrich Weigand266db7f2013-07-08 20:20:51 +000021 bool isDarwin, MCContext &Ctx) {
22 return new (Ctx) PPCMCExpr(Kind, Expr, isDarwin);
Ulrich Weigand41789de2013-05-23 22:26:41 +000023}
24
25void PPCMCExpr::PrintImpl(raw_ostream &OS) const {
Ulrich Weigand96e65782013-06-20 16:23:52 +000026 if (isDarwinSyntax()) {
27 switch (Kind) {
28 default: llvm_unreachable("Invalid kind!");
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000029 case VK_PPC_LO: OS << "lo16"; break;
Ulrich Weigande67c5652013-06-21 14:42:49 +000030 case VK_PPC_HI: OS << "hi16"; break;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000031 case VK_PPC_HA: OS << "ha16"; break;
Ulrich Weigand96e65782013-06-20 16:23:52 +000032 }
Ulrich Weigand41789de2013-05-23 22:26:41 +000033
Ulrich Weigand96e65782013-06-20 16:23:52 +000034 OS << '(';
35 getSubExpr()->print(OS);
36 OS << ')';
37 } else {
38 getSubExpr()->print(OS);
39
40 switch (Kind) {
41 default: llvm_unreachable("Invalid kind!");
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000042 case VK_PPC_LO: OS << "@l"; break;
Ulrich Weigande67c5652013-06-21 14:42:49 +000043 case VK_PPC_HI: OS << "@h"; break;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000044 case VK_PPC_HA: OS << "@ha"; break;
Ulrich Weigande9126f52013-06-21 14:43:42 +000045 case VK_PPC_HIGHER: OS << "@higher"; break;
46 case VK_PPC_HIGHERA: OS << "@highera"; break;
47 case VK_PPC_HIGHEST: OS << "@highest"; break;
48 case VK_PPC_HIGHESTA: OS << "@highesta"; break;
Ulrich Weigand96e65782013-06-20 16:23:52 +000049 }
50 }
Ulrich Weigand41789de2013-05-23 22:26:41 +000051}
52
53bool
54PPCMCExpr::EvaluateAsRelocatableImpl(MCValue &Res,
55 const MCAsmLayout *Layout) const {
56 MCValue Value;
57
Rafael Espindola3d5d4642014-03-12 16:55:59 +000058 if (!getSubExpr()->EvaluateAsRelocatable(Value, Layout))
Ulrich Weigand41789de2013-05-23 22:26:41 +000059 return false;
60
61 if (Value.isAbsolute()) {
62 int64_t Result = Value.getConstant();
63 switch (Kind) {
64 default:
65 llvm_unreachable("Invalid kind!");
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000066 case VK_PPC_LO:
Ulrich Weigand41789de2013-05-23 22:26:41 +000067 Result = Result & 0xffff;
68 break;
Ulrich Weigande67c5652013-06-21 14:42:49 +000069 case VK_PPC_HI:
70 Result = (Result >> 16) & 0xffff;
71 break;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000072 case VK_PPC_HA:
Ulrich Weigande9126f52013-06-21 14:43:42 +000073 Result = ((Result + 0x8000) >> 16) & 0xffff;
74 break;
75 case VK_PPC_HIGHER:
76 Result = (Result >> 32) & 0xffff;
77 break;
78 case VK_PPC_HIGHERA:
79 Result = ((Result + 0x8000) >> 32) & 0xffff;
80 break;
81 case VK_PPC_HIGHEST:
82 Result = (Result >> 48) & 0xffff;
83 break;
84 case VK_PPC_HIGHESTA:
85 Result = ((Result + 0x8000) >> 48) & 0xffff;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000086 break;
Ulrich Weigand41789de2013-05-23 22:26:41 +000087 }
88 Res = MCValue::get(Result);
Alexey Samsonova3a037d2013-08-28 08:30:47 +000089 } else {
Rafael Espindola3d5d4642014-03-12 16:55:59 +000090 if (!Layout)
91 return false;
92
Ulrich Weigand41789de2013-05-23 22:26:41 +000093 MCContext &Context = Layout->getAssembler().getContext();
94 const MCSymbolRefExpr *Sym = Value.getSymA();
95 MCSymbolRefExpr::VariantKind Modifier = Sym->getKind();
96 if (Modifier != MCSymbolRefExpr::VK_None)
97 return false;
98 switch (Kind) {
99 default:
100 llvm_unreachable("Invalid kind!");
Ulrich Weigandd51c09f2013-06-21 14:42:20 +0000101 case VK_PPC_LO:
102 Modifier = MCSymbolRefExpr::VK_PPC_LO;
Ulrich Weigand41789de2013-05-23 22:26:41 +0000103 break;
Ulrich Weigande67c5652013-06-21 14:42:49 +0000104 case VK_PPC_HI:
105 Modifier = MCSymbolRefExpr::VK_PPC_HI;
106 break;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +0000107 case VK_PPC_HA:
108 Modifier = MCSymbolRefExpr::VK_PPC_HA;
Ulrich Weigand41789de2013-05-23 22:26:41 +0000109 break;
Ulrich Weigande9126f52013-06-21 14:43:42 +0000110 case VK_PPC_HIGHERA:
111 Modifier = MCSymbolRefExpr::VK_PPC_HIGHERA;
112 break;
113 case VK_PPC_HIGHER:
114 Modifier = MCSymbolRefExpr::VK_PPC_HIGHER;
115 break;
116 case VK_PPC_HIGHEST:
117 Modifier = MCSymbolRefExpr::VK_PPC_HIGHEST;
118 break;
119 case VK_PPC_HIGHESTA:
120 Modifier = MCSymbolRefExpr::VK_PPC_HIGHESTA;
121 break;
Ulrich Weigand41789de2013-05-23 22:26:41 +0000122 }
123 Sym = MCSymbolRefExpr::Create(&Sym->getSymbol(), Modifier, Context);
124 Res = MCValue::get(Sym, Value.getSymB(), Value.getConstant());
Alexey Samsonova3a037d2013-08-28 08:30:47 +0000125 }
Ulrich Weigand41789de2013-05-23 22:26:41 +0000126
127 return true;
128}
129
130// FIXME: This basically copies MCObjectStreamer::AddValueSymbols. Perhaps
131// that method should be made public?
132static void AddValueSymbols_(const MCExpr *Value, MCAssembler *Asm) {
133 switch (Value->getKind()) {
134 case MCExpr::Target:
135 llvm_unreachable("Can't handle nested target expr!");
136
137 case MCExpr::Constant:
138 break;
139
140 case MCExpr::Binary: {
141 const MCBinaryExpr *BE = cast<MCBinaryExpr>(Value);
142 AddValueSymbols_(BE->getLHS(), Asm);
143 AddValueSymbols_(BE->getRHS(), Asm);
144 break;
145 }
146
147 case MCExpr::SymbolRef:
148 Asm->getOrCreateSymbolData(cast<MCSymbolRefExpr>(Value)->getSymbol());
149 break;
150
151 case MCExpr::Unary:
152 AddValueSymbols_(cast<MCUnaryExpr>(Value)->getSubExpr(), Asm);
153 break;
154 }
155}
156
157void PPCMCExpr::AddValueSymbols(MCAssembler *Asm) const {
158 AddValueSymbols_(getSubExpr(), Asm);
159}