blob: 8a346b422d2d9f18a169ca39c6ec3c64484823db [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,
20 MCContext &Ctx) {
Ulrich Weigand96e65782013-06-20 16:23:52 +000021 int AssemblerDialect = Ctx.getAsmInfo()->getAssemblerDialect();
22 return new (Ctx) PPCMCExpr(Kind, Expr, AssemblerDialect);
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 Weigand96e65782013-06-20 16:23:52 +000045 }
46 }
Ulrich Weigand41789de2013-05-23 22:26:41 +000047}
48
49bool
50PPCMCExpr::EvaluateAsRelocatableImpl(MCValue &Res,
51 const MCAsmLayout *Layout) const {
52 MCValue Value;
53
54 if (!getSubExpr()->EvaluateAsRelocatable(Value, *Layout))
55 return false;
56
57 if (Value.isAbsolute()) {
58 int64_t Result = Value.getConstant();
59 switch (Kind) {
60 default:
61 llvm_unreachable("Invalid kind!");
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000062 case VK_PPC_LO:
Ulrich Weigand41789de2013-05-23 22:26:41 +000063 Result = Result & 0xffff;
64 break;
Ulrich Weigande67c5652013-06-21 14:42:49 +000065 case VK_PPC_HI:
66 Result = (Result >> 16) & 0xffff;
67 break;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000068 case VK_PPC_HA:
69 Result = ((Result >> 16) + ((Result & 0x8000) ? 1 : 0)) & 0xffff;
70 break;
Ulrich Weigand41789de2013-05-23 22:26:41 +000071 }
72 Res = MCValue::get(Result);
73 } else {
74 MCContext &Context = Layout->getAssembler().getContext();
75 const MCSymbolRefExpr *Sym = Value.getSymA();
76 MCSymbolRefExpr::VariantKind Modifier = Sym->getKind();
77 if (Modifier != MCSymbolRefExpr::VK_None)
78 return false;
79 switch (Kind) {
80 default:
81 llvm_unreachable("Invalid kind!");
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000082 case VK_PPC_LO:
83 Modifier = MCSymbolRefExpr::VK_PPC_LO;
Ulrich Weigand41789de2013-05-23 22:26:41 +000084 break;
Ulrich Weigande67c5652013-06-21 14:42:49 +000085 case VK_PPC_HI:
86 Modifier = MCSymbolRefExpr::VK_PPC_HI;
87 break;
Ulrich Weigandd51c09f2013-06-21 14:42:20 +000088 case VK_PPC_HA:
89 Modifier = MCSymbolRefExpr::VK_PPC_HA;
Ulrich Weigand41789de2013-05-23 22:26:41 +000090 break;
91 }
92 Sym = MCSymbolRefExpr::Create(&Sym->getSymbol(), Modifier, Context);
93 Res = MCValue::get(Sym, Value.getSymB(), Value.getConstant());
94 }
95
96 return true;
97}
98
99// FIXME: This basically copies MCObjectStreamer::AddValueSymbols. Perhaps
100// that method should be made public?
101static void AddValueSymbols_(const MCExpr *Value, MCAssembler *Asm) {
102 switch (Value->getKind()) {
103 case MCExpr::Target:
104 llvm_unreachable("Can't handle nested target expr!");
105
106 case MCExpr::Constant:
107 break;
108
109 case MCExpr::Binary: {
110 const MCBinaryExpr *BE = cast<MCBinaryExpr>(Value);
111 AddValueSymbols_(BE->getLHS(), Asm);
112 AddValueSymbols_(BE->getRHS(), Asm);
113 break;
114 }
115
116 case MCExpr::SymbolRef:
117 Asm->getOrCreateSymbolData(cast<MCSymbolRefExpr>(Value)->getSymbol());
118 break;
119
120 case MCExpr::Unary:
121 AddValueSymbols_(cast<MCUnaryExpr>(Value)->getSubExpr(), Asm);
122 break;
123 }
124}
125
126void PPCMCExpr::AddValueSymbols(MCAssembler *Asm) const {
127 AddValueSymbols_(getSubExpr(), Asm);
128}