1*9880d681SAndroid Build Coastguard Worker //===-- HexagonAsmParser.cpp - Parse Hexagon asm to MCInst instructions----===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker // The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker
10*9880d681SAndroid Build Coastguard Worker #define DEBUG_TYPE "mcasmparser"
11*9880d681SAndroid Build Coastguard Worker
12*9880d681SAndroid Build Coastguard Worker #include "Hexagon.h"
13*9880d681SAndroid Build Coastguard Worker #include "HexagonRegisterInfo.h"
14*9880d681SAndroid Build Coastguard Worker #include "HexagonTargetStreamer.h"
15*9880d681SAndroid Build Coastguard Worker #include "MCTargetDesc/HexagonBaseInfo.h"
16*9880d681SAndroid Build Coastguard Worker #include "MCTargetDesc/HexagonMCAsmInfo.h"
17*9880d681SAndroid Build Coastguard Worker #include "MCTargetDesc/HexagonMCChecker.h"
18*9880d681SAndroid Build Coastguard Worker #include "MCTargetDesc/HexagonMCELFStreamer.h"
19*9880d681SAndroid Build Coastguard Worker #include "MCTargetDesc/HexagonMCExpr.h"
20*9880d681SAndroid Build Coastguard Worker #include "MCTargetDesc/HexagonMCShuffler.h"
21*9880d681SAndroid Build Coastguard Worker #include "MCTargetDesc/HexagonMCTargetDesc.h"
22*9880d681SAndroid Build Coastguard Worker #include "MCTargetDesc/HexagonShuffler.h"
23*9880d681SAndroid Build Coastguard Worker #include "llvm/ADT/SmallVector.h"
24*9880d681SAndroid Build Coastguard Worker #include "llvm/ADT/StringExtras.h"
25*9880d681SAndroid Build Coastguard Worker #include "llvm/ADT/Twine.h"
26*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCContext.h"
27*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCELFStreamer.h"
28*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCExpr.h"
29*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCInst.h"
30*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCParser/MCAsmLexer.h"
31*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCParser/MCAsmParser.h"
32*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCParser/MCParsedAsmOperand.h"
33*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCParser/MCTargetAsmParser.h"
34*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCSectionELF.h"
35*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCStreamer.h"
36*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCSubtargetInfo.h"
37*9880d681SAndroid Build Coastguard Worker #include "llvm/MC/MCValue.h"
38*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/CommandLine.h"
39*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Debug.h"
40*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/ELF.h"
41*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Format.h"
42*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/MemoryBuffer.h"
43*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/SourceMgr.h"
44*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/TargetRegistry.h"
45*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/raw_ostream.h"
46*9880d681SAndroid Build Coastguard Worker
47*9880d681SAndroid Build Coastguard Worker using namespace llvm;
48*9880d681SAndroid Build Coastguard Worker
49*9880d681SAndroid Build Coastguard Worker static cl::opt<bool> EnableFutureRegs("mfuture-regs",
50*9880d681SAndroid Build Coastguard Worker cl::desc("Enable future registers"));
51*9880d681SAndroid Build Coastguard Worker
52*9880d681SAndroid Build Coastguard Worker static cl::opt<bool> WarnMissingParenthesis("mwarn-missing-parenthesis",
53*9880d681SAndroid Build Coastguard Worker cl::desc("Warn for missing parenthesis around predicate registers"),
54*9880d681SAndroid Build Coastguard Worker cl::init(true));
55*9880d681SAndroid Build Coastguard Worker static cl::opt<bool> ErrorMissingParenthesis("merror-missing-parenthesis",
56*9880d681SAndroid Build Coastguard Worker cl::desc("Error for missing parenthesis around predicate registers"),
57*9880d681SAndroid Build Coastguard Worker cl::init(false));
58*9880d681SAndroid Build Coastguard Worker static cl::opt<bool> WarnSignedMismatch("mwarn-sign-mismatch",
59*9880d681SAndroid Build Coastguard Worker cl::desc("Warn for mismatching a signed and unsigned value"),
60*9880d681SAndroid Build Coastguard Worker cl::init(true));
61*9880d681SAndroid Build Coastguard Worker static cl::opt<bool> WarnNoncontigiousRegister("mwarn-noncontigious-register",
62*9880d681SAndroid Build Coastguard Worker cl::desc("Warn for register names that arent contigious"),
63*9880d681SAndroid Build Coastguard Worker cl::init(true));
64*9880d681SAndroid Build Coastguard Worker static cl::opt<bool> ErrorNoncontigiousRegister("merror-noncontigious-register",
65*9880d681SAndroid Build Coastguard Worker cl::desc("Error for register names that aren't contigious"),
66*9880d681SAndroid Build Coastguard Worker cl::init(false));
67*9880d681SAndroid Build Coastguard Worker
68*9880d681SAndroid Build Coastguard Worker
69*9880d681SAndroid Build Coastguard Worker namespace {
70*9880d681SAndroid Build Coastguard Worker struct HexagonOperand;
71*9880d681SAndroid Build Coastguard Worker
72*9880d681SAndroid Build Coastguard Worker class HexagonAsmParser : public MCTargetAsmParser {
73*9880d681SAndroid Build Coastguard Worker
getTargetStreamer()74*9880d681SAndroid Build Coastguard Worker HexagonTargetStreamer &getTargetStreamer() {
75*9880d681SAndroid Build Coastguard Worker MCTargetStreamer &TS = *Parser.getStreamer().getTargetStreamer();
76*9880d681SAndroid Build Coastguard Worker return static_cast<HexagonTargetStreamer &>(TS);
77*9880d681SAndroid Build Coastguard Worker }
78*9880d681SAndroid Build Coastguard Worker
79*9880d681SAndroid Build Coastguard Worker MCAsmParser &Parser;
80*9880d681SAndroid Build Coastguard Worker MCAssembler *Assembler;
81*9880d681SAndroid Build Coastguard Worker MCInstrInfo const &MCII;
82*9880d681SAndroid Build Coastguard Worker MCInst MCB;
83*9880d681SAndroid Build Coastguard Worker bool InBrackets;
84*9880d681SAndroid Build Coastguard Worker
getParser() const85*9880d681SAndroid Build Coastguard Worker MCAsmParser &getParser() const { return Parser; }
getAssembler() const86*9880d681SAndroid Build Coastguard Worker MCAssembler *getAssembler() const { return Assembler; }
getLexer() const87*9880d681SAndroid Build Coastguard Worker MCAsmLexer &getLexer() const { return Parser.getLexer(); }
88*9880d681SAndroid Build Coastguard Worker
equalIsAsmAssignment()89*9880d681SAndroid Build Coastguard Worker bool equalIsAsmAssignment() override { return false; }
90*9880d681SAndroid Build Coastguard Worker bool isLabel(AsmToken &Token) override;
91*9880d681SAndroid Build Coastguard Worker
Warning(SMLoc L,const Twine & Msg)92*9880d681SAndroid Build Coastguard Worker void Warning(SMLoc L, const Twine &Msg) { Parser.Warning(L, Msg); }
Error(SMLoc L,const Twine & Msg)93*9880d681SAndroid Build Coastguard Worker bool Error(SMLoc L, const Twine &Msg) { return Parser.Error(L, Msg); }
94*9880d681SAndroid Build Coastguard Worker bool ParseDirectiveFalign(unsigned Size, SMLoc L);
95*9880d681SAndroid Build Coastguard Worker
96*9880d681SAndroid Build Coastguard Worker virtual bool ParseRegister(unsigned &RegNo,
97*9880d681SAndroid Build Coastguard Worker SMLoc &StartLoc,
98*9880d681SAndroid Build Coastguard Worker SMLoc &EndLoc) override;
99*9880d681SAndroid Build Coastguard Worker bool ParseDirectiveSubsection(SMLoc L);
100*9880d681SAndroid Build Coastguard Worker bool ParseDirectiveValue(unsigned Size, SMLoc L);
101*9880d681SAndroid Build Coastguard Worker bool ParseDirectiveComm(bool IsLocal, SMLoc L);
102*9880d681SAndroid Build Coastguard Worker bool RegisterMatchesArch(unsigned MatchNum) const;
103*9880d681SAndroid Build Coastguard Worker
104*9880d681SAndroid Build Coastguard Worker bool matchBundleOptions();
105*9880d681SAndroid Build Coastguard Worker bool handleNoncontigiousRegister(bool Contigious, SMLoc &Loc);
106*9880d681SAndroid Build Coastguard Worker bool finishBundle(SMLoc IDLoc, MCStreamer &Out);
107*9880d681SAndroid Build Coastguard Worker void canonicalizeImmediates(MCInst &MCI);
108*9880d681SAndroid Build Coastguard Worker bool matchOneInstruction(MCInst &MCB, SMLoc IDLoc,
109*9880d681SAndroid Build Coastguard Worker OperandVector &InstOperands, uint64_t &ErrorInfo,
110*9880d681SAndroid Build Coastguard Worker bool MatchingInlineAsm);
111*9880d681SAndroid Build Coastguard Worker
112*9880d681SAndroid Build Coastguard Worker bool MatchAndEmitInstruction(SMLoc IDLoc, unsigned &Opcode,
113*9880d681SAndroid Build Coastguard Worker OperandVector &Operands, MCStreamer &Out,
114*9880d681SAndroid Build Coastguard Worker uint64_t &ErrorInfo, bool MatchingInlineAsm) override;
115*9880d681SAndroid Build Coastguard Worker
116*9880d681SAndroid Build Coastguard Worker unsigned validateTargetOperandClass(MCParsedAsmOperand &Op, unsigned Kind) override;
117*9880d681SAndroid Build Coastguard Worker void OutOfRange(SMLoc IDLoc, long long Val, long long Max);
118*9880d681SAndroid Build Coastguard Worker int processInstruction(MCInst &Inst, OperandVector const &Operands,
119*9880d681SAndroid Build Coastguard Worker SMLoc IDLoc);
120*9880d681SAndroid Build Coastguard Worker
121*9880d681SAndroid Build Coastguard Worker // Check if we have an assembler and, if so, set the ELF e_header flags.
chksetELFHeaderEFlags(unsigned flags)122*9880d681SAndroid Build Coastguard Worker void chksetELFHeaderEFlags(unsigned flags) {
123*9880d681SAndroid Build Coastguard Worker if (getAssembler())
124*9880d681SAndroid Build Coastguard Worker getAssembler()->setELFHeaderEFlags(flags);
125*9880d681SAndroid Build Coastguard Worker }
126*9880d681SAndroid Build Coastguard Worker
127*9880d681SAndroid Build Coastguard Worker unsigned matchRegister(StringRef Name);
128*9880d681SAndroid Build Coastguard Worker
129*9880d681SAndroid Build Coastguard Worker /// @name Auto-generated Match Functions
130*9880d681SAndroid Build Coastguard Worker /// {
131*9880d681SAndroid Build Coastguard Worker
132*9880d681SAndroid Build Coastguard Worker #define GET_ASSEMBLER_HEADER
133*9880d681SAndroid Build Coastguard Worker #include "HexagonGenAsmMatcher.inc"
134*9880d681SAndroid Build Coastguard Worker
135*9880d681SAndroid Build Coastguard Worker /// }
136*9880d681SAndroid Build Coastguard Worker
137*9880d681SAndroid Build Coastguard Worker public:
HexagonAsmParser(const MCSubtargetInfo & _STI,MCAsmParser & _Parser,const MCInstrInfo & MII,const MCTargetOptions & Options)138*9880d681SAndroid Build Coastguard Worker HexagonAsmParser(const MCSubtargetInfo &_STI, MCAsmParser &_Parser,
139*9880d681SAndroid Build Coastguard Worker const MCInstrInfo &MII, const MCTargetOptions &Options)
140*9880d681SAndroid Build Coastguard Worker : MCTargetAsmParser(Options, _STI), Parser(_Parser),
141*9880d681SAndroid Build Coastguard Worker MCII (MII), MCB(HexagonMCInstrInfo::createBundle()), InBrackets(false) {
142*9880d681SAndroid Build Coastguard Worker setAvailableFeatures(ComputeAvailableFeatures(getSTI().getFeatureBits()));
143*9880d681SAndroid Build Coastguard Worker
144*9880d681SAndroid Build Coastguard Worker MCAsmParserExtension::Initialize(_Parser);
145*9880d681SAndroid Build Coastguard Worker
146*9880d681SAndroid Build Coastguard Worker Assembler = nullptr;
147*9880d681SAndroid Build Coastguard Worker // FIXME: need better way to detect AsmStreamer (upstream removed getKind())
148*9880d681SAndroid Build Coastguard Worker if (!Parser.getStreamer().hasRawTextSupport()) {
149*9880d681SAndroid Build Coastguard Worker MCELFStreamer *MES = static_cast<MCELFStreamer *>(&Parser.getStreamer());
150*9880d681SAndroid Build Coastguard Worker Assembler = &MES->getAssembler();
151*9880d681SAndroid Build Coastguard Worker }
152*9880d681SAndroid Build Coastguard Worker }
153*9880d681SAndroid Build Coastguard Worker
154*9880d681SAndroid Build Coastguard Worker bool splitIdentifier(OperandVector &Operands);
155*9880d681SAndroid Build Coastguard Worker bool parseOperand(OperandVector &Operands);
156*9880d681SAndroid Build Coastguard Worker bool parseInstruction(OperandVector &Operands);
157*9880d681SAndroid Build Coastguard Worker bool implicitExpressionLocation(OperandVector &Operands);
158*9880d681SAndroid Build Coastguard Worker bool parseExpressionOrOperand(OperandVector &Operands);
159*9880d681SAndroid Build Coastguard Worker bool parseExpression(MCExpr const *& Expr);
ParseInstruction(ParseInstructionInfo & Info,StringRef Name,SMLoc NameLoc,OperandVector & Operands)160*9880d681SAndroid Build Coastguard Worker virtual bool ParseInstruction(ParseInstructionInfo &Info, StringRef Name,
161*9880d681SAndroid Build Coastguard Worker SMLoc NameLoc, OperandVector &Operands) override
162*9880d681SAndroid Build Coastguard Worker {
163*9880d681SAndroid Build Coastguard Worker llvm_unreachable("Unimplemented");
164*9880d681SAndroid Build Coastguard Worker }
165*9880d681SAndroid Build Coastguard Worker virtual bool ParseInstruction(ParseInstructionInfo &Info, StringRef Name,
166*9880d681SAndroid Build Coastguard Worker AsmToken ID, OperandVector &Operands) override;
167*9880d681SAndroid Build Coastguard Worker
168*9880d681SAndroid Build Coastguard Worker virtual bool ParseDirective(AsmToken DirectiveID) override;
169*9880d681SAndroid Build Coastguard Worker };
170*9880d681SAndroid Build Coastguard Worker
171*9880d681SAndroid Build Coastguard Worker /// HexagonOperand - Instances of this class represent a parsed Hexagon machine
172*9880d681SAndroid Build Coastguard Worker /// instruction.
173*9880d681SAndroid Build Coastguard Worker struct HexagonOperand : public MCParsedAsmOperand {
174*9880d681SAndroid Build Coastguard Worker enum KindTy { Token, Immediate, Register } Kind;
175*9880d681SAndroid Build Coastguard Worker
176*9880d681SAndroid Build Coastguard Worker SMLoc StartLoc, EndLoc;
177*9880d681SAndroid Build Coastguard Worker
178*9880d681SAndroid Build Coastguard Worker struct TokTy {
179*9880d681SAndroid Build Coastguard Worker const char *Data;
180*9880d681SAndroid Build Coastguard Worker unsigned Length;
181*9880d681SAndroid Build Coastguard Worker };
182*9880d681SAndroid Build Coastguard Worker
183*9880d681SAndroid Build Coastguard Worker struct RegTy {
184*9880d681SAndroid Build Coastguard Worker unsigned RegNum;
185*9880d681SAndroid Build Coastguard Worker };
186*9880d681SAndroid Build Coastguard Worker
187*9880d681SAndroid Build Coastguard Worker struct ImmTy {
188*9880d681SAndroid Build Coastguard Worker const MCExpr *Val;
189*9880d681SAndroid Build Coastguard Worker };
190*9880d681SAndroid Build Coastguard Worker
191*9880d681SAndroid Build Coastguard Worker struct InstTy {
192*9880d681SAndroid Build Coastguard Worker OperandVector *SubInsts;
193*9880d681SAndroid Build Coastguard Worker };
194*9880d681SAndroid Build Coastguard Worker
195*9880d681SAndroid Build Coastguard Worker union {
196*9880d681SAndroid Build Coastguard Worker struct TokTy Tok;
197*9880d681SAndroid Build Coastguard Worker struct RegTy Reg;
198*9880d681SAndroid Build Coastguard Worker struct ImmTy Imm;
199*9880d681SAndroid Build Coastguard Worker };
200*9880d681SAndroid Build Coastguard Worker
HexagonOperand__anon6bb6924c0111::HexagonOperand201*9880d681SAndroid Build Coastguard Worker HexagonOperand(KindTy K) : MCParsedAsmOperand(), Kind(K) {}
202*9880d681SAndroid Build Coastguard Worker
203*9880d681SAndroid Build Coastguard Worker public:
HexagonOperand__anon6bb6924c0111::HexagonOperand204*9880d681SAndroid Build Coastguard Worker HexagonOperand(const HexagonOperand &o) : MCParsedAsmOperand() {
205*9880d681SAndroid Build Coastguard Worker Kind = o.Kind;
206*9880d681SAndroid Build Coastguard Worker StartLoc = o.StartLoc;
207*9880d681SAndroid Build Coastguard Worker EndLoc = o.EndLoc;
208*9880d681SAndroid Build Coastguard Worker switch (Kind) {
209*9880d681SAndroid Build Coastguard Worker case Register:
210*9880d681SAndroid Build Coastguard Worker Reg = o.Reg;
211*9880d681SAndroid Build Coastguard Worker break;
212*9880d681SAndroid Build Coastguard Worker case Immediate:
213*9880d681SAndroid Build Coastguard Worker Imm = o.Imm;
214*9880d681SAndroid Build Coastguard Worker break;
215*9880d681SAndroid Build Coastguard Worker case Token:
216*9880d681SAndroid Build Coastguard Worker Tok = o.Tok;
217*9880d681SAndroid Build Coastguard Worker break;
218*9880d681SAndroid Build Coastguard Worker }
219*9880d681SAndroid Build Coastguard Worker }
220*9880d681SAndroid Build Coastguard Worker
221*9880d681SAndroid Build Coastguard Worker /// getStartLoc - Get the location of the first token of this operand.
getStartLoc__anon6bb6924c0111::HexagonOperand222*9880d681SAndroid Build Coastguard Worker SMLoc getStartLoc() const { return StartLoc; }
223*9880d681SAndroid Build Coastguard Worker
224*9880d681SAndroid Build Coastguard Worker /// getEndLoc - Get the location of the last token of this operand.
getEndLoc__anon6bb6924c0111::HexagonOperand225*9880d681SAndroid Build Coastguard Worker SMLoc getEndLoc() const { return EndLoc; }
226*9880d681SAndroid Build Coastguard Worker
getReg__anon6bb6924c0111::HexagonOperand227*9880d681SAndroid Build Coastguard Worker unsigned getReg() const {
228*9880d681SAndroid Build Coastguard Worker assert(Kind == Register && "Invalid access!");
229*9880d681SAndroid Build Coastguard Worker return Reg.RegNum;
230*9880d681SAndroid Build Coastguard Worker }
231*9880d681SAndroid Build Coastguard Worker
getImm__anon6bb6924c0111::HexagonOperand232*9880d681SAndroid Build Coastguard Worker const MCExpr *getImm() const {
233*9880d681SAndroid Build Coastguard Worker assert(Kind == Immediate && "Invalid access!");
234*9880d681SAndroid Build Coastguard Worker return Imm.Val;
235*9880d681SAndroid Build Coastguard Worker }
236*9880d681SAndroid Build Coastguard Worker
isToken__anon6bb6924c0111::HexagonOperand237*9880d681SAndroid Build Coastguard Worker bool isToken() const { return Kind == Token; }
isImm__anon6bb6924c0111::HexagonOperand238*9880d681SAndroid Build Coastguard Worker bool isImm() const { return Kind == Immediate; }
isMem__anon6bb6924c0111::HexagonOperand239*9880d681SAndroid Build Coastguard Worker bool isMem() const { llvm_unreachable("No isMem"); }
isReg__anon6bb6924c0111::HexagonOperand240*9880d681SAndroid Build Coastguard Worker bool isReg() const { return Kind == Register; }
241*9880d681SAndroid Build Coastguard Worker
CheckImmRange__anon6bb6924c0111::HexagonOperand242*9880d681SAndroid Build Coastguard Worker bool CheckImmRange(int immBits, int zeroBits, bool isSigned,
243*9880d681SAndroid Build Coastguard Worker bool isRelocatable, bool Extendable) const {
244*9880d681SAndroid Build Coastguard Worker if (Kind == Immediate) {
245*9880d681SAndroid Build Coastguard Worker const MCExpr *myMCExpr = &HexagonMCInstrInfo::getExpr(*getImm());
246*9880d681SAndroid Build Coastguard Worker if (HexagonMCInstrInfo::mustExtend(*Imm.Val) && !Extendable)
247*9880d681SAndroid Build Coastguard Worker return false;
248*9880d681SAndroid Build Coastguard Worker int64_t Res;
249*9880d681SAndroid Build Coastguard Worker if (myMCExpr->evaluateAsAbsolute(Res)) {
250*9880d681SAndroid Build Coastguard Worker int bits = immBits + zeroBits;
251*9880d681SAndroid Build Coastguard Worker // Field bit range is zerobits + bits
252*9880d681SAndroid Build Coastguard Worker // zeroBits must be 0
253*9880d681SAndroid Build Coastguard Worker if (Res & ((1 << zeroBits) - 1))
254*9880d681SAndroid Build Coastguard Worker return false;
255*9880d681SAndroid Build Coastguard Worker if (isSigned) {
256*9880d681SAndroid Build Coastguard Worker if (Res < (1LL << (bits - 1)) && Res >= -(1LL << (bits - 1)))
257*9880d681SAndroid Build Coastguard Worker return true;
258*9880d681SAndroid Build Coastguard Worker } else {
259*9880d681SAndroid Build Coastguard Worker if (bits == 64)
260*9880d681SAndroid Build Coastguard Worker return true;
261*9880d681SAndroid Build Coastguard Worker if (Res >= 0)
262*9880d681SAndroid Build Coastguard Worker return ((uint64_t)Res < (uint64_t)(1ULL << bits)) ? true : false;
263*9880d681SAndroid Build Coastguard Worker else {
264*9880d681SAndroid Build Coastguard Worker const int64_t high_bit_set = 1ULL << 63;
265*9880d681SAndroid Build Coastguard Worker const uint64_t mask = (high_bit_set >> (63 - bits));
266*9880d681SAndroid Build Coastguard Worker return (((uint64_t)Res & mask) == mask) ? true : false;
267*9880d681SAndroid Build Coastguard Worker }
268*9880d681SAndroid Build Coastguard Worker }
269*9880d681SAndroid Build Coastguard Worker } else if (myMCExpr->getKind() == MCExpr::SymbolRef && isRelocatable)
270*9880d681SAndroid Build Coastguard Worker return true;
271*9880d681SAndroid Build Coastguard Worker else if (myMCExpr->getKind() == MCExpr::Binary ||
272*9880d681SAndroid Build Coastguard Worker myMCExpr->getKind() == MCExpr::Unary)
273*9880d681SAndroid Build Coastguard Worker return true;
274*9880d681SAndroid Build Coastguard Worker }
275*9880d681SAndroid Build Coastguard Worker return false;
276*9880d681SAndroid Build Coastguard Worker }
277*9880d681SAndroid Build Coastguard Worker
isf32Ext__anon6bb6924c0111::HexagonOperand278*9880d681SAndroid Build Coastguard Worker bool isf32Ext() const { return false; }
iss32Imm__anon6bb6924c0111::HexagonOperand279*9880d681SAndroid Build Coastguard Worker bool iss32Imm() const { return CheckImmRange(32, 0, true, true, false); }
iss23_2Imm__anon6bb6924c0111::HexagonOperand280*9880d681SAndroid Build Coastguard Worker bool iss23_2Imm() const { return CheckImmRange(23, 2, true, true, false); }
iss8Imm__anon6bb6924c0111::HexagonOperand281*9880d681SAndroid Build Coastguard Worker bool iss8Imm() const { return CheckImmRange(8, 0, true, false, false); }
iss8Imm64__anon6bb6924c0111::HexagonOperand282*9880d681SAndroid Build Coastguard Worker bool iss8Imm64() const { return CheckImmRange(8, 0, true, true, false); }
iss7Imm__anon6bb6924c0111::HexagonOperand283*9880d681SAndroid Build Coastguard Worker bool iss7Imm() const { return CheckImmRange(7, 0, true, false, false); }
iss6Imm__anon6bb6924c0111::HexagonOperand284*9880d681SAndroid Build Coastguard Worker bool iss6Imm() const { return CheckImmRange(6, 0, true, false, false); }
iss4Imm__anon6bb6924c0111::HexagonOperand285*9880d681SAndroid Build Coastguard Worker bool iss4Imm() const { return CheckImmRange(4, 0, true, false, false); }
iss4_0Imm__anon6bb6924c0111::HexagonOperand286*9880d681SAndroid Build Coastguard Worker bool iss4_0Imm() const { return CheckImmRange(4, 0, true, false, false); }
iss4_1Imm__anon6bb6924c0111::HexagonOperand287*9880d681SAndroid Build Coastguard Worker bool iss4_1Imm() const { return CheckImmRange(4, 1, true, false, false); }
iss4_2Imm__anon6bb6924c0111::HexagonOperand288*9880d681SAndroid Build Coastguard Worker bool iss4_2Imm() const { return CheckImmRange(4, 2, true, false, false); }
iss4_3Imm__anon6bb6924c0111::HexagonOperand289*9880d681SAndroid Build Coastguard Worker bool iss4_3Imm() const { return CheckImmRange(4, 3, true, false, false); }
iss4_6Imm__anon6bb6924c0111::HexagonOperand290*9880d681SAndroid Build Coastguard Worker bool iss4_6Imm() const { return CheckImmRange(4, 0, true, false, false); }
iss3_6Imm__anon6bb6924c0111::HexagonOperand291*9880d681SAndroid Build Coastguard Worker bool iss3_6Imm() const { return CheckImmRange(3, 0, true, false, false); }
iss3Imm__anon6bb6924c0111::HexagonOperand292*9880d681SAndroid Build Coastguard Worker bool iss3Imm() const { return CheckImmRange(3, 0, true, false, false); }
293*9880d681SAndroid Build Coastguard Worker
isu64Imm__anon6bb6924c0111::HexagonOperand294*9880d681SAndroid Build Coastguard Worker bool isu64Imm() const { return CheckImmRange(64, 0, false, true, true); }
isu32Imm__anon6bb6924c0111::HexagonOperand295*9880d681SAndroid Build Coastguard Worker bool isu32Imm() const { return CheckImmRange(32, 0, false, true, false); }
isu26_6Imm__anon6bb6924c0111::HexagonOperand296*9880d681SAndroid Build Coastguard Worker bool isu26_6Imm() const { return CheckImmRange(26, 6, false, true, false); }
isu16Imm__anon6bb6924c0111::HexagonOperand297*9880d681SAndroid Build Coastguard Worker bool isu16Imm() const { return CheckImmRange(16, 0, false, true, false); }
isu16_0Imm__anon6bb6924c0111::HexagonOperand298*9880d681SAndroid Build Coastguard Worker bool isu16_0Imm() const { return CheckImmRange(16, 0, false, true, false); }
isu16_1Imm__anon6bb6924c0111::HexagonOperand299*9880d681SAndroid Build Coastguard Worker bool isu16_1Imm() const { return CheckImmRange(16, 1, false, true, false); }
isu16_2Imm__anon6bb6924c0111::HexagonOperand300*9880d681SAndroid Build Coastguard Worker bool isu16_2Imm() const { return CheckImmRange(16, 2, false, true, false); }
isu16_3Imm__anon6bb6924c0111::HexagonOperand301*9880d681SAndroid Build Coastguard Worker bool isu16_3Imm() const { return CheckImmRange(16, 3, false, true, false); }
isu11_3Imm__anon6bb6924c0111::HexagonOperand302*9880d681SAndroid Build Coastguard Worker bool isu11_3Imm() const { return CheckImmRange(11, 3, false, false, false); }
isu6_0Imm__anon6bb6924c0111::HexagonOperand303*9880d681SAndroid Build Coastguard Worker bool isu6_0Imm() const { return CheckImmRange(6, 0, false, false, false); }
isu6_1Imm__anon6bb6924c0111::HexagonOperand304*9880d681SAndroid Build Coastguard Worker bool isu6_1Imm() const { return CheckImmRange(6, 1, false, false, false); }
isu6_2Imm__anon6bb6924c0111::HexagonOperand305*9880d681SAndroid Build Coastguard Worker bool isu6_2Imm() const { return CheckImmRange(6, 2, false, false, false); }
isu6_3Imm__anon6bb6924c0111::HexagonOperand306*9880d681SAndroid Build Coastguard Worker bool isu6_3Imm() const { return CheckImmRange(6, 3, false, false, false); }
isu10Imm__anon6bb6924c0111::HexagonOperand307*9880d681SAndroid Build Coastguard Worker bool isu10Imm() const { return CheckImmRange(10, 0, false, false, false); }
isu9Imm__anon6bb6924c0111::HexagonOperand308*9880d681SAndroid Build Coastguard Worker bool isu9Imm() const { return CheckImmRange(9, 0, false, false, false); }
isu8Imm__anon6bb6924c0111::HexagonOperand309*9880d681SAndroid Build Coastguard Worker bool isu8Imm() const { return CheckImmRange(8, 0, false, false, false); }
isu7Imm__anon6bb6924c0111::HexagonOperand310*9880d681SAndroid Build Coastguard Worker bool isu7Imm() const { return CheckImmRange(7, 0, false, false, false); }
isu6Imm__anon6bb6924c0111::HexagonOperand311*9880d681SAndroid Build Coastguard Worker bool isu6Imm() const { return CheckImmRange(6, 0, false, false, false); }
isu5Imm__anon6bb6924c0111::HexagonOperand312*9880d681SAndroid Build Coastguard Worker bool isu5Imm() const { return CheckImmRange(5, 0, false, false, false); }
isu4Imm__anon6bb6924c0111::HexagonOperand313*9880d681SAndroid Build Coastguard Worker bool isu4Imm() const { return CheckImmRange(4, 0, false, false, false); }
isu3Imm__anon6bb6924c0111::HexagonOperand314*9880d681SAndroid Build Coastguard Worker bool isu3Imm() const { return CheckImmRange(3, 0, false, false, false); }
isu2Imm__anon6bb6924c0111::HexagonOperand315*9880d681SAndroid Build Coastguard Worker bool isu2Imm() const { return CheckImmRange(2, 0, false, false, false); }
isu1Imm__anon6bb6924c0111::HexagonOperand316*9880d681SAndroid Build Coastguard Worker bool isu1Imm() const { return CheckImmRange(1, 0, false, false, false); }
317*9880d681SAndroid Build Coastguard Worker
ism6Imm__anon6bb6924c0111::HexagonOperand318*9880d681SAndroid Build Coastguard Worker bool ism6Imm() const { return CheckImmRange(6, 0, false, false, false); }
isn8Imm__anon6bb6924c0111::HexagonOperand319*9880d681SAndroid Build Coastguard Worker bool isn8Imm() const { return CheckImmRange(8, 0, false, false, false); }
320*9880d681SAndroid Build Coastguard Worker
iss16Ext__anon6bb6924c0111::HexagonOperand321*9880d681SAndroid Build Coastguard Worker bool iss16Ext() const { return CheckImmRange(16 + 26, 0, true, true, true); }
iss12Ext__anon6bb6924c0111::HexagonOperand322*9880d681SAndroid Build Coastguard Worker bool iss12Ext() const { return CheckImmRange(12 + 26, 0, true, true, true); }
iss10Ext__anon6bb6924c0111::HexagonOperand323*9880d681SAndroid Build Coastguard Worker bool iss10Ext() const { return CheckImmRange(10 + 26, 0, true, true, true); }
iss9Ext__anon6bb6924c0111::HexagonOperand324*9880d681SAndroid Build Coastguard Worker bool iss9Ext() const { return CheckImmRange(9 + 26, 0, true, true, true); }
iss8Ext__anon6bb6924c0111::HexagonOperand325*9880d681SAndroid Build Coastguard Worker bool iss8Ext() const { return CheckImmRange(8 + 26, 0, true, true, true); }
iss7Ext__anon6bb6924c0111::HexagonOperand326*9880d681SAndroid Build Coastguard Worker bool iss7Ext() const { return CheckImmRange(7 + 26, 0, true, true, true); }
iss6Ext__anon6bb6924c0111::HexagonOperand327*9880d681SAndroid Build Coastguard Worker bool iss6Ext() const { return CheckImmRange(6 + 26, 0, true, true, true); }
iss11_0Ext__anon6bb6924c0111::HexagonOperand328*9880d681SAndroid Build Coastguard Worker bool iss11_0Ext() const {
329*9880d681SAndroid Build Coastguard Worker return CheckImmRange(11 + 26, 0, true, true, true);
330*9880d681SAndroid Build Coastguard Worker }
iss11_1Ext__anon6bb6924c0111::HexagonOperand331*9880d681SAndroid Build Coastguard Worker bool iss11_1Ext() const {
332*9880d681SAndroid Build Coastguard Worker return CheckImmRange(11 + 26, 1, true, true, true);
333*9880d681SAndroid Build Coastguard Worker }
iss11_2Ext__anon6bb6924c0111::HexagonOperand334*9880d681SAndroid Build Coastguard Worker bool iss11_2Ext() const {
335*9880d681SAndroid Build Coastguard Worker return CheckImmRange(11 + 26, 2, true, true, true);
336*9880d681SAndroid Build Coastguard Worker }
iss11_3Ext__anon6bb6924c0111::HexagonOperand337*9880d681SAndroid Build Coastguard Worker bool iss11_3Ext() const {
338*9880d681SAndroid Build Coastguard Worker return CheckImmRange(11 + 26, 3, true, true, true);
339*9880d681SAndroid Build Coastguard Worker }
340*9880d681SAndroid Build Coastguard Worker
isu6Ext__anon6bb6924c0111::HexagonOperand341*9880d681SAndroid Build Coastguard Worker bool isu6Ext() const { return CheckImmRange(6 + 26, 0, false, true, true); }
isu7Ext__anon6bb6924c0111::HexagonOperand342*9880d681SAndroid Build Coastguard Worker bool isu7Ext() const { return CheckImmRange(7 + 26, 0, false, true, true); }
isu8Ext__anon6bb6924c0111::HexagonOperand343*9880d681SAndroid Build Coastguard Worker bool isu8Ext() const { return CheckImmRange(8 + 26, 0, false, true, true); }
isu9Ext__anon6bb6924c0111::HexagonOperand344*9880d681SAndroid Build Coastguard Worker bool isu9Ext() const { return CheckImmRange(9 + 26, 0, false, true, true); }
isu10Ext__anon6bb6924c0111::HexagonOperand345*9880d681SAndroid Build Coastguard Worker bool isu10Ext() const { return CheckImmRange(10 + 26, 0, false, true, true); }
isu6_0Ext__anon6bb6924c0111::HexagonOperand346*9880d681SAndroid Build Coastguard Worker bool isu6_0Ext() const { return CheckImmRange(6 + 26, 0, false, true, true); }
isu6_1Ext__anon6bb6924c0111::HexagonOperand347*9880d681SAndroid Build Coastguard Worker bool isu6_1Ext() const { return CheckImmRange(6 + 26, 1, false, true, true); }
isu6_2Ext__anon6bb6924c0111::HexagonOperand348*9880d681SAndroid Build Coastguard Worker bool isu6_2Ext() const { return CheckImmRange(6 + 26, 2, false, true, true); }
isu6_3Ext__anon6bb6924c0111::HexagonOperand349*9880d681SAndroid Build Coastguard Worker bool isu6_3Ext() const { return CheckImmRange(6 + 26, 3, false, true, true); }
isu32MustExt__anon6bb6924c0111::HexagonOperand350*9880d681SAndroid Build Coastguard Worker bool isu32MustExt() const { return isImm(); }
351*9880d681SAndroid Build Coastguard Worker
addRegOperands__anon6bb6924c0111::HexagonOperand352*9880d681SAndroid Build Coastguard Worker void addRegOperands(MCInst &Inst, unsigned N) const {
353*9880d681SAndroid Build Coastguard Worker assert(N == 1 && "Invalid number of operands!");
354*9880d681SAndroid Build Coastguard Worker Inst.addOperand(MCOperand::createReg(getReg()));
355*9880d681SAndroid Build Coastguard Worker }
356*9880d681SAndroid Build Coastguard Worker
addImmOperands__anon6bb6924c0111::HexagonOperand357*9880d681SAndroid Build Coastguard Worker void addImmOperands(MCInst &Inst, unsigned N) const {
358*9880d681SAndroid Build Coastguard Worker assert(N == 1 && "Invalid number of operands!");
359*9880d681SAndroid Build Coastguard Worker Inst.addOperand(MCOperand::createExpr(getImm()));
360*9880d681SAndroid Build Coastguard Worker }
361*9880d681SAndroid Build Coastguard Worker
addSignedImmOperands__anon6bb6924c0111::HexagonOperand362*9880d681SAndroid Build Coastguard Worker void addSignedImmOperands(MCInst &Inst, unsigned N) const {
363*9880d681SAndroid Build Coastguard Worker assert(N == 1 && "Invalid number of operands!");
364*9880d681SAndroid Build Coastguard Worker HexagonMCExpr *Expr =
365*9880d681SAndroid Build Coastguard Worker const_cast<HexagonMCExpr *>(cast<HexagonMCExpr>(getImm()));
366*9880d681SAndroid Build Coastguard Worker int64_t Value;
367*9880d681SAndroid Build Coastguard Worker if (!Expr->evaluateAsAbsolute(Value)) {
368*9880d681SAndroid Build Coastguard Worker Inst.addOperand(MCOperand::createExpr(Expr));
369*9880d681SAndroid Build Coastguard Worker return;
370*9880d681SAndroid Build Coastguard Worker }
371*9880d681SAndroid Build Coastguard Worker int64_t Extended = SignExtend64(Value, 32);
372*9880d681SAndroid Build Coastguard Worker if ((Extended < 0) != (Value < 0))
373*9880d681SAndroid Build Coastguard Worker Expr->setSignMismatch();
374*9880d681SAndroid Build Coastguard Worker Inst.addOperand(MCOperand::createExpr(Expr));
375*9880d681SAndroid Build Coastguard Worker }
376*9880d681SAndroid Build Coastguard Worker
addf32ExtOperands__anon6bb6924c0111::HexagonOperand377*9880d681SAndroid Build Coastguard Worker void addf32ExtOperands(MCInst &Inst, unsigned N) const {
378*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
379*9880d681SAndroid Build Coastguard Worker }
380*9880d681SAndroid Build Coastguard Worker
adds32ImmOperands__anon6bb6924c0111::HexagonOperand381*9880d681SAndroid Build Coastguard Worker void adds32ImmOperands(MCInst &Inst, unsigned N) const {
382*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
383*9880d681SAndroid Build Coastguard Worker }
adds23_2ImmOperands__anon6bb6924c0111::HexagonOperand384*9880d681SAndroid Build Coastguard Worker void adds23_2ImmOperands(MCInst &Inst, unsigned N) const {
385*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
386*9880d681SAndroid Build Coastguard Worker }
adds8ImmOperands__anon6bb6924c0111::HexagonOperand387*9880d681SAndroid Build Coastguard Worker void adds8ImmOperands(MCInst &Inst, unsigned N) const {
388*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
389*9880d681SAndroid Build Coastguard Worker }
adds8Imm64Operands__anon6bb6924c0111::HexagonOperand390*9880d681SAndroid Build Coastguard Worker void adds8Imm64Operands(MCInst &Inst, unsigned N) const {
391*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
392*9880d681SAndroid Build Coastguard Worker }
adds6ImmOperands__anon6bb6924c0111::HexagonOperand393*9880d681SAndroid Build Coastguard Worker void adds6ImmOperands(MCInst &Inst, unsigned N) const {
394*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
395*9880d681SAndroid Build Coastguard Worker }
adds4ImmOperands__anon6bb6924c0111::HexagonOperand396*9880d681SAndroid Build Coastguard Worker void adds4ImmOperands(MCInst &Inst, unsigned N) const {
397*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
398*9880d681SAndroid Build Coastguard Worker }
adds4_0ImmOperands__anon6bb6924c0111::HexagonOperand399*9880d681SAndroid Build Coastguard Worker void adds4_0ImmOperands(MCInst &Inst, unsigned N) const {
400*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
401*9880d681SAndroid Build Coastguard Worker }
adds4_1ImmOperands__anon6bb6924c0111::HexagonOperand402*9880d681SAndroid Build Coastguard Worker void adds4_1ImmOperands(MCInst &Inst, unsigned N) const {
403*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
404*9880d681SAndroid Build Coastguard Worker }
adds4_2ImmOperands__anon6bb6924c0111::HexagonOperand405*9880d681SAndroid Build Coastguard Worker void adds4_2ImmOperands(MCInst &Inst, unsigned N) const {
406*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
407*9880d681SAndroid Build Coastguard Worker }
adds4_3ImmOperands__anon6bb6924c0111::HexagonOperand408*9880d681SAndroid Build Coastguard Worker void adds4_3ImmOperands(MCInst &Inst, unsigned N) const {
409*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
410*9880d681SAndroid Build Coastguard Worker }
adds3ImmOperands__anon6bb6924c0111::HexagonOperand411*9880d681SAndroid Build Coastguard Worker void adds3ImmOperands(MCInst &Inst, unsigned N) const {
412*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
413*9880d681SAndroid Build Coastguard Worker }
414*9880d681SAndroid Build Coastguard Worker
addu64ImmOperands__anon6bb6924c0111::HexagonOperand415*9880d681SAndroid Build Coastguard Worker void addu64ImmOperands(MCInst &Inst, unsigned N) const {
416*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
417*9880d681SAndroid Build Coastguard Worker }
addu32ImmOperands__anon6bb6924c0111::HexagonOperand418*9880d681SAndroid Build Coastguard Worker void addu32ImmOperands(MCInst &Inst, unsigned N) const {
419*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
420*9880d681SAndroid Build Coastguard Worker }
addu26_6ImmOperands__anon6bb6924c0111::HexagonOperand421*9880d681SAndroid Build Coastguard Worker void addu26_6ImmOperands(MCInst &Inst, unsigned N) const {
422*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
423*9880d681SAndroid Build Coastguard Worker }
addu16ImmOperands__anon6bb6924c0111::HexagonOperand424*9880d681SAndroid Build Coastguard Worker void addu16ImmOperands(MCInst &Inst, unsigned N) const {
425*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
426*9880d681SAndroid Build Coastguard Worker }
addu16_0ImmOperands__anon6bb6924c0111::HexagonOperand427*9880d681SAndroid Build Coastguard Worker void addu16_0ImmOperands(MCInst &Inst, unsigned N) const {
428*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
429*9880d681SAndroid Build Coastguard Worker }
addu16_1ImmOperands__anon6bb6924c0111::HexagonOperand430*9880d681SAndroid Build Coastguard Worker void addu16_1ImmOperands(MCInst &Inst, unsigned N) const {
431*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
432*9880d681SAndroid Build Coastguard Worker }
addu16_2ImmOperands__anon6bb6924c0111::HexagonOperand433*9880d681SAndroid Build Coastguard Worker void addu16_2ImmOperands(MCInst &Inst, unsigned N) const {
434*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
435*9880d681SAndroid Build Coastguard Worker }
addu16_3ImmOperands__anon6bb6924c0111::HexagonOperand436*9880d681SAndroid Build Coastguard Worker void addu16_3ImmOperands(MCInst &Inst, unsigned N) const {
437*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
438*9880d681SAndroid Build Coastguard Worker }
addu11_3ImmOperands__anon6bb6924c0111::HexagonOperand439*9880d681SAndroid Build Coastguard Worker void addu11_3ImmOperands(MCInst &Inst, unsigned N) const {
440*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
441*9880d681SAndroid Build Coastguard Worker }
addu10ImmOperands__anon6bb6924c0111::HexagonOperand442*9880d681SAndroid Build Coastguard Worker void addu10ImmOperands(MCInst &Inst, unsigned N) const {
443*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
444*9880d681SAndroid Build Coastguard Worker }
addu9ImmOperands__anon6bb6924c0111::HexagonOperand445*9880d681SAndroid Build Coastguard Worker void addu9ImmOperands(MCInst &Inst, unsigned N) const {
446*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
447*9880d681SAndroid Build Coastguard Worker }
addu8ImmOperands__anon6bb6924c0111::HexagonOperand448*9880d681SAndroid Build Coastguard Worker void addu8ImmOperands(MCInst &Inst, unsigned N) const {
449*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
450*9880d681SAndroid Build Coastguard Worker }
addu7ImmOperands__anon6bb6924c0111::HexagonOperand451*9880d681SAndroid Build Coastguard Worker void addu7ImmOperands(MCInst &Inst, unsigned N) const {
452*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
453*9880d681SAndroid Build Coastguard Worker }
addu6ImmOperands__anon6bb6924c0111::HexagonOperand454*9880d681SAndroid Build Coastguard Worker void addu6ImmOperands(MCInst &Inst, unsigned N) const {
455*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
456*9880d681SAndroid Build Coastguard Worker }
addu6_0ImmOperands__anon6bb6924c0111::HexagonOperand457*9880d681SAndroid Build Coastguard Worker void addu6_0ImmOperands(MCInst &Inst, unsigned N) const {
458*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
459*9880d681SAndroid Build Coastguard Worker }
addu6_1ImmOperands__anon6bb6924c0111::HexagonOperand460*9880d681SAndroid Build Coastguard Worker void addu6_1ImmOperands(MCInst &Inst, unsigned N) const {
461*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
462*9880d681SAndroid Build Coastguard Worker }
addu6_2ImmOperands__anon6bb6924c0111::HexagonOperand463*9880d681SAndroid Build Coastguard Worker void addu6_2ImmOperands(MCInst &Inst, unsigned N) const {
464*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
465*9880d681SAndroid Build Coastguard Worker }
addu6_3ImmOperands__anon6bb6924c0111::HexagonOperand466*9880d681SAndroid Build Coastguard Worker void addu6_3ImmOperands(MCInst &Inst, unsigned N) const {
467*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
468*9880d681SAndroid Build Coastguard Worker }
addu5ImmOperands__anon6bb6924c0111::HexagonOperand469*9880d681SAndroid Build Coastguard Worker void addu5ImmOperands(MCInst &Inst, unsigned N) const {
470*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
471*9880d681SAndroid Build Coastguard Worker }
addu4ImmOperands__anon6bb6924c0111::HexagonOperand472*9880d681SAndroid Build Coastguard Worker void addu4ImmOperands(MCInst &Inst, unsigned N) const {
473*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
474*9880d681SAndroid Build Coastguard Worker }
addu3ImmOperands__anon6bb6924c0111::HexagonOperand475*9880d681SAndroid Build Coastguard Worker void addu3ImmOperands(MCInst &Inst, unsigned N) const {
476*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
477*9880d681SAndroid Build Coastguard Worker }
addu2ImmOperands__anon6bb6924c0111::HexagonOperand478*9880d681SAndroid Build Coastguard Worker void addu2ImmOperands(MCInst &Inst, unsigned N) const {
479*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
480*9880d681SAndroid Build Coastguard Worker }
addu1ImmOperands__anon6bb6924c0111::HexagonOperand481*9880d681SAndroid Build Coastguard Worker void addu1ImmOperands(MCInst &Inst, unsigned N) const {
482*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
483*9880d681SAndroid Build Coastguard Worker }
484*9880d681SAndroid Build Coastguard Worker
addm6ImmOperands__anon6bb6924c0111::HexagonOperand485*9880d681SAndroid Build Coastguard Worker void addm6ImmOperands(MCInst &Inst, unsigned N) const {
486*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
487*9880d681SAndroid Build Coastguard Worker }
addn8ImmOperands__anon6bb6924c0111::HexagonOperand488*9880d681SAndroid Build Coastguard Worker void addn8ImmOperands(MCInst &Inst, unsigned N) const {
489*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
490*9880d681SAndroid Build Coastguard Worker }
491*9880d681SAndroid Build Coastguard Worker
adds16ExtOperands__anon6bb6924c0111::HexagonOperand492*9880d681SAndroid Build Coastguard Worker void adds16ExtOperands(MCInst &Inst, unsigned N) const {
493*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
494*9880d681SAndroid Build Coastguard Worker }
adds12ExtOperands__anon6bb6924c0111::HexagonOperand495*9880d681SAndroid Build Coastguard Worker void adds12ExtOperands(MCInst &Inst, unsigned N) const {
496*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
497*9880d681SAndroid Build Coastguard Worker }
adds10ExtOperands__anon6bb6924c0111::HexagonOperand498*9880d681SAndroid Build Coastguard Worker void adds10ExtOperands(MCInst &Inst, unsigned N) const {
499*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
500*9880d681SAndroid Build Coastguard Worker }
adds9ExtOperands__anon6bb6924c0111::HexagonOperand501*9880d681SAndroid Build Coastguard Worker void adds9ExtOperands(MCInst &Inst, unsigned N) const {
502*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
503*9880d681SAndroid Build Coastguard Worker }
adds8ExtOperands__anon6bb6924c0111::HexagonOperand504*9880d681SAndroid Build Coastguard Worker void adds8ExtOperands(MCInst &Inst, unsigned N) const {
505*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
506*9880d681SAndroid Build Coastguard Worker }
adds6ExtOperands__anon6bb6924c0111::HexagonOperand507*9880d681SAndroid Build Coastguard Worker void adds6ExtOperands(MCInst &Inst, unsigned N) const {
508*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
509*9880d681SAndroid Build Coastguard Worker }
adds11_0ExtOperands__anon6bb6924c0111::HexagonOperand510*9880d681SAndroid Build Coastguard Worker void adds11_0ExtOperands(MCInst &Inst, unsigned N) const {
511*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
512*9880d681SAndroid Build Coastguard Worker }
adds11_1ExtOperands__anon6bb6924c0111::HexagonOperand513*9880d681SAndroid Build Coastguard Worker void adds11_1ExtOperands(MCInst &Inst, unsigned N) const {
514*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
515*9880d681SAndroid Build Coastguard Worker }
adds11_2ExtOperands__anon6bb6924c0111::HexagonOperand516*9880d681SAndroid Build Coastguard Worker void adds11_2ExtOperands(MCInst &Inst, unsigned N) const {
517*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
518*9880d681SAndroid Build Coastguard Worker }
adds11_3ExtOperands__anon6bb6924c0111::HexagonOperand519*9880d681SAndroid Build Coastguard Worker void adds11_3ExtOperands(MCInst &Inst, unsigned N) const {
520*9880d681SAndroid Build Coastguard Worker addSignedImmOperands(Inst, N);
521*9880d681SAndroid Build Coastguard Worker }
522*9880d681SAndroid Build Coastguard Worker
addu6ExtOperands__anon6bb6924c0111::HexagonOperand523*9880d681SAndroid Build Coastguard Worker void addu6ExtOperands(MCInst &Inst, unsigned N) const {
524*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
525*9880d681SAndroid Build Coastguard Worker }
addu7ExtOperands__anon6bb6924c0111::HexagonOperand526*9880d681SAndroid Build Coastguard Worker void addu7ExtOperands(MCInst &Inst, unsigned N) const {
527*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
528*9880d681SAndroid Build Coastguard Worker }
addu8ExtOperands__anon6bb6924c0111::HexagonOperand529*9880d681SAndroid Build Coastguard Worker void addu8ExtOperands(MCInst &Inst, unsigned N) const {
530*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
531*9880d681SAndroid Build Coastguard Worker }
addu9ExtOperands__anon6bb6924c0111::HexagonOperand532*9880d681SAndroid Build Coastguard Worker void addu9ExtOperands(MCInst &Inst, unsigned N) const {
533*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
534*9880d681SAndroid Build Coastguard Worker }
addu10ExtOperands__anon6bb6924c0111::HexagonOperand535*9880d681SAndroid Build Coastguard Worker void addu10ExtOperands(MCInst &Inst, unsigned N) const {
536*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
537*9880d681SAndroid Build Coastguard Worker }
addu6_0ExtOperands__anon6bb6924c0111::HexagonOperand538*9880d681SAndroid Build Coastguard Worker void addu6_0ExtOperands(MCInst &Inst, unsigned N) const {
539*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
540*9880d681SAndroid Build Coastguard Worker }
addu6_1ExtOperands__anon6bb6924c0111::HexagonOperand541*9880d681SAndroid Build Coastguard Worker void addu6_1ExtOperands(MCInst &Inst, unsigned N) const {
542*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
543*9880d681SAndroid Build Coastguard Worker }
addu6_2ExtOperands__anon6bb6924c0111::HexagonOperand544*9880d681SAndroid Build Coastguard Worker void addu6_2ExtOperands(MCInst &Inst, unsigned N) const {
545*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
546*9880d681SAndroid Build Coastguard Worker }
addu6_3ExtOperands__anon6bb6924c0111::HexagonOperand547*9880d681SAndroid Build Coastguard Worker void addu6_3ExtOperands(MCInst &Inst, unsigned N) const {
548*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
549*9880d681SAndroid Build Coastguard Worker }
addu32MustExtOperands__anon6bb6924c0111::HexagonOperand550*9880d681SAndroid Build Coastguard Worker void addu32MustExtOperands(MCInst &Inst, unsigned N) const {
551*9880d681SAndroid Build Coastguard Worker addImmOperands(Inst, N);
552*9880d681SAndroid Build Coastguard Worker }
553*9880d681SAndroid Build Coastguard Worker
adds4_6ImmOperands__anon6bb6924c0111::HexagonOperand554*9880d681SAndroid Build Coastguard Worker void adds4_6ImmOperands(MCInst &Inst, unsigned N) const {
555*9880d681SAndroid Build Coastguard Worker assert(N == 1 && "Invalid number of operands!");
556*9880d681SAndroid Build Coastguard Worker const MCConstantExpr *CE =
557*9880d681SAndroid Build Coastguard Worker dyn_cast<MCConstantExpr>(&HexagonMCInstrInfo::getExpr(*getImm()));
558*9880d681SAndroid Build Coastguard Worker Inst.addOperand(MCOperand::createImm(CE->getValue() * 64));
559*9880d681SAndroid Build Coastguard Worker }
560*9880d681SAndroid Build Coastguard Worker
adds3_6ImmOperands__anon6bb6924c0111::HexagonOperand561*9880d681SAndroid Build Coastguard Worker void adds3_6ImmOperands(MCInst &Inst, unsigned N) const {
562*9880d681SAndroid Build Coastguard Worker assert(N == 1 && "Invalid number of operands!");
563*9880d681SAndroid Build Coastguard Worker const MCConstantExpr *CE =
564*9880d681SAndroid Build Coastguard Worker dyn_cast<MCConstantExpr>(&HexagonMCInstrInfo::getExpr(*getImm()));
565*9880d681SAndroid Build Coastguard Worker Inst.addOperand(MCOperand::createImm(CE->getValue() * 64));
566*9880d681SAndroid Build Coastguard Worker }
567*9880d681SAndroid Build Coastguard Worker
getToken__anon6bb6924c0111::HexagonOperand568*9880d681SAndroid Build Coastguard Worker StringRef getToken() const {
569*9880d681SAndroid Build Coastguard Worker assert(Kind == Token && "Invalid access!");
570*9880d681SAndroid Build Coastguard Worker return StringRef(Tok.Data, Tok.Length);
571*9880d681SAndroid Build Coastguard Worker }
572*9880d681SAndroid Build Coastguard Worker
573*9880d681SAndroid Build Coastguard Worker virtual void print(raw_ostream &OS) const;
574*9880d681SAndroid Build Coastguard Worker
CreateToken__anon6bb6924c0111::HexagonOperand575*9880d681SAndroid Build Coastguard Worker static std::unique_ptr<HexagonOperand> CreateToken(StringRef Str, SMLoc S) {
576*9880d681SAndroid Build Coastguard Worker HexagonOperand *Op = new HexagonOperand(Token);
577*9880d681SAndroid Build Coastguard Worker Op->Tok.Data = Str.data();
578*9880d681SAndroid Build Coastguard Worker Op->Tok.Length = Str.size();
579*9880d681SAndroid Build Coastguard Worker Op->StartLoc = S;
580*9880d681SAndroid Build Coastguard Worker Op->EndLoc = S;
581*9880d681SAndroid Build Coastguard Worker return std::unique_ptr<HexagonOperand>(Op);
582*9880d681SAndroid Build Coastguard Worker }
583*9880d681SAndroid Build Coastguard Worker
CreateReg__anon6bb6924c0111::HexagonOperand584*9880d681SAndroid Build Coastguard Worker static std::unique_ptr<HexagonOperand> CreateReg(unsigned RegNum, SMLoc S,
585*9880d681SAndroid Build Coastguard Worker SMLoc E) {
586*9880d681SAndroid Build Coastguard Worker HexagonOperand *Op = new HexagonOperand(Register);
587*9880d681SAndroid Build Coastguard Worker Op->Reg.RegNum = RegNum;
588*9880d681SAndroid Build Coastguard Worker Op->StartLoc = S;
589*9880d681SAndroid Build Coastguard Worker Op->EndLoc = E;
590*9880d681SAndroid Build Coastguard Worker return std::unique_ptr<HexagonOperand>(Op);
591*9880d681SAndroid Build Coastguard Worker }
592*9880d681SAndroid Build Coastguard Worker
CreateImm__anon6bb6924c0111::HexagonOperand593*9880d681SAndroid Build Coastguard Worker static std::unique_ptr<HexagonOperand> CreateImm(const MCExpr *Val, SMLoc S,
594*9880d681SAndroid Build Coastguard Worker SMLoc E) {
595*9880d681SAndroid Build Coastguard Worker HexagonOperand *Op = new HexagonOperand(Immediate);
596*9880d681SAndroid Build Coastguard Worker Op->Imm.Val = Val;
597*9880d681SAndroid Build Coastguard Worker Op->StartLoc = S;
598*9880d681SAndroid Build Coastguard Worker Op->EndLoc = E;
599*9880d681SAndroid Build Coastguard Worker return std::unique_ptr<HexagonOperand>(Op);
600*9880d681SAndroid Build Coastguard Worker }
601*9880d681SAndroid Build Coastguard Worker };
602*9880d681SAndroid Build Coastguard Worker
603*9880d681SAndroid Build Coastguard Worker } // end anonymous namespace.
604*9880d681SAndroid Build Coastguard Worker
print(raw_ostream & OS) const605*9880d681SAndroid Build Coastguard Worker void HexagonOperand::print(raw_ostream &OS) const {
606*9880d681SAndroid Build Coastguard Worker switch (Kind) {
607*9880d681SAndroid Build Coastguard Worker case Immediate:
608*9880d681SAndroid Build Coastguard Worker getImm()->print(OS, nullptr);
609*9880d681SAndroid Build Coastguard Worker break;
610*9880d681SAndroid Build Coastguard Worker case Register:
611*9880d681SAndroid Build Coastguard Worker OS << "<register R";
612*9880d681SAndroid Build Coastguard Worker OS << getReg() << ">";
613*9880d681SAndroid Build Coastguard Worker break;
614*9880d681SAndroid Build Coastguard Worker case Token:
615*9880d681SAndroid Build Coastguard Worker OS << "'" << getToken() << "'";
616*9880d681SAndroid Build Coastguard Worker break;
617*9880d681SAndroid Build Coastguard Worker }
618*9880d681SAndroid Build Coastguard Worker }
619*9880d681SAndroid Build Coastguard Worker
finishBundle(SMLoc IDLoc,MCStreamer & Out)620*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::finishBundle(SMLoc IDLoc, MCStreamer &Out) {
621*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << "Bundle:");
622*9880d681SAndroid Build Coastguard Worker DEBUG(MCB.dump_pretty(dbgs()));
623*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << "--\n");
624*9880d681SAndroid Build Coastguard Worker
625*9880d681SAndroid Build Coastguard Worker // Check the bundle for errors.
626*9880d681SAndroid Build Coastguard Worker const MCRegisterInfo *RI = getContext().getRegisterInfo();
627*9880d681SAndroid Build Coastguard Worker HexagonMCChecker Check(MCII, getSTI(), MCB, MCB, *RI);
628*9880d681SAndroid Build Coastguard Worker
629*9880d681SAndroid Build Coastguard Worker bool CheckOk = HexagonMCInstrInfo::canonicalizePacket(MCII, getSTI(),
630*9880d681SAndroid Build Coastguard Worker getContext(), MCB,
631*9880d681SAndroid Build Coastguard Worker &Check);
632*9880d681SAndroid Build Coastguard Worker
633*9880d681SAndroid Build Coastguard Worker while (Check.getNextErrInfo() == true) {
634*9880d681SAndroid Build Coastguard Worker unsigned Reg = Check.getErrRegister();
635*9880d681SAndroid Build Coastguard Worker Twine R(RI->getName(Reg));
636*9880d681SAndroid Build Coastguard Worker
637*9880d681SAndroid Build Coastguard Worker uint64_t Err = Check.getError();
638*9880d681SAndroid Build Coastguard Worker if (Err != HexagonMCErrInfo::CHECK_SUCCESS) {
639*9880d681SAndroid Build Coastguard Worker if (HexagonMCErrInfo::CHECK_ERROR_BRANCHES & Err)
640*9880d681SAndroid Build Coastguard Worker Error(IDLoc,
641*9880d681SAndroid Build Coastguard Worker "unconditional branch cannot precede another branch in packet");
642*9880d681SAndroid Build Coastguard Worker
643*9880d681SAndroid Build Coastguard Worker if (HexagonMCErrInfo::CHECK_ERROR_NEWP & Err ||
644*9880d681SAndroid Build Coastguard Worker HexagonMCErrInfo::CHECK_ERROR_NEWV & Err)
645*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "register `" + R +
646*9880d681SAndroid Build Coastguard Worker "' used with `.new' "
647*9880d681SAndroid Build Coastguard Worker "but not validly modified in the same packet");
648*9880d681SAndroid Build Coastguard Worker
649*9880d681SAndroid Build Coastguard Worker if (HexagonMCErrInfo::CHECK_ERROR_REGISTERS & Err)
650*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "register `" + R + "' modified more than once");
651*9880d681SAndroid Build Coastguard Worker
652*9880d681SAndroid Build Coastguard Worker if (HexagonMCErrInfo::CHECK_ERROR_READONLY & Err)
653*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "cannot write to read-only register `" + R + "'");
654*9880d681SAndroid Build Coastguard Worker
655*9880d681SAndroid Build Coastguard Worker if (HexagonMCErrInfo::CHECK_ERROR_LOOP & Err)
656*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "loop-setup and some branch instructions "
657*9880d681SAndroid Build Coastguard Worker "cannot be in the same packet");
658*9880d681SAndroid Build Coastguard Worker
659*9880d681SAndroid Build Coastguard Worker if (HexagonMCErrInfo::CHECK_ERROR_ENDLOOP & Err) {
660*9880d681SAndroid Build Coastguard Worker Twine N(HexagonMCInstrInfo::isInnerLoop(MCB) ? '0' : '1');
661*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "packet marked with `:endloop" + N + "' " +
662*9880d681SAndroid Build Coastguard Worker "cannot contain instructions that modify register " +
663*9880d681SAndroid Build Coastguard Worker "`" + R + "'");
664*9880d681SAndroid Build Coastguard Worker }
665*9880d681SAndroid Build Coastguard Worker
666*9880d681SAndroid Build Coastguard Worker if (HexagonMCErrInfo::CHECK_ERROR_SOLO & Err)
667*9880d681SAndroid Build Coastguard Worker Error(IDLoc,
668*9880d681SAndroid Build Coastguard Worker "instruction cannot appear in packet with other instructions");
669*9880d681SAndroid Build Coastguard Worker
670*9880d681SAndroid Build Coastguard Worker if (HexagonMCErrInfo::CHECK_ERROR_NOSLOTS & Err)
671*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "too many slots used in packet");
672*9880d681SAndroid Build Coastguard Worker
673*9880d681SAndroid Build Coastguard Worker if (Err & HexagonMCErrInfo::CHECK_ERROR_SHUFFLE) {
674*9880d681SAndroid Build Coastguard Worker uint64_t Erm = Check.getShuffleError();
675*9880d681SAndroid Build Coastguard Worker
676*9880d681SAndroid Build Coastguard Worker if (HexagonShuffler::SHUFFLE_ERROR_INVALID == Erm)
677*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "invalid instruction packet");
678*9880d681SAndroid Build Coastguard Worker else if (HexagonShuffler::SHUFFLE_ERROR_STORES == Erm)
679*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "invalid instruction packet: too many stores");
680*9880d681SAndroid Build Coastguard Worker else if (HexagonShuffler::SHUFFLE_ERROR_LOADS == Erm)
681*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "invalid instruction packet: too many loads");
682*9880d681SAndroid Build Coastguard Worker else if (HexagonShuffler::SHUFFLE_ERROR_BRANCHES == Erm)
683*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "too many branches in packet");
684*9880d681SAndroid Build Coastguard Worker else if (HexagonShuffler::SHUFFLE_ERROR_NOSLOTS == Erm)
685*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "invalid instruction packet: out of slots");
686*9880d681SAndroid Build Coastguard Worker else if (HexagonShuffler::SHUFFLE_ERROR_SLOTS == Erm)
687*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "invalid instruction packet: slot error");
688*9880d681SAndroid Build Coastguard Worker else if (HexagonShuffler::SHUFFLE_ERROR_ERRATA2 == Erm)
689*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "v60 packet violation");
690*9880d681SAndroid Build Coastguard Worker else if (HexagonShuffler::SHUFFLE_ERROR_STORE_LOAD_CONFLICT == Erm)
691*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "slot 0 instruction does not allow slot 1 store");
692*9880d681SAndroid Build Coastguard Worker else
693*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "unknown error in instruction packet");
694*9880d681SAndroid Build Coastguard Worker }
695*9880d681SAndroid Build Coastguard Worker }
696*9880d681SAndroid Build Coastguard Worker
697*9880d681SAndroid Build Coastguard Worker unsigned Warn = Check.getWarning();
698*9880d681SAndroid Build Coastguard Worker if (Warn != HexagonMCErrInfo::CHECK_SUCCESS) {
699*9880d681SAndroid Build Coastguard Worker if (HexagonMCErrInfo::CHECK_WARN_CURRENT & Warn)
700*9880d681SAndroid Build Coastguard Worker Warning(IDLoc, "register `" + R + "' used with `.cur' "
701*9880d681SAndroid Build Coastguard Worker "but not used in the same packet");
702*9880d681SAndroid Build Coastguard Worker else if (HexagonMCErrInfo::CHECK_WARN_TEMPORARY & Warn)
703*9880d681SAndroid Build Coastguard Worker Warning(IDLoc, "register `" + R + "' used with `.tmp' "
704*9880d681SAndroid Build Coastguard Worker "but not used in the same packet");
705*9880d681SAndroid Build Coastguard Worker }
706*9880d681SAndroid Build Coastguard Worker }
707*9880d681SAndroid Build Coastguard Worker
708*9880d681SAndroid Build Coastguard Worker if (CheckOk) {
709*9880d681SAndroid Build Coastguard Worker MCB.setLoc(IDLoc);
710*9880d681SAndroid Build Coastguard Worker if (HexagonMCInstrInfo::bundleSize(MCB) == 0) {
711*9880d681SAndroid Build Coastguard Worker assert(!HexagonMCInstrInfo::isInnerLoop(MCB));
712*9880d681SAndroid Build Coastguard Worker assert(!HexagonMCInstrInfo::isOuterLoop(MCB));
713*9880d681SAndroid Build Coastguard Worker // Empty packets are valid yet aren't emitted
714*9880d681SAndroid Build Coastguard Worker return false;
715*9880d681SAndroid Build Coastguard Worker }
716*9880d681SAndroid Build Coastguard Worker Out.EmitInstruction(MCB, getSTI());
717*9880d681SAndroid Build Coastguard Worker } else {
718*9880d681SAndroid Build Coastguard Worker // If compounding and duplexing didn't reduce the size below
719*9880d681SAndroid Build Coastguard Worker // 4 or less we have a packet that is too big.
720*9880d681SAndroid Build Coastguard Worker if (HexagonMCInstrInfo::bundleSize(MCB) > HEXAGON_PACKET_SIZE) {
721*9880d681SAndroid Build Coastguard Worker Error(IDLoc, "invalid instruction packet: out of slots");
722*9880d681SAndroid Build Coastguard Worker return true; // Error
723*9880d681SAndroid Build Coastguard Worker }
724*9880d681SAndroid Build Coastguard Worker }
725*9880d681SAndroid Build Coastguard Worker
726*9880d681SAndroid Build Coastguard Worker return false; // No error
727*9880d681SAndroid Build Coastguard Worker }
728*9880d681SAndroid Build Coastguard Worker
matchBundleOptions()729*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::matchBundleOptions() {
730*9880d681SAndroid Build Coastguard Worker MCAsmParser &Parser = getParser();
731*9880d681SAndroid Build Coastguard Worker while (true) {
732*9880d681SAndroid Build Coastguard Worker if (!Parser.getTok().is(AsmToken::Colon))
733*9880d681SAndroid Build Coastguard Worker return false;
734*9880d681SAndroid Build Coastguard Worker Lex();
735*9880d681SAndroid Build Coastguard Worker StringRef Option = Parser.getTok().getString();
736*9880d681SAndroid Build Coastguard Worker if (Option.compare_lower("endloop0") == 0)
737*9880d681SAndroid Build Coastguard Worker HexagonMCInstrInfo::setInnerLoop(MCB);
738*9880d681SAndroid Build Coastguard Worker else if (Option.compare_lower("endloop1") == 0)
739*9880d681SAndroid Build Coastguard Worker HexagonMCInstrInfo::setOuterLoop(MCB);
740*9880d681SAndroid Build Coastguard Worker else if (Option.compare_lower("mem_noshuf") == 0)
741*9880d681SAndroid Build Coastguard Worker HexagonMCInstrInfo::setMemReorderDisabled(MCB);
742*9880d681SAndroid Build Coastguard Worker else if (Option.compare_lower("mem_shuf") == 0)
743*9880d681SAndroid Build Coastguard Worker HexagonMCInstrInfo::setMemStoreReorderEnabled(MCB);
744*9880d681SAndroid Build Coastguard Worker else
745*9880d681SAndroid Build Coastguard Worker return true;
746*9880d681SAndroid Build Coastguard Worker Lex();
747*9880d681SAndroid Build Coastguard Worker }
748*9880d681SAndroid Build Coastguard Worker }
749*9880d681SAndroid Build Coastguard Worker
750*9880d681SAndroid Build Coastguard Worker // For instruction aliases, immediates are generated rather than
751*9880d681SAndroid Build Coastguard Worker // MCConstantExpr. Convert them for uniform MCExpr.
752*9880d681SAndroid Build Coastguard Worker // Also check for signed/unsigned mismatches and warn
canonicalizeImmediates(MCInst & MCI)753*9880d681SAndroid Build Coastguard Worker void HexagonAsmParser::canonicalizeImmediates(MCInst &MCI) {
754*9880d681SAndroid Build Coastguard Worker MCInst NewInst;
755*9880d681SAndroid Build Coastguard Worker NewInst.setOpcode(MCI.getOpcode());
756*9880d681SAndroid Build Coastguard Worker for (MCOperand &I : MCI)
757*9880d681SAndroid Build Coastguard Worker if (I.isImm()) {
758*9880d681SAndroid Build Coastguard Worker int64_t Value (I.getImm());
759*9880d681SAndroid Build Coastguard Worker NewInst.addOperand(MCOperand::createExpr(HexagonMCExpr::create(
760*9880d681SAndroid Build Coastguard Worker MCConstantExpr::create(Value, getContext()), getContext())));
761*9880d681SAndroid Build Coastguard Worker }
762*9880d681SAndroid Build Coastguard Worker else {
763*9880d681SAndroid Build Coastguard Worker if (I.isExpr() && cast<HexagonMCExpr>(I.getExpr())->signMismatch() &&
764*9880d681SAndroid Build Coastguard Worker WarnSignedMismatch)
765*9880d681SAndroid Build Coastguard Worker Warning (MCI.getLoc(), "Signed/Unsigned mismatch");
766*9880d681SAndroid Build Coastguard Worker NewInst.addOperand(I);
767*9880d681SAndroid Build Coastguard Worker }
768*9880d681SAndroid Build Coastguard Worker MCI = NewInst;
769*9880d681SAndroid Build Coastguard Worker }
770*9880d681SAndroid Build Coastguard Worker
matchOneInstruction(MCInst & MCI,SMLoc IDLoc,OperandVector & InstOperands,uint64_t & ErrorInfo,bool MatchingInlineAsm)771*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::matchOneInstruction(MCInst &MCI, SMLoc IDLoc,
772*9880d681SAndroid Build Coastguard Worker OperandVector &InstOperands,
773*9880d681SAndroid Build Coastguard Worker uint64_t &ErrorInfo,
774*9880d681SAndroid Build Coastguard Worker bool MatchingInlineAsm) {
775*9880d681SAndroid Build Coastguard Worker // Perform matching with tablegen asmmatcher generated function
776*9880d681SAndroid Build Coastguard Worker int result =
777*9880d681SAndroid Build Coastguard Worker MatchInstructionImpl(InstOperands, MCI, ErrorInfo, MatchingInlineAsm);
778*9880d681SAndroid Build Coastguard Worker if (result == Match_Success) {
779*9880d681SAndroid Build Coastguard Worker MCI.setLoc(IDLoc);
780*9880d681SAndroid Build Coastguard Worker canonicalizeImmediates(MCI);
781*9880d681SAndroid Build Coastguard Worker result = processInstruction(MCI, InstOperands, IDLoc);
782*9880d681SAndroid Build Coastguard Worker
783*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << "Insn:");
784*9880d681SAndroid Build Coastguard Worker DEBUG(MCI.dump_pretty(dbgs()));
785*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << "\n\n");
786*9880d681SAndroid Build Coastguard Worker
787*9880d681SAndroid Build Coastguard Worker MCI.setLoc(IDLoc);
788*9880d681SAndroid Build Coastguard Worker }
789*9880d681SAndroid Build Coastguard Worker
790*9880d681SAndroid Build Coastguard Worker // Create instruction operand for bundle instruction
791*9880d681SAndroid Build Coastguard Worker // Break this into a separate function Code here is less readable
792*9880d681SAndroid Build Coastguard Worker // Think about how to get an instruction error to report correctly.
793*9880d681SAndroid Build Coastguard Worker // SMLoc will return the "{"
794*9880d681SAndroid Build Coastguard Worker switch (result) {
795*9880d681SAndroid Build Coastguard Worker default:
796*9880d681SAndroid Build Coastguard Worker break;
797*9880d681SAndroid Build Coastguard Worker case Match_Success:
798*9880d681SAndroid Build Coastguard Worker return false;
799*9880d681SAndroid Build Coastguard Worker case Match_MissingFeature:
800*9880d681SAndroid Build Coastguard Worker return Error(IDLoc, "invalid instruction");
801*9880d681SAndroid Build Coastguard Worker case Match_MnemonicFail:
802*9880d681SAndroid Build Coastguard Worker return Error(IDLoc, "unrecognized instruction");
803*9880d681SAndroid Build Coastguard Worker case Match_InvalidOperand:
804*9880d681SAndroid Build Coastguard Worker SMLoc ErrorLoc = IDLoc;
805*9880d681SAndroid Build Coastguard Worker if (ErrorInfo != ~0U) {
806*9880d681SAndroid Build Coastguard Worker if (ErrorInfo >= InstOperands.size())
807*9880d681SAndroid Build Coastguard Worker return Error(IDLoc, "too few operands for instruction");
808*9880d681SAndroid Build Coastguard Worker
809*9880d681SAndroid Build Coastguard Worker ErrorLoc = (static_cast<HexagonOperand *>(InstOperands[ErrorInfo].get()))
810*9880d681SAndroid Build Coastguard Worker ->getStartLoc();
811*9880d681SAndroid Build Coastguard Worker if (ErrorLoc == SMLoc())
812*9880d681SAndroid Build Coastguard Worker ErrorLoc = IDLoc;
813*9880d681SAndroid Build Coastguard Worker }
814*9880d681SAndroid Build Coastguard Worker return Error(ErrorLoc, "invalid operand for instruction");
815*9880d681SAndroid Build Coastguard Worker }
816*9880d681SAndroid Build Coastguard Worker llvm_unreachable("Implement any new match types added!");
817*9880d681SAndroid Build Coastguard Worker }
818*9880d681SAndroid Build Coastguard Worker
MatchAndEmitInstruction(SMLoc IDLoc,unsigned & Opcode,OperandVector & Operands,MCStreamer & Out,uint64_t & ErrorInfo,bool MatchingInlineAsm)819*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::MatchAndEmitInstruction(SMLoc IDLoc, unsigned &Opcode,
820*9880d681SAndroid Build Coastguard Worker OperandVector &Operands,
821*9880d681SAndroid Build Coastguard Worker MCStreamer &Out,
822*9880d681SAndroid Build Coastguard Worker uint64_t &ErrorInfo,
823*9880d681SAndroid Build Coastguard Worker bool MatchingInlineAsm) {
824*9880d681SAndroid Build Coastguard Worker if (!InBrackets) {
825*9880d681SAndroid Build Coastguard Worker MCB.clear();
826*9880d681SAndroid Build Coastguard Worker MCB.addOperand(MCOperand::createImm(0));
827*9880d681SAndroid Build Coastguard Worker }
828*9880d681SAndroid Build Coastguard Worker HexagonOperand &FirstOperand = static_cast<HexagonOperand &>(*Operands[0]);
829*9880d681SAndroid Build Coastguard Worker if (FirstOperand.isToken() && FirstOperand.getToken() == "{") {
830*9880d681SAndroid Build Coastguard Worker assert(Operands.size() == 1 && "Brackets should be by themselves");
831*9880d681SAndroid Build Coastguard Worker if (InBrackets) {
832*9880d681SAndroid Build Coastguard Worker getParser().Error(IDLoc, "Already in a packet");
833*9880d681SAndroid Build Coastguard Worker return true;
834*9880d681SAndroid Build Coastguard Worker }
835*9880d681SAndroid Build Coastguard Worker InBrackets = true;
836*9880d681SAndroid Build Coastguard Worker return false;
837*9880d681SAndroid Build Coastguard Worker }
838*9880d681SAndroid Build Coastguard Worker if (FirstOperand.isToken() && FirstOperand.getToken() == "}") {
839*9880d681SAndroid Build Coastguard Worker assert(Operands.size() == 1 && "Brackets should be by themselves");
840*9880d681SAndroid Build Coastguard Worker if (!InBrackets) {
841*9880d681SAndroid Build Coastguard Worker getParser().Error(IDLoc, "Not in a packet");
842*9880d681SAndroid Build Coastguard Worker return true;
843*9880d681SAndroid Build Coastguard Worker }
844*9880d681SAndroid Build Coastguard Worker InBrackets = false;
845*9880d681SAndroid Build Coastguard Worker if (matchBundleOptions())
846*9880d681SAndroid Build Coastguard Worker return true;
847*9880d681SAndroid Build Coastguard Worker return finishBundle(IDLoc, Out);
848*9880d681SAndroid Build Coastguard Worker }
849*9880d681SAndroid Build Coastguard Worker MCInst *SubInst = new (getParser().getContext()) MCInst;
850*9880d681SAndroid Build Coastguard Worker if (matchOneInstruction(*SubInst, IDLoc, Operands, ErrorInfo,
851*9880d681SAndroid Build Coastguard Worker MatchingInlineAsm))
852*9880d681SAndroid Build Coastguard Worker return true;
853*9880d681SAndroid Build Coastguard Worker HexagonMCInstrInfo::extendIfNeeded(
854*9880d681SAndroid Build Coastguard Worker getParser().getContext(), MCII, MCB, *SubInst);
855*9880d681SAndroid Build Coastguard Worker MCB.addOperand(MCOperand::createInst(SubInst));
856*9880d681SAndroid Build Coastguard Worker if (!InBrackets)
857*9880d681SAndroid Build Coastguard Worker return finishBundle(IDLoc, Out);
858*9880d681SAndroid Build Coastguard Worker return false;
859*9880d681SAndroid Build Coastguard Worker }
860*9880d681SAndroid Build Coastguard Worker
861*9880d681SAndroid Build Coastguard Worker /// ParseDirective parses the Hexagon specific directives
ParseDirective(AsmToken DirectiveID)862*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::ParseDirective(AsmToken DirectiveID) {
863*9880d681SAndroid Build Coastguard Worker StringRef IDVal = DirectiveID.getIdentifier();
864*9880d681SAndroid Build Coastguard Worker if ((IDVal.lower() == ".word") || (IDVal.lower() == ".4byte"))
865*9880d681SAndroid Build Coastguard Worker return ParseDirectiveValue(4, DirectiveID.getLoc());
866*9880d681SAndroid Build Coastguard Worker if (IDVal.lower() == ".short" || IDVal.lower() == ".hword" ||
867*9880d681SAndroid Build Coastguard Worker IDVal.lower() == ".half")
868*9880d681SAndroid Build Coastguard Worker return ParseDirectiveValue(2, DirectiveID.getLoc());
869*9880d681SAndroid Build Coastguard Worker if (IDVal.lower() == ".falign")
870*9880d681SAndroid Build Coastguard Worker return ParseDirectiveFalign(256, DirectiveID.getLoc());
871*9880d681SAndroid Build Coastguard Worker if ((IDVal.lower() == ".lcomm") || (IDVal.lower() == ".lcommon"))
872*9880d681SAndroid Build Coastguard Worker return ParseDirectiveComm(true, DirectiveID.getLoc());
873*9880d681SAndroid Build Coastguard Worker if ((IDVal.lower() == ".comm") || (IDVal.lower() == ".common"))
874*9880d681SAndroid Build Coastguard Worker return ParseDirectiveComm(false, DirectiveID.getLoc());
875*9880d681SAndroid Build Coastguard Worker if (IDVal.lower() == ".subsection")
876*9880d681SAndroid Build Coastguard Worker return ParseDirectiveSubsection(DirectiveID.getLoc());
877*9880d681SAndroid Build Coastguard Worker
878*9880d681SAndroid Build Coastguard Worker return true;
879*9880d681SAndroid Build Coastguard Worker }
ParseDirectiveSubsection(SMLoc L)880*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::ParseDirectiveSubsection(SMLoc L) {
881*9880d681SAndroid Build Coastguard Worker const MCExpr *Subsection = 0;
882*9880d681SAndroid Build Coastguard Worker int64_t Res;
883*9880d681SAndroid Build Coastguard Worker
884*9880d681SAndroid Build Coastguard Worker assert((getLexer().isNot(AsmToken::EndOfStatement)) &&
885*9880d681SAndroid Build Coastguard Worker "Invalid subsection directive");
886*9880d681SAndroid Build Coastguard Worker getParser().parseExpression(Subsection);
887*9880d681SAndroid Build Coastguard Worker
888*9880d681SAndroid Build Coastguard Worker if (!Subsection->evaluateAsAbsolute(Res))
889*9880d681SAndroid Build Coastguard Worker return Error(L, "Cannot evaluate subsection number");
890*9880d681SAndroid Build Coastguard Worker
891*9880d681SAndroid Build Coastguard Worker if (getLexer().isNot(AsmToken::EndOfStatement))
892*9880d681SAndroid Build Coastguard Worker return TokError("unexpected token in directive");
893*9880d681SAndroid Build Coastguard Worker
894*9880d681SAndroid Build Coastguard Worker // 0-8192 is the hard-coded range in MCObjectStreamper.cpp, this keeps the
895*9880d681SAndroid Build Coastguard Worker // negative subsections together and in the same order but at the opposite
896*9880d681SAndroid Build Coastguard Worker // end of the section. Only legacy hexagon-gcc created assembly code
897*9880d681SAndroid Build Coastguard Worker // used negative subsections.
898*9880d681SAndroid Build Coastguard Worker if ((Res < 0) && (Res > -8193))
899*9880d681SAndroid Build Coastguard Worker Subsection = HexagonMCExpr::create(
900*9880d681SAndroid Build Coastguard Worker MCConstantExpr::create(8192 + Res, getContext()), getContext());
901*9880d681SAndroid Build Coastguard Worker
902*9880d681SAndroid Build Coastguard Worker getStreamer().SubSection(Subsection);
903*9880d681SAndroid Build Coastguard Worker return false;
904*9880d681SAndroid Build Coastguard Worker }
905*9880d681SAndroid Build Coastguard Worker
906*9880d681SAndroid Build Coastguard Worker /// ::= .falign [expression]
ParseDirectiveFalign(unsigned Size,SMLoc L)907*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::ParseDirectiveFalign(unsigned Size, SMLoc L) {
908*9880d681SAndroid Build Coastguard Worker
909*9880d681SAndroid Build Coastguard Worker int64_t MaxBytesToFill = 15;
910*9880d681SAndroid Build Coastguard Worker
911*9880d681SAndroid Build Coastguard Worker // if there is an arguement
912*9880d681SAndroid Build Coastguard Worker if (getLexer().isNot(AsmToken::EndOfStatement)) {
913*9880d681SAndroid Build Coastguard Worker const MCExpr *Value;
914*9880d681SAndroid Build Coastguard Worker SMLoc ExprLoc = L;
915*9880d681SAndroid Build Coastguard Worker
916*9880d681SAndroid Build Coastguard Worker // Make sure we have a number (false is returned if expression is a number)
917*9880d681SAndroid Build Coastguard Worker if (getParser().parseExpression(Value) == false) {
918*9880d681SAndroid Build Coastguard Worker // Make sure this is a number that is in range
919*9880d681SAndroid Build Coastguard Worker const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(Value);
920*9880d681SAndroid Build Coastguard Worker uint64_t IntValue = MCE->getValue();
921*9880d681SAndroid Build Coastguard Worker if (!isUIntN(Size, IntValue) && !isIntN(Size, IntValue))
922*9880d681SAndroid Build Coastguard Worker return Error(ExprLoc, "literal value out of range (256) for falign");
923*9880d681SAndroid Build Coastguard Worker MaxBytesToFill = IntValue;
924*9880d681SAndroid Build Coastguard Worker Lex();
925*9880d681SAndroid Build Coastguard Worker } else {
926*9880d681SAndroid Build Coastguard Worker return Error(ExprLoc, "not a valid expression for falign directive");
927*9880d681SAndroid Build Coastguard Worker }
928*9880d681SAndroid Build Coastguard Worker }
929*9880d681SAndroid Build Coastguard Worker
930*9880d681SAndroid Build Coastguard Worker getTargetStreamer().emitFAlign(16, MaxBytesToFill);
931*9880d681SAndroid Build Coastguard Worker Lex();
932*9880d681SAndroid Build Coastguard Worker
933*9880d681SAndroid Build Coastguard Worker return false;
934*9880d681SAndroid Build Coastguard Worker }
935*9880d681SAndroid Build Coastguard Worker
936*9880d681SAndroid Build Coastguard Worker /// ::= .word [ expression (, expression)* ]
ParseDirectiveValue(unsigned Size,SMLoc L)937*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::ParseDirectiveValue(unsigned Size, SMLoc L) {
938*9880d681SAndroid Build Coastguard Worker if (getLexer().isNot(AsmToken::EndOfStatement)) {
939*9880d681SAndroid Build Coastguard Worker
940*9880d681SAndroid Build Coastguard Worker for (;;) {
941*9880d681SAndroid Build Coastguard Worker const MCExpr *Value;
942*9880d681SAndroid Build Coastguard Worker SMLoc ExprLoc = L;
943*9880d681SAndroid Build Coastguard Worker if (getParser().parseExpression(Value))
944*9880d681SAndroid Build Coastguard Worker return true;
945*9880d681SAndroid Build Coastguard Worker
946*9880d681SAndroid Build Coastguard Worker // Special case constant expressions to match code generator.
947*9880d681SAndroid Build Coastguard Worker if (const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(Value)) {
948*9880d681SAndroid Build Coastguard Worker assert(Size <= 8 && "Invalid size");
949*9880d681SAndroid Build Coastguard Worker uint64_t IntValue = MCE->getValue();
950*9880d681SAndroid Build Coastguard Worker if (!isUIntN(8 * Size, IntValue) && !isIntN(8 * Size, IntValue))
951*9880d681SAndroid Build Coastguard Worker return Error(ExprLoc, "literal value out of range for directive");
952*9880d681SAndroid Build Coastguard Worker getStreamer().EmitIntValue(IntValue, Size);
953*9880d681SAndroid Build Coastguard Worker } else
954*9880d681SAndroid Build Coastguard Worker getStreamer().EmitValue(Value, Size);
955*9880d681SAndroid Build Coastguard Worker
956*9880d681SAndroid Build Coastguard Worker if (getLexer().is(AsmToken::EndOfStatement))
957*9880d681SAndroid Build Coastguard Worker break;
958*9880d681SAndroid Build Coastguard Worker
959*9880d681SAndroid Build Coastguard Worker // FIXME: Improve diagnostic.
960*9880d681SAndroid Build Coastguard Worker if (getLexer().isNot(AsmToken::Comma))
961*9880d681SAndroid Build Coastguard Worker return TokError("unexpected token in directive");
962*9880d681SAndroid Build Coastguard Worker Lex();
963*9880d681SAndroid Build Coastguard Worker }
964*9880d681SAndroid Build Coastguard Worker }
965*9880d681SAndroid Build Coastguard Worker
966*9880d681SAndroid Build Coastguard Worker Lex();
967*9880d681SAndroid Build Coastguard Worker return false;
968*9880d681SAndroid Build Coastguard Worker }
969*9880d681SAndroid Build Coastguard Worker
970*9880d681SAndroid Build Coastguard Worker // This is largely a copy of AsmParser's ParseDirectiveComm extended to
971*9880d681SAndroid Build Coastguard Worker // accept a 3rd argument, AccessAlignment which indicates the smallest
972*9880d681SAndroid Build Coastguard Worker // memory access made to the symbol, expressed in bytes. If no
973*9880d681SAndroid Build Coastguard Worker // AccessAlignment is specified it defaults to the Alignment Value.
974*9880d681SAndroid Build Coastguard Worker // Hexagon's .lcomm:
975*9880d681SAndroid Build Coastguard Worker // .lcomm Symbol, Length, Alignment, AccessAlignment
ParseDirectiveComm(bool IsLocal,SMLoc Loc)976*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::ParseDirectiveComm(bool IsLocal, SMLoc Loc) {
977*9880d681SAndroid Build Coastguard Worker // FIXME: need better way to detect if AsmStreamer (upstream removed
978*9880d681SAndroid Build Coastguard Worker // getKind())
979*9880d681SAndroid Build Coastguard Worker if (getStreamer().hasRawTextSupport())
980*9880d681SAndroid Build Coastguard Worker return true; // Only object file output requires special treatment.
981*9880d681SAndroid Build Coastguard Worker
982*9880d681SAndroid Build Coastguard Worker StringRef Name;
983*9880d681SAndroid Build Coastguard Worker if (getParser().parseIdentifier(Name))
984*9880d681SAndroid Build Coastguard Worker return TokError("expected identifier in directive");
985*9880d681SAndroid Build Coastguard Worker // Handle the identifier as the key symbol.
986*9880d681SAndroid Build Coastguard Worker MCSymbol *Sym = getContext().getOrCreateSymbol(Name);
987*9880d681SAndroid Build Coastguard Worker
988*9880d681SAndroid Build Coastguard Worker if (getLexer().isNot(AsmToken::Comma))
989*9880d681SAndroid Build Coastguard Worker return TokError("unexpected token in directive");
990*9880d681SAndroid Build Coastguard Worker Lex();
991*9880d681SAndroid Build Coastguard Worker
992*9880d681SAndroid Build Coastguard Worker int64_t Size;
993*9880d681SAndroid Build Coastguard Worker SMLoc SizeLoc = getLexer().getLoc();
994*9880d681SAndroid Build Coastguard Worker if (getParser().parseAbsoluteExpression(Size))
995*9880d681SAndroid Build Coastguard Worker return true;
996*9880d681SAndroid Build Coastguard Worker
997*9880d681SAndroid Build Coastguard Worker int64_t ByteAlignment = 1;
998*9880d681SAndroid Build Coastguard Worker SMLoc ByteAlignmentLoc;
999*9880d681SAndroid Build Coastguard Worker if (getLexer().is(AsmToken::Comma)) {
1000*9880d681SAndroid Build Coastguard Worker Lex();
1001*9880d681SAndroid Build Coastguard Worker ByteAlignmentLoc = getLexer().getLoc();
1002*9880d681SAndroid Build Coastguard Worker if (getParser().parseAbsoluteExpression(ByteAlignment))
1003*9880d681SAndroid Build Coastguard Worker return true;
1004*9880d681SAndroid Build Coastguard Worker if (!isPowerOf2_64(ByteAlignment))
1005*9880d681SAndroid Build Coastguard Worker return Error(ByteAlignmentLoc, "alignment must be a power of 2");
1006*9880d681SAndroid Build Coastguard Worker }
1007*9880d681SAndroid Build Coastguard Worker
1008*9880d681SAndroid Build Coastguard Worker int64_t AccessAlignment = 0;
1009*9880d681SAndroid Build Coastguard Worker if (getLexer().is(AsmToken::Comma)) {
1010*9880d681SAndroid Build Coastguard Worker // The optional access argument specifies the size of the smallest memory
1011*9880d681SAndroid Build Coastguard Worker // access to be made to the symbol, expressed in bytes.
1012*9880d681SAndroid Build Coastguard Worker SMLoc AccessAlignmentLoc;
1013*9880d681SAndroid Build Coastguard Worker Lex();
1014*9880d681SAndroid Build Coastguard Worker AccessAlignmentLoc = getLexer().getLoc();
1015*9880d681SAndroid Build Coastguard Worker if (getParser().parseAbsoluteExpression(AccessAlignment))
1016*9880d681SAndroid Build Coastguard Worker return true;
1017*9880d681SAndroid Build Coastguard Worker
1018*9880d681SAndroid Build Coastguard Worker if (!isPowerOf2_64(AccessAlignment))
1019*9880d681SAndroid Build Coastguard Worker return Error(AccessAlignmentLoc, "access alignment must be a power of 2");
1020*9880d681SAndroid Build Coastguard Worker }
1021*9880d681SAndroid Build Coastguard Worker
1022*9880d681SAndroid Build Coastguard Worker if (getLexer().isNot(AsmToken::EndOfStatement))
1023*9880d681SAndroid Build Coastguard Worker return TokError("unexpected token in '.comm' or '.lcomm' directive");
1024*9880d681SAndroid Build Coastguard Worker
1025*9880d681SAndroid Build Coastguard Worker Lex();
1026*9880d681SAndroid Build Coastguard Worker
1027*9880d681SAndroid Build Coastguard Worker // NOTE: a size of zero for a .comm should create a undefined symbol
1028*9880d681SAndroid Build Coastguard Worker // but a size of .lcomm creates a bss symbol of size zero.
1029*9880d681SAndroid Build Coastguard Worker if (Size < 0)
1030*9880d681SAndroid Build Coastguard Worker return Error(SizeLoc, "invalid '.comm' or '.lcomm' directive size, can't "
1031*9880d681SAndroid Build Coastguard Worker "be less than zero");
1032*9880d681SAndroid Build Coastguard Worker
1033*9880d681SAndroid Build Coastguard Worker // NOTE: The alignment in the directive is a power of 2 value, the assembler
1034*9880d681SAndroid Build Coastguard Worker // may internally end up wanting an alignment in bytes.
1035*9880d681SAndroid Build Coastguard Worker // FIXME: Diagnose overflow.
1036*9880d681SAndroid Build Coastguard Worker if (ByteAlignment < 0)
1037*9880d681SAndroid Build Coastguard Worker return Error(ByteAlignmentLoc, "invalid '.comm' or '.lcomm' directive "
1038*9880d681SAndroid Build Coastguard Worker "alignment, can't be less than zero");
1039*9880d681SAndroid Build Coastguard Worker
1040*9880d681SAndroid Build Coastguard Worker if (!Sym->isUndefined())
1041*9880d681SAndroid Build Coastguard Worker return Error(Loc, "invalid symbol redefinition");
1042*9880d681SAndroid Build Coastguard Worker
1043*9880d681SAndroid Build Coastguard Worker HexagonMCELFStreamer &HexagonELFStreamer =
1044*9880d681SAndroid Build Coastguard Worker static_cast<HexagonMCELFStreamer &>(getStreamer());
1045*9880d681SAndroid Build Coastguard Worker if (IsLocal) {
1046*9880d681SAndroid Build Coastguard Worker HexagonELFStreamer.HexagonMCEmitLocalCommonSymbol(Sym, Size, ByteAlignment,
1047*9880d681SAndroid Build Coastguard Worker AccessAlignment);
1048*9880d681SAndroid Build Coastguard Worker return false;
1049*9880d681SAndroid Build Coastguard Worker }
1050*9880d681SAndroid Build Coastguard Worker
1051*9880d681SAndroid Build Coastguard Worker HexagonELFStreamer.HexagonMCEmitCommonSymbol(Sym, Size, ByteAlignment,
1052*9880d681SAndroid Build Coastguard Worker AccessAlignment);
1053*9880d681SAndroid Build Coastguard Worker return false;
1054*9880d681SAndroid Build Coastguard Worker }
1055*9880d681SAndroid Build Coastguard Worker
1056*9880d681SAndroid Build Coastguard Worker // validate register against architecture
RegisterMatchesArch(unsigned MatchNum) const1057*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::RegisterMatchesArch(unsigned MatchNum) const {
1058*9880d681SAndroid Build Coastguard Worker return true;
1059*9880d681SAndroid Build Coastguard Worker }
1060*9880d681SAndroid Build Coastguard Worker
1061*9880d681SAndroid Build Coastguard Worker // extern "C" void LLVMInitializeHexagonAsmLexer();
1062*9880d681SAndroid Build Coastguard Worker
1063*9880d681SAndroid Build Coastguard Worker /// Force static initialization.
LLVMInitializeHexagonAsmParser()1064*9880d681SAndroid Build Coastguard Worker extern "C" void LLVMInitializeHexagonAsmParser() {
1065*9880d681SAndroid Build Coastguard Worker RegisterMCAsmParser<HexagonAsmParser> X(TheHexagonTarget);
1066*9880d681SAndroid Build Coastguard Worker }
1067*9880d681SAndroid Build Coastguard Worker
1068*9880d681SAndroid Build Coastguard Worker #define GET_MATCHER_IMPLEMENTATION
1069*9880d681SAndroid Build Coastguard Worker #define GET_REGISTER_MATCHER
1070*9880d681SAndroid Build Coastguard Worker #include "HexagonGenAsmMatcher.inc"
1071*9880d681SAndroid Build Coastguard Worker
1072*9880d681SAndroid Build Coastguard Worker namespace {
previousEqual(OperandVector & Operands,size_t Index,StringRef String)1073*9880d681SAndroid Build Coastguard Worker bool previousEqual(OperandVector &Operands, size_t Index, StringRef String) {
1074*9880d681SAndroid Build Coastguard Worker if (Index >= Operands.size())
1075*9880d681SAndroid Build Coastguard Worker return false;
1076*9880d681SAndroid Build Coastguard Worker MCParsedAsmOperand &Operand = *Operands[Operands.size() - Index - 1];
1077*9880d681SAndroid Build Coastguard Worker if (!Operand.isToken())
1078*9880d681SAndroid Build Coastguard Worker return false;
1079*9880d681SAndroid Build Coastguard Worker return static_cast<HexagonOperand &>(Operand).getToken().equals_lower(String);
1080*9880d681SAndroid Build Coastguard Worker }
previousIsLoop(OperandVector & Operands,size_t Index)1081*9880d681SAndroid Build Coastguard Worker bool previousIsLoop(OperandVector &Operands, size_t Index) {
1082*9880d681SAndroid Build Coastguard Worker return previousEqual(Operands, Index, "loop0") ||
1083*9880d681SAndroid Build Coastguard Worker previousEqual(Operands, Index, "loop1") ||
1084*9880d681SAndroid Build Coastguard Worker previousEqual(Operands, Index, "sp1loop0") ||
1085*9880d681SAndroid Build Coastguard Worker previousEqual(Operands, Index, "sp2loop0") ||
1086*9880d681SAndroid Build Coastguard Worker previousEqual(Operands, Index, "sp3loop0");
1087*9880d681SAndroid Build Coastguard Worker }
1088*9880d681SAndroid Build Coastguard Worker }
1089*9880d681SAndroid Build Coastguard Worker
splitIdentifier(OperandVector & Operands)1090*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::splitIdentifier(OperandVector &Operands) {
1091*9880d681SAndroid Build Coastguard Worker AsmToken const &Token = getParser().getTok();
1092*9880d681SAndroid Build Coastguard Worker StringRef String = Token.getString();
1093*9880d681SAndroid Build Coastguard Worker SMLoc Loc = Token.getLoc();
1094*9880d681SAndroid Build Coastguard Worker Lex();
1095*9880d681SAndroid Build Coastguard Worker do {
1096*9880d681SAndroid Build Coastguard Worker std::pair<StringRef, StringRef> HeadTail = String.split('.');
1097*9880d681SAndroid Build Coastguard Worker if (!HeadTail.first.empty())
1098*9880d681SAndroid Build Coastguard Worker Operands.push_back(HexagonOperand::CreateToken(HeadTail.first, Loc));
1099*9880d681SAndroid Build Coastguard Worker if (!HeadTail.second.empty())
1100*9880d681SAndroid Build Coastguard Worker Operands.push_back(HexagonOperand::CreateToken(
1101*9880d681SAndroid Build Coastguard Worker String.substr(HeadTail.first.size(), 1), Loc));
1102*9880d681SAndroid Build Coastguard Worker String = HeadTail.second;
1103*9880d681SAndroid Build Coastguard Worker } while (!String.empty());
1104*9880d681SAndroid Build Coastguard Worker return false;
1105*9880d681SAndroid Build Coastguard Worker }
1106*9880d681SAndroid Build Coastguard Worker
parseOperand(OperandVector & Operands)1107*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::parseOperand(OperandVector &Operands) {
1108*9880d681SAndroid Build Coastguard Worker unsigned Register;
1109*9880d681SAndroid Build Coastguard Worker SMLoc Begin;
1110*9880d681SAndroid Build Coastguard Worker SMLoc End;
1111*9880d681SAndroid Build Coastguard Worker MCAsmLexer &Lexer = getLexer();
1112*9880d681SAndroid Build Coastguard Worker if (!ParseRegister(Register, Begin, End)) {
1113*9880d681SAndroid Build Coastguard Worker if (!ErrorMissingParenthesis)
1114*9880d681SAndroid Build Coastguard Worker switch (Register) {
1115*9880d681SAndroid Build Coastguard Worker default:
1116*9880d681SAndroid Build Coastguard Worker break;
1117*9880d681SAndroid Build Coastguard Worker case Hexagon::P0:
1118*9880d681SAndroid Build Coastguard Worker case Hexagon::P1:
1119*9880d681SAndroid Build Coastguard Worker case Hexagon::P2:
1120*9880d681SAndroid Build Coastguard Worker case Hexagon::P3:
1121*9880d681SAndroid Build Coastguard Worker if (previousEqual(Operands, 0, "if")) {
1122*9880d681SAndroid Build Coastguard Worker if (WarnMissingParenthesis)
1123*9880d681SAndroid Build Coastguard Worker Warning (Begin, "Missing parenthesis around predicate register");
1124*9880d681SAndroid Build Coastguard Worker static char const *LParen = "(";
1125*9880d681SAndroid Build Coastguard Worker static char const *RParen = ")";
1126*9880d681SAndroid Build Coastguard Worker Operands.push_back(HexagonOperand::CreateToken(LParen, Begin));
1127*9880d681SAndroid Build Coastguard Worker Operands.push_back(HexagonOperand::CreateReg(Register, Begin, End));
1128*9880d681SAndroid Build Coastguard Worker const AsmToken &MaybeDotNew = Lexer.getTok();
1129*9880d681SAndroid Build Coastguard Worker if (MaybeDotNew.is(AsmToken::TokenKind::Identifier) &&
1130*9880d681SAndroid Build Coastguard Worker MaybeDotNew.getString().equals_lower(".new"))
1131*9880d681SAndroid Build Coastguard Worker splitIdentifier(Operands);
1132*9880d681SAndroid Build Coastguard Worker Operands.push_back(HexagonOperand::CreateToken(RParen, Begin));
1133*9880d681SAndroid Build Coastguard Worker return false;
1134*9880d681SAndroid Build Coastguard Worker }
1135*9880d681SAndroid Build Coastguard Worker if (previousEqual(Operands, 0, "!") &&
1136*9880d681SAndroid Build Coastguard Worker previousEqual(Operands, 1, "if")) {
1137*9880d681SAndroid Build Coastguard Worker if (WarnMissingParenthesis)
1138*9880d681SAndroid Build Coastguard Worker Warning (Begin, "Missing parenthesis around predicate register");
1139*9880d681SAndroid Build Coastguard Worker static char const *LParen = "(";
1140*9880d681SAndroid Build Coastguard Worker static char const *RParen = ")";
1141*9880d681SAndroid Build Coastguard Worker Operands.insert(Operands.end () - 1,
1142*9880d681SAndroid Build Coastguard Worker HexagonOperand::CreateToken(LParen, Begin));
1143*9880d681SAndroid Build Coastguard Worker Operands.push_back(HexagonOperand::CreateReg(Register, Begin, End));
1144*9880d681SAndroid Build Coastguard Worker const AsmToken &MaybeDotNew = Lexer.getTok();
1145*9880d681SAndroid Build Coastguard Worker if (MaybeDotNew.is(AsmToken::TokenKind::Identifier) &&
1146*9880d681SAndroid Build Coastguard Worker MaybeDotNew.getString().equals_lower(".new"))
1147*9880d681SAndroid Build Coastguard Worker splitIdentifier(Operands);
1148*9880d681SAndroid Build Coastguard Worker Operands.push_back(HexagonOperand::CreateToken(RParen, Begin));
1149*9880d681SAndroid Build Coastguard Worker return false;
1150*9880d681SAndroid Build Coastguard Worker }
1151*9880d681SAndroid Build Coastguard Worker break;
1152*9880d681SAndroid Build Coastguard Worker }
1153*9880d681SAndroid Build Coastguard Worker Operands.push_back(HexagonOperand::CreateReg(
1154*9880d681SAndroid Build Coastguard Worker Register, Begin, End));
1155*9880d681SAndroid Build Coastguard Worker return false;
1156*9880d681SAndroid Build Coastguard Worker }
1157*9880d681SAndroid Build Coastguard Worker return splitIdentifier(Operands);
1158*9880d681SAndroid Build Coastguard Worker }
1159*9880d681SAndroid Build Coastguard Worker
isLabel(AsmToken & Token)1160*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::isLabel(AsmToken &Token) {
1161*9880d681SAndroid Build Coastguard Worker MCAsmLexer &Lexer = getLexer();
1162*9880d681SAndroid Build Coastguard Worker AsmToken const &Second = Lexer.getTok();
1163*9880d681SAndroid Build Coastguard Worker AsmToken Third = Lexer.peekTok();
1164*9880d681SAndroid Build Coastguard Worker StringRef String = Token.getString();
1165*9880d681SAndroid Build Coastguard Worker if (Token.is(AsmToken::TokenKind::LCurly) ||
1166*9880d681SAndroid Build Coastguard Worker Token.is(AsmToken::TokenKind::RCurly))
1167*9880d681SAndroid Build Coastguard Worker return false;
1168*9880d681SAndroid Build Coastguard Worker if (!Token.is(AsmToken::TokenKind::Identifier))
1169*9880d681SAndroid Build Coastguard Worker return true;
1170*9880d681SAndroid Build Coastguard Worker if (!matchRegister(String.lower()))
1171*9880d681SAndroid Build Coastguard Worker return true;
1172*9880d681SAndroid Build Coastguard Worker (void)Second;
1173*9880d681SAndroid Build Coastguard Worker assert(Second.is(AsmToken::Colon));
1174*9880d681SAndroid Build Coastguard Worker StringRef Raw (String.data(), Third.getString().data() - String.data() +
1175*9880d681SAndroid Build Coastguard Worker Third.getString().size());
1176*9880d681SAndroid Build Coastguard Worker std::string Collapsed = Raw;
1177*9880d681SAndroid Build Coastguard Worker Collapsed.erase(std::remove_if(Collapsed.begin(), Collapsed.end(), isspace),
1178*9880d681SAndroid Build Coastguard Worker Collapsed.end());
1179*9880d681SAndroid Build Coastguard Worker StringRef Whole = Collapsed;
1180*9880d681SAndroid Build Coastguard Worker std::pair<StringRef, StringRef> DotSplit = Whole.split('.');
1181*9880d681SAndroid Build Coastguard Worker if (!matchRegister(DotSplit.first.lower()))
1182*9880d681SAndroid Build Coastguard Worker return true;
1183*9880d681SAndroid Build Coastguard Worker return false;
1184*9880d681SAndroid Build Coastguard Worker }
1185*9880d681SAndroid Build Coastguard Worker
handleNoncontigiousRegister(bool Contigious,SMLoc & Loc)1186*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::handleNoncontigiousRegister(bool Contigious, SMLoc &Loc) {
1187*9880d681SAndroid Build Coastguard Worker if (!Contigious && ErrorNoncontigiousRegister) {
1188*9880d681SAndroid Build Coastguard Worker Error(Loc, "Register name is not contigious");
1189*9880d681SAndroid Build Coastguard Worker return true;
1190*9880d681SAndroid Build Coastguard Worker }
1191*9880d681SAndroid Build Coastguard Worker if (!Contigious && WarnNoncontigiousRegister)
1192*9880d681SAndroid Build Coastguard Worker Warning(Loc, "Register name is not contigious");
1193*9880d681SAndroid Build Coastguard Worker return false;
1194*9880d681SAndroid Build Coastguard Worker }
1195*9880d681SAndroid Build Coastguard Worker
ParseRegister(unsigned & RegNo,SMLoc & StartLoc,SMLoc & EndLoc)1196*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::ParseRegister(unsigned &RegNo, SMLoc &StartLoc, SMLoc &EndLoc) {
1197*9880d681SAndroid Build Coastguard Worker MCAsmLexer &Lexer = getLexer();
1198*9880d681SAndroid Build Coastguard Worker StartLoc = getLexer().getLoc();
1199*9880d681SAndroid Build Coastguard Worker SmallVector<AsmToken, 5> Lookahead;
1200*9880d681SAndroid Build Coastguard Worker StringRef RawString(Lexer.getTok().getString().data(), 0);
1201*9880d681SAndroid Build Coastguard Worker bool Again = Lexer.is(AsmToken::Identifier);
1202*9880d681SAndroid Build Coastguard Worker bool NeededWorkaround = false;
1203*9880d681SAndroid Build Coastguard Worker while (Again) {
1204*9880d681SAndroid Build Coastguard Worker AsmToken const &Token = Lexer.getTok();
1205*9880d681SAndroid Build Coastguard Worker RawString = StringRef(RawString.data(),
1206*9880d681SAndroid Build Coastguard Worker Token.getString().data() - RawString.data () +
1207*9880d681SAndroid Build Coastguard Worker Token.getString().size());
1208*9880d681SAndroid Build Coastguard Worker Lookahead.push_back(Token);
1209*9880d681SAndroid Build Coastguard Worker Lexer.Lex();
1210*9880d681SAndroid Build Coastguard Worker bool Contigious = Lexer.getTok().getString().data() ==
1211*9880d681SAndroid Build Coastguard Worker Lookahead.back().getString().data() +
1212*9880d681SAndroid Build Coastguard Worker Lookahead.back().getString().size();
1213*9880d681SAndroid Build Coastguard Worker bool Type = Lexer.is(AsmToken::Identifier) || Lexer.is(AsmToken::Dot) ||
1214*9880d681SAndroid Build Coastguard Worker Lexer.is(AsmToken::Integer) || Lexer.is(AsmToken::Real) ||
1215*9880d681SAndroid Build Coastguard Worker Lexer.is(AsmToken::Colon);
1216*9880d681SAndroid Build Coastguard Worker bool Workaround = Lexer.is(AsmToken::Colon) ||
1217*9880d681SAndroid Build Coastguard Worker Lookahead.back().is(AsmToken::Colon);
1218*9880d681SAndroid Build Coastguard Worker Again = (Contigious && Type) || (Workaround && Type);
1219*9880d681SAndroid Build Coastguard Worker NeededWorkaround = NeededWorkaround || (Again && !(Contigious && Type));
1220*9880d681SAndroid Build Coastguard Worker }
1221*9880d681SAndroid Build Coastguard Worker std::string Collapsed = RawString;
1222*9880d681SAndroid Build Coastguard Worker Collapsed.erase(std::remove_if(Collapsed.begin(), Collapsed.end(), isspace),
1223*9880d681SAndroid Build Coastguard Worker Collapsed.end());
1224*9880d681SAndroid Build Coastguard Worker StringRef FullString = Collapsed;
1225*9880d681SAndroid Build Coastguard Worker std::pair<StringRef, StringRef> DotSplit = FullString.split('.');
1226*9880d681SAndroid Build Coastguard Worker unsigned DotReg = matchRegister(DotSplit.first.lower());
1227*9880d681SAndroid Build Coastguard Worker if (DotReg != Hexagon::NoRegister && RegisterMatchesArch(DotReg)) {
1228*9880d681SAndroid Build Coastguard Worker if (DotSplit.second.empty()) {
1229*9880d681SAndroid Build Coastguard Worker RegNo = DotReg;
1230*9880d681SAndroid Build Coastguard Worker EndLoc = Lexer.getLoc();
1231*9880d681SAndroid Build Coastguard Worker if (handleNoncontigiousRegister(!NeededWorkaround, StartLoc))
1232*9880d681SAndroid Build Coastguard Worker return true;
1233*9880d681SAndroid Build Coastguard Worker return false;
1234*9880d681SAndroid Build Coastguard Worker } else {
1235*9880d681SAndroid Build Coastguard Worker RegNo = DotReg;
1236*9880d681SAndroid Build Coastguard Worker size_t First = RawString.find('.');
1237*9880d681SAndroid Build Coastguard Worker StringRef DotString (RawString.data() + First, RawString.size() - First);
1238*9880d681SAndroid Build Coastguard Worker Lexer.UnLex(AsmToken(AsmToken::Identifier, DotString));
1239*9880d681SAndroid Build Coastguard Worker EndLoc = Lexer.getLoc();
1240*9880d681SAndroid Build Coastguard Worker if (handleNoncontigiousRegister(!NeededWorkaround, StartLoc))
1241*9880d681SAndroid Build Coastguard Worker return true;
1242*9880d681SAndroid Build Coastguard Worker return false;
1243*9880d681SAndroid Build Coastguard Worker }
1244*9880d681SAndroid Build Coastguard Worker }
1245*9880d681SAndroid Build Coastguard Worker std::pair<StringRef, StringRef> ColonSplit = StringRef(FullString).split(':');
1246*9880d681SAndroid Build Coastguard Worker unsigned ColonReg = matchRegister(ColonSplit.first.lower());
1247*9880d681SAndroid Build Coastguard Worker if (ColonReg != Hexagon::NoRegister && RegisterMatchesArch(DotReg)) {
1248*9880d681SAndroid Build Coastguard Worker Lexer.UnLex(Lookahead.back());
1249*9880d681SAndroid Build Coastguard Worker Lookahead.pop_back();
1250*9880d681SAndroid Build Coastguard Worker Lexer.UnLex(Lookahead.back());
1251*9880d681SAndroid Build Coastguard Worker Lookahead.pop_back();
1252*9880d681SAndroid Build Coastguard Worker RegNo = ColonReg;
1253*9880d681SAndroid Build Coastguard Worker EndLoc = Lexer.getLoc();
1254*9880d681SAndroid Build Coastguard Worker if (handleNoncontigiousRegister(!NeededWorkaround, StartLoc))
1255*9880d681SAndroid Build Coastguard Worker return true;
1256*9880d681SAndroid Build Coastguard Worker return false;
1257*9880d681SAndroid Build Coastguard Worker }
1258*9880d681SAndroid Build Coastguard Worker while (!Lookahead.empty()) {
1259*9880d681SAndroid Build Coastguard Worker Lexer.UnLex(Lookahead.back());
1260*9880d681SAndroid Build Coastguard Worker Lookahead.pop_back();
1261*9880d681SAndroid Build Coastguard Worker }
1262*9880d681SAndroid Build Coastguard Worker return true;
1263*9880d681SAndroid Build Coastguard Worker }
1264*9880d681SAndroid Build Coastguard Worker
implicitExpressionLocation(OperandVector & Operands)1265*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::implicitExpressionLocation(OperandVector &Operands) {
1266*9880d681SAndroid Build Coastguard Worker if (previousEqual(Operands, 0, "call"))
1267*9880d681SAndroid Build Coastguard Worker return true;
1268*9880d681SAndroid Build Coastguard Worker if (previousEqual(Operands, 0, "jump"))
1269*9880d681SAndroid Build Coastguard Worker if (!getLexer().getTok().is(AsmToken::Colon))
1270*9880d681SAndroid Build Coastguard Worker return true;
1271*9880d681SAndroid Build Coastguard Worker if (previousEqual(Operands, 0, "(") && previousIsLoop(Operands, 1))
1272*9880d681SAndroid Build Coastguard Worker return true;
1273*9880d681SAndroid Build Coastguard Worker if (previousEqual(Operands, 1, ":") && previousEqual(Operands, 2, "jump") &&
1274*9880d681SAndroid Build Coastguard Worker (previousEqual(Operands, 0, "nt") || previousEqual(Operands, 0, "t")))
1275*9880d681SAndroid Build Coastguard Worker return true;
1276*9880d681SAndroid Build Coastguard Worker return false;
1277*9880d681SAndroid Build Coastguard Worker }
1278*9880d681SAndroid Build Coastguard Worker
parseExpression(MCExpr const * & Expr)1279*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::parseExpression(MCExpr const *& Expr) {
1280*9880d681SAndroid Build Coastguard Worker llvm::SmallVector<AsmToken, 4> Tokens;
1281*9880d681SAndroid Build Coastguard Worker MCAsmLexer &Lexer = getLexer();
1282*9880d681SAndroid Build Coastguard Worker bool Done = false;
1283*9880d681SAndroid Build Coastguard Worker static char const * Comma = ",";
1284*9880d681SAndroid Build Coastguard Worker do {
1285*9880d681SAndroid Build Coastguard Worker Tokens.emplace_back (Lexer.getTok());
1286*9880d681SAndroid Build Coastguard Worker Lex();
1287*9880d681SAndroid Build Coastguard Worker switch (Tokens.back().getKind())
1288*9880d681SAndroid Build Coastguard Worker {
1289*9880d681SAndroid Build Coastguard Worker case AsmToken::TokenKind::Hash:
1290*9880d681SAndroid Build Coastguard Worker if (Tokens.size () > 1)
1291*9880d681SAndroid Build Coastguard Worker if ((Tokens.end () - 2)->getKind() == AsmToken::TokenKind::Plus) {
1292*9880d681SAndroid Build Coastguard Worker Tokens.insert(Tokens.end() - 2,
1293*9880d681SAndroid Build Coastguard Worker AsmToken(AsmToken::TokenKind::Comma, Comma));
1294*9880d681SAndroid Build Coastguard Worker Done = true;
1295*9880d681SAndroid Build Coastguard Worker }
1296*9880d681SAndroid Build Coastguard Worker break;
1297*9880d681SAndroid Build Coastguard Worker case AsmToken::TokenKind::RCurly:
1298*9880d681SAndroid Build Coastguard Worker case AsmToken::TokenKind::EndOfStatement:
1299*9880d681SAndroid Build Coastguard Worker case AsmToken::TokenKind::Eof:
1300*9880d681SAndroid Build Coastguard Worker Done = true;
1301*9880d681SAndroid Build Coastguard Worker break;
1302*9880d681SAndroid Build Coastguard Worker default:
1303*9880d681SAndroid Build Coastguard Worker break;
1304*9880d681SAndroid Build Coastguard Worker }
1305*9880d681SAndroid Build Coastguard Worker } while (!Done);
1306*9880d681SAndroid Build Coastguard Worker while (!Tokens.empty()) {
1307*9880d681SAndroid Build Coastguard Worker Lexer.UnLex(Tokens.back());
1308*9880d681SAndroid Build Coastguard Worker Tokens.pop_back();
1309*9880d681SAndroid Build Coastguard Worker }
1310*9880d681SAndroid Build Coastguard Worker return getParser().parseExpression(Expr);
1311*9880d681SAndroid Build Coastguard Worker }
1312*9880d681SAndroid Build Coastguard Worker
parseExpressionOrOperand(OperandVector & Operands)1313*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::parseExpressionOrOperand(OperandVector &Operands) {
1314*9880d681SAndroid Build Coastguard Worker if (implicitExpressionLocation(Operands)) {
1315*9880d681SAndroid Build Coastguard Worker MCAsmParser &Parser = getParser();
1316*9880d681SAndroid Build Coastguard Worker SMLoc Loc = Parser.getLexer().getLoc();
1317*9880d681SAndroid Build Coastguard Worker MCExpr const *Expr = nullptr;
1318*9880d681SAndroid Build Coastguard Worker bool Error = parseExpression(Expr);
1319*9880d681SAndroid Build Coastguard Worker Expr = HexagonMCExpr::create(Expr, getContext());
1320*9880d681SAndroid Build Coastguard Worker if (!Error)
1321*9880d681SAndroid Build Coastguard Worker Operands.push_back(HexagonOperand::CreateImm(Expr, Loc, Loc));
1322*9880d681SAndroid Build Coastguard Worker return Error;
1323*9880d681SAndroid Build Coastguard Worker }
1324*9880d681SAndroid Build Coastguard Worker return parseOperand(Operands);
1325*9880d681SAndroid Build Coastguard Worker }
1326*9880d681SAndroid Build Coastguard Worker
1327*9880d681SAndroid Build Coastguard Worker /// Parse an instruction.
parseInstruction(OperandVector & Operands)1328*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::parseInstruction(OperandVector &Operands) {
1329*9880d681SAndroid Build Coastguard Worker MCAsmParser &Parser = getParser();
1330*9880d681SAndroid Build Coastguard Worker MCAsmLexer &Lexer = getLexer();
1331*9880d681SAndroid Build Coastguard Worker while (true) {
1332*9880d681SAndroid Build Coastguard Worker AsmToken const &Token = Parser.getTok();
1333*9880d681SAndroid Build Coastguard Worker switch (Token.getKind()) {
1334*9880d681SAndroid Build Coastguard Worker case AsmToken::EndOfStatement: {
1335*9880d681SAndroid Build Coastguard Worker Lex();
1336*9880d681SAndroid Build Coastguard Worker return false;
1337*9880d681SAndroid Build Coastguard Worker }
1338*9880d681SAndroid Build Coastguard Worker case AsmToken::LCurly: {
1339*9880d681SAndroid Build Coastguard Worker if (!Operands.empty())
1340*9880d681SAndroid Build Coastguard Worker return true;
1341*9880d681SAndroid Build Coastguard Worker Operands.push_back(
1342*9880d681SAndroid Build Coastguard Worker HexagonOperand::CreateToken(Token.getString(), Token.getLoc()));
1343*9880d681SAndroid Build Coastguard Worker Lex();
1344*9880d681SAndroid Build Coastguard Worker return false;
1345*9880d681SAndroid Build Coastguard Worker }
1346*9880d681SAndroid Build Coastguard Worker case AsmToken::RCurly: {
1347*9880d681SAndroid Build Coastguard Worker if (Operands.empty()) {
1348*9880d681SAndroid Build Coastguard Worker Operands.push_back(
1349*9880d681SAndroid Build Coastguard Worker HexagonOperand::CreateToken(Token.getString(), Token.getLoc()));
1350*9880d681SAndroid Build Coastguard Worker Lex();
1351*9880d681SAndroid Build Coastguard Worker }
1352*9880d681SAndroid Build Coastguard Worker return false;
1353*9880d681SAndroid Build Coastguard Worker }
1354*9880d681SAndroid Build Coastguard Worker case AsmToken::Comma: {
1355*9880d681SAndroid Build Coastguard Worker Lex();
1356*9880d681SAndroid Build Coastguard Worker continue;
1357*9880d681SAndroid Build Coastguard Worker }
1358*9880d681SAndroid Build Coastguard Worker case AsmToken::EqualEqual:
1359*9880d681SAndroid Build Coastguard Worker case AsmToken::ExclaimEqual:
1360*9880d681SAndroid Build Coastguard Worker case AsmToken::GreaterEqual:
1361*9880d681SAndroid Build Coastguard Worker case AsmToken::GreaterGreater:
1362*9880d681SAndroid Build Coastguard Worker case AsmToken::LessEqual:
1363*9880d681SAndroid Build Coastguard Worker case AsmToken::LessLess: {
1364*9880d681SAndroid Build Coastguard Worker Operands.push_back(HexagonOperand::CreateToken(
1365*9880d681SAndroid Build Coastguard Worker Token.getString().substr(0, 1), Token.getLoc()));
1366*9880d681SAndroid Build Coastguard Worker Operands.push_back(HexagonOperand::CreateToken(
1367*9880d681SAndroid Build Coastguard Worker Token.getString().substr(1, 1), Token.getLoc()));
1368*9880d681SAndroid Build Coastguard Worker Lex();
1369*9880d681SAndroid Build Coastguard Worker continue;
1370*9880d681SAndroid Build Coastguard Worker }
1371*9880d681SAndroid Build Coastguard Worker case AsmToken::Hash: {
1372*9880d681SAndroid Build Coastguard Worker bool MustNotExtend = false;
1373*9880d681SAndroid Build Coastguard Worker bool ImplicitExpression = implicitExpressionLocation(Operands);
1374*9880d681SAndroid Build Coastguard Worker SMLoc ExprLoc = Lexer.getLoc();
1375*9880d681SAndroid Build Coastguard Worker if (!ImplicitExpression)
1376*9880d681SAndroid Build Coastguard Worker Operands.push_back(
1377*9880d681SAndroid Build Coastguard Worker HexagonOperand::CreateToken(Token.getString(), Token.getLoc()));
1378*9880d681SAndroid Build Coastguard Worker Lex();
1379*9880d681SAndroid Build Coastguard Worker bool MustExtend = false;
1380*9880d681SAndroid Build Coastguard Worker bool HiOnly = false;
1381*9880d681SAndroid Build Coastguard Worker bool LoOnly = false;
1382*9880d681SAndroid Build Coastguard Worker if (Lexer.is(AsmToken::Hash)) {
1383*9880d681SAndroid Build Coastguard Worker Lex();
1384*9880d681SAndroid Build Coastguard Worker MustExtend = true;
1385*9880d681SAndroid Build Coastguard Worker } else if (ImplicitExpression)
1386*9880d681SAndroid Build Coastguard Worker MustNotExtend = true;
1387*9880d681SAndroid Build Coastguard Worker AsmToken const &Token = Parser.getTok();
1388*9880d681SAndroid Build Coastguard Worker if (Token.is(AsmToken::Identifier)) {
1389*9880d681SAndroid Build Coastguard Worker StringRef String = Token.getString();
1390*9880d681SAndroid Build Coastguard Worker if (String.lower() == "hi") {
1391*9880d681SAndroid Build Coastguard Worker HiOnly = true;
1392*9880d681SAndroid Build Coastguard Worker } else if (String.lower() == "lo") {
1393*9880d681SAndroid Build Coastguard Worker LoOnly = true;
1394*9880d681SAndroid Build Coastguard Worker }
1395*9880d681SAndroid Build Coastguard Worker if (HiOnly || LoOnly) {
1396*9880d681SAndroid Build Coastguard Worker AsmToken LParen = Lexer.peekTok();
1397*9880d681SAndroid Build Coastguard Worker if (!LParen.is(AsmToken::LParen)) {
1398*9880d681SAndroid Build Coastguard Worker HiOnly = false;
1399*9880d681SAndroid Build Coastguard Worker LoOnly = false;
1400*9880d681SAndroid Build Coastguard Worker } else {
1401*9880d681SAndroid Build Coastguard Worker Lex();
1402*9880d681SAndroid Build Coastguard Worker }
1403*9880d681SAndroid Build Coastguard Worker }
1404*9880d681SAndroid Build Coastguard Worker }
1405*9880d681SAndroid Build Coastguard Worker MCExpr const *Expr = nullptr;
1406*9880d681SAndroid Build Coastguard Worker if (parseExpression(Expr))
1407*9880d681SAndroid Build Coastguard Worker return true;
1408*9880d681SAndroid Build Coastguard Worker int64_t Value;
1409*9880d681SAndroid Build Coastguard Worker MCContext &Context = Parser.getContext();
1410*9880d681SAndroid Build Coastguard Worker assert(Expr != nullptr);
1411*9880d681SAndroid Build Coastguard Worker if (Expr->evaluateAsAbsolute(Value)) {
1412*9880d681SAndroid Build Coastguard Worker if (HiOnly)
1413*9880d681SAndroid Build Coastguard Worker Expr = MCBinaryExpr::createLShr(
1414*9880d681SAndroid Build Coastguard Worker Expr, MCConstantExpr::create(16, Context), Context);
1415*9880d681SAndroid Build Coastguard Worker if (HiOnly || LoOnly)
1416*9880d681SAndroid Build Coastguard Worker Expr = MCBinaryExpr::createAnd(Expr,
1417*9880d681SAndroid Build Coastguard Worker MCConstantExpr::create(0xffff, Context),
1418*9880d681SAndroid Build Coastguard Worker Context);
1419*9880d681SAndroid Build Coastguard Worker } else {
1420*9880d681SAndroid Build Coastguard Worker MCValue Value;
1421*9880d681SAndroid Build Coastguard Worker if (Expr->evaluateAsRelocatable(Value, nullptr, nullptr)) {
1422*9880d681SAndroid Build Coastguard Worker if (!Value.isAbsolute()) {
1423*9880d681SAndroid Build Coastguard Worker switch(Value.getAccessVariant()) {
1424*9880d681SAndroid Build Coastguard Worker case MCSymbolRefExpr::VariantKind::VK_TPREL:
1425*9880d681SAndroid Build Coastguard Worker case MCSymbolRefExpr::VariantKind::VK_DTPREL:
1426*9880d681SAndroid Build Coastguard Worker // Don't lazy extend these expression variants
1427*9880d681SAndroid Build Coastguard Worker MustNotExtend = !MustExtend;
1428*9880d681SAndroid Build Coastguard Worker break;
1429*9880d681SAndroid Build Coastguard Worker default:
1430*9880d681SAndroid Build Coastguard Worker break;
1431*9880d681SAndroid Build Coastguard Worker }
1432*9880d681SAndroid Build Coastguard Worker }
1433*9880d681SAndroid Build Coastguard Worker }
1434*9880d681SAndroid Build Coastguard Worker }
1435*9880d681SAndroid Build Coastguard Worker Expr = HexagonMCExpr::create(Expr, Context);
1436*9880d681SAndroid Build Coastguard Worker HexagonMCInstrInfo::setMustNotExtend(*Expr, MustNotExtend);
1437*9880d681SAndroid Build Coastguard Worker HexagonMCInstrInfo::setMustExtend(*Expr, MustExtend);
1438*9880d681SAndroid Build Coastguard Worker std::unique_ptr<HexagonOperand> Operand =
1439*9880d681SAndroid Build Coastguard Worker HexagonOperand::CreateImm(Expr, ExprLoc, ExprLoc);
1440*9880d681SAndroid Build Coastguard Worker Operands.push_back(std::move(Operand));
1441*9880d681SAndroid Build Coastguard Worker continue;
1442*9880d681SAndroid Build Coastguard Worker }
1443*9880d681SAndroid Build Coastguard Worker default:
1444*9880d681SAndroid Build Coastguard Worker break;
1445*9880d681SAndroid Build Coastguard Worker }
1446*9880d681SAndroid Build Coastguard Worker if (parseExpressionOrOperand(Operands))
1447*9880d681SAndroid Build Coastguard Worker return true;
1448*9880d681SAndroid Build Coastguard Worker }
1449*9880d681SAndroid Build Coastguard Worker }
1450*9880d681SAndroid Build Coastguard Worker
ParseInstruction(ParseInstructionInfo & Info,StringRef Name,AsmToken ID,OperandVector & Operands)1451*9880d681SAndroid Build Coastguard Worker bool HexagonAsmParser::ParseInstruction(ParseInstructionInfo &Info,
1452*9880d681SAndroid Build Coastguard Worker StringRef Name,
1453*9880d681SAndroid Build Coastguard Worker AsmToken ID,
1454*9880d681SAndroid Build Coastguard Worker OperandVector &Operands) {
1455*9880d681SAndroid Build Coastguard Worker getLexer().UnLex(ID);
1456*9880d681SAndroid Build Coastguard Worker return parseInstruction(Operands);
1457*9880d681SAndroid Build Coastguard Worker }
1458*9880d681SAndroid Build Coastguard Worker
1459*9880d681SAndroid Build Coastguard Worker namespace {
makeCombineInst(int opCode,MCOperand & Rdd,MCOperand & MO1,MCOperand & MO2)1460*9880d681SAndroid Build Coastguard Worker MCInst makeCombineInst(int opCode, MCOperand &Rdd,
1461*9880d681SAndroid Build Coastguard Worker MCOperand &MO1, MCOperand &MO2) {
1462*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
1463*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(opCode);
1464*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rdd);
1465*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(MO1);
1466*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(MO2);
1467*9880d681SAndroid Build Coastguard Worker
1468*9880d681SAndroid Build Coastguard Worker return TmpInst;
1469*9880d681SAndroid Build Coastguard Worker }
1470*9880d681SAndroid Build Coastguard Worker }
1471*9880d681SAndroid Build Coastguard Worker
1472*9880d681SAndroid Build Coastguard Worker // Define this matcher function after the auto-generated include so we
1473*9880d681SAndroid Build Coastguard Worker // have the match class enum definitions.
validateTargetOperandClass(MCParsedAsmOperand & AsmOp,unsigned Kind)1474*9880d681SAndroid Build Coastguard Worker unsigned HexagonAsmParser::validateTargetOperandClass(MCParsedAsmOperand &AsmOp,
1475*9880d681SAndroid Build Coastguard Worker unsigned Kind) {
1476*9880d681SAndroid Build Coastguard Worker HexagonOperand *Op = static_cast<HexagonOperand *>(&AsmOp);
1477*9880d681SAndroid Build Coastguard Worker
1478*9880d681SAndroid Build Coastguard Worker switch (Kind) {
1479*9880d681SAndroid Build Coastguard Worker case MCK_0: {
1480*9880d681SAndroid Build Coastguard Worker int64_t Value;
1481*9880d681SAndroid Build Coastguard Worker return Op->isImm() && Op->Imm.Val->evaluateAsAbsolute(Value) && Value == 0
1482*9880d681SAndroid Build Coastguard Worker ? Match_Success
1483*9880d681SAndroid Build Coastguard Worker : Match_InvalidOperand;
1484*9880d681SAndroid Build Coastguard Worker }
1485*9880d681SAndroid Build Coastguard Worker case MCK_1: {
1486*9880d681SAndroid Build Coastguard Worker int64_t Value;
1487*9880d681SAndroid Build Coastguard Worker return Op->isImm() && Op->Imm.Val->evaluateAsAbsolute(Value) && Value == 1
1488*9880d681SAndroid Build Coastguard Worker ? Match_Success
1489*9880d681SAndroid Build Coastguard Worker : Match_InvalidOperand;
1490*9880d681SAndroid Build Coastguard Worker }
1491*9880d681SAndroid Build Coastguard Worker case MCK__MINUS_1: {
1492*9880d681SAndroid Build Coastguard Worker int64_t Value;
1493*9880d681SAndroid Build Coastguard Worker return Op->isImm() && Op->Imm.Val->evaluateAsAbsolute(Value) && Value == -1
1494*9880d681SAndroid Build Coastguard Worker ? Match_Success
1495*9880d681SAndroid Build Coastguard Worker : Match_InvalidOperand;
1496*9880d681SAndroid Build Coastguard Worker }
1497*9880d681SAndroid Build Coastguard Worker }
1498*9880d681SAndroid Build Coastguard Worker if (Op->Kind == HexagonOperand::Token && Kind != InvalidMatchClass) {
1499*9880d681SAndroid Build Coastguard Worker StringRef myStringRef = StringRef(Op->Tok.Data, Op->Tok.Length);
1500*9880d681SAndroid Build Coastguard Worker if (matchTokenString(myStringRef.lower()) == (MatchClassKind)Kind)
1501*9880d681SAndroid Build Coastguard Worker return Match_Success;
1502*9880d681SAndroid Build Coastguard Worker if (matchTokenString(myStringRef.upper()) == (MatchClassKind)Kind)
1503*9880d681SAndroid Build Coastguard Worker return Match_Success;
1504*9880d681SAndroid Build Coastguard Worker }
1505*9880d681SAndroid Build Coastguard Worker
1506*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << "Unmatched Operand:");
1507*9880d681SAndroid Build Coastguard Worker DEBUG(Op->dump());
1508*9880d681SAndroid Build Coastguard Worker DEBUG(dbgs() << "\n");
1509*9880d681SAndroid Build Coastguard Worker
1510*9880d681SAndroid Build Coastguard Worker return Match_InvalidOperand;
1511*9880d681SAndroid Build Coastguard Worker }
1512*9880d681SAndroid Build Coastguard Worker
OutOfRange(SMLoc IDLoc,long long Val,long long Max)1513*9880d681SAndroid Build Coastguard Worker void HexagonAsmParser::OutOfRange(SMLoc IDLoc, long long Val, long long Max) {
1514*9880d681SAndroid Build Coastguard Worker std::string errStr;
1515*9880d681SAndroid Build Coastguard Worker raw_string_ostream ES(errStr);
1516*9880d681SAndroid Build Coastguard Worker ES << "value " << Val << "(" << format_hex(Val, 0) << ") out of range: ";
1517*9880d681SAndroid Build Coastguard Worker if (Max >= 0)
1518*9880d681SAndroid Build Coastguard Worker ES << "0-" << Max;
1519*9880d681SAndroid Build Coastguard Worker else
1520*9880d681SAndroid Build Coastguard Worker ES << Max << "-" << (-Max - 1);
1521*9880d681SAndroid Build Coastguard Worker Error(IDLoc, ES.str().c_str());
1522*9880d681SAndroid Build Coastguard Worker }
1523*9880d681SAndroid Build Coastguard Worker
processInstruction(MCInst & Inst,OperandVector const & Operands,SMLoc IDLoc)1524*9880d681SAndroid Build Coastguard Worker int HexagonAsmParser::processInstruction(MCInst &Inst,
1525*9880d681SAndroid Build Coastguard Worker OperandVector const &Operands,
1526*9880d681SAndroid Build Coastguard Worker SMLoc IDLoc) {
1527*9880d681SAndroid Build Coastguard Worker MCContext &Context = getParser().getContext();
1528*9880d681SAndroid Build Coastguard Worker const MCRegisterInfo *RI = getContext().getRegisterInfo();
1529*9880d681SAndroid Build Coastguard Worker std::string r = "r";
1530*9880d681SAndroid Build Coastguard Worker std::string v = "v";
1531*9880d681SAndroid Build Coastguard Worker std::string Colon = ":";
1532*9880d681SAndroid Build Coastguard Worker
1533*9880d681SAndroid Build Coastguard Worker bool is32bit = false; // used to distinguish between CONST32 and CONST64
1534*9880d681SAndroid Build Coastguard Worker switch (Inst.getOpcode()) {
1535*9880d681SAndroid Build Coastguard Worker default:
1536*9880d681SAndroid Build Coastguard Worker break;
1537*9880d681SAndroid Build Coastguard Worker
1538*9880d681SAndroid Build Coastguard Worker case Hexagon::A2_iconst: {
1539*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::A2_addi);
1540*9880d681SAndroid Build Coastguard Worker MCOperand Reg = Inst.getOperand(0);
1541*9880d681SAndroid Build Coastguard Worker MCOperand S16 = Inst.getOperand(1);
1542*9880d681SAndroid Build Coastguard Worker HexagonMCInstrInfo::setMustNotExtend(*S16.getExpr());
1543*9880d681SAndroid Build Coastguard Worker HexagonMCInstrInfo::setS23_2_reloc(*S16.getExpr());
1544*9880d681SAndroid Build Coastguard Worker Inst.clear();
1545*9880d681SAndroid Build Coastguard Worker Inst.addOperand(Reg);
1546*9880d681SAndroid Build Coastguard Worker Inst.addOperand(MCOperand::createReg(Hexagon::R0));
1547*9880d681SAndroid Build Coastguard Worker Inst.addOperand(S16);
1548*9880d681SAndroid Build Coastguard Worker break;
1549*9880d681SAndroid Build Coastguard Worker }
1550*9880d681SAndroid Build Coastguard Worker case Hexagon::M4_mpyrr_addr:
1551*9880d681SAndroid Build Coastguard Worker case Hexagon::S4_addi_asl_ri:
1552*9880d681SAndroid Build Coastguard Worker case Hexagon::S4_addi_lsr_ri:
1553*9880d681SAndroid Build Coastguard Worker case Hexagon::S4_andi_asl_ri:
1554*9880d681SAndroid Build Coastguard Worker case Hexagon::S4_andi_lsr_ri:
1555*9880d681SAndroid Build Coastguard Worker case Hexagon::S4_ori_asl_ri:
1556*9880d681SAndroid Build Coastguard Worker case Hexagon::S4_ori_lsr_ri:
1557*9880d681SAndroid Build Coastguard Worker case Hexagon::S4_or_andix:
1558*9880d681SAndroid Build Coastguard Worker case Hexagon::S4_subi_asl_ri:
1559*9880d681SAndroid Build Coastguard Worker case Hexagon::S4_subi_lsr_ri: {
1560*9880d681SAndroid Build Coastguard Worker MCOperand &Ry = Inst.getOperand(0);
1561*9880d681SAndroid Build Coastguard Worker MCOperand &src = Inst.getOperand(2);
1562*9880d681SAndroid Build Coastguard Worker if (RI->getEncodingValue(Ry.getReg()) != RI->getEncodingValue(src.getReg()))
1563*9880d681SAndroid Build Coastguard Worker return Match_InvalidOperand;
1564*9880d681SAndroid Build Coastguard Worker break;
1565*9880d681SAndroid Build Coastguard Worker }
1566*9880d681SAndroid Build Coastguard Worker
1567*9880d681SAndroid Build Coastguard Worker case Hexagon::C2_cmpgei: {
1568*9880d681SAndroid Build Coastguard Worker MCOperand &MO = Inst.getOperand(2);
1569*9880d681SAndroid Build Coastguard Worker MO.setExpr(HexagonMCExpr::create(MCBinaryExpr::createSub(
1570*9880d681SAndroid Build Coastguard Worker MO.getExpr(), MCConstantExpr::create(1, Context), Context), Context));
1571*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::C2_cmpgti);
1572*9880d681SAndroid Build Coastguard Worker break;
1573*9880d681SAndroid Build Coastguard Worker }
1574*9880d681SAndroid Build Coastguard Worker
1575*9880d681SAndroid Build Coastguard Worker case Hexagon::C2_cmpgeui: {
1576*9880d681SAndroid Build Coastguard Worker MCOperand &MO = Inst.getOperand(2);
1577*9880d681SAndroid Build Coastguard Worker int64_t Value;
1578*9880d681SAndroid Build Coastguard Worker bool Success = MO.getExpr()->evaluateAsAbsolute(Value);
1579*9880d681SAndroid Build Coastguard Worker (void)Success;
1580*9880d681SAndroid Build Coastguard Worker assert(Success && "Assured by matcher");
1581*9880d681SAndroid Build Coastguard Worker if (Value == 0) {
1582*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
1583*9880d681SAndroid Build Coastguard Worker MCOperand &Pd = Inst.getOperand(0);
1584*9880d681SAndroid Build Coastguard Worker MCOperand &Rt = Inst.getOperand(1);
1585*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::C2_cmpeq);
1586*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Pd);
1587*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rt);
1588*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rt);
1589*9880d681SAndroid Build Coastguard Worker Inst = TmpInst;
1590*9880d681SAndroid Build Coastguard Worker } else {
1591*9880d681SAndroid Build Coastguard Worker MO.setExpr(HexagonMCExpr::create(MCBinaryExpr::createSub(
1592*9880d681SAndroid Build Coastguard Worker MO.getExpr(), MCConstantExpr::create(1, Context), Context), Context));
1593*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::C2_cmpgtui);
1594*9880d681SAndroid Build Coastguard Worker }
1595*9880d681SAndroid Build Coastguard Worker break;
1596*9880d681SAndroid Build Coastguard Worker }
1597*9880d681SAndroid Build Coastguard Worker
1598*9880d681SAndroid Build Coastguard Worker // Translate a "$Rdd = $Rss" to "$Rdd = combine($Rs, $Rt)"
1599*9880d681SAndroid Build Coastguard Worker case Hexagon::A2_tfrp: {
1600*9880d681SAndroid Build Coastguard Worker MCOperand &MO = Inst.getOperand(1);
1601*9880d681SAndroid Build Coastguard Worker unsigned int RegPairNum = RI->getEncodingValue(MO.getReg());
1602*9880d681SAndroid Build Coastguard Worker std::string R1 = r + llvm::utostr(RegPairNum + 1);
1603*9880d681SAndroid Build Coastguard Worker StringRef Reg1(R1);
1604*9880d681SAndroid Build Coastguard Worker MO.setReg(matchRegister(Reg1));
1605*9880d681SAndroid Build Coastguard Worker // Add a new operand for the second register in the pair.
1606*9880d681SAndroid Build Coastguard Worker std::string R2 = r + llvm::utostr(RegPairNum);
1607*9880d681SAndroid Build Coastguard Worker StringRef Reg2(R2);
1608*9880d681SAndroid Build Coastguard Worker Inst.addOperand(MCOperand::createReg(matchRegister(Reg2)));
1609*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::A2_combinew);
1610*9880d681SAndroid Build Coastguard Worker break;
1611*9880d681SAndroid Build Coastguard Worker }
1612*9880d681SAndroid Build Coastguard Worker
1613*9880d681SAndroid Build Coastguard Worker case Hexagon::A2_tfrpt:
1614*9880d681SAndroid Build Coastguard Worker case Hexagon::A2_tfrpf: {
1615*9880d681SAndroid Build Coastguard Worker MCOperand &MO = Inst.getOperand(2);
1616*9880d681SAndroid Build Coastguard Worker unsigned int RegPairNum = RI->getEncodingValue(MO.getReg());
1617*9880d681SAndroid Build Coastguard Worker std::string R1 = r + llvm::utostr(RegPairNum + 1);
1618*9880d681SAndroid Build Coastguard Worker StringRef Reg1(R1);
1619*9880d681SAndroid Build Coastguard Worker MO.setReg(matchRegister(Reg1));
1620*9880d681SAndroid Build Coastguard Worker // Add a new operand for the second register in the pair.
1621*9880d681SAndroid Build Coastguard Worker std::string R2 = r + llvm::utostr(RegPairNum);
1622*9880d681SAndroid Build Coastguard Worker StringRef Reg2(R2);
1623*9880d681SAndroid Build Coastguard Worker Inst.addOperand(MCOperand::createReg(matchRegister(Reg2)));
1624*9880d681SAndroid Build Coastguard Worker Inst.setOpcode((Inst.getOpcode() == Hexagon::A2_tfrpt)
1625*9880d681SAndroid Build Coastguard Worker ? Hexagon::C2_ccombinewt
1626*9880d681SAndroid Build Coastguard Worker : Hexagon::C2_ccombinewf);
1627*9880d681SAndroid Build Coastguard Worker break;
1628*9880d681SAndroid Build Coastguard Worker }
1629*9880d681SAndroid Build Coastguard Worker case Hexagon::A2_tfrptnew:
1630*9880d681SAndroid Build Coastguard Worker case Hexagon::A2_tfrpfnew: {
1631*9880d681SAndroid Build Coastguard Worker MCOperand &MO = Inst.getOperand(2);
1632*9880d681SAndroid Build Coastguard Worker unsigned int RegPairNum = RI->getEncodingValue(MO.getReg());
1633*9880d681SAndroid Build Coastguard Worker std::string R1 = r + llvm::utostr(RegPairNum + 1);
1634*9880d681SAndroid Build Coastguard Worker StringRef Reg1(R1);
1635*9880d681SAndroid Build Coastguard Worker MO.setReg(matchRegister(Reg1));
1636*9880d681SAndroid Build Coastguard Worker // Add a new operand for the second register in the pair.
1637*9880d681SAndroid Build Coastguard Worker std::string R2 = r + llvm::utostr(RegPairNum);
1638*9880d681SAndroid Build Coastguard Worker StringRef Reg2(R2);
1639*9880d681SAndroid Build Coastguard Worker Inst.addOperand(MCOperand::createReg(matchRegister(Reg2)));
1640*9880d681SAndroid Build Coastguard Worker Inst.setOpcode((Inst.getOpcode() == Hexagon::A2_tfrptnew)
1641*9880d681SAndroid Build Coastguard Worker ? Hexagon::C2_ccombinewnewt
1642*9880d681SAndroid Build Coastguard Worker : Hexagon::C2_ccombinewnewf);
1643*9880d681SAndroid Build Coastguard Worker break;
1644*9880d681SAndroid Build Coastguard Worker }
1645*9880d681SAndroid Build Coastguard Worker
1646*9880d681SAndroid Build Coastguard Worker // Translate a "$Vdd = $Vss" to "$Vdd = vcombine($Vs, $Vt)"
1647*9880d681SAndroid Build Coastguard Worker case Hexagon::HEXAGON_V6_vassignpair: {
1648*9880d681SAndroid Build Coastguard Worker MCOperand &MO = Inst.getOperand(1);
1649*9880d681SAndroid Build Coastguard Worker unsigned int RegPairNum = RI->getEncodingValue(MO.getReg());
1650*9880d681SAndroid Build Coastguard Worker std::string R1 = v + llvm::utostr(RegPairNum + 1);
1651*9880d681SAndroid Build Coastguard Worker MO.setReg(MatchRegisterName(R1));
1652*9880d681SAndroid Build Coastguard Worker // Add a new operand for the second register in the pair.
1653*9880d681SAndroid Build Coastguard Worker std::string R2 = v + llvm::utostr(RegPairNum);
1654*9880d681SAndroid Build Coastguard Worker Inst.addOperand(MCOperand::createReg(MatchRegisterName(R2)));
1655*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::V6_vcombine);
1656*9880d681SAndroid Build Coastguard Worker break;
1657*9880d681SAndroid Build Coastguard Worker }
1658*9880d681SAndroid Build Coastguard Worker
1659*9880d681SAndroid Build Coastguard Worker // Translate a "$Rx = CONST32(#imm)" to "$Rx = memw(gp+#LABEL) "
1660*9880d681SAndroid Build Coastguard Worker case Hexagon::CONST32:
1661*9880d681SAndroid Build Coastguard Worker case Hexagon::CONST32_Float_Real:
1662*9880d681SAndroid Build Coastguard Worker case Hexagon::CONST32_Int_Real:
1663*9880d681SAndroid Build Coastguard Worker case Hexagon::FCONST32_nsdata:
1664*9880d681SAndroid Build Coastguard Worker is32bit = true;
1665*9880d681SAndroid Build Coastguard Worker // Translate a "$Rx:y = CONST64(#imm)" to "$Rx:y = memd(gp+#LABEL) "
1666*9880d681SAndroid Build Coastguard Worker case Hexagon::CONST64_Float_Real:
1667*9880d681SAndroid Build Coastguard Worker case Hexagon::CONST64_Int_Real:
1668*9880d681SAndroid Build Coastguard Worker
1669*9880d681SAndroid Build Coastguard Worker // FIXME: need better way to detect AsmStreamer (upstream removed getKind())
1670*9880d681SAndroid Build Coastguard Worker if (!Parser.getStreamer().hasRawTextSupport()) {
1671*9880d681SAndroid Build Coastguard Worker MCELFStreamer *MES = static_cast<MCELFStreamer *>(&Parser.getStreamer());
1672*9880d681SAndroid Build Coastguard Worker MCOperand &MO_1 = Inst.getOperand(1);
1673*9880d681SAndroid Build Coastguard Worker MCOperand &MO_0 = Inst.getOperand(0);
1674*9880d681SAndroid Build Coastguard Worker
1675*9880d681SAndroid Build Coastguard Worker // push section onto section stack
1676*9880d681SAndroid Build Coastguard Worker MES->PushSection();
1677*9880d681SAndroid Build Coastguard Worker
1678*9880d681SAndroid Build Coastguard Worker std::string myCharStr;
1679*9880d681SAndroid Build Coastguard Worker MCSectionELF *mySection;
1680*9880d681SAndroid Build Coastguard Worker
1681*9880d681SAndroid Build Coastguard Worker // check if this as an immediate or a symbol
1682*9880d681SAndroid Build Coastguard Worker int64_t Value;
1683*9880d681SAndroid Build Coastguard Worker bool Absolute = MO_1.getExpr()->evaluateAsAbsolute(Value);
1684*9880d681SAndroid Build Coastguard Worker if (Absolute) {
1685*9880d681SAndroid Build Coastguard Worker // Create a new section - one for each constant
1686*9880d681SAndroid Build Coastguard Worker // Some or all of the zeros are replaced with the given immediate.
1687*9880d681SAndroid Build Coastguard Worker if (is32bit) {
1688*9880d681SAndroid Build Coastguard Worker std::string myImmStr = utohexstr(static_cast<uint32_t>(Value));
1689*9880d681SAndroid Build Coastguard Worker myCharStr = StringRef(".gnu.linkonce.l4.CONST_00000000")
1690*9880d681SAndroid Build Coastguard Worker .drop_back(myImmStr.size())
1691*9880d681SAndroid Build Coastguard Worker .str() +
1692*9880d681SAndroid Build Coastguard Worker myImmStr;
1693*9880d681SAndroid Build Coastguard Worker } else {
1694*9880d681SAndroid Build Coastguard Worker std::string myImmStr = utohexstr(Value);
1695*9880d681SAndroid Build Coastguard Worker myCharStr = StringRef(".gnu.linkonce.l8.CONST_0000000000000000")
1696*9880d681SAndroid Build Coastguard Worker .drop_back(myImmStr.size())
1697*9880d681SAndroid Build Coastguard Worker .str() +
1698*9880d681SAndroid Build Coastguard Worker myImmStr;
1699*9880d681SAndroid Build Coastguard Worker }
1700*9880d681SAndroid Build Coastguard Worker
1701*9880d681SAndroid Build Coastguard Worker mySection = getContext().getELFSection(myCharStr, ELF::SHT_PROGBITS,
1702*9880d681SAndroid Build Coastguard Worker ELF::SHF_ALLOC | ELF::SHF_WRITE);
1703*9880d681SAndroid Build Coastguard Worker } else if (MO_1.isExpr()) {
1704*9880d681SAndroid Build Coastguard Worker // .lita - for expressions
1705*9880d681SAndroid Build Coastguard Worker myCharStr = ".lita";
1706*9880d681SAndroid Build Coastguard Worker mySection = getContext().getELFSection(myCharStr, ELF::SHT_PROGBITS,
1707*9880d681SAndroid Build Coastguard Worker ELF::SHF_ALLOC | ELF::SHF_WRITE);
1708*9880d681SAndroid Build Coastguard Worker } else
1709*9880d681SAndroid Build Coastguard Worker llvm_unreachable("unexpected type of machine operand!");
1710*9880d681SAndroid Build Coastguard Worker
1711*9880d681SAndroid Build Coastguard Worker MES->SwitchSection(mySection);
1712*9880d681SAndroid Build Coastguard Worker unsigned byteSize = is32bit ? 4 : 8;
1713*9880d681SAndroid Build Coastguard Worker getStreamer().EmitCodeAlignment(byteSize, byteSize);
1714*9880d681SAndroid Build Coastguard Worker
1715*9880d681SAndroid Build Coastguard Worker MCSymbol *Sym;
1716*9880d681SAndroid Build Coastguard Worker
1717*9880d681SAndroid Build Coastguard Worker // for symbols, get rid of prepended ".gnu.linkonce.lx."
1718*9880d681SAndroid Build Coastguard Worker
1719*9880d681SAndroid Build Coastguard Worker // emit symbol if needed
1720*9880d681SAndroid Build Coastguard Worker if (Absolute) {
1721*9880d681SAndroid Build Coastguard Worker Sym = getContext().getOrCreateSymbol(StringRef(myCharStr.c_str() + 16));
1722*9880d681SAndroid Build Coastguard Worker if (Sym->isUndefined()) {
1723*9880d681SAndroid Build Coastguard Worker getStreamer().EmitLabel(Sym);
1724*9880d681SAndroid Build Coastguard Worker getStreamer().EmitSymbolAttribute(Sym, MCSA_Global);
1725*9880d681SAndroid Build Coastguard Worker getStreamer().EmitIntValue(Value, byteSize);
1726*9880d681SAndroid Build Coastguard Worker }
1727*9880d681SAndroid Build Coastguard Worker } else if (MO_1.isExpr()) {
1728*9880d681SAndroid Build Coastguard Worker const char *StringStart = 0;
1729*9880d681SAndroid Build Coastguard Worker const char *StringEnd = 0;
1730*9880d681SAndroid Build Coastguard Worker if (*Operands[4]->getStartLoc().getPointer() == '#') {
1731*9880d681SAndroid Build Coastguard Worker StringStart = Operands[5]->getStartLoc().getPointer();
1732*9880d681SAndroid Build Coastguard Worker StringEnd = Operands[6]->getStartLoc().getPointer();
1733*9880d681SAndroid Build Coastguard Worker } else { // no pound
1734*9880d681SAndroid Build Coastguard Worker StringStart = Operands[4]->getStartLoc().getPointer();
1735*9880d681SAndroid Build Coastguard Worker StringEnd = Operands[5]->getStartLoc().getPointer();
1736*9880d681SAndroid Build Coastguard Worker }
1737*9880d681SAndroid Build Coastguard Worker
1738*9880d681SAndroid Build Coastguard Worker unsigned size = StringEnd - StringStart;
1739*9880d681SAndroid Build Coastguard Worker std::string DotConst = ".CONST_";
1740*9880d681SAndroid Build Coastguard Worker Sym = getContext().getOrCreateSymbol(DotConst +
1741*9880d681SAndroid Build Coastguard Worker StringRef(StringStart, size));
1742*9880d681SAndroid Build Coastguard Worker
1743*9880d681SAndroid Build Coastguard Worker if (Sym->isUndefined()) {
1744*9880d681SAndroid Build Coastguard Worker // case where symbol is not yet defined: emit symbol
1745*9880d681SAndroid Build Coastguard Worker getStreamer().EmitLabel(Sym);
1746*9880d681SAndroid Build Coastguard Worker getStreamer().EmitSymbolAttribute(Sym, MCSA_Local);
1747*9880d681SAndroid Build Coastguard Worker getStreamer().EmitValue(MO_1.getExpr(), 4);
1748*9880d681SAndroid Build Coastguard Worker }
1749*9880d681SAndroid Build Coastguard Worker } else
1750*9880d681SAndroid Build Coastguard Worker llvm_unreachable("unexpected type of machine operand!");
1751*9880d681SAndroid Build Coastguard Worker
1752*9880d681SAndroid Build Coastguard Worker MES->PopSection();
1753*9880d681SAndroid Build Coastguard Worker
1754*9880d681SAndroid Build Coastguard Worker if (Sym) {
1755*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
1756*9880d681SAndroid Build Coastguard Worker if (is32bit) // 32 bit
1757*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::L2_loadrigp);
1758*9880d681SAndroid Build Coastguard Worker else // 64 bit
1759*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::L2_loadrdgp);
1760*9880d681SAndroid Build Coastguard Worker
1761*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(MO_0);
1762*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(
1763*9880d681SAndroid Build Coastguard Worker MCOperand::createExpr(MCSymbolRefExpr::create(Sym, getContext())));
1764*9880d681SAndroid Build Coastguard Worker Inst = TmpInst;
1765*9880d681SAndroid Build Coastguard Worker }
1766*9880d681SAndroid Build Coastguard Worker }
1767*9880d681SAndroid Build Coastguard Worker break;
1768*9880d681SAndroid Build Coastguard Worker
1769*9880d681SAndroid Build Coastguard Worker // Translate a "$Rdd = #-imm" to "$Rdd = combine(#[-1,0], #-imm)"
1770*9880d681SAndroid Build Coastguard Worker case Hexagon::A2_tfrpi: {
1771*9880d681SAndroid Build Coastguard Worker MCOperand &Rdd = Inst.getOperand(0);
1772*9880d681SAndroid Build Coastguard Worker MCOperand &MO = Inst.getOperand(1);
1773*9880d681SAndroid Build Coastguard Worker int64_t Value;
1774*9880d681SAndroid Build Coastguard Worker int sVal = (MO.getExpr()->evaluateAsAbsolute(Value) && Value < 0) ? -1 : 0;
1775*9880d681SAndroid Build Coastguard Worker MCOperand imm(MCOperand::createExpr(
1776*9880d681SAndroid Build Coastguard Worker HexagonMCExpr::create(MCConstantExpr::create(sVal, Context), Context)));
1777*9880d681SAndroid Build Coastguard Worker Inst = makeCombineInst(Hexagon::A2_combineii, Rdd, imm, MO);
1778*9880d681SAndroid Build Coastguard Worker break;
1779*9880d681SAndroid Build Coastguard Worker }
1780*9880d681SAndroid Build Coastguard Worker
1781*9880d681SAndroid Build Coastguard Worker // Translate a "$Rdd = [#]#imm" to "$Rdd = combine(#, [#]#imm)"
1782*9880d681SAndroid Build Coastguard Worker case Hexagon::TFRI64_V4: {
1783*9880d681SAndroid Build Coastguard Worker MCOperand &Rdd = Inst.getOperand(0);
1784*9880d681SAndroid Build Coastguard Worker MCOperand &MO = Inst.getOperand(1);
1785*9880d681SAndroid Build Coastguard Worker int64_t Value;
1786*9880d681SAndroid Build Coastguard Worker if (MO.getExpr()->evaluateAsAbsolute(Value)) {
1787*9880d681SAndroid Build Coastguard Worker int s8 = Hi_32(Value);
1788*9880d681SAndroid Build Coastguard Worker if (!isInt<8>(s8))
1789*9880d681SAndroid Build Coastguard Worker OutOfRange(IDLoc, s8, -128);
1790*9880d681SAndroid Build Coastguard Worker MCOperand imm(MCOperand::createExpr(HexagonMCExpr::create(
1791*9880d681SAndroid Build Coastguard Worker MCConstantExpr::create(s8, Context), Context))); // upper 32
1792*9880d681SAndroid Build Coastguard Worker auto Expr = HexagonMCExpr::create(
1793*9880d681SAndroid Build Coastguard Worker MCConstantExpr::create(Lo_32(Value), Context), Context);
1794*9880d681SAndroid Build Coastguard Worker HexagonMCInstrInfo::setMustExtend(*Expr, HexagonMCInstrInfo::mustExtend(*MO.getExpr()));
1795*9880d681SAndroid Build Coastguard Worker MCOperand imm2(MCOperand::createExpr(Expr)); // lower 32
1796*9880d681SAndroid Build Coastguard Worker Inst = makeCombineInst(Hexagon::A4_combineii, Rdd, imm, imm2);
1797*9880d681SAndroid Build Coastguard Worker } else {
1798*9880d681SAndroid Build Coastguard Worker MCOperand imm(MCOperand::createExpr(HexagonMCExpr::create(
1799*9880d681SAndroid Build Coastguard Worker MCConstantExpr::create(0, Context), Context))); // upper 32
1800*9880d681SAndroid Build Coastguard Worker Inst = makeCombineInst(Hexagon::A4_combineii, Rdd, imm, MO);
1801*9880d681SAndroid Build Coastguard Worker }
1802*9880d681SAndroid Build Coastguard Worker break;
1803*9880d681SAndroid Build Coastguard Worker }
1804*9880d681SAndroid Build Coastguard Worker
1805*9880d681SAndroid Build Coastguard Worker // Handle $Rdd = combine(##imm, #imm)"
1806*9880d681SAndroid Build Coastguard Worker case Hexagon::TFRI64_V2_ext: {
1807*9880d681SAndroid Build Coastguard Worker MCOperand &Rdd = Inst.getOperand(0);
1808*9880d681SAndroid Build Coastguard Worker MCOperand &MO1 = Inst.getOperand(1);
1809*9880d681SAndroid Build Coastguard Worker MCOperand &MO2 = Inst.getOperand(2);
1810*9880d681SAndroid Build Coastguard Worker int64_t Value;
1811*9880d681SAndroid Build Coastguard Worker if (MO2.getExpr()->evaluateAsAbsolute(Value)) {
1812*9880d681SAndroid Build Coastguard Worker int s8 = Value;
1813*9880d681SAndroid Build Coastguard Worker if (s8 < -128 || s8 > 127)
1814*9880d681SAndroid Build Coastguard Worker OutOfRange(IDLoc, s8, -128);
1815*9880d681SAndroid Build Coastguard Worker }
1816*9880d681SAndroid Build Coastguard Worker Inst = makeCombineInst(Hexagon::A2_combineii, Rdd, MO1, MO2);
1817*9880d681SAndroid Build Coastguard Worker break;
1818*9880d681SAndroid Build Coastguard Worker }
1819*9880d681SAndroid Build Coastguard Worker
1820*9880d681SAndroid Build Coastguard Worker // Handle $Rdd = combine(#imm, ##imm)"
1821*9880d681SAndroid Build Coastguard Worker case Hexagon::A4_combineii: {
1822*9880d681SAndroid Build Coastguard Worker MCOperand &Rdd = Inst.getOperand(0);
1823*9880d681SAndroid Build Coastguard Worker MCOperand &MO1 = Inst.getOperand(1);
1824*9880d681SAndroid Build Coastguard Worker int64_t Value;
1825*9880d681SAndroid Build Coastguard Worker if (MO1.getExpr()->evaluateAsAbsolute(Value)) {
1826*9880d681SAndroid Build Coastguard Worker int s8 = Value;
1827*9880d681SAndroid Build Coastguard Worker if (s8 < -128 || s8 > 127)
1828*9880d681SAndroid Build Coastguard Worker OutOfRange(IDLoc, s8, -128);
1829*9880d681SAndroid Build Coastguard Worker }
1830*9880d681SAndroid Build Coastguard Worker MCOperand &MO2 = Inst.getOperand(2);
1831*9880d681SAndroid Build Coastguard Worker Inst = makeCombineInst(Hexagon::A4_combineii, Rdd, MO1, MO2);
1832*9880d681SAndroid Build Coastguard Worker break;
1833*9880d681SAndroid Build Coastguard Worker }
1834*9880d681SAndroid Build Coastguard Worker
1835*9880d681SAndroid Build Coastguard Worker case Hexagon::S2_tableidxb_goodsyntax: {
1836*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::S2_tableidxb);
1837*9880d681SAndroid Build Coastguard Worker break;
1838*9880d681SAndroid Build Coastguard Worker }
1839*9880d681SAndroid Build Coastguard Worker
1840*9880d681SAndroid Build Coastguard Worker case Hexagon::S2_tableidxh_goodsyntax: {
1841*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
1842*9880d681SAndroid Build Coastguard Worker MCOperand &Rx = Inst.getOperand(0);
1843*9880d681SAndroid Build Coastguard Worker MCOperand &_dst_ = Inst.getOperand(1);
1844*9880d681SAndroid Build Coastguard Worker MCOperand &Rs = Inst.getOperand(2);
1845*9880d681SAndroid Build Coastguard Worker MCOperand &Imm4 = Inst.getOperand(3);
1846*9880d681SAndroid Build Coastguard Worker MCOperand &Imm6 = Inst.getOperand(4);
1847*9880d681SAndroid Build Coastguard Worker Imm6.setExpr(HexagonMCExpr::create(MCBinaryExpr::createSub(
1848*9880d681SAndroid Build Coastguard Worker Imm6.getExpr(), MCConstantExpr::create(1, Context), Context), Context));
1849*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::S2_tableidxh);
1850*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rx);
1851*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(_dst_);
1852*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rs);
1853*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Imm4);
1854*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Imm6);
1855*9880d681SAndroid Build Coastguard Worker Inst = TmpInst;
1856*9880d681SAndroid Build Coastguard Worker break;
1857*9880d681SAndroid Build Coastguard Worker }
1858*9880d681SAndroid Build Coastguard Worker
1859*9880d681SAndroid Build Coastguard Worker case Hexagon::S2_tableidxw_goodsyntax: {
1860*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
1861*9880d681SAndroid Build Coastguard Worker MCOperand &Rx = Inst.getOperand(0);
1862*9880d681SAndroid Build Coastguard Worker MCOperand &_dst_ = Inst.getOperand(1);
1863*9880d681SAndroid Build Coastguard Worker MCOperand &Rs = Inst.getOperand(2);
1864*9880d681SAndroid Build Coastguard Worker MCOperand &Imm4 = Inst.getOperand(3);
1865*9880d681SAndroid Build Coastguard Worker MCOperand &Imm6 = Inst.getOperand(4);
1866*9880d681SAndroid Build Coastguard Worker Imm6.setExpr(HexagonMCExpr::create(MCBinaryExpr::createSub(
1867*9880d681SAndroid Build Coastguard Worker Imm6.getExpr(), MCConstantExpr::create(2, Context), Context), Context));
1868*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::S2_tableidxw);
1869*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rx);
1870*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(_dst_);
1871*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rs);
1872*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Imm4);
1873*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Imm6);
1874*9880d681SAndroid Build Coastguard Worker Inst = TmpInst;
1875*9880d681SAndroid Build Coastguard Worker break;
1876*9880d681SAndroid Build Coastguard Worker }
1877*9880d681SAndroid Build Coastguard Worker
1878*9880d681SAndroid Build Coastguard Worker case Hexagon::S2_tableidxd_goodsyntax: {
1879*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
1880*9880d681SAndroid Build Coastguard Worker MCOperand &Rx = Inst.getOperand(0);
1881*9880d681SAndroid Build Coastguard Worker MCOperand &_dst_ = Inst.getOperand(1);
1882*9880d681SAndroid Build Coastguard Worker MCOperand &Rs = Inst.getOperand(2);
1883*9880d681SAndroid Build Coastguard Worker MCOperand &Imm4 = Inst.getOperand(3);
1884*9880d681SAndroid Build Coastguard Worker MCOperand &Imm6 = Inst.getOperand(4);
1885*9880d681SAndroid Build Coastguard Worker Imm6.setExpr(HexagonMCExpr::create(MCBinaryExpr::createSub(
1886*9880d681SAndroid Build Coastguard Worker Imm6.getExpr(), MCConstantExpr::create(3, Context), Context), Context));
1887*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::S2_tableidxd);
1888*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rx);
1889*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(_dst_);
1890*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rs);
1891*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Imm4);
1892*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Imm6);
1893*9880d681SAndroid Build Coastguard Worker Inst = TmpInst;
1894*9880d681SAndroid Build Coastguard Worker break;
1895*9880d681SAndroid Build Coastguard Worker }
1896*9880d681SAndroid Build Coastguard Worker
1897*9880d681SAndroid Build Coastguard Worker case Hexagon::M2_mpyui: {
1898*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::M2_mpyi);
1899*9880d681SAndroid Build Coastguard Worker break;
1900*9880d681SAndroid Build Coastguard Worker }
1901*9880d681SAndroid Build Coastguard Worker case Hexagon::M2_mpysmi: {
1902*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
1903*9880d681SAndroid Build Coastguard Worker MCOperand &Rd = Inst.getOperand(0);
1904*9880d681SAndroid Build Coastguard Worker MCOperand &Rs = Inst.getOperand(1);
1905*9880d681SAndroid Build Coastguard Worker MCOperand &Imm = Inst.getOperand(2);
1906*9880d681SAndroid Build Coastguard Worker int64_t Value;
1907*9880d681SAndroid Build Coastguard Worker MCExpr const &Expr = *Imm.getExpr();
1908*9880d681SAndroid Build Coastguard Worker bool Absolute = Expr.evaluateAsAbsolute(Value);
1909*9880d681SAndroid Build Coastguard Worker assert(Absolute);
1910*9880d681SAndroid Build Coastguard Worker (void)Absolute;
1911*9880d681SAndroid Build Coastguard Worker if (!HexagonMCInstrInfo::mustExtend(Expr)) {
1912*9880d681SAndroid Build Coastguard Worker if (Value < 0 && Value > -256) {
1913*9880d681SAndroid Build Coastguard Worker Imm.setExpr(HexagonMCExpr::create(
1914*9880d681SAndroid Build Coastguard Worker MCConstantExpr::create(Value * -1, Context), Context));
1915*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::M2_mpysin);
1916*9880d681SAndroid Build Coastguard Worker } else if (Value < 256 && Value >= 0)
1917*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::M2_mpysip);
1918*9880d681SAndroid Build Coastguard Worker else
1919*9880d681SAndroid Build Coastguard Worker return Match_InvalidOperand;
1920*9880d681SAndroid Build Coastguard Worker } else {
1921*9880d681SAndroid Build Coastguard Worker if (Value >= 0)
1922*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::M2_mpysip);
1923*9880d681SAndroid Build Coastguard Worker else
1924*9880d681SAndroid Build Coastguard Worker return Match_InvalidOperand;
1925*9880d681SAndroid Build Coastguard Worker }
1926*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rd);
1927*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rs);
1928*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Imm);
1929*9880d681SAndroid Build Coastguard Worker Inst = TmpInst;
1930*9880d681SAndroid Build Coastguard Worker break;
1931*9880d681SAndroid Build Coastguard Worker }
1932*9880d681SAndroid Build Coastguard Worker
1933*9880d681SAndroid Build Coastguard Worker case Hexagon::S2_asr_i_r_rnd_goodsyntax: {
1934*9880d681SAndroid Build Coastguard Worker MCOperand &Imm = Inst.getOperand(2);
1935*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
1936*9880d681SAndroid Build Coastguard Worker int64_t Value;
1937*9880d681SAndroid Build Coastguard Worker bool Absolute = Imm.getExpr()->evaluateAsAbsolute(Value);
1938*9880d681SAndroid Build Coastguard Worker assert(Absolute);
1939*9880d681SAndroid Build Coastguard Worker (void)Absolute;
1940*9880d681SAndroid Build Coastguard Worker if (Value == 0) { // convert to $Rd = $Rs
1941*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::A2_tfr);
1942*9880d681SAndroid Build Coastguard Worker MCOperand &Rd = Inst.getOperand(0);
1943*9880d681SAndroid Build Coastguard Worker MCOperand &Rs = Inst.getOperand(1);
1944*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rd);
1945*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rs);
1946*9880d681SAndroid Build Coastguard Worker } else {
1947*9880d681SAndroid Build Coastguard Worker Imm.setExpr(HexagonMCExpr::create(
1948*9880d681SAndroid Build Coastguard Worker MCBinaryExpr::createSub(Imm.getExpr(),
1949*9880d681SAndroid Build Coastguard Worker MCConstantExpr::create(1, Context), Context),
1950*9880d681SAndroid Build Coastguard Worker Context));
1951*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::S2_asr_i_r_rnd);
1952*9880d681SAndroid Build Coastguard Worker MCOperand &Rd = Inst.getOperand(0);
1953*9880d681SAndroid Build Coastguard Worker MCOperand &Rs = Inst.getOperand(1);
1954*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rd);
1955*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rs);
1956*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Imm);
1957*9880d681SAndroid Build Coastguard Worker }
1958*9880d681SAndroid Build Coastguard Worker Inst = TmpInst;
1959*9880d681SAndroid Build Coastguard Worker break;
1960*9880d681SAndroid Build Coastguard Worker }
1961*9880d681SAndroid Build Coastguard Worker
1962*9880d681SAndroid Build Coastguard Worker case Hexagon::S2_asr_i_p_rnd_goodsyntax: {
1963*9880d681SAndroid Build Coastguard Worker MCOperand &Rdd = Inst.getOperand(0);
1964*9880d681SAndroid Build Coastguard Worker MCOperand &Rss = Inst.getOperand(1);
1965*9880d681SAndroid Build Coastguard Worker MCOperand &Imm = Inst.getOperand(2);
1966*9880d681SAndroid Build Coastguard Worker int64_t Value;
1967*9880d681SAndroid Build Coastguard Worker bool Absolute = Imm.getExpr()->evaluateAsAbsolute(Value);
1968*9880d681SAndroid Build Coastguard Worker assert(Absolute);
1969*9880d681SAndroid Build Coastguard Worker (void)Absolute;
1970*9880d681SAndroid Build Coastguard Worker if (Value == 0) { // convert to $Rdd = combine ($Rs[0], $Rs[1])
1971*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
1972*9880d681SAndroid Build Coastguard Worker unsigned int RegPairNum = RI->getEncodingValue(Rss.getReg());
1973*9880d681SAndroid Build Coastguard Worker std::string R1 = r + llvm::utostr(RegPairNum + 1);
1974*9880d681SAndroid Build Coastguard Worker StringRef Reg1(R1);
1975*9880d681SAndroid Build Coastguard Worker Rss.setReg(matchRegister(Reg1));
1976*9880d681SAndroid Build Coastguard Worker // Add a new operand for the second register in the pair.
1977*9880d681SAndroid Build Coastguard Worker std::string R2 = r + llvm::utostr(RegPairNum);
1978*9880d681SAndroid Build Coastguard Worker StringRef Reg2(R2);
1979*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::A2_combinew);
1980*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rdd);
1981*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rss);
1982*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(MCOperand::createReg(matchRegister(Reg2)));
1983*9880d681SAndroid Build Coastguard Worker Inst = TmpInst;
1984*9880d681SAndroid Build Coastguard Worker } else {
1985*9880d681SAndroid Build Coastguard Worker Imm.setExpr(HexagonMCExpr::create(
1986*9880d681SAndroid Build Coastguard Worker MCBinaryExpr::createSub(Imm.getExpr(),
1987*9880d681SAndroid Build Coastguard Worker MCConstantExpr::create(1, Context), Context),
1988*9880d681SAndroid Build Coastguard Worker Context));
1989*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::S2_asr_i_p_rnd);
1990*9880d681SAndroid Build Coastguard Worker }
1991*9880d681SAndroid Build Coastguard Worker break;
1992*9880d681SAndroid Build Coastguard Worker }
1993*9880d681SAndroid Build Coastguard Worker
1994*9880d681SAndroid Build Coastguard Worker case Hexagon::A4_boundscheck: {
1995*9880d681SAndroid Build Coastguard Worker MCOperand &Rs = Inst.getOperand(1);
1996*9880d681SAndroid Build Coastguard Worker unsigned int RegNum = RI->getEncodingValue(Rs.getReg());
1997*9880d681SAndroid Build Coastguard Worker if (RegNum & 1) { // Odd mapped to raw:hi, regpair is rodd:odd-1, like r3:2
1998*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::A4_boundscheck_hi);
1999*9880d681SAndroid Build Coastguard Worker std::string Name =
2000*9880d681SAndroid Build Coastguard Worker r + llvm::utostr(RegNum) + Colon + llvm::utostr(RegNum - 1);
2001*9880d681SAndroid Build Coastguard Worker StringRef RegPair = Name;
2002*9880d681SAndroid Build Coastguard Worker Rs.setReg(matchRegister(RegPair));
2003*9880d681SAndroid Build Coastguard Worker } else { // raw:lo
2004*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::A4_boundscheck_lo);
2005*9880d681SAndroid Build Coastguard Worker std::string Name =
2006*9880d681SAndroid Build Coastguard Worker r + llvm::utostr(RegNum + 1) + Colon + llvm::utostr(RegNum);
2007*9880d681SAndroid Build Coastguard Worker StringRef RegPair = Name;
2008*9880d681SAndroid Build Coastguard Worker Rs.setReg(matchRegister(RegPair));
2009*9880d681SAndroid Build Coastguard Worker }
2010*9880d681SAndroid Build Coastguard Worker break;
2011*9880d681SAndroid Build Coastguard Worker }
2012*9880d681SAndroid Build Coastguard Worker
2013*9880d681SAndroid Build Coastguard Worker case Hexagon::A2_addsp: {
2014*9880d681SAndroid Build Coastguard Worker MCOperand &Rs = Inst.getOperand(1);
2015*9880d681SAndroid Build Coastguard Worker unsigned int RegNum = RI->getEncodingValue(Rs.getReg());
2016*9880d681SAndroid Build Coastguard Worker if (RegNum & 1) { // Odd mapped to raw:hi
2017*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::A2_addsph);
2018*9880d681SAndroid Build Coastguard Worker std::string Name =
2019*9880d681SAndroid Build Coastguard Worker r + llvm::utostr(RegNum) + Colon + llvm::utostr(RegNum - 1);
2020*9880d681SAndroid Build Coastguard Worker StringRef RegPair = Name;
2021*9880d681SAndroid Build Coastguard Worker Rs.setReg(matchRegister(RegPair));
2022*9880d681SAndroid Build Coastguard Worker } else { // Even mapped raw:lo
2023*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::A2_addspl);
2024*9880d681SAndroid Build Coastguard Worker std::string Name =
2025*9880d681SAndroid Build Coastguard Worker r + llvm::utostr(RegNum + 1) + Colon + llvm::utostr(RegNum);
2026*9880d681SAndroid Build Coastguard Worker StringRef RegPair = Name;
2027*9880d681SAndroid Build Coastguard Worker Rs.setReg(matchRegister(RegPair));
2028*9880d681SAndroid Build Coastguard Worker }
2029*9880d681SAndroid Build Coastguard Worker break;
2030*9880d681SAndroid Build Coastguard Worker }
2031*9880d681SAndroid Build Coastguard Worker
2032*9880d681SAndroid Build Coastguard Worker case Hexagon::M2_vrcmpys_s1: {
2033*9880d681SAndroid Build Coastguard Worker MCOperand &Rt = Inst.getOperand(2);
2034*9880d681SAndroid Build Coastguard Worker unsigned int RegNum = RI->getEncodingValue(Rt.getReg());
2035*9880d681SAndroid Build Coastguard Worker if (RegNum & 1) { // Odd mapped to sat:raw:hi
2036*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::M2_vrcmpys_s1_h);
2037*9880d681SAndroid Build Coastguard Worker std::string Name =
2038*9880d681SAndroid Build Coastguard Worker r + llvm::utostr(RegNum) + Colon + llvm::utostr(RegNum - 1);
2039*9880d681SAndroid Build Coastguard Worker StringRef RegPair = Name;
2040*9880d681SAndroid Build Coastguard Worker Rt.setReg(matchRegister(RegPair));
2041*9880d681SAndroid Build Coastguard Worker } else { // Even mapped sat:raw:lo
2042*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::M2_vrcmpys_s1_l);
2043*9880d681SAndroid Build Coastguard Worker std::string Name =
2044*9880d681SAndroid Build Coastguard Worker r + llvm::utostr(RegNum + 1) + Colon + llvm::utostr(RegNum);
2045*9880d681SAndroid Build Coastguard Worker StringRef RegPair = Name;
2046*9880d681SAndroid Build Coastguard Worker Rt.setReg(matchRegister(RegPair));
2047*9880d681SAndroid Build Coastguard Worker }
2048*9880d681SAndroid Build Coastguard Worker break;
2049*9880d681SAndroid Build Coastguard Worker }
2050*9880d681SAndroid Build Coastguard Worker
2051*9880d681SAndroid Build Coastguard Worker case Hexagon::M2_vrcmpys_acc_s1: {
2052*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
2053*9880d681SAndroid Build Coastguard Worker MCOperand &Rxx = Inst.getOperand(0);
2054*9880d681SAndroid Build Coastguard Worker MCOperand &Rss = Inst.getOperand(2);
2055*9880d681SAndroid Build Coastguard Worker MCOperand &Rt = Inst.getOperand(3);
2056*9880d681SAndroid Build Coastguard Worker unsigned int RegNum = RI->getEncodingValue(Rt.getReg());
2057*9880d681SAndroid Build Coastguard Worker if (RegNum & 1) { // Odd mapped to sat:raw:hi
2058*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::M2_vrcmpys_acc_s1_h);
2059*9880d681SAndroid Build Coastguard Worker std::string Name =
2060*9880d681SAndroid Build Coastguard Worker r + llvm::utostr(RegNum) + Colon + llvm::utostr(RegNum - 1);
2061*9880d681SAndroid Build Coastguard Worker StringRef RegPair = Name;
2062*9880d681SAndroid Build Coastguard Worker Rt.setReg(matchRegister(RegPair));
2063*9880d681SAndroid Build Coastguard Worker } else { // Even mapped sat:raw:lo
2064*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::M2_vrcmpys_acc_s1_l);
2065*9880d681SAndroid Build Coastguard Worker std::string Name =
2066*9880d681SAndroid Build Coastguard Worker r + llvm::utostr(RegNum + 1) + Colon + llvm::utostr(RegNum);
2067*9880d681SAndroid Build Coastguard Worker StringRef RegPair = Name;
2068*9880d681SAndroid Build Coastguard Worker Rt.setReg(matchRegister(RegPair));
2069*9880d681SAndroid Build Coastguard Worker }
2070*9880d681SAndroid Build Coastguard Worker // Registers are in different positions
2071*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rxx);
2072*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rxx);
2073*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rss);
2074*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rt);
2075*9880d681SAndroid Build Coastguard Worker Inst = TmpInst;
2076*9880d681SAndroid Build Coastguard Worker break;
2077*9880d681SAndroid Build Coastguard Worker }
2078*9880d681SAndroid Build Coastguard Worker
2079*9880d681SAndroid Build Coastguard Worker case Hexagon::M2_vrcmpys_s1rp: {
2080*9880d681SAndroid Build Coastguard Worker MCOperand &Rt = Inst.getOperand(2);
2081*9880d681SAndroid Build Coastguard Worker unsigned int RegNum = RI->getEncodingValue(Rt.getReg());
2082*9880d681SAndroid Build Coastguard Worker if (RegNum & 1) { // Odd mapped to rnd:sat:raw:hi
2083*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::M2_vrcmpys_s1rp_h);
2084*9880d681SAndroid Build Coastguard Worker std::string Name =
2085*9880d681SAndroid Build Coastguard Worker r + llvm::utostr(RegNum) + Colon + llvm::utostr(RegNum - 1);
2086*9880d681SAndroid Build Coastguard Worker StringRef RegPair = Name;
2087*9880d681SAndroid Build Coastguard Worker Rt.setReg(matchRegister(RegPair));
2088*9880d681SAndroid Build Coastguard Worker } else { // Even mapped rnd:sat:raw:lo
2089*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::M2_vrcmpys_s1rp_l);
2090*9880d681SAndroid Build Coastguard Worker std::string Name =
2091*9880d681SAndroid Build Coastguard Worker r + llvm::utostr(RegNum + 1) + Colon + llvm::utostr(RegNum);
2092*9880d681SAndroid Build Coastguard Worker StringRef RegPair = Name;
2093*9880d681SAndroid Build Coastguard Worker Rt.setReg(matchRegister(RegPair));
2094*9880d681SAndroid Build Coastguard Worker }
2095*9880d681SAndroid Build Coastguard Worker break;
2096*9880d681SAndroid Build Coastguard Worker }
2097*9880d681SAndroid Build Coastguard Worker
2098*9880d681SAndroid Build Coastguard Worker case Hexagon::S5_asrhub_rnd_sat_goodsyntax: {
2099*9880d681SAndroid Build Coastguard Worker MCOperand &Imm = Inst.getOperand(2);
2100*9880d681SAndroid Build Coastguard Worker int64_t Value;
2101*9880d681SAndroid Build Coastguard Worker bool Absolute = Imm.getExpr()->evaluateAsAbsolute(Value);
2102*9880d681SAndroid Build Coastguard Worker assert(Absolute);
2103*9880d681SAndroid Build Coastguard Worker (void)Absolute;
2104*9880d681SAndroid Build Coastguard Worker if (Value == 0)
2105*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::S2_vsathub);
2106*9880d681SAndroid Build Coastguard Worker else {
2107*9880d681SAndroid Build Coastguard Worker Imm.setExpr(HexagonMCExpr::create(
2108*9880d681SAndroid Build Coastguard Worker MCBinaryExpr::createSub(Imm.getExpr(),
2109*9880d681SAndroid Build Coastguard Worker MCConstantExpr::create(1, Context), Context),
2110*9880d681SAndroid Build Coastguard Worker Context));
2111*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::S5_asrhub_rnd_sat);
2112*9880d681SAndroid Build Coastguard Worker }
2113*9880d681SAndroid Build Coastguard Worker break;
2114*9880d681SAndroid Build Coastguard Worker }
2115*9880d681SAndroid Build Coastguard Worker
2116*9880d681SAndroid Build Coastguard Worker case Hexagon::S5_vasrhrnd_goodsyntax: {
2117*9880d681SAndroid Build Coastguard Worker MCOperand &Rdd = Inst.getOperand(0);
2118*9880d681SAndroid Build Coastguard Worker MCOperand &Rss = Inst.getOperand(1);
2119*9880d681SAndroid Build Coastguard Worker MCOperand &Imm = Inst.getOperand(2);
2120*9880d681SAndroid Build Coastguard Worker int64_t Value;
2121*9880d681SAndroid Build Coastguard Worker bool Absolute = Imm.getExpr()->evaluateAsAbsolute(Value);
2122*9880d681SAndroid Build Coastguard Worker assert(Absolute);
2123*9880d681SAndroid Build Coastguard Worker (void)Absolute;
2124*9880d681SAndroid Build Coastguard Worker if (Value == 0) {
2125*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
2126*9880d681SAndroid Build Coastguard Worker unsigned int RegPairNum = RI->getEncodingValue(Rss.getReg());
2127*9880d681SAndroid Build Coastguard Worker std::string R1 = r + llvm::utostr(RegPairNum + 1);
2128*9880d681SAndroid Build Coastguard Worker StringRef Reg1(R1);
2129*9880d681SAndroid Build Coastguard Worker Rss.setReg(matchRegister(Reg1));
2130*9880d681SAndroid Build Coastguard Worker // Add a new operand for the second register in the pair.
2131*9880d681SAndroid Build Coastguard Worker std::string R2 = r + llvm::utostr(RegPairNum);
2132*9880d681SAndroid Build Coastguard Worker StringRef Reg2(R2);
2133*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::A2_combinew);
2134*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rdd);
2135*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rss);
2136*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(MCOperand::createReg(matchRegister(Reg2)));
2137*9880d681SAndroid Build Coastguard Worker Inst = TmpInst;
2138*9880d681SAndroid Build Coastguard Worker } else {
2139*9880d681SAndroid Build Coastguard Worker Imm.setExpr(HexagonMCExpr::create(
2140*9880d681SAndroid Build Coastguard Worker MCBinaryExpr::createSub(Imm.getExpr(),
2141*9880d681SAndroid Build Coastguard Worker MCConstantExpr::create(1, Context), Context),
2142*9880d681SAndroid Build Coastguard Worker Context));
2143*9880d681SAndroid Build Coastguard Worker Inst.setOpcode(Hexagon::S5_vasrhrnd);
2144*9880d681SAndroid Build Coastguard Worker }
2145*9880d681SAndroid Build Coastguard Worker break;
2146*9880d681SAndroid Build Coastguard Worker }
2147*9880d681SAndroid Build Coastguard Worker
2148*9880d681SAndroid Build Coastguard Worker case Hexagon::A2_not: {
2149*9880d681SAndroid Build Coastguard Worker MCInst TmpInst;
2150*9880d681SAndroid Build Coastguard Worker MCOperand &Rd = Inst.getOperand(0);
2151*9880d681SAndroid Build Coastguard Worker MCOperand &Rs = Inst.getOperand(1);
2152*9880d681SAndroid Build Coastguard Worker TmpInst.setOpcode(Hexagon::A2_subri);
2153*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rd);
2154*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(MCOperand::createExpr(
2155*9880d681SAndroid Build Coastguard Worker HexagonMCExpr::create(MCConstantExpr::create(-1, Context), Context)));
2156*9880d681SAndroid Build Coastguard Worker TmpInst.addOperand(Rs);
2157*9880d681SAndroid Build Coastguard Worker Inst = TmpInst;
2158*9880d681SAndroid Build Coastguard Worker break;
2159*9880d681SAndroid Build Coastguard Worker }
2160*9880d681SAndroid Build Coastguard Worker } // switch
2161*9880d681SAndroid Build Coastguard Worker
2162*9880d681SAndroid Build Coastguard Worker return Match_Success;
2163*9880d681SAndroid Build Coastguard Worker }
2164*9880d681SAndroid Build Coastguard Worker
2165*9880d681SAndroid Build Coastguard Worker
matchRegister(StringRef Name)2166*9880d681SAndroid Build Coastguard Worker unsigned HexagonAsmParser::matchRegister(StringRef Name) {
2167*9880d681SAndroid Build Coastguard Worker if (unsigned Reg = MatchRegisterName(Name))
2168*9880d681SAndroid Build Coastguard Worker return Reg;
2169*9880d681SAndroid Build Coastguard Worker return MatchRegisterAltName(Name);
2170*9880d681SAndroid Build Coastguard Worker }
2171