2014-02-05 02:41:57 +08:00
|
|
|
//===-- MipsMCExpr.cpp - Mips specific MC expression classes --------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "MipsMCExpr.h"
|
2014-03-04 18:07:28 +08:00
|
|
|
#include "llvm/MC/MCAsmInfo.h"
|
2014-02-05 02:41:57 +08:00
|
|
|
#include "llvm/MC/MCAssembler.h"
|
|
|
|
#include "llvm/MC/MCContext.h"
|
2014-06-25 23:29:54 +08:00
|
|
|
#include "llvm/MC/MCObjectStreamer.h"
|
2014-02-05 02:41:57 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
2014-04-22 10:41:26 +08:00
|
|
|
#define DEBUG_TYPE "mipsmcexpr"
|
|
|
|
|
2014-04-03 18:37:45 +08:00
|
|
|
bool MipsMCExpr::isSupportedBinaryExpr(MCSymbolRefExpr::VariantKind VK,
|
|
|
|
const MCBinaryExpr *BE) {
|
|
|
|
switch (VK) {
|
|
|
|
case MCSymbolRefExpr::VK_Mips_ABS_LO:
|
|
|
|
case MCSymbolRefExpr::VK_Mips_ABS_HI:
|
|
|
|
case MCSymbolRefExpr::VK_Mips_HIGHER:
|
|
|
|
case MCSymbolRefExpr::VK_Mips_HIGHEST:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We support expressions of the form "(sym1 binop1 sym2) binop2 const",
|
|
|
|
// where "binop2 const" is optional.
|
|
|
|
if (isa<MCBinaryExpr>(BE->getLHS())) {
|
|
|
|
if (!isa<MCConstantExpr>(BE->getRHS()))
|
|
|
|
return false;
|
|
|
|
BE = cast<MCBinaryExpr>(BE->getLHS());
|
|
|
|
}
|
|
|
|
return (isa<MCSymbolRefExpr>(BE->getLHS())
|
|
|
|
&& isa<MCSymbolRefExpr>(BE->getRHS()));
|
|
|
|
}
|
|
|
|
|
2014-02-05 02:41:57 +08:00
|
|
|
const MipsMCExpr*
|
2014-04-03 18:37:45 +08:00
|
|
|
MipsMCExpr::Create(MCSymbolRefExpr::VariantKind VK, const MCExpr *Expr,
|
2014-02-05 02:41:57 +08:00
|
|
|
MCContext &Ctx) {
|
2014-04-03 18:37:45 +08:00
|
|
|
VariantKind Kind;
|
|
|
|
switch (VK) {
|
|
|
|
case MCSymbolRefExpr::VK_Mips_ABS_LO:
|
|
|
|
Kind = VK_Mips_LO;
|
|
|
|
break;
|
|
|
|
case MCSymbolRefExpr::VK_Mips_ABS_HI:
|
|
|
|
Kind = VK_Mips_HI;
|
|
|
|
break;
|
|
|
|
case MCSymbolRefExpr::VK_Mips_HIGHER:
|
|
|
|
Kind = VK_Mips_HIGHER;
|
|
|
|
break;
|
|
|
|
case MCSymbolRefExpr::VK_Mips_HIGHEST:
|
|
|
|
Kind = VK_Mips_HIGHEST;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
llvm_unreachable("Invalid kind!");
|
|
|
|
}
|
|
|
|
|
2014-02-05 02:41:57 +08:00
|
|
|
return new (Ctx) MipsMCExpr(Kind, Expr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MipsMCExpr::PrintImpl(raw_ostream &OS) const {
|
|
|
|
switch (Kind) {
|
|
|
|
default: llvm_unreachable("Invalid kind!");
|
2014-04-03 18:37:45 +08:00
|
|
|
case VK_Mips_LO: OS << "%lo"; break;
|
|
|
|
case VK_Mips_HI: OS << "%hi"; break;
|
|
|
|
case VK_Mips_HIGHER: OS << "%higher"; break;
|
|
|
|
case VK_Mips_HIGHEST: OS << "%highest"; break;
|
2014-02-05 02:41:57 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
OS << '(';
|
|
|
|
Expr->print(OS);
|
|
|
|
OS << ')';
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
MipsMCExpr::EvaluateAsRelocatableImpl(MCValue &Res,
|
2014-08-10 19:35:12 +08:00
|
|
|
const MCAsmLayout *Layout,
|
|
|
|
const MCFixup *Fixup) const {
|
|
|
|
return getSubExpr()->EvaluateAsRelocatable(Res, Layout, Fixup);
|
2014-02-05 02:41:57 +08:00
|
|
|
}
|
|
|
|
|
2014-06-25 23:45:33 +08:00
|
|
|
void MipsMCExpr::visitUsedExpr(MCStreamer &Streamer) const {
|
2014-06-25 23:29:54 +08:00
|
|
|
Streamer.visitUsedExpr(*getSubExpr());
|
2014-02-05 02:41:57 +08:00
|
|
|
}
|