1 //==- AArch64AsmParser.cpp - Parse AArch64 assembly to MCInst instructions -==//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #include "MCTargetDesc/AArch64AddressingModes.h"
11 #include "MCTargetDesc/AArch64MCExpr.h"
12 #include "MCTargetDesc/AArch64TargetStreamer.h"
13 #include "Utils/AArch64BaseInfo.h"
14 #include "llvm/ADT/APInt.h"
15 #include "llvm/ADT/STLExtras.h"
16 #include "llvm/ADT/SmallString.h"
17 #include "llvm/ADT/SmallVector.h"
18 #include "llvm/ADT/StringSwitch.h"
19 #include "llvm/ADT/Twine.h"
20 #include "llvm/MC/MCContext.h"
21 #include "llvm/MC/MCExpr.h"
22 #include "llvm/MC/MCInst.h"
23 #include "llvm/MC/MCObjectFileInfo.h"
24 #include "llvm/MC/MCParser/MCAsmLexer.h"
25 #include "llvm/MC/MCParser/MCAsmParser.h"
26 #include "llvm/MC/MCParser/MCParsedAsmOperand.h"
27 #include "llvm/MC/MCRegisterInfo.h"
28 #include "llvm/MC/MCStreamer.h"
29 #include "llvm/MC/MCSubtargetInfo.h"
30 #include "llvm/MC/MCSymbol.h"
31 #include "llvm/MC/MCTargetAsmParser.h"
32 #include "llvm/Support/ErrorHandling.h"
33 #include "llvm/Support/SourceMgr.h"
34 #include "llvm/Support/TargetRegistry.h"
35 #include "llvm/Support/raw_ostream.h"
36 #include <cstdio>
37 using namespace llvm;
38 
39 namespace {
40 
41 class AArch64Operand;
42 
43 class AArch64AsmParser : public MCTargetAsmParser {
44 private:
45   StringRef Mnemonic; ///< Instruction mnemonic.
46 
47   // Map of register aliases registers via the .req directive.
48   StringMap<std::pair<bool, unsigned> > RegisterReqs;
49 
getTargetStreamer()50   AArch64TargetStreamer &getTargetStreamer() {
51     MCTargetStreamer &TS = *getParser().getStreamer().getTargetStreamer();
52     return static_cast<AArch64TargetStreamer &>(TS);
53   }
54 
getLoc() const55   SMLoc getLoc() const { return getParser().getTok().getLoc(); }
56 
57   bool parseSysAlias(StringRef Name, SMLoc NameLoc, OperandVector &Operands);
58   AArch64CC::CondCode parseCondCodeString(StringRef Cond);
59   bool parseCondCode(OperandVector &Operands, bool invertCondCode);
60   unsigned matchRegisterNameAlias(StringRef Name, bool isVector);
61   int tryParseRegister();
62   int tryMatchVectorRegister(StringRef &Kind, bool expected);
63   bool parseRegister(OperandVector &Operands);
64   bool parseSymbolicImmVal(const MCExpr *&ImmVal);
65   bool parseVectorList(OperandVector &Operands);
66   bool parseOperand(OperandVector &Operands, bool isCondCode,
67                     bool invertCondCode);
68 
Warning(SMLoc L,const Twine & Msg)69   void Warning(SMLoc L, const Twine &Msg) { getParser().Warning(L, Msg); }
Error(SMLoc L,const Twine & Msg)70   bool Error(SMLoc L, const Twine &Msg) { return getParser().Error(L, Msg); }
71   bool showMatchError(SMLoc Loc, unsigned ErrCode);
72 
73   bool parseDirectiveWord(unsigned Size, SMLoc L);
74   bool parseDirectiveInst(SMLoc L);
75 
76   bool parseDirectiveTLSDescCall(SMLoc L);
77 
78   bool parseDirectiveLOH(StringRef LOH, SMLoc L);
79   bool parseDirectiveLtorg(SMLoc L);
80 
81   bool parseDirectiveReq(StringRef Name, SMLoc L);
82   bool parseDirectiveUnreq(SMLoc L);
83 
84   bool validateInstruction(MCInst &Inst, SmallVectorImpl<SMLoc> &Loc);
85   bool MatchAndEmitInstruction(SMLoc IDLoc, unsigned &Opcode,
86                                OperandVector &Operands, MCStreamer &Out,
87                                uint64_t &ErrorInfo,
88                                bool MatchingInlineAsm) override;
89 /// @name Auto-generated Match Functions
90 /// {
91 
92 #define GET_ASSEMBLER_HEADER
93 #include "AArch64GenAsmMatcher.inc"
94 
95   /// }
96 
97   OperandMatchResultTy tryParseOptionalShiftExtend(OperandVector &Operands);
98   OperandMatchResultTy tryParseBarrierOperand(OperandVector &Operands);
99   OperandMatchResultTy tryParseMRSSystemRegister(OperandVector &Operands);
100   OperandMatchResultTy tryParseSysReg(OperandVector &Operands);
101   OperandMatchResultTy tryParseSysCROperand(OperandVector &Operands);
102   OperandMatchResultTy tryParsePrefetch(OperandVector &Operands);
103   OperandMatchResultTy tryParsePSBHint(OperandVector &Operands);
104   OperandMatchResultTy tryParseAdrpLabel(OperandVector &Operands);
105   OperandMatchResultTy tryParseAdrLabel(OperandVector &Operands);
106   OperandMatchResultTy tryParseFPImm(OperandVector &Operands);
107   OperandMatchResultTy tryParseAddSubImm(OperandVector &Operands);
108   OperandMatchResultTy tryParseGPR64sp0Operand(OperandVector &Operands);
109   bool tryParseVectorRegister(OperandVector &Operands);
110   OperandMatchResultTy tryParseGPRSeqPair(OperandVector &Operands);
111 
112 public:
113   enum AArch64MatchResultTy {
114     Match_InvalidSuffix = FIRST_TARGET_MATCH_RESULT_TY,
115 #define GET_OPERAND_DIAGNOSTIC_TYPES
116 #include "AArch64GenAsmMatcher.inc"
117   };
AArch64AsmParser(const MCSubtargetInfo & STI,MCAsmParser & Parser,const MCInstrInfo & MII,const MCTargetOptions & Options)118   AArch64AsmParser(const MCSubtargetInfo &STI, MCAsmParser &Parser,
119                    const MCInstrInfo &MII, const MCTargetOptions &Options)
120     : MCTargetAsmParser(Options, STI) {
121     MCAsmParserExtension::Initialize(Parser);
122     MCStreamer &S = getParser().getStreamer();
123     if (S.getTargetStreamer() == nullptr)
124       new AArch64TargetStreamer(S);
125 
126     // Initialize the set of available features.
127     setAvailableFeatures(ComputeAvailableFeatures(getSTI().getFeatureBits()));
128   }
129 
130   bool ParseInstruction(ParseInstructionInfo &Info, StringRef Name,
131                         SMLoc NameLoc, OperandVector &Operands) override;
132   bool ParseRegister(unsigned &RegNo, SMLoc &StartLoc, SMLoc &EndLoc) override;
133   bool ParseDirective(AsmToken DirectiveID) override;
134   unsigned validateTargetOperandClass(MCParsedAsmOperand &Op,
135                                       unsigned Kind) override;
136 
137   static bool classifySymbolRef(const MCExpr *Expr,
138                                 AArch64MCExpr::VariantKind &ELFRefKind,
139                                 MCSymbolRefExpr::VariantKind &DarwinRefKind,
140                                 int64_t &Addend);
141 };
142 } // end anonymous namespace
143 
144 namespace {
145 
146 /// AArch64Operand - Instances of this class represent a parsed AArch64 machine
147 /// instruction.
148 class AArch64Operand : public MCParsedAsmOperand {
149 private:
150   enum KindTy {
151     k_Immediate,
152     k_ShiftedImm,
153     k_CondCode,
154     k_Register,
155     k_VectorList,
156     k_VectorIndex,
157     k_Token,
158     k_SysReg,
159     k_SysCR,
160     k_Prefetch,
161     k_ShiftExtend,
162     k_FPImm,
163     k_Barrier,
164     k_PSBHint,
165   } Kind;
166 
167   SMLoc StartLoc, EndLoc;
168 
169   struct TokOp {
170     const char *Data;
171     unsigned Length;
172     bool IsSuffix; // Is the operand actually a suffix on the mnemonic.
173   };
174 
175   struct RegOp {
176     unsigned RegNum;
177     bool isVector;
178   };
179 
180   struct VectorListOp {
181     unsigned RegNum;
182     unsigned Count;
183     unsigned NumElements;
184     unsigned ElementKind;
185   };
186 
187   struct VectorIndexOp {
188     unsigned Val;
189   };
190 
191   struct ImmOp {
192     const MCExpr *Val;
193   };
194 
195   struct ShiftedImmOp {
196     const MCExpr *Val;
197     unsigned ShiftAmount;
198   };
199 
200   struct CondCodeOp {
201     AArch64CC::CondCode Code;
202   };
203 
204   struct FPImmOp {
205     unsigned Val; // Encoded 8-bit representation.
206   };
207 
208   struct BarrierOp {
209     unsigned Val; // Not the enum since not all values have names.
210     const char *Data;
211     unsigned Length;
212   };
213 
214   struct SysRegOp {
215     const char *Data;
216     unsigned Length;
217     uint32_t MRSReg;
218     uint32_t MSRReg;
219     uint32_t PStateField;
220   };
221 
222   struct SysCRImmOp {
223     unsigned Val;
224   };
225 
226   struct PrefetchOp {
227     unsigned Val;
228     const char *Data;
229     unsigned Length;
230   };
231 
232   struct PSBHintOp {
233     unsigned Val;
234     const char *Data;
235     unsigned Length;
236   };
237 
238   struct ShiftExtendOp {
239     AArch64_AM::ShiftExtendType Type;
240     unsigned Amount;
241     bool HasExplicitAmount;
242   };
243 
244   struct ExtendOp {
245     unsigned Val;
246   };
247 
248   union {
249     struct TokOp Tok;
250     struct RegOp Reg;
251     struct VectorListOp VectorList;
252     struct VectorIndexOp VectorIndex;
253     struct ImmOp Imm;
254     struct ShiftedImmOp ShiftedImm;
255     struct CondCodeOp CondCode;
256     struct FPImmOp FPImm;
257     struct BarrierOp Barrier;
258     struct SysRegOp SysReg;
259     struct SysCRImmOp SysCRImm;
260     struct PrefetchOp Prefetch;
261     struct PSBHintOp PSBHint;
262     struct ShiftExtendOp ShiftExtend;
263   };
264 
265   // Keep the MCContext around as the MCExprs may need manipulated during
266   // the add<>Operands() calls.
267   MCContext &Ctx;
268 
269 public:
AArch64Operand(KindTy K,MCContext & Ctx)270   AArch64Operand(KindTy K, MCContext &Ctx) : Kind(K), Ctx(Ctx) {}
271 
AArch64Operand(const AArch64Operand & o)272   AArch64Operand(const AArch64Operand &o) : MCParsedAsmOperand(), Ctx(o.Ctx) {
273     Kind = o.Kind;
274     StartLoc = o.StartLoc;
275     EndLoc = o.EndLoc;
276     switch (Kind) {
277     case k_Token:
278       Tok = o.Tok;
279       break;
280     case k_Immediate:
281       Imm = o.Imm;
282       break;
283     case k_ShiftedImm:
284       ShiftedImm = o.ShiftedImm;
285       break;
286     case k_CondCode:
287       CondCode = o.CondCode;
288       break;
289     case k_FPImm:
290       FPImm = o.FPImm;
291       break;
292     case k_Barrier:
293       Barrier = o.Barrier;
294       break;
295     case k_Register:
296       Reg = o.Reg;
297       break;
298     case k_VectorList:
299       VectorList = o.VectorList;
300       break;
301     case k_VectorIndex:
302       VectorIndex = o.VectorIndex;
303       break;
304     case k_SysReg:
305       SysReg = o.SysReg;
306       break;
307     case k_SysCR:
308       SysCRImm = o.SysCRImm;
309       break;
310     case k_Prefetch:
311       Prefetch = o.Prefetch;
312       break;
313     case k_PSBHint:
314       PSBHint = o.PSBHint;
315       break;
316     case k_ShiftExtend:
317       ShiftExtend = o.ShiftExtend;
318       break;
319     }
320   }
321 
322   /// getStartLoc - Get the location of the first token of this operand.
getStartLoc() const323   SMLoc getStartLoc() const override { return StartLoc; }
324   /// getEndLoc - Get the location of the last token of this operand.
getEndLoc() const325   SMLoc getEndLoc() const override { return EndLoc; }
326 
getToken() const327   StringRef getToken() const {
328     assert(Kind == k_Token && "Invalid access!");
329     return StringRef(Tok.Data, Tok.Length);
330   }
331 
isTokenSuffix() const332   bool isTokenSuffix() const {
333     assert(Kind == k_Token && "Invalid access!");
334     return Tok.IsSuffix;
335   }
336 
getImm() const337   const MCExpr *getImm() const {
338     assert(Kind == k_Immediate && "Invalid access!");
339     return Imm.Val;
340   }
341 
getShiftedImmVal() const342   const MCExpr *getShiftedImmVal() const {
343     assert(Kind == k_ShiftedImm && "Invalid access!");
344     return ShiftedImm.Val;
345   }
346 
getShiftedImmShift() const347   unsigned getShiftedImmShift() const {
348     assert(Kind == k_ShiftedImm && "Invalid access!");
349     return ShiftedImm.ShiftAmount;
350   }
351 
getCondCode() const352   AArch64CC::CondCode getCondCode() const {
353     assert(Kind == k_CondCode && "Invalid access!");
354     return CondCode.Code;
355   }
356 
getFPImm() const357   unsigned getFPImm() const {
358     assert(Kind == k_FPImm && "Invalid access!");
359     return FPImm.Val;
360   }
361 
getBarrier() const362   unsigned getBarrier() const {
363     assert(Kind == k_Barrier && "Invalid access!");
364     return Barrier.Val;
365   }
366 
getBarrierName() const367   StringRef getBarrierName() const {
368     assert(Kind == k_Barrier && "Invalid access!");
369     return StringRef(Barrier.Data, Barrier.Length);
370   }
371 
getReg() const372   unsigned getReg() const override {
373     assert(Kind == k_Register && "Invalid access!");
374     return Reg.RegNum;
375   }
376 
getVectorListStart() const377   unsigned getVectorListStart() const {
378     assert(Kind == k_VectorList && "Invalid access!");
379     return VectorList.RegNum;
380   }
381 
getVectorListCount() const382   unsigned getVectorListCount() const {
383     assert(Kind == k_VectorList && "Invalid access!");
384     return VectorList.Count;
385   }
386 
getVectorIndex() const387   unsigned getVectorIndex() const {
388     assert(Kind == k_VectorIndex && "Invalid access!");
389     return VectorIndex.Val;
390   }
391 
getSysReg() const392   StringRef getSysReg() const {
393     assert(Kind == k_SysReg && "Invalid access!");
394     return StringRef(SysReg.Data, SysReg.Length);
395   }
396 
getSysCR() const397   unsigned getSysCR() const {
398     assert(Kind == k_SysCR && "Invalid access!");
399     return SysCRImm.Val;
400   }
401 
getPrefetch() const402   unsigned getPrefetch() const {
403     assert(Kind == k_Prefetch && "Invalid access!");
404     return Prefetch.Val;
405   }
406 
getPSBHint() const407   unsigned getPSBHint() const {
408     assert(Kind == k_PSBHint && "Invalid access!");
409     return PSBHint.Val;
410   }
411 
getPSBHintName() const412   StringRef getPSBHintName() const {
413     assert(Kind == k_PSBHint && "Invalid access!");
414     return StringRef(PSBHint.Data, PSBHint.Length);
415   }
416 
getPrefetchName() const417   StringRef getPrefetchName() const {
418     assert(Kind == k_Prefetch && "Invalid access!");
419     return StringRef(Prefetch.Data, Prefetch.Length);
420   }
421 
getShiftExtendType() const422   AArch64_AM::ShiftExtendType getShiftExtendType() const {
423     assert(Kind == k_ShiftExtend && "Invalid access!");
424     return ShiftExtend.Type;
425   }
426 
getShiftExtendAmount() const427   unsigned getShiftExtendAmount() const {
428     assert(Kind == k_ShiftExtend && "Invalid access!");
429     return ShiftExtend.Amount;
430   }
431 
hasShiftExtendAmount() const432   bool hasShiftExtendAmount() const {
433     assert(Kind == k_ShiftExtend && "Invalid access!");
434     return ShiftExtend.HasExplicitAmount;
435   }
436 
isImm() const437   bool isImm() const override { return Kind == k_Immediate; }
isMem() const438   bool isMem() const override { return false; }
isSImm9() const439   bool isSImm9() const {
440     if (!isImm())
441       return false;
442     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
443     if (!MCE)
444       return false;
445     int64_t Val = MCE->getValue();
446     return (Val >= -256 && Val < 256);
447   }
isSImm7s4() const448   bool isSImm7s4() const {
449     if (!isImm())
450       return false;
451     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
452     if (!MCE)
453       return false;
454     int64_t Val = MCE->getValue();
455     return (Val >= -256 && Val <= 252 && (Val & 3) == 0);
456   }
isSImm7s8() const457   bool isSImm7s8() const {
458     if (!isImm())
459       return false;
460     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
461     if (!MCE)
462       return false;
463     int64_t Val = MCE->getValue();
464     return (Val >= -512 && Val <= 504 && (Val & 7) == 0);
465   }
isSImm7s16() const466   bool isSImm7s16() const {
467     if (!isImm())
468       return false;
469     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
470     if (!MCE)
471       return false;
472     int64_t Val = MCE->getValue();
473     return (Val >= -1024 && Val <= 1008 && (Val & 15) == 0);
474   }
475 
isSymbolicUImm12Offset(const MCExpr * Expr,unsigned Scale) const476   bool isSymbolicUImm12Offset(const MCExpr *Expr, unsigned Scale) const {
477     AArch64MCExpr::VariantKind ELFRefKind;
478     MCSymbolRefExpr::VariantKind DarwinRefKind;
479     int64_t Addend;
480     if (!AArch64AsmParser::classifySymbolRef(Expr, ELFRefKind, DarwinRefKind,
481                                            Addend)) {
482       // If we don't understand the expression, assume the best and
483       // let the fixup and relocation code deal with it.
484       return true;
485     }
486 
487     if (DarwinRefKind == MCSymbolRefExpr::VK_PAGEOFF ||
488         ELFRefKind == AArch64MCExpr::VK_LO12 ||
489         ELFRefKind == AArch64MCExpr::VK_GOT_LO12 ||
490         ELFRefKind == AArch64MCExpr::VK_DTPREL_LO12 ||
491         ELFRefKind == AArch64MCExpr::VK_DTPREL_LO12_NC ||
492         ELFRefKind == AArch64MCExpr::VK_TPREL_LO12 ||
493         ELFRefKind == AArch64MCExpr::VK_TPREL_LO12_NC ||
494         ELFRefKind == AArch64MCExpr::VK_GOTTPREL_LO12_NC ||
495         ELFRefKind == AArch64MCExpr::VK_TLSDESC_LO12) {
496       // Note that we don't range-check the addend. It's adjusted modulo page
497       // size when converted, so there is no "out of range" condition when using
498       // @pageoff.
499       return Addend >= 0 && (Addend % Scale) == 0;
500     } else if (DarwinRefKind == MCSymbolRefExpr::VK_GOTPAGEOFF ||
501                DarwinRefKind == MCSymbolRefExpr::VK_TLVPPAGEOFF) {
502       // @gotpageoff/@tlvppageoff can only be used directly, not with an addend.
503       return Addend == 0;
504     }
505 
506     return false;
507   }
508 
isUImm12Offset() const509   template <int Scale> bool isUImm12Offset() const {
510     if (!isImm())
511       return false;
512 
513     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
514     if (!MCE)
515       return isSymbolicUImm12Offset(getImm(), Scale);
516 
517     int64_t Val = MCE->getValue();
518     return (Val % Scale) == 0 && Val >= 0 && (Val / Scale) < 0x1000;
519   }
520 
isImm0_1() const521   bool isImm0_1() const {
522     if (!isImm())
523       return false;
524     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
525     if (!MCE)
526       return false;
527     int64_t Val = MCE->getValue();
528     return (Val >= 0 && Val < 2);
529   }
isImm0_7() const530   bool isImm0_7() const {
531     if (!isImm())
532       return false;
533     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
534     if (!MCE)
535       return false;
536     int64_t Val = MCE->getValue();
537     return (Val >= 0 && Val < 8);
538   }
isImm1_8() const539   bool isImm1_8() const {
540     if (!isImm())
541       return false;
542     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
543     if (!MCE)
544       return false;
545     int64_t Val = MCE->getValue();
546     return (Val > 0 && Val < 9);
547   }
isImm0_15() const548   bool isImm0_15() const {
549     if (!isImm())
550       return false;
551     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
552     if (!MCE)
553       return false;
554     int64_t Val = MCE->getValue();
555     return (Val >= 0 && Val < 16);
556   }
isImm1_16() const557   bool isImm1_16() const {
558     if (!isImm())
559       return false;
560     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
561     if (!MCE)
562       return false;
563     int64_t Val = MCE->getValue();
564     return (Val > 0 && Val < 17);
565   }
isImm0_31() const566   bool isImm0_31() const {
567     if (!isImm())
568       return false;
569     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
570     if (!MCE)
571       return false;
572     int64_t Val = MCE->getValue();
573     return (Val >= 0 && Val < 32);
574   }
isImm1_31() const575   bool isImm1_31() const {
576     if (!isImm())
577       return false;
578     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
579     if (!MCE)
580       return false;
581     int64_t Val = MCE->getValue();
582     return (Val >= 1 && Val < 32);
583   }
isImm1_32() const584   bool isImm1_32() const {
585     if (!isImm())
586       return false;
587     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
588     if (!MCE)
589       return false;
590     int64_t Val = MCE->getValue();
591     return (Val >= 1 && Val < 33);
592   }
isImm0_63() const593   bool isImm0_63() const {
594     if (!isImm())
595       return false;
596     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
597     if (!MCE)
598       return false;
599     int64_t Val = MCE->getValue();
600     return (Val >= 0 && Val < 64);
601   }
isImm1_63() const602   bool isImm1_63() const {
603     if (!isImm())
604       return false;
605     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
606     if (!MCE)
607       return false;
608     int64_t Val = MCE->getValue();
609     return (Val >= 1 && Val < 64);
610   }
isImm1_64() const611   bool isImm1_64() const {
612     if (!isImm())
613       return false;
614     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
615     if (!MCE)
616       return false;
617     int64_t Val = MCE->getValue();
618     return (Val >= 1 && Val < 65);
619   }
isImm0_127() const620   bool isImm0_127() const {
621     if (!isImm())
622       return false;
623     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
624     if (!MCE)
625       return false;
626     int64_t Val = MCE->getValue();
627     return (Val >= 0 && Val < 128);
628   }
isImm0_255() const629   bool isImm0_255() const {
630     if (!isImm())
631       return false;
632     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
633     if (!MCE)
634       return false;
635     int64_t Val = MCE->getValue();
636     return (Val >= 0 && Val < 256);
637   }
isImm0_65535() const638   bool isImm0_65535() const {
639     if (!isImm())
640       return false;
641     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
642     if (!MCE)
643       return false;
644     int64_t Val = MCE->getValue();
645     return (Val >= 0 && Val < 65536);
646   }
isImm32_63() const647   bool isImm32_63() const {
648     if (!isImm())
649       return false;
650     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
651     if (!MCE)
652       return false;
653     int64_t Val = MCE->getValue();
654     return (Val >= 32 && Val < 64);
655   }
isLogicalImm32() const656   bool isLogicalImm32() const {
657     if (!isImm())
658       return false;
659     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
660     if (!MCE)
661       return false;
662     int64_t Val = MCE->getValue();
663     if (Val >> 32 != 0 && Val >> 32 != ~0LL)
664       return false;
665     Val &= 0xFFFFFFFF;
666     return AArch64_AM::isLogicalImmediate(Val, 32);
667   }
isLogicalImm64() const668   bool isLogicalImm64() const {
669     if (!isImm())
670       return false;
671     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
672     if (!MCE)
673       return false;
674     return AArch64_AM::isLogicalImmediate(MCE->getValue(), 64);
675   }
isLogicalImm32Not() const676   bool isLogicalImm32Not() const {
677     if (!isImm())
678       return false;
679     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
680     if (!MCE)
681       return false;
682     int64_t Val = ~MCE->getValue() & 0xFFFFFFFF;
683     return AArch64_AM::isLogicalImmediate(Val, 32);
684   }
isLogicalImm64Not() const685   bool isLogicalImm64Not() const {
686     if (!isImm())
687       return false;
688     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
689     if (!MCE)
690       return false;
691     return AArch64_AM::isLogicalImmediate(~MCE->getValue(), 64);
692   }
isShiftedImm() const693   bool isShiftedImm() const { return Kind == k_ShiftedImm; }
isAddSubImm() const694   bool isAddSubImm() const {
695     if (!isShiftedImm() && !isImm())
696       return false;
697 
698     const MCExpr *Expr;
699 
700     // An ADD/SUB shifter is either 'lsl #0' or 'lsl #12'.
701     if (isShiftedImm()) {
702       unsigned Shift = ShiftedImm.ShiftAmount;
703       Expr = ShiftedImm.Val;
704       if (Shift != 0 && Shift != 12)
705         return false;
706     } else {
707       Expr = getImm();
708     }
709 
710     AArch64MCExpr::VariantKind ELFRefKind;
711     MCSymbolRefExpr::VariantKind DarwinRefKind;
712     int64_t Addend;
713     if (AArch64AsmParser::classifySymbolRef(Expr, ELFRefKind,
714                                           DarwinRefKind, Addend)) {
715       return DarwinRefKind == MCSymbolRefExpr::VK_PAGEOFF
716           || DarwinRefKind == MCSymbolRefExpr::VK_TLVPPAGEOFF
717           || (DarwinRefKind == MCSymbolRefExpr::VK_GOTPAGEOFF && Addend == 0)
718           || ELFRefKind == AArch64MCExpr::VK_LO12
719           || ELFRefKind == AArch64MCExpr::VK_DTPREL_HI12
720           || ELFRefKind == AArch64MCExpr::VK_DTPREL_LO12
721           || ELFRefKind == AArch64MCExpr::VK_DTPREL_LO12_NC
722           || ELFRefKind == AArch64MCExpr::VK_TPREL_HI12
723           || ELFRefKind == AArch64MCExpr::VK_TPREL_LO12
724           || ELFRefKind == AArch64MCExpr::VK_TPREL_LO12_NC
725           || ELFRefKind == AArch64MCExpr::VK_TLSDESC_LO12;
726     }
727 
728     // Otherwise it should be a real immediate in range:
729     const MCConstantExpr *CE = cast<MCConstantExpr>(Expr);
730     return CE->getValue() >= 0 && CE->getValue() <= 0xfff;
731   }
isAddSubImmNeg() const732   bool isAddSubImmNeg() const {
733     if (!isShiftedImm() && !isImm())
734       return false;
735 
736     const MCExpr *Expr;
737 
738     // An ADD/SUB shifter is either 'lsl #0' or 'lsl #12'.
739     if (isShiftedImm()) {
740       unsigned Shift = ShiftedImm.ShiftAmount;
741       Expr = ShiftedImm.Val;
742       if (Shift != 0 && Shift != 12)
743         return false;
744     } else
745       Expr = getImm();
746 
747     // Otherwise it should be a real negative immediate in range:
748     const MCConstantExpr *CE = dyn_cast<MCConstantExpr>(Expr);
749     return CE != nullptr && CE->getValue() < 0 && -CE->getValue() <= 0xfff;
750   }
isCondCode() const751   bool isCondCode() const { return Kind == k_CondCode; }
isSIMDImmType10() const752   bool isSIMDImmType10() const {
753     if (!isImm())
754       return false;
755     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
756     if (!MCE)
757       return false;
758     return AArch64_AM::isAdvSIMDModImmType10(MCE->getValue());
759   }
isBranchTarget26() const760   bool isBranchTarget26() const {
761     if (!isImm())
762       return false;
763     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
764     if (!MCE)
765       return true;
766     int64_t Val = MCE->getValue();
767     if (Val & 0x3)
768       return false;
769     return (Val >= -(0x2000000 << 2) && Val <= (0x1ffffff << 2));
770   }
isPCRelLabel19() const771   bool isPCRelLabel19() const {
772     if (!isImm())
773       return false;
774     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
775     if (!MCE)
776       return true;
777     int64_t Val = MCE->getValue();
778     if (Val & 0x3)
779       return false;
780     return (Val >= -(0x40000 << 2) && Val <= (0x3ffff << 2));
781   }
isBranchTarget14() const782   bool isBranchTarget14() const {
783     if (!isImm())
784       return false;
785     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
786     if (!MCE)
787       return true;
788     int64_t Val = MCE->getValue();
789     if (Val & 0x3)
790       return false;
791     return (Val >= -(0x2000 << 2) && Val <= (0x1fff << 2));
792   }
793 
794   bool
isMovWSymbol(ArrayRef<AArch64MCExpr::VariantKind> AllowedModifiers) const795   isMovWSymbol(ArrayRef<AArch64MCExpr::VariantKind> AllowedModifiers) const {
796     if (!isImm())
797       return false;
798 
799     AArch64MCExpr::VariantKind ELFRefKind;
800     MCSymbolRefExpr::VariantKind DarwinRefKind;
801     int64_t Addend;
802     if (!AArch64AsmParser::classifySymbolRef(getImm(), ELFRefKind,
803                                              DarwinRefKind, Addend)) {
804       return false;
805     }
806     if (DarwinRefKind != MCSymbolRefExpr::VK_None)
807       return false;
808 
809     for (unsigned i = 0; i != AllowedModifiers.size(); ++i) {
810       if (ELFRefKind == AllowedModifiers[i])
811         return Addend == 0;
812     }
813 
814     return false;
815   }
816 
isMovZSymbolG3() const817   bool isMovZSymbolG3() const {
818     return isMovWSymbol(AArch64MCExpr::VK_ABS_G3);
819   }
820 
isMovZSymbolG2() const821   bool isMovZSymbolG2() const {
822     return isMovWSymbol({AArch64MCExpr::VK_ABS_G2, AArch64MCExpr::VK_ABS_G2_S,
823                          AArch64MCExpr::VK_TPREL_G2,
824                          AArch64MCExpr::VK_DTPREL_G2});
825   }
826 
isMovZSymbolG1() const827   bool isMovZSymbolG1() const {
828     return isMovWSymbol({
829         AArch64MCExpr::VK_ABS_G1, AArch64MCExpr::VK_ABS_G1_S,
830         AArch64MCExpr::VK_GOTTPREL_G1, AArch64MCExpr::VK_TPREL_G1,
831         AArch64MCExpr::VK_DTPREL_G1,
832     });
833   }
834 
isMovZSymbolG0() const835   bool isMovZSymbolG0() const {
836     return isMovWSymbol({AArch64MCExpr::VK_ABS_G0, AArch64MCExpr::VK_ABS_G0_S,
837                          AArch64MCExpr::VK_TPREL_G0,
838                          AArch64MCExpr::VK_DTPREL_G0});
839   }
840 
isMovKSymbolG3() const841   bool isMovKSymbolG3() const {
842     return isMovWSymbol(AArch64MCExpr::VK_ABS_G3);
843   }
844 
isMovKSymbolG2() const845   bool isMovKSymbolG2() const {
846     return isMovWSymbol(AArch64MCExpr::VK_ABS_G2_NC);
847   }
848 
isMovKSymbolG1() const849   bool isMovKSymbolG1() const {
850     return isMovWSymbol({AArch64MCExpr::VK_ABS_G1_NC,
851                          AArch64MCExpr::VK_TPREL_G1_NC,
852                          AArch64MCExpr::VK_DTPREL_G1_NC});
853   }
854 
isMovKSymbolG0() const855   bool isMovKSymbolG0() const {
856     return isMovWSymbol(
857         {AArch64MCExpr::VK_ABS_G0_NC, AArch64MCExpr::VK_GOTTPREL_G0_NC,
858          AArch64MCExpr::VK_TPREL_G0_NC, AArch64MCExpr::VK_DTPREL_G0_NC});
859   }
860 
861   template<int RegWidth, int Shift>
isMOVZMovAlias() const862   bool isMOVZMovAlias() const {
863     if (!isImm()) return false;
864 
865     const MCConstantExpr *CE = dyn_cast<MCConstantExpr>(getImm());
866     if (!CE) return false;
867     uint64_t Value = CE->getValue();
868 
869     if (RegWidth == 32)
870       Value &= 0xffffffffULL;
871 
872     // "lsl #0" takes precedence: in practice this only affects "#0, lsl #0".
873     if (Value == 0 && Shift != 0)
874       return false;
875 
876     return (Value & ~(0xffffULL << Shift)) == 0;
877   }
878 
879   template<int RegWidth, int Shift>
isMOVNMovAlias() const880   bool isMOVNMovAlias() const {
881     if (!isImm()) return false;
882 
883     const MCConstantExpr *CE = dyn_cast<MCConstantExpr>(getImm());
884     if (!CE) return false;
885     uint64_t Value = CE->getValue();
886 
887     // MOVZ takes precedence over MOVN.
888     for (int MOVZShift = 0; MOVZShift <= 48; MOVZShift += 16)
889       if ((Value & ~(0xffffULL << MOVZShift)) == 0)
890         return false;
891 
892     Value = ~Value;
893     if (RegWidth == 32)
894       Value &= 0xffffffffULL;
895 
896     return (Value & ~(0xffffULL << Shift)) == 0;
897   }
898 
isFPImm() const899   bool isFPImm() const { return Kind == k_FPImm; }
isBarrier() const900   bool isBarrier() const { return Kind == k_Barrier; }
isSysReg() const901   bool isSysReg() const { return Kind == k_SysReg; }
isMRSSystemRegister() const902   bool isMRSSystemRegister() const {
903     if (!isSysReg()) return false;
904 
905     return SysReg.MRSReg != -1U;
906   }
isMSRSystemRegister() const907   bool isMSRSystemRegister() const {
908     if (!isSysReg()) return false;
909     return SysReg.MSRReg != -1U;
910   }
isSystemPStateFieldWithImm0_1() const911   bool isSystemPStateFieldWithImm0_1() const {
912     if (!isSysReg()) return false;
913     return (SysReg.PStateField == AArch64PState::PAN ||
914             SysReg.PStateField == AArch64PState::UAO);
915   }
isSystemPStateFieldWithImm0_15() const916   bool isSystemPStateFieldWithImm0_15() const {
917     if (!isSysReg() || isSystemPStateFieldWithImm0_1()) return false;
918     return SysReg.PStateField != -1U;
919   }
isReg() const920   bool isReg() const override { return Kind == k_Register && !Reg.isVector; }
isVectorReg() const921   bool isVectorReg() const { return Kind == k_Register && Reg.isVector; }
isVectorRegLo() const922   bool isVectorRegLo() const {
923     return Kind == k_Register && Reg.isVector &&
924            AArch64MCRegisterClasses[AArch64::FPR128_loRegClassID].contains(
925                Reg.RegNum);
926   }
isGPR32as64() const927   bool isGPR32as64() const {
928     return Kind == k_Register && !Reg.isVector &&
929       AArch64MCRegisterClasses[AArch64::GPR64RegClassID].contains(Reg.RegNum);
930   }
isWSeqPair() const931   bool isWSeqPair() const {
932     return Kind == k_Register && !Reg.isVector &&
933            AArch64MCRegisterClasses[AArch64::WSeqPairsClassRegClassID].contains(
934                Reg.RegNum);
935   }
isXSeqPair() const936   bool isXSeqPair() const {
937     return Kind == k_Register && !Reg.isVector &&
938            AArch64MCRegisterClasses[AArch64::XSeqPairsClassRegClassID].contains(
939                Reg.RegNum);
940   }
941 
isGPR64sp0() const942   bool isGPR64sp0() const {
943     return Kind == k_Register && !Reg.isVector &&
944       AArch64MCRegisterClasses[AArch64::GPR64spRegClassID].contains(Reg.RegNum);
945   }
946 
947   /// Is this a vector list with the type implicit (presumably attached to the
948   /// instruction itself)?
isImplicitlyTypedVectorList() const949   template <unsigned NumRegs> bool isImplicitlyTypedVectorList() const {
950     return Kind == k_VectorList && VectorList.Count == NumRegs &&
951            !VectorList.ElementKind;
952   }
953 
954   template <unsigned NumRegs, unsigned NumElements, char ElementKind>
isTypedVectorList() const955   bool isTypedVectorList() const {
956     if (Kind != k_VectorList)
957       return false;
958     if (VectorList.Count != NumRegs)
959       return false;
960     if (VectorList.ElementKind != ElementKind)
961       return false;
962     return VectorList.NumElements == NumElements;
963   }
964 
isVectorIndex1() const965   bool isVectorIndex1() const {
966     return Kind == k_VectorIndex && VectorIndex.Val == 1;
967   }
isVectorIndexB() const968   bool isVectorIndexB() const {
969     return Kind == k_VectorIndex && VectorIndex.Val < 16;
970   }
isVectorIndexH() const971   bool isVectorIndexH() const {
972     return Kind == k_VectorIndex && VectorIndex.Val < 8;
973   }
isVectorIndexS() const974   bool isVectorIndexS() const {
975     return Kind == k_VectorIndex && VectorIndex.Val < 4;
976   }
isVectorIndexD() const977   bool isVectorIndexD() const {
978     return Kind == k_VectorIndex && VectorIndex.Val < 2;
979   }
isToken() const980   bool isToken() const override { return Kind == k_Token; }
isTokenEqual(StringRef Str) const981   bool isTokenEqual(StringRef Str) const {
982     return Kind == k_Token && getToken() == Str;
983   }
isSysCR() const984   bool isSysCR() const { return Kind == k_SysCR; }
isPrefetch() const985   bool isPrefetch() const { return Kind == k_Prefetch; }
isPSBHint() const986   bool isPSBHint() const { return Kind == k_PSBHint; }
isShiftExtend() const987   bool isShiftExtend() const { return Kind == k_ShiftExtend; }
isShifter() const988   bool isShifter() const {
989     if (!isShiftExtend())
990       return false;
991 
992     AArch64_AM::ShiftExtendType ST = getShiftExtendType();
993     return (ST == AArch64_AM::LSL || ST == AArch64_AM::LSR ||
994             ST == AArch64_AM::ASR || ST == AArch64_AM::ROR ||
995             ST == AArch64_AM::MSL);
996   }
isExtend() const997   bool isExtend() const {
998     if (!isShiftExtend())
999       return false;
1000 
1001     AArch64_AM::ShiftExtendType ET = getShiftExtendType();
1002     return (ET == AArch64_AM::UXTB || ET == AArch64_AM::SXTB ||
1003             ET == AArch64_AM::UXTH || ET == AArch64_AM::SXTH ||
1004             ET == AArch64_AM::UXTW || ET == AArch64_AM::SXTW ||
1005             ET == AArch64_AM::UXTX || ET == AArch64_AM::SXTX ||
1006             ET == AArch64_AM::LSL) &&
1007            getShiftExtendAmount() <= 4;
1008   }
1009 
isExtend64() const1010   bool isExtend64() const {
1011     if (!isExtend())
1012       return false;
1013     // UXTX and SXTX require a 64-bit source register (the ExtendLSL64 class).
1014     AArch64_AM::ShiftExtendType ET = getShiftExtendType();
1015     return ET != AArch64_AM::UXTX && ET != AArch64_AM::SXTX;
1016   }
isExtendLSL64() const1017   bool isExtendLSL64() const {
1018     if (!isExtend())
1019       return false;
1020     AArch64_AM::ShiftExtendType ET = getShiftExtendType();
1021     return (ET == AArch64_AM::UXTX || ET == AArch64_AM::SXTX ||
1022             ET == AArch64_AM::LSL) &&
1023            getShiftExtendAmount() <= 4;
1024   }
1025 
isMemXExtend() const1026   template<int Width> bool isMemXExtend() const {
1027     if (!isExtend())
1028       return false;
1029     AArch64_AM::ShiftExtendType ET = getShiftExtendType();
1030     return (ET == AArch64_AM::LSL || ET == AArch64_AM::SXTX) &&
1031            (getShiftExtendAmount() == Log2_32(Width / 8) ||
1032             getShiftExtendAmount() == 0);
1033   }
1034 
isMemWExtend() const1035   template<int Width> bool isMemWExtend() const {
1036     if (!isExtend())
1037       return false;
1038     AArch64_AM::ShiftExtendType ET = getShiftExtendType();
1039     return (ET == AArch64_AM::UXTW || ET == AArch64_AM::SXTW) &&
1040            (getShiftExtendAmount() == Log2_32(Width / 8) ||
1041             getShiftExtendAmount() == 0);
1042   }
1043 
1044   template <unsigned width>
isArithmeticShifter() const1045   bool isArithmeticShifter() const {
1046     if (!isShifter())
1047       return false;
1048 
1049     // An arithmetic shifter is LSL, LSR, or ASR.
1050     AArch64_AM::ShiftExtendType ST = getShiftExtendType();
1051     return (ST == AArch64_AM::LSL || ST == AArch64_AM::LSR ||
1052             ST == AArch64_AM::ASR) && getShiftExtendAmount() < width;
1053   }
1054 
1055   template <unsigned width>
isLogicalShifter() const1056   bool isLogicalShifter() const {
1057     if (!isShifter())
1058       return false;
1059 
1060     // A logical shifter is LSL, LSR, ASR or ROR.
1061     AArch64_AM::ShiftExtendType ST = getShiftExtendType();
1062     return (ST == AArch64_AM::LSL || ST == AArch64_AM::LSR ||
1063             ST == AArch64_AM::ASR || ST == AArch64_AM::ROR) &&
1064            getShiftExtendAmount() < width;
1065   }
1066 
isMovImm32Shifter() const1067   bool isMovImm32Shifter() const {
1068     if (!isShifter())
1069       return false;
1070 
1071     // A MOVi shifter is LSL of 0, 16, 32, or 48.
1072     AArch64_AM::ShiftExtendType ST = getShiftExtendType();
1073     if (ST != AArch64_AM::LSL)
1074       return false;
1075     uint64_t Val = getShiftExtendAmount();
1076     return (Val == 0 || Val == 16);
1077   }
1078 
isMovImm64Shifter() const1079   bool isMovImm64Shifter() const {
1080     if (!isShifter())
1081       return false;
1082 
1083     // A MOVi shifter is LSL of 0 or 16.
1084     AArch64_AM::ShiftExtendType ST = getShiftExtendType();
1085     if (ST != AArch64_AM::LSL)
1086       return false;
1087     uint64_t Val = getShiftExtendAmount();
1088     return (Val == 0 || Val == 16 || Val == 32 || Val == 48);
1089   }
1090 
isLogicalVecShifter() const1091   bool isLogicalVecShifter() const {
1092     if (!isShifter())
1093       return false;
1094 
1095     // A logical vector shifter is a left shift by 0, 8, 16, or 24.
1096     unsigned Shift = getShiftExtendAmount();
1097     return getShiftExtendType() == AArch64_AM::LSL &&
1098            (Shift == 0 || Shift == 8 || Shift == 16 || Shift == 24);
1099   }
1100 
isLogicalVecHalfWordShifter() const1101   bool isLogicalVecHalfWordShifter() const {
1102     if (!isLogicalVecShifter())
1103       return false;
1104 
1105     // A logical vector shifter is a left shift by 0 or 8.
1106     unsigned Shift = getShiftExtendAmount();
1107     return getShiftExtendType() == AArch64_AM::LSL &&
1108            (Shift == 0 || Shift == 8);
1109   }
1110 
isMoveVecShifter() const1111   bool isMoveVecShifter() const {
1112     if (!isShiftExtend())
1113       return false;
1114 
1115     // A logical vector shifter is a left shift by 8 or 16.
1116     unsigned Shift = getShiftExtendAmount();
1117     return getShiftExtendType() == AArch64_AM::MSL &&
1118            (Shift == 8 || Shift == 16);
1119   }
1120 
1121   // Fallback unscaled operands are for aliases of LDR/STR that fall back
1122   // to LDUR/STUR when the offset is not legal for the former but is for
1123   // the latter. As such, in addition to checking for being a legal unscaled
1124   // address, also check that it is not a legal scaled address. This avoids
1125   // ambiguity in the matcher.
1126   template<int Width>
isSImm9OffsetFB() const1127   bool isSImm9OffsetFB() const {
1128     return isSImm9() && !isUImm12Offset<Width / 8>();
1129   }
1130 
isAdrpLabel() const1131   bool isAdrpLabel() const {
1132     // Validation was handled during parsing, so we just sanity check that
1133     // something didn't go haywire.
1134     if (!isImm())
1135         return false;
1136 
1137     if (const MCConstantExpr *CE = dyn_cast<MCConstantExpr>(Imm.Val)) {
1138       int64_t Val = CE->getValue();
1139       int64_t Min = - (4096 * (1LL << (21 - 1)));
1140       int64_t Max = 4096 * ((1LL << (21 - 1)) - 1);
1141       return (Val % 4096) == 0 && Val >= Min && Val <= Max;
1142     }
1143 
1144     return true;
1145   }
1146 
isAdrLabel() const1147   bool isAdrLabel() const {
1148     // Validation was handled during parsing, so we just sanity check that
1149     // something didn't go haywire.
1150     if (!isImm())
1151         return false;
1152 
1153     if (const MCConstantExpr *CE = dyn_cast<MCConstantExpr>(Imm.Val)) {
1154       int64_t Val = CE->getValue();
1155       int64_t Min = - (1LL << (21 - 1));
1156       int64_t Max = ((1LL << (21 - 1)) - 1);
1157       return Val >= Min && Val <= Max;
1158     }
1159 
1160     return true;
1161   }
1162 
addExpr(MCInst & Inst,const MCExpr * Expr) const1163   void addExpr(MCInst &Inst, const MCExpr *Expr) const {
1164     // Add as immediates when possible.  Null MCExpr = 0.
1165     if (!Expr)
1166       Inst.addOperand(MCOperand::createImm(0));
1167     else if (const MCConstantExpr *CE = dyn_cast<MCConstantExpr>(Expr))
1168       Inst.addOperand(MCOperand::createImm(CE->getValue()));
1169     else
1170       Inst.addOperand(MCOperand::createExpr(Expr));
1171   }
1172 
addRegOperands(MCInst & Inst,unsigned N) const1173   void addRegOperands(MCInst &Inst, unsigned N) const {
1174     assert(N == 1 && "Invalid number of operands!");
1175     Inst.addOperand(MCOperand::createReg(getReg()));
1176   }
1177 
addGPR32as64Operands(MCInst & Inst,unsigned N) const1178   void addGPR32as64Operands(MCInst &Inst, unsigned N) const {
1179     assert(N == 1 && "Invalid number of operands!");
1180     assert(
1181         AArch64MCRegisterClasses[AArch64::GPR64RegClassID].contains(getReg()));
1182 
1183     const MCRegisterInfo *RI = Ctx.getRegisterInfo();
1184     uint32_t Reg = RI->getRegClass(AArch64::GPR32RegClassID).getRegister(
1185         RI->getEncodingValue(getReg()));
1186 
1187     Inst.addOperand(MCOperand::createReg(Reg));
1188   }
1189 
addVectorReg64Operands(MCInst & Inst,unsigned N) const1190   void addVectorReg64Operands(MCInst &Inst, unsigned N) const {
1191     assert(N == 1 && "Invalid number of operands!");
1192     assert(
1193         AArch64MCRegisterClasses[AArch64::FPR128RegClassID].contains(getReg()));
1194     Inst.addOperand(MCOperand::createReg(AArch64::D0 + getReg() - AArch64::Q0));
1195   }
1196 
addVectorReg128Operands(MCInst & Inst,unsigned N) const1197   void addVectorReg128Operands(MCInst &Inst, unsigned N) const {
1198     assert(N == 1 && "Invalid number of operands!");
1199     assert(
1200         AArch64MCRegisterClasses[AArch64::FPR128RegClassID].contains(getReg()));
1201     Inst.addOperand(MCOperand::createReg(getReg()));
1202   }
1203 
addVectorRegLoOperands(MCInst & Inst,unsigned N) const1204   void addVectorRegLoOperands(MCInst &Inst, unsigned N) const {
1205     assert(N == 1 && "Invalid number of operands!");
1206     Inst.addOperand(MCOperand::createReg(getReg()));
1207   }
1208 
1209   template <unsigned NumRegs>
addVectorList64Operands(MCInst & Inst,unsigned N) const1210   void addVectorList64Operands(MCInst &Inst, unsigned N) const {
1211     assert(N == 1 && "Invalid number of operands!");
1212     static const unsigned FirstRegs[] = { AArch64::D0,
1213                                           AArch64::D0_D1,
1214                                           AArch64::D0_D1_D2,
1215                                           AArch64::D0_D1_D2_D3 };
1216     unsigned FirstReg = FirstRegs[NumRegs - 1];
1217 
1218     Inst.addOperand(
1219         MCOperand::createReg(FirstReg + getVectorListStart() - AArch64::Q0));
1220   }
1221 
1222   template <unsigned NumRegs>
addVectorList128Operands(MCInst & Inst,unsigned N) const1223   void addVectorList128Operands(MCInst &Inst, unsigned N) const {
1224     assert(N == 1 && "Invalid number of operands!");
1225     static const unsigned FirstRegs[] = { AArch64::Q0,
1226                                           AArch64::Q0_Q1,
1227                                           AArch64::Q0_Q1_Q2,
1228                                           AArch64::Q0_Q1_Q2_Q3 };
1229     unsigned FirstReg = FirstRegs[NumRegs - 1];
1230 
1231     Inst.addOperand(
1232         MCOperand::createReg(FirstReg + getVectorListStart() - AArch64::Q0));
1233   }
1234 
addVectorIndex1Operands(MCInst & Inst,unsigned N) const1235   void addVectorIndex1Operands(MCInst &Inst, unsigned N) const {
1236     assert(N == 1 && "Invalid number of operands!");
1237     Inst.addOperand(MCOperand::createImm(getVectorIndex()));
1238   }
1239 
addVectorIndexBOperands(MCInst & Inst,unsigned N) const1240   void addVectorIndexBOperands(MCInst &Inst, unsigned N) const {
1241     assert(N == 1 && "Invalid number of operands!");
1242     Inst.addOperand(MCOperand::createImm(getVectorIndex()));
1243   }
1244 
addVectorIndexHOperands(MCInst & Inst,unsigned N) const1245   void addVectorIndexHOperands(MCInst &Inst, unsigned N) const {
1246     assert(N == 1 && "Invalid number of operands!");
1247     Inst.addOperand(MCOperand::createImm(getVectorIndex()));
1248   }
1249 
addVectorIndexSOperands(MCInst & Inst,unsigned N) const1250   void addVectorIndexSOperands(MCInst &Inst, unsigned N) const {
1251     assert(N == 1 && "Invalid number of operands!");
1252     Inst.addOperand(MCOperand::createImm(getVectorIndex()));
1253   }
1254 
addVectorIndexDOperands(MCInst & Inst,unsigned N) const1255   void addVectorIndexDOperands(MCInst &Inst, unsigned N) const {
1256     assert(N == 1 && "Invalid number of operands!");
1257     Inst.addOperand(MCOperand::createImm(getVectorIndex()));
1258   }
1259 
addImmOperands(MCInst & Inst,unsigned N) const1260   void addImmOperands(MCInst &Inst, unsigned N) const {
1261     assert(N == 1 && "Invalid number of operands!");
1262     // If this is a pageoff symrefexpr with an addend, adjust the addend
1263     // to be only the page-offset portion. Otherwise, just add the expr
1264     // as-is.
1265     addExpr(Inst, getImm());
1266   }
1267 
addAddSubImmOperands(MCInst & Inst,unsigned N) const1268   void addAddSubImmOperands(MCInst &Inst, unsigned N) const {
1269     assert(N == 2 && "Invalid number of operands!");
1270     if (isShiftedImm()) {
1271       addExpr(Inst, getShiftedImmVal());
1272       Inst.addOperand(MCOperand::createImm(getShiftedImmShift()));
1273     } else {
1274       addExpr(Inst, getImm());
1275       Inst.addOperand(MCOperand::createImm(0));
1276     }
1277   }
1278 
addAddSubImmNegOperands(MCInst & Inst,unsigned N) const1279   void addAddSubImmNegOperands(MCInst &Inst, unsigned N) const {
1280     assert(N == 2 && "Invalid number of operands!");
1281 
1282     const MCExpr *MCE = isShiftedImm() ? getShiftedImmVal() : getImm();
1283     const MCConstantExpr *CE = cast<MCConstantExpr>(MCE);
1284     int64_t Val = -CE->getValue();
1285     unsigned ShiftAmt = isShiftedImm() ? ShiftedImm.ShiftAmount : 0;
1286 
1287     Inst.addOperand(MCOperand::createImm(Val));
1288     Inst.addOperand(MCOperand::createImm(ShiftAmt));
1289   }
1290 
addCondCodeOperands(MCInst & Inst,unsigned N) const1291   void addCondCodeOperands(MCInst &Inst, unsigned N) const {
1292     assert(N == 1 && "Invalid number of operands!");
1293     Inst.addOperand(MCOperand::createImm(getCondCode()));
1294   }
1295 
addAdrpLabelOperands(MCInst & Inst,unsigned N) const1296   void addAdrpLabelOperands(MCInst &Inst, unsigned N) const {
1297     assert(N == 1 && "Invalid number of operands!");
1298     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
1299     if (!MCE)
1300       addExpr(Inst, getImm());
1301     else
1302       Inst.addOperand(MCOperand::createImm(MCE->getValue() >> 12));
1303   }
1304 
addAdrLabelOperands(MCInst & Inst,unsigned N) const1305   void addAdrLabelOperands(MCInst &Inst, unsigned N) const {
1306     addImmOperands(Inst, N);
1307   }
1308 
1309   template<int Scale>
addUImm12OffsetOperands(MCInst & Inst,unsigned N) const1310   void addUImm12OffsetOperands(MCInst &Inst, unsigned N) const {
1311     assert(N == 1 && "Invalid number of operands!");
1312     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
1313 
1314     if (!MCE) {
1315       Inst.addOperand(MCOperand::createExpr(getImm()));
1316       return;
1317     }
1318     Inst.addOperand(MCOperand::createImm(MCE->getValue() / Scale));
1319   }
1320 
addSImm9Operands(MCInst & Inst,unsigned N) const1321   void addSImm9Operands(MCInst &Inst, unsigned N) const {
1322     assert(N == 1 && "Invalid number of operands!");
1323     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1324     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1325   }
1326 
addSImm7s4Operands(MCInst & Inst,unsigned N) const1327   void addSImm7s4Operands(MCInst &Inst, unsigned N) const {
1328     assert(N == 1 && "Invalid number of operands!");
1329     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1330     Inst.addOperand(MCOperand::createImm(MCE->getValue() / 4));
1331   }
1332 
addSImm7s8Operands(MCInst & Inst,unsigned N) const1333   void addSImm7s8Operands(MCInst &Inst, unsigned N) const {
1334     assert(N == 1 && "Invalid number of operands!");
1335     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1336     Inst.addOperand(MCOperand::createImm(MCE->getValue() / 8));
1337   }
1338 
addSImm7s16Operands(MCInst & Inst,unsigned N) const1339   void addSImm7s16Operands(MCInst &Inst, unsigned N) const {
1340     assert(N == 1 && "Invalid number of operands!");
1341     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1342     Inst.addOperand(MCOperand::createImm(MCE->getValue() / 16));
1343   }
1344 
addImm0_1Operands(MCInst & Inst,unsigned N) const1345   void addImm0_1Operands(MCInst &Inst, unsigned N) const {
1346     assert(N == 1 && "Invalid number of operands!");
1347     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1348     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1349   }
1350 
addImm0_7Operands(MCInst & Inst,unsigned N) const1351   void addImm0_7Operands(MCInst &Inst, unsigned N) const {
1352     assert(N == 1 && "Invalid number of operands!");
1353     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1354     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1355   }
1356 
addImm1_8Operands(MCInst & Inst,unsigned N) const1357   void addImm1_8Operands(MCInst &Inst, unsigned N) const {
1358     assert(N == 1 && "Invalid number of operands!");
1359     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1360     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1361   }
1362 
addImm0_15Operands(MCInst & Inst,unsigned N) const1363   void addImm0_15Operands(MCInst &Inst, unsigned N) const {
1364     assert(N == 1 && "Invalid number of operands!");
1365     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1366     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1367   }
1368 
addImm1_16Operands(MCInst & Inst,unsigned N) const1369   void addImm1_16Operands(MCInst &Inst, unsigned N) const {
1370     assert(N == 1 && "Invalid number of operands!");
1371     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1372     assert(MCE && "Invalid constant immediate operand!");
1373     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1374   }
1375 
addImm0_31Operands(MCInst & Inst,unsigned N) const1376   void addImm0_31Operands(MCInst &Inst, unsigned N) const {
1377     assert(N == 1 && "Invalid number of operands!");
1378     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1379     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1380   }
1381 
addImm1_31Operands(MCInst & Inst,unsigned N) const1382   void addImm1_31Operands(MCInst &Inst, unsigned N) const {
1383     assert(N == 1 && "Invalid number of operands!");
1384     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1385     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1386   }
1387 
addImm1_32Operands(MCInst & Inst,unsigned N) const1388   void addImm1_32Operands(MCInst &Inst, unsigned N) const {
1389     assert(N == 1 && "Invalid number of operands!");
1390     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1391     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1392   }
1393 
addImm0_63Operands(MCInst & Inst,unsigned N) const1394   void addImm0_63Operands(MCInst &Inst, unsigned N) const {
1395     assert(N == 1 && "Invalid number of operands!");
1396     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1397     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1398   }
1399 
addImm1_63Operands(MCInst & Inst,unsigned N) const1400   void addImm1_63Operands(MCInst &Inst, unsigned N) const {
1401     assert(N == 1 && "Invalid number of operands!");
1402     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1403     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1404   }
1405 
addImm1_64Operands(MCInst & Inst,unsigned N) const1406   void addImm1_64Operands(MCInst &Inst, unsigned N) const {
1407     assert(N == 1 && "Invalid number of operands!");
1408     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1409     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1410   }
1411 
addImm0_127Operands(MCInst & Inst,unsigned N) const1412   void addImm0_127Operands(MCInst &Inst, unsigned N) const {
1413     assert(N == 1 && "Invalid number of operands!");
1414     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1415     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1416   }
1417 
addImm0_255Operands(MCInst & Inst,unsigned N) const1418   void addImm0_255Operands(MCInst &Inst, unsigned N) const {
1419     assert(N == 1 && "Invalid number of operands!");
1420     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1421     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1422   }
1423 
addImm0_65535Operands(MCInst & Inst,unsigned N) const1424   void addImm0_65535Operands(MCInst &Inst, unsigned N) const {
1425     assert(N == 1 && "Invalid number of operands!");
1426     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1427     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1428   }
1429 
addImm32_63Operands(MCInst & Inst,unsigned N) const1430   void addImm32_63Operands(MCInst &Inst, unsigned N) const {
1431     assert(N == 1 && "Invalid number of operands!");
1432     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1433     Inst.addOperand(MCOperand::createImm(MCE->getValue()));
1434   }
1435 
addLogicalImm32Operands(MCInst & Inst,unsigned N) const1436   void addLogicalImm32Operands(MCInst &Inst, unsigned N) const {
1437     assert(N == 1 && "Invalid number of operands!");
1438     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1439     uint64_t encoding =
1440         AArch64_AM::encodeLogicalImmediate(MCE->getValue() & 0xFFFFFFFF, 32);
1441     Inst.addOperand(MCOperand::createImm(encoding));
1442   }
1443 
addLogicalImm64Operands(MCInst & Inst,unsigned N) const1444   void addLogicalImm64Operands(MCInst &Inst, unsigned N) const {
1445     assert(N == 1 && "Invalid number of operands!");
1446     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1447     uint64_t encoding = AArch64_AM::encodeLogicalImmediate(MCE->getValue(), 64);
1448     Inst.addOperand(MCOperand::createImm(encoding));
1449   }
1450 
addLogicalImm32NotOperands(MCInst & Inst,unsigned N) const1451   void addLogicalImm32NotOperands(MCInst &Inst, unsigned N) const {
1452     assert(N == 1 && "Invalid number of operands!");
1453     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1454     int64_t Val = ~MCE->getValue() & 0xFFFFFFFF;
1455     uint64_t encoding = AArch64_AM::encodeLogicalImmediate(Val, 32);
1456     Inst.addOperand(MCOperand::createImm(encoding));
1457   }
1458 
addLogicalImm64NotOperands(MCInst & Inst,unsigned N) const1459   void addLogicalImm64NotOperands(MCInst &Inst, unsigned N) const {
1460     assert(N == 1 && "Invalid number of operands!");
1461     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1462     uint64_t encoding =
1463         AArch64_AM::encodeLogicalImmediate(~MCE->getValue(), 64);
1464     Inst.addOperand(MCOperand::createImm(encoding));
1465   }
1466 
addSIMDImmType10Operands(MCInst & Inst,unsigned N) const1467   void addSIMDImmType10Operands(MCInst &Inst, unsigned N) const {
1468     assert(N == 1 && "Invalid number of operands!");
1469     const MCConstantExpr *MCE = cast<MCConstantExpr>(getImm());
1470     uint64_t encoding = AArch64_AM::encodeAdvSIMDModImmType10(MCE->getValue());
1471     Inst.addOperand(MCOperand::createImm(encoding));
1472   }
1473 
addBranchTarget26Operands(MCInst & Inst,unsigned N) const1474   void addBranchTarget26Operands(MCInst &Inst, unsigned N) const {
1475     // Branch operands don't encode the low bits, so shift them off
1476     // here. If it's a label, however, just put it on directly as there's
1477     // not enough information now to do anything.
1478     assert(N == 1 && "Invalid number of operands!");
1479     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
1480     if (!MCE) {
1481       addExpr(Inst, getImm());
1482       return;
1483     }
1484     assert(MCE && "Invalid constant immediate operand!");
1485     Inst.addOperand(MCOperand::createImm(MCE->getValue() >> 2));
1486   }
1487 
addPCRelLabel19Operands(MCInst & Inst,unsigned N) const1488   void addPCRelLabel19Operands(MCInst &Inst, unsigned N) const {
1489     // Branch operands don't encode the low bits, so shift them off
1490     // here. If it's a label, however, just put it on directly as there's
1491     // not enough information now to do anything.
1492     assert(N == 1 && "Invalid number of operands!");
1493     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
1494     if (!MCE) {
1495       addExpr(Inst, getImm());
1496       return;
1497     }
1498     assert(MCE && "Invalid constant immediate operand!");
1499     Inst.addOperand(MCOperand::createImm(MCE->getValue() >> 2));
1500   }
1501 
addBranchTarget14Operands(MCInst & Inst,unsigned N) const1502   void addBranchTarget14Operands(MCInst &Inst, unsigned N) const {
1503     // Branch operands don't encode the low bits, so shift them off
1504     // here. If it's a label, however, just put it on directly as there's
1505     // not enough information now to do anything.
1506     assert(N == 1 && "Invalid number of operands!");
1507     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(getImm());
1508     if (!MCE) {
1509       addExpr(Inst, getImm());
1510       return;
1511     }
1512     assert(MCE && "Invalid constant immediate operand!");
1513     Inst.addOperand(MCOperand::createImm(MCE->getValue() >> 2));
1514   }
1515 
addFPImmOperands(MCInst & Inst,unsigned N) const1516   void addFPImmOperands(MCInst &Inst, unsigned N) const {
1517     assert(N == 1 && "Invalid number of operands!");
1518     Inst.addOperand(MCOperand::createImm(getFPImm()));
1519   }
1520 
addBarrierOperands(MCInst & Inst,unsigned N) const1521   void addBarrierOperands(MCInst &Inst, unsigned N) const {
1522     assert(N == 1 && "Invalid number of operands!");
1523     Inst.addOperand(MCOperand::createImm(getBarrier()));
1524   }
1525 
addMRSSystemRegisterOperands(MCInst & Inst,unsigned N) const1526   void addMRSSystemRegisterOperands(MCInst &Inst, unsigned N) const {
1527     assert(N == 1 && "Invalid number of operands!");
1528 
1529     Inst.addOperand(MCOperand::createImm(SysReg.MRSReg));
1530   }
1531 
addMSRSystemRegisterOperands(MCInst & Inst,unsigned N) const1532   void addMSRSystemRegisterOperands(MCInst &Inst, unsigned N) const {
1533     assert(N == 1 && "Invalid number of operands!");
1534 
1535     Inst.addOperand(MCOperand::createImm(SysReg.MSRReg));
1536   }
1537 
addSystemPStateFieldWithImm0_1Operands(MCInst & Inst,unsigned N) const1538   void addSystemPStateFieldWithImm0_1Operands(MCInst &Inst, unsigned N) const {
1539     assert(N == 1 && "Invalid number of operands!");
1540 
1541     Inst.addOperand(MCOperand::createImm(SysReg.PStateField));
1542   }
1543 
addSystemPStateFieldWithImm0_15Operands(MCInst & Inst,unsigned N) const1544   void addSystemPStateFieldWithImm0_15Operands(MCInst &Inst, unsigned N) const {
1545     assert(N == 1 && "Invalid number of operands!");
1546 
1547     Inst.addOperand(MCOperand::createImm(SysReg.PStateField));
1548   }
1549 
addSysCROperands(MCInst & Inst,unsigned N) const1550   void addSysCROperands(MCInst &Inst, unsigned N) const {
1551     assert(N == 1 && "Invalid number of operands!");
1552     Inst.addOperand(MCOperand::createImm(getSysCR()));
1553   }
1554 
addPrefetchOperands(MCInst & Inst,unsigned N) const1555   void addPrefetchOperands(MCInst &Inst, unsigned N) const {
1556     assert(N == 1 && "Invalid number of operands!");
1557     Inst.addOperand(MCOperand::createImm(getPrefetch()));
1558   }
1559 
addPSBHintOperands(MCInst & Inst,unsigned N) const1560   void addPSBHintOperands(MCInst &Inst, unsigned N) const {
1561     assert(N == 1 && "Invalid number of operands!");
1562     Inst.addOperand(MCOperand::createImm(getPSBHint()));
1563   }
1564 
addShifterOperands(MCInst & Inst,unsigned N) const1565   void addShifterOperands(MCInst &Inst, unsigned N) const {
1566     assert(N == 1 && "Invalid number of operands!");
1567     unsigned Imm =
1568         AArch64_AM::getShifterImm(getShiftExtendType(), getShiftExtendAmount());
1569     Inst.addOperand(MCOperand::createImm(Imm));
1570   }
1571 
addExtendOperands(MCInst & Inst,unsigned N) const1572   void addExtendOperands(MCInst &Inst, unsigned N) const {
1573     assert(N == 1 && "Invalid number of operands!");
1574     AArch64_AM::ShiftExtendType ET = getShiftExtendType();
1575     if (ET == AArch64_AM::LSL) ET = AArch64_AM::UXTW;
1576     unsigned Imm = AArch64_AM::getArithExtendImm(ET, getShiftExtendAmount());
1577     Inst.addOperand(MCOperand::createImm(Imm));
1578   }
1579 
addExtend64Operands(MCInst & Inst,unsigned N) const1580   void addExtend64Operands(MCInst &Inst, unsigned N) const {
1581     assert(N == 1 && "Invalid number of operands!");
1582     AArch64_AM::ShiftExtendType ET = getShiftExtendType();
1583     if (ET == AArch64_AM::LSL) ET = AArch64_AM::UXTX;
1584     unsigned Imm = AArch64_AM::getArithExtendImm(ET, getShiftExtendAmount());
1585     Inst.addOperand(MCOperand::createImm(Imm));
1586   }
1587 
addMemExtendOperands(MCInst & Inst,unsigned N) const1588   void addMemExtendOperands(MCInst &Inst, unsigned N) const {
1589     assert(N == 2 && "Invalid number of operands!");
1590     AArch64_AM::ShiftExtendType ET = getShiftExtendType();
1591     bool IsSigned = ET == AArch64_AM::SXTW || ET == AArch64_AM::SXTX;
1592     Inst.addOperand(MCOperand::createImm(IsSigned));
1593     Inst.addOperand(MCOperand::createImm(getShiftExtendAmount() != 0));
1594   }
1595 
1596   // For 8-bit load/store instructions with a register offset, both the
1597   // "DoShift" and "NoShift" variants have a shift of 0. Because of this,
1598   // they're disambiguated by whether the shift was explicit or implicit rather
1599   // than its size.
addMemExtend8Operands(MCInst & Inst,unsigned N) const1600   void addMemExtend8Operands(MCInst &Inst, unsigned N) const {
1601     assert(N == 2 && "Invalid number of operands!");
1602     AArch64_AM::ShiftExtendType ET = getShiftExtendType();
1603     bool IsSigned = ET == AArch64_AM::SXTW || ET == AArch64_AM::SXTX;
1604     Inst.addOperand(MCOperand::createImm(IsSigned));
1605     Inst.addOperand(MCOperand::createImm(hasShiftExtendAmount()));
1606   }
1607 
1608   template<int Shift>
addMOVZMovAliasOperands(MCInst & Inst,unsigned N) const1609   void addMOVZMovAliasOperands(MCInst &Inst, unsigned N) const {
1610     assert(N == 1 && "Invalid number of operands!");
1611 
1612     const MCConstantExpr *CE = cast<MCConstantExpr>(getImm());
1613     uint64_t Value = CE->getValue();
1614     Inst.addOperand(MCOperand::createImm((Value >> Shift) & 0xffff));
1615   }
1616 
1617   template<int Shift>
addMOVNMovAliasOperands(MCInst & Inst,unsigned N) const1618   void addMOVNMovAliasOperands(MCInst &Inst, unsigned N) const {
1619     assert(N == 1 && "Invalid number of operands!");
1620 
1621     const MCConstantExpr *CE = cast<MCConstantExpr>(getImm());
1622     uint64_t Value = CE->getValue();
1623     Inst.addOperand(MCOperand::createImm((~Value >> Shift) & 0xffff));
1624   }
1625 
1626   void print(raw_ostream &OS) const override;
1627 
1628   static std::unique_ptr<AArch64Operand>
CreateToken(StringRef Str,bool IsSuffix,SMLoc S,MCContext & Ctx)1629   CreateToken(StringRef Str, bool IsSuffix, SMLoc S, MCContext &Ctx) {
1630     auto Op = make_unique<AArch64Operand>(k_Token, Ctx);
1631     Op->Tok.Data = Str.data();
1632     Op->Tok.Length = Str.size();
1633     Op->Tok.IsSuffix = IsSuffix;
1634     Op->StartLoc = S;
1635     Op->EndLoc = S;
1636     return Op;
1637   }
1638 
1639   static std::unique_ptr<AArch64Operand>
CreateReg(unsigned RegNum,bool isVector,SMLoc S,SMLoc E,MCContext & Ctx)1640   CreateReg(unsigned RegNum, bool isVector, SMLoc S, SMLoc E, MCContext &Ctx) {
1641     auto Op = make_unique<AArch64Operand>(k_Register, Ctx);
1642     Op->Reg.RegNum = RegNum;
1643     Op->Reg.isVector = isVector;
1644     Op->StartLoc = S;
1645     Op->EndLoc = E;
1646     return Op;
1647   }
1648 
1649   static std::unique_ptr<AArch64Operand>
CreateVectorList(unsigned RegNum,unsigned Count,unsigned NumElements,char ElementKind,SMLoc S,SMLoc E,MCContext & Ctx)1650   CreateVectorList(unsigned RegNum, unsigned Count, unsigned NumElements,
1651                    char ElementKind, SMLoc S, SMLoc E, MCContext &Ctx) {
1652     auto Op = make_unique<AArch64Operand>(k_VectorList, Ctx);
1653     Op->VectorList.RegNum = RegNum;
1654     Op->VectorList.Count = Count;
1655     Op->VectorList.NumElements = NumElements;
1656     Op->VectorList.ElementKind = ElementKind;
1657     Op->StartLoc = S;
1658     Op->EndLoc = E;
1659     return Op;
1660   }
1661 
1662   static std::unique_ptr<AArch64Operand>
CreateVectorIndex(unsigned Idx,SMLoc S,SMLoc E,MCContext & Ctx)1663   CreateVectorIndex(unsigned Idx, SMLoc S, SMLoc E, MCContext &Ctx) {
1664     auto Op = make_unique<AArch64Operand>(k_VectorIndex, Ctx);
1665     Op->VectorIndex.Val = Idx;
1666     Op->StartLoc = S;
1667     Op->EndLoc = E;
1668     return Op;
1669   }
1670 
CreateImm(const MCExpr * Val,SMLoc S,SMLoc E,MCContext & Ctx)1671   static std::unique_ptr<AArch64Operand> CreateImm(const MCExpr *Val, SMLoc S,
1672                                                    SMLoc E, MCContext &Ctx) {
1673     auto Op = make_unique<AArch64Operand>(k_Immediate, Ctx);
1674     Op->Imm.Val = Val;
1675     Op->StartLoc = S;
1676     Op->EndLoc = E;
1677     return Op;
1678   }
1679 
CreateShiftedImm(const MCExpr * Val,unsigned ShiftAmount,SMLoc S,SMLoc E,MCContext & Ctx)1680   static std::unique_ptr<AArch64Operand> CreateShiftedImm(const MCExpr *Val,
1681                                                           unsigned ShiftAmount,
1682                                                           SMLoc S, SMLoc E,
1683                                                           MCContext &Ctx) {
1684     auto Op = make_unique<AArch64Operand>(k_ShiftedImm, Ctx);
1685     Op->ShiftedImm .Val = Val;
1686     Op->ShiftedImm.ShiftAmount = ShiftAmount;
1687     Op->StartLoc = S;
1688     Op->EndLoc = E;
1689     return Op;
1690   }
1691 
1692   static std::unique_ptr<AArch64Operand>
CreateCondCode(AArch64CC::CondCode Code,SMLoc S,SMLoc E,MCContext & Ctx)1693   CreateCondCode(AArch64CC::CondCode Code, SMLoc S, SMLoc E, MCContext &Ctx) {
1694     auto Op = make_unique<AArch64Operand>(k_CondCode, Ctx);
1695     Op->CondCode.Code = Code;
1696     Op->StartLoc = S;
1697     Op->EndLoc = E;
1698     return Op;
1699   }
1700 
CreateFPImm(unsigned Val,SMLoc S,MCContext & Ctx)1701   static std::unique_ptr<AArch64Operand> CreateFPImm(unsigned Val, SMLoc S,
1702                                                      MCContext &Ctx) {
1703     auto Op = make_unique<AArch64Operand>(k_FPImm, Ctx);
1704     Op->FPImm.Val = Val;
1705     Op->StartLoc = S;
1706     Op->EndLoc = S;
1707     return Op;
1708   }
1709 
CreateBarrier(unsigned Val,StringRef Str,SMLoc S,MCContext & Ctx)1710   static std::unique_ptr<AArch64Operand> CreateBarrier(unsigned Val,
1711                                                        StringRef Str,
1712                                                        SMLoc S,
1713                                                        MCContext &Ctx) {
1714     auto Op = make_unique<AArch64Operand>(k_Barrier, Ctx);
1715     Op->Barrier.Val = Val;
1716     Op->Barrier.Data = Str.data();
1717     Op->Barrier.Length = Str.size();
1718     Op->StartLoc = S;
1719     Op->EndLoc = S;
1720     return Op;
1721   }
1722 
CreateSysReg(StringRef Str,SMLoc S,uint32_t MRSReg,uint32_t MSRReg,uint32_t PStateField,MCContext & Ctx)1723   static std::unique_ptr<AArch64Operand> CreateSysReg(StringRef Str, SMLoc S,
1724                                                       uint32_t MRSReg,
1725                                                       uint32_t MSRReg,
1726                                                       uint32_t PStateField,
1727                                                       MCContext &Ctx) {
1728     auto Op = make_unique<AArch64Operand>(k_SysReg, Ctx);
1729     Op->SysReg.Data = Str.data();
1730     Op->SysReg.Length = Str.size();
1731     Op->SysReg.MRSReg = MRSReg;
1732     Op->SysReg.MSRReg = MSRReg;
1733     Op->SysReg.PStateField = PStateField;
1734     Op->StartLoc = S;
1735     Op->EndLoc = S;
1736     return Op;
1737   }
1738 
CreateSysCR(unsigned Val,SMLoc S,SMLoc E,MCContext & Ctx)1739   static std::unique_ptr<AArch64Operand> CreateSysCR(unsigned Val, SMLoc S,
1740                                                      SMLoc E, MCContext &Ctx) {
1741     auto Op = make_unique<AArch64Operand>(k_SysCR, Ctx);
1742     Op->SysCRImm.Val = Val;
1743     Op->StartLoc = S;
1744     Op->EndLoc = E;
1745     return Op;
1746   }
1747 
CreatePrefetch(unsigned Val,StringRef Str,SMLoc S,MCContext & Ctx)1748   static std::unique_ptr<AArch64Operand> CreatePrefetch(unsigned Val,
1749                                                         StringRef Str,
1750                                                         SMLoc S,
1751                                                         MCContext &Ctx) {
1752     auto Op = make_unique<AArch64Operand>(k_Prefetch, Ctx);
1753     Op->Prefetch.Val = Val;
1754     Op->Barrier.Data = Str.data();
1755     Op->Barrier.Length = Str.size();
1756     Op->StartLoc = S;
1757     Op->EndLoc = S;
1758     return Op;
1759   }
1760 
CreatePSBHint(unsigned Val,StringRef Str,SMLoc S,MCContext & Ctx)1761   static std::unique_ptr<AArch64Operand> CreatePSBHint(unsigned Val,
1762                                                        StringRef Str,
1763                                                        SMLoc S,
1764                                                        MCContext &Ctx) {
1765     auto Op = make_unique<AArch64Operand>(k_PSBHint, Ctx);
1766     Op->PSBHint.Val = Val;
1767     Op->PSBHint.Data = Str.data();
1768     Op->PSBHint.Length = Str.size();
1769     Op->StartLoc = S;
1770     Op->EndLoc = S;
1771     return Op;
1772   }
1773 
1774   static std::unique_ptr<AArch64Operand>
CreateShiftExtend(AArch64_AM::ShiftExtendType ShOp,unsigned Val,bool HasExplicitAmount,SMLoc S,SMLoc E,MCContext & Ctx)1775   CreateShiftExtend(AArch64_AM::ShiftExtendType ShOp, unsigned Val,
1776                     bool HasExplicitAmount, SMLoc S, SMLoc E, MCContext &Ctx) {
1777     auto Op = make_unique<AArch64Operand>(k_ShiftExtend, Ctx);
1778     Op->ShiftExtend.Type = ShOp;
1779     Op->ShiftExtend.Amount = Val;
1780     Op->ShiftExtend.HasExplicitAmount = HasExplicitAmount;
1781     Op->StartLoc = S;
1782     Op->EndLoc = E;
1783     return Op;
1784   }
1785 };
1786 
1787 } // end anonymous namespace.
1788 
print(raw_ostream & OS) const1789 void AArch64Operand::print(raw_ostream &OS) const {
1790   switch (Kind) {
1791   case k_FPImm:
1792     OS << "<fpimm " << getFPImm() << "("
1793        << AArch64_AM::getFPImmFloat(getFPImm()) << ") >";
1794     break;
1795   case k_Barrier: {
1796     StringRef Name = getBarrierName();
1797     if (!Name.empty())
1798       OS << "<barrier " << Name << ">";
1799     else
1800       OS << "<barrier invalid #" << getBarrier() << ">";
1801     break;
1802   }
1803   case k_Immediate:
1804     OS << *getImm();
1805     break;
1806   case k_ShiftedImm: {
1807     unsigned Shift = getShiftedImmShift();
1808     OS << "<shiftedimm ";
1809     OS << *getShiftedImmVal();
1810     OS << ", lsl #" << AArch64_AM::getShiftValue(Shift) << ">";
1811     break;
1812   }
1813   case k_CondCode:
1814     OS << "<condcode " << getCondCode() << ">";
1815     break;
1816   case k_Register:
1817     OS << "<register " << getReg() << ">";
1818     break;
1819   case k_VectorList: {
1820     OS << "<vectorlist ";
1821     unsigned Reg = getVectorListStart();
1822     for (unsigned i = 0, e = getVectorListCount(); i != e; ++i)
1823       OS << Reg + i << " ";
1824     OS << ">";
1825     break;
1826   }
1827   case k_VectorIndex:
1828     OS << "<vectorindex " << getVectorIndex() << ">";
1829     break;
1830   case k_SysReg:
1831     OS << "<sysreg: " << getSysReg() << '>';
1832     break;
1833   case k_Token:
1834     OS << "'" << getToken() << "'";
1835     break;
1836   case k_SysCR:
1837     OS << "c" << getSysCR();
1838     break;
1839   case k_Prefetch: {
1840     StringRef Name = getPrefetchName();
1841     if (!Name.empty())
1842       OS << "<prfop " << Name << ">";
1843     else
1844       OS << "<prfop invalid #" << getPrefetch() << ">";
1845     break;
1846   }
1847   case k_PSBHint: {
1848     OS << getPSBHintName();
1849     break;
1850   }
1851   case k_ShiftExtend: {
1852     OS << "<" << AArch64_AM::getShiftExtendName(getShiftExtendType()) << " #"
1853        << getShiftExtendAmount();
1854     if (!hasShiftExtendAmount())
1855       OS << "<imp>";
1856     OS << '>';
1857     break;
1858   }
1859   }
1860 }
1861 
1862 /// @name Auto-generated Match Functions
1863 /// {
1864 
1865 static unsigned MatchRegisterName(StringRef Name);
1866 
1867 /// }
1868 
matchVectorRegName(StringRef Name)1869 static unsigned matchVectorRegName(StringRef Name) {
1870   return StringSwitch<unsigned>(Name.lower())
1871       .Case("v0", AArch64::Q0)
1872       .Case("v1", AArch64::Q1)
1873       .Case("v2", AArch64::Q2)
1874       .Case("v3", AArch64::Q3)
1875       .Case("v4", AArch64::Q4)
1876       .Case("v5", AArch64::Q5)
1877       .Case("v6", AArch64::Q6)
1878       .Case("v7", AArch64::Q7)
1879       .Case("v8", AArch64::Q8)
1880       .Case("v9", AArch64::Q9)
1881       .Case("v10", AArch64::Q10)
1882       .Case("v11", AArch64::Q11)
1883       .Case("v12", AArch64::Q12)
1884       .Case("v13", AArch64::Q13)
1885       .Case("v14", AArch64::Q14)
1886       .Case("v15", AArch64::Q15)
1887       .Case("v16", AArch64::Q16)
1888       .Case("v17", AArch64::Q17)
1889       .Case("v18", AArch64::Q18)
1890       .Case("v19", AArch64::Q19)
1891       .Case("v20", AArch64::Q20)
1892       .Case("v21", AArch64::Q21)
1893       .Case("v22", AArch64::Q22)
1894       .Case("v23", AArch64::Q23)
1895       .Case("v24", AArch64::Q24)
1896       .Case("v25", AArch64::Q25)
1897       .Case("v26", AArch64::Q26)
1898       .Case("v27", AArch64::Q27)
1899       .Case("v28", AArch64::Q28)
1900       .Case("v29", AArch64::Q29)
1901       .Case("v30", AArch64::Q30)
1902       .Case("v31", AArch64::Q31)
1903       .Default(0);
1904 }
1905 
isValidVectorKind(StringRef Name)1906 static bool isValidVectorKind(StringRef Name) {
1907   return StringSwitch<bool>(Name.lower())
1908       .Case(".8b", true)
1909       .Case(".16b", true)
1910       .Case(".4h", true)
1911       .Case(".8h", true)
1912       .Case(".2s", true)
1913       .Case(".4s", true)
1914       .Case(".1d", true)
1915       .Case(".2d", true)
1916       .Case(".1q", true)
1917       // Accept the width neutral ones, too, for verbose syntax. If those
1918       // aren't used in the right places, the token operand won't match so
1919       // all will work out.
1920       .Case(".b", true)
1921       .Case(".h", true)
1922       .Case(".s", true)
1923       .Case(".d", true)
1924       // Needed for fp16 scalar pairwise reductions
1925       .Case(".2h", true)
1926       .Default(false);
1927 }
1928 
parseValidVectorKind(StringRef Name,unsigned & NumElements,char & ElementKind)1929 static void parseValidVectorKind(StringRef Name, unsigned &NumElements,
1930                                  char &ElementKind) {
1931   assert(isValidVectorKind(Name));
1932 
1933   ElementKind = Name.lower()[Name.size() - 1];
1934   NumElements = 0;
1935 
1936   if (Name.size() == 2)
1937     return;
1938 
1939   // Parse the lane count
1940   Name = Name.drop_front();
1941   while (isdigit(Name.front())) {
1942     NumElements = 10 * NumElements + (Name.front() - '0');
1943     Name = Name.drop_front();
1944   }
1945 }
1946 
ParseRegister(unsigned & RegNo,SMLoc & StartLoc,SMLoc & EndLoc)1947 bool AArch64AsmParser::ParseRegister(unsigned &RegNo, SMLoc &StartLoc,
1948                                      SMLoc &EndLoc) {
1949   StartLoc = getLoc();
1950   RegNo = tryParseRegister();
1951   EndLoc = SMLoc::getFromPointer(getLoc().getPointer() - 1);
1952   return (RegNo == (unsigned)-1);
1953 }
1954 
1955 // Matches a register name or register alias previously defined by '.req'
matchRegisterNameAlias(StringRef Name,bool isVector)1956 unsigned AArch64AsmParser::matchRegisterNameAlias(StringRef Name,
1957                                                   bool isVector) {
1958   unsigned RegNum = isVector ? matchVectorRegName(Name)
1959                              : MatchRegisterName(Name);
1960 
1961   if (RegNum == 0) {
1962     // Check for aliases registered via .req. Canonicalize to lower case.
1963     // That's more consistent since register names are case insensitive, and
1964     // it's how the original entry was passed in from MC/MCParser/AsmParser.
1965     auto Entry = RegisterReqs.find(Name.lower());
1966     if (Entry == RegisterReqs.end())
1967       return 0;
1968     // set RegNum if the match is the right kind of register
1969     if (isVector == Entry->getValue().first)
1970       RegNum = Entry->getValue().second;
1971   }
1972   return RegNum;
1973 }
1974 
1975 /// tryParseRegister - Try to parse a register name. The token must be an
1976 /// Identifier when called, and if it is a register name the token is eaten and
1977 /// the register is added to the operand list.
tryParseRegister()1978 int AArch64AsmParser::tryParseRegister() {
1979   MCAsmParser &Parser = getParser();
1980   const AsmToken &Tok = Parser.getTok();
1981   assert(Tok.is(AsmToken::Identifier) && "Token is not an Identifier");
1982 
1983   std::string lowerCase = Tok.getString().lower();
1984   unsigned RegNum = matchRegisterNameAlias(lowerCase, false);
1985   // Also handle a few aliases of registers.
1986   if (RegNum == 0)
1987     RegNum = StringSwitch<unsigned>(lowerCase)
1988                  .Case("fp",  AArch64::FP)
1989                  .Case("lr",  AArch64::LR)
1990                  .Case("x31", AArch64::XZR)
1991                  .Case("w31", AArch64::WZR)
1992                  .Default(0);
1993 
1994   if (RegNum == 0)
1995     return -1;
1996 
1997   Parser.Lex(); // Eat identifier token.
1998   return RegNum;
1999 }
2000 
2001 /// tryMatchVectorRegister - Try to parse a vector register name with optional
2002 /// kind specifier. If it is a register specifier, eat the token and return it.
tryMatchVectorRegister(StringRef & Kind,bool expected)2003 int AArch64AsmParser::tryMatchVectorRegister(StringRef &Kind, bool expected) {
2004   MCAsmParser &Parser = getParser();
2005   if (Parser.getTok().isNot(AsmToken::Identifier)) {
2006     TokError("vector register expected");
2007     return -1;
2008   }
2009 
2010   StringRef Name = Parser.getTok().getString();
2011   // If there is a kind specifier, it's separated from the register name by
2012   // a '.'.
2013   size_t Start = 0, Next = Name.find('.');
2014   StringRef Head = Name.slice(Start, Next);
2015   unsigned RegNum = matchRegisterNameAlias(Head, true);
2016 
2017   if (RegNum) {
2018     if (Next != StringRef::npos) {
2019       Kind = Name.slice(Next, StringRef::npos);
2020       if (!isValidVectorKind(Kind)) {
2021         TokError("invalid vector kind qualifier");
2022         return -1;
2023       }
2024     }
2025     Parser.Lex(); // Eat the register token.
2026     return RegNum;
2027   }
2028 
2029   if (expected)
2030     TokError("vector register expected");
2031   return -1;
2032 }
2033 
2034 /// tryParseSysCROperand - Try to parse a system instruction CR operand name.
2035 AArch64AsmParser::OperandMatchResultTy
tryParseSysCROperand(OperandVector & Operands)2036 AArch64AsmParser::tryParseSysCROperand(OperandVector &Operands) {
2037   MCAsmParser &Parser = getParser();
2038   SMLoc S = getLoc();
2039 
2040   if (Parser.getTok().isNot(AsmToken::Identifier)) {
2041     Error(S, "Expected cN operand where 0 <= N <= 15");
2042     return MatchOperand_ParseFail;
2043   }
2044 
2045   StringRef Tok = Parser.getTok().getIdentifier();
2046   if (Tok[0] != 'c' && Tok[0] != 'C') {
2047     Error(S, "Expected cN operand where 0 <= N <= 15");
2048     return MatchOperand_ParseFail;
2049   }
2050 
2051   uint32_t CRNum;
2052   bool BadNum = Tok.drop_front().getAsInteger(10, CRNum);
2053   if (BadNum || CRNum > 15) {
2054     Error(S, "Expected cN operand where 0 <= N <= 15");
2055     return MatchOperand_ParseFail;
2056   }
2057 
2058   Parser.Lex(); // Eat identifier token.
2059   Operands.push_back(
2060       AArch64Operand::CreateSysCR(CRNum, S, getLoc(), getContext()));
2061   return MatchOperand_Success;
2062 }
2063 
2064 /// tryParsePrefetch - Try to parse a prefetch operand.
2065 AArch64AsmParser::OperandMatchResultTy
tryParsePrefetch(OperandVector & Operands)2066 AArch64AsmParser::tryParsePrefetch(OperandVector &Operands) {
2067   MCAsmParser &Parser = getParser();
2068   SMLoc S = getLoc();
2069   const AsmToken &Tok = Parser.getTok();
2070   // Either an identifier for named values or a 5-bit immediate.
2071   bool Hash = Tok.is(AsmToken::Hash);
2072   if (Hash || Tok.is(AsmToken::Integer)) {
2073     if (Hash)
2074       Parser.Lex(); // Eat hash token.
2075     const MCExpr *ImmVal;
2076     if (getParser().parseExpression(ImmVal))
2077       return MatchOperand_ParseFail;
2078 
2079     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(ImmVal);
2080     if (!MCE) {
2081       TokError("immediate value expected for prefetch operand");
2082       return MatchOperand_ParseFail;
2083     }
2084     unsigned prfop = MCE->getValue();
2085     if (prfop > 31) {
2086       TokError("prefetch operand out of range, [0,31] expected");
2087       return MatchOperand_ParseFail;
2088     }
2089 
2090     bool Valid;
2091     auto Mapper = AArch64PRFM::PRFMMapper();
2092     StringRef Name =
2093         Mapper.toString(MCE->getValue(), getSTI().getFeatureBits(), Valid);
2094     Operands.push_back(AArch64Operand::CreatePrefetch(prfop, Name,
2095                                                       S, getContext()));
2096     return MatchOperand_Success;
2097   }
2098 
2099   if (Tok.isNot(AsmToken::Identifier)) {
2100     TokError("pre-fetch hint expected");
2101     return MatchOperand_ParseFail;
2102   }
2103 
2104   bool Valid;
2105   auto Mapper = AArch64PRFM::PRFMMapper();
2106   unsigned prfop =
2107       Mapper.fromString(Tok.getString(), getSTI().getFeatureBits(), Valid);
2108   if (!Valid) {
2109     TokError("pre-fetch hint expected");
2110     return MatchOperand_ParseFail;
2111   }
2112 
2113   Parser.Lex(); // Eat identifier token.
2114   Operands.push_back(AArch64Operand::CreatePrefetch(prfop, Tok.getString(),
2115                                                     S, getContext()));
2116   return MatchOperand_Success;
2117 }
2118 
2119 /// tryParsePSBHint - Try to parse a PSB operand, mapped to Hint command
2120 AArch64AsmParser::OperandMatchResultTy
tryParsePSBHint(OperandVector & Operands)2121 AArch64AsmParser::tryParsePSBHint(OperandVector &Operands) {
2122   MCAsmParser &Parser = getParser();
2123   SMLoc S = getLoc();
2124   const AsmToken &Tok = Parser.getTok();
2125   if (Tok.isNot(AsmToken::Identifier)) {
2126     TokError("invalid operand for instruction");
2127     return MatchOperand_ParseFail;
2128   }
2129 
2130   bool Valid;
2131   auto Mapper = AArch64PSBHint::PSBHintMapper();
2132   unsigned psbhint =
2133       Mapper.fromString(Tok.getString(), getSTI().getFeatureBits(), Valid);
2134   if (!Valid) {
2135     TokError("invalid operand for instruction");
2136     return MatchOperand_ParseFail;
2137   }
2138 
2139   Parser.Lex(); // Eat identifier token.
2140   Operands.push_back(AArch64Operand::CreatePSBHint(psbhint, Tok.getString(),
2141                                                    S, getContext()));
2142   return MatchOperand_Success;
2143 }
2144 
2145 /// tryParseAdrpLabel - Parse and validate a source label for the ADRP
2146 /// instruction.
2147 AArch64AsmParser::OperandMatchResultTy
tryParseAdrpLabel(OperandVector & Operands)2148 AArch64AsmParser::tryParseAdrpLabel(OperandVector &Operands) {
2149   MCAsmParser &Parser = getParser();
2150   SMLoc S = getLoc();
2151   const MCExpr *Expr;
2152 
2153   if (Parser.getTok().is(AsmToken::Hash)) {
2154     Parser.Lex(); // Eat hash token.
2155   }
2156 
2157   if (parseSymbolicImmVal(Expr))
2158     return MatchOperand_ParseFail;
2159 
2160   AArch64MCExpr::VariantKind ELFRefKind;
2161   MCSymbolRefExpr::VariantKind DarwinRefKind;
2162   int64_t Addend;
2163   if (classifySymbolRef(Expr, ELFRefKind, DarwinRefKind, Addend)) {
2164     if (DarwinRefKind == MCSymbolRefExpr::VK_None &&
2165         ELFRefKind == AArch64MCExpr::VK_INVALID) {
2166       // No modifier was specified at all; this is the syntax for an ELF basic
2167       // ADRP relocation (unfortunately).
2168       Expr =
2169           AArch64MCExpr::create(Expr, AArch64MCExpr::VK_ABS_PAGE, getContext());
2170     } else if ((DarwinRefKind == MCSymbolRefExpr::VK_GOTPAGE ||
2171                 DarwinRefKind == MCSymbolRefExpr::VK_TLVPPAGE) &&
2172                Addend != 0) {
2173       Error(S, "gotpage label reference not allowed an addend");
2174       return MatchOperand_ParseFail;
2175     } else if (DarwinRefKind != MCSymbolRefExpr::VK_PAGE &&
2176                DarwinRefKind != MCSymbolRefExpr::VK_GOTPAGE &&
2177                DarwinRefKind != MCSymbolRefExpr::VK_TLVPPAGE &&
2178                ELFRefKind != AArch64MCExpr::VK_GOT_PAGE &&
2179                ELFRefKind != AArch64MCExpr::VK_GOTTPREL_PAGE &&
2180                ELFRefKind != AArch64MCExpr::VK_TLSDESC_PAGE) {
2181       // The operand must be an @page or @gotpage qualified symbolref.
2182       Error(S, "page or gotpage label reference expected");
2183       return MatchOperand_ParseFail;
2184     }
2185   }
2186 
2187   // We have either a label reference possibly with addend or an immediate. The
2188   // addend is a raw value here. The linker will adjust it to only reference the
2189   // page.
2190   SMLoc E = SMLoc::getFromPointer(getLoc().getPointer() - 1);
2191   Operands.push_back(AArch64Operand::CreateImm(Expr, S, E, getContext()));
2192 
2193   return MatchOperand_Success;
2194 }
2195 
2196 /// tryParseAdrLabel - Parse and validate a source label for the ADR
2197 /// instruction.
2198 AArch64AsmParser::OperandMatchResultTy
tryParseAdrLabel(OperandVector & Operands)2199 AArch64AsmParser::tryParseAdrLabel(OperandVector &Operands) {
2200   MCAsmParser &Parser = getParser();
2201   SMLoc S = getLoc();
2202   const MCExpr *Expr;
2203 
2204   if (Parser.getTok().is(AsmToken::Hash)) {
2205     Parser.Lex(); // Eat hash token.
2206   }
2207 
2208   if (getParser().parseExpression(Expr))
2209     return MatchOperand_ParseFail;
2210 
2211   SMLoc E = SMLoc::getFromPointer(getLoc().getPointer() - 1);
2212   Operands.push_back(AArch64Operand::CreateImm(Expr, S, E, getContext()));
2213 
2214   return MatchOperand_Success;
2215 }
2216 
2217 /// tryParseFPImm - A floating point immediate expression operand.
2218 AArch64AsmParser::OperandMatchResultTy
tryParseFPImm(OperandVector & Operands)2219 AArch64AsmParser::tryParseFPImm(OperandVector &Operands) {
2220   MCAsmParser &Parser = getParser();
2221   SMLoc S = getLoc();
2222 
2223   bool Hash = false;
2224   if (Parser.getTok().is(AsmToken::Hash)) {
2225     Parser.Lex(); // Eat '#'
2226     Hash = true;
2227   }
2228 
2229   // Handle negation, as that still comes through as a separate token.
2230   bool isNegative = false;
2231   if (Parser.getTok().is(AsmToken::Minus)) {
2232     isNegative = true;
2233     Parser.Lex();
2234   }
2235   const AsmToken &Tok = Parser.getTok();
2236   if (Tok.is(AsmToken::Real)) {
2237     APFloat RealVal(APFloat::IEEEdouble, Tok.getString());
2238     if (isNegative)
2239       RealVal.changeSign();
2240 
2241     uint64_t IntVal = RealVal.bitcastToAPInt().getZExtValue();
2242     int Val = AArch64_AM::getFP64Imm(APInt(64, IntVal));
2243     Parser.Lex(); // Eat the token.
2244     // Check for out of range values. As an exception, we let Zero through,
2245     // as we handle that special case in post-processing before matching in
2246     // order to use the zero register for it.
2247     if (Val == -1 && !RealVal.isPosZero()) {
2248       TokError("expected compatible register or floating-point constant");
2249       return MatchOperand_ParseFail;
2250     }
2251     Operands.push_back(AArch64Operand::CreateFPImm(Val, S, getContext()));
2252     return MatchOperand_Success;
2253   }
2254   if (Tok.is(AsmToken::Integer)) {
2255     int64_t Val;
2256     if (!isNegative && Tok.getString().startswith("0x")) {
2257       Val = Tok.getIntVal();
2258       if (Val > 255 || Val < 0) {
2259         TokError("encoded floating point value out of range");
2260         return MatchOperand_ParseFail;
2261       }
2262     } else {
2263       APFloat RealVal(APFloat::IEEEdouble, Tok.getString());
2264       uint64_t IntVal = RealVal.bitcastToAPInt().getZExtValue();
2265       // If we had a '-' in front, toggle the sign bit.
2266       IntVal ^= (uint64_t)isNegative << 63;
2267       Val = AArch64_AM::getFP64Imm(APInt(64, IntVal));
2268     }
2269     Parser.Lex(); // Eat the token.
2270     Operands.push_back(AArch64Operand::CreateFPImm(Val, S, getContext()));
2271     return MatchOperand_Success;
2272   }
2273 
2274   if (!Hash)
2275     return MatchOperand_NoMatch;
2276 
2277   TokError("invalid floating point immediate");
2278   return MatchOperand_ParseFail;
2279 }
2280 
2281 /// tryParseAddSubImm - Parse ADD/SUB shifted immediate operand
2282 AArch64AsmParser::OperandMatchResultTy
tryParseAddSubImm(OperandVector & Operands)2283 AArch64AsmParser::tryParseAddSubImm(OperandVector &Operands) {
2284   MCAsmParser &Parser = getParser();
2285   SMLoc S = getLoc();
2286 
2287   if (Parser.getTok().is(AsmToken::Hash))
2288     Parser.Lex(); // Eat '#'
2289   else if (Parser.getTok().isNot(AsmToken::Integer))
2290     // Operand should start from # or should be integer, emit error otherwise.
2291     return MatchOperand_NoMatch;
2292 
2293   const MCExpr *Imm;
2294   if (parseSymbolicImmVal(Imm))
2295     return MatchOperand_ParseFail;
2296   else if (Parser.getTok().isNot(AsmToken::Comma)) {
2297     uint64_t ShiftAmount = 0;
2298     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(Imm);
2299     if (MCE) {
2300       int64_t Val = MCE->getValue();
2301       if (Val > 0xfff && (Val & 0xfff) == 0) {
2302         Imm = MCConstantExpr::create(Val >> 12, getContext());
2303         ShiftAmount = 12;
2304       }
2305     }
2306     SMLoc E = Parser.getTok().getLoc();
2307     Operands.push_back(AArch64Operand::CreateShiftedImm(Imm, ShiftAmount, S, E,
2308                                                         getContext()));
2309     return MatchOperand_Success;
2310   }
2311 
2312   // Eat ','
2313   Parser.Lex();
2314 
2315   // The optional operand must be "lsl #N" where N is non-negative.
2316   if (!Parser.getTok().is(AsmToken::Identifier) ||
2317       !Parser.getTok().getIdentifier().equals_lower("lsl")) {
2318     Error(Parser.getTok().getLoc(), "only 'lsl #+N' valid after immediate");
2319     return MatchOperand_ParseFail;
2320   }
2321 
2322   // Eat 'lsl'
2323   Parser.Lex();
2324 
2325   if (Parser.getTok().is(AsmToken::Hash)) {
2326     Parser.Lex();
2327   }
2328 
2329   if (Parser.getTok().isNot(AsmToken::Integer)) {
2330     Error(Parser.getTok().getLoc(), "only 'lsl #+N' valid after immediate");
2331     return MatchOperand_ParseFail;
2332   }
2333 
2334   int64_t ShiftAmount = Parser.getTok().getIntVal();
2335 
2336   if (ShiftAmount < 0) {
2337     Error(Parser.getTok().getLoc(), "positive shift amount required");
2338     return MatchOperand_ParseFail;
2339   }
2340   Parser.Lex(); // Eat the number
2341 
2342   SMLoc E = Parser.getTok().getLoc();
2343   Operands.push_back(AArch64Operand::CreateShiftedImm(Imm, ShiftAmount,
2344                                                       S, E, getContext()));
2345   return MatchOperand_Success;
2346 }
2347 
2348 /// parseCondCodeString - Parse a Condition Code string.
parseCondCodeString(StringRef Cond)2349 AArch64CC::CondCode AArch64AsmParser::parseCondCodeString(StringRef Cond) {
2350   AArch64CC::CondCode CC = StringSwitch<AArch64CC::CondCode>(Cond.lower())
2351                     .Case("eq", AArch64CC::EQ)
2352                     .Case("ne", AArch64CC::NE)
2353                     .Case("cs", AArch64CC::HS)
2354                     .Case("hs", AArch64CC::HS)
2355                     .Case("cc", AArch64CC::LO)
2356                     .Case("lo", AArch64CC::LO)
2357                     .Case("mi", AArch64CC::MI)
2358                     .Case("pl", AArch64CC::PL)
2359                     .Case("vs", AArch64CC::VS)
2360                     .Case("vc", AArch64CC::VC)
2361                     .Case("hi", AArch64CC::HI)
2362                     .Case("ls", AArch64CC::LS)
2363                     .Case("ge", AArch64CC::GE)
2364                     .Case("lt", AArch64CC::LT)
2365                     .Case("gt", AArch64CC::GT)
2366                     .Case("le", AArch64CC::LE)
2367                     .Case("al", AArch64CC::AL)
2368                     .Case("nv", AArch64CC::NV)
2369                     .Default(AArch64CC::Invalid);
2370   return CC;
2371 }
2372 
2373 /// parseCondCode - Parse a Condition Code operand.
parseCondCode(OperandVector & Operands,bool invertCondCode)2374 bool AArch64AsmParser::parseCondCode(OperandVector &Operands,
2375                                      bool invertCondCode) {
2376   MCAsmParser &Parser = getParser();
2377   SMLoc S = getLoc();
2378   const AsmToken &Tok = Parser.getTok();
2379   assert(Tok.is(AsmToken::Identifier) && "Token is not an Identifier");
2380 
2381   StringRef Cond = Tok.getString();
2382   AArch64CC::CondCode CC = parseCondCodeString(Cond);
2383   if (CC == AArch64CC::Invalid)
2384     return TokError("invalid condition code");
2385   Parser.Lex(); // Eat identifier token.
2386 
2387   if (invertCondCode) {
2388     if (CC == AArch64CC::AL || CC == AArch64CC::NV)
2389       return TokError("condition codes AL and NV are invalid for this instruction");
2390     CC = AArch64CC::getInvertedCondCode(AArch64CC::CondCode(CC));
2391   }
2392 
2393   Operands.push_back(
2394       AArch64Operand::CreateCondCode(CC, S, getLoc(), getContext()));
2395   return false;
2396 }
2397 
2398 /// tryParseOptionalShift - Some operands take an optional shift argument. Parse
2399 /// them if present.
2400 AArch64AsmParser::OperandMatchResultTy
tryParseOptionalShiftExtend(OperandVector & Operands)2401 AArch64AsmParser::tryParseOptionalShiftExtend(OperandVector &Operands) {
2402   MCAsmParser &Parser = getParser();
2403   const AsmToken &Tok = Parser.getTok();
2404   std::string LowerID = Tok.getString().lower();
2405   AArch64_AM::ShiftExtendType ShOp =
2406       StringSwitch<AArch64_AM::ShiftExtendType>(LowerID)
2407           .Case("lsl", AArch64_AM::LSL)
2408           .Case("lsr", AArch64_AM::LSR)
2409           .Case("asr", AArch64_AM::ASR)
2410           .Case("ror", AArch64_AM::ROR)
2411           .Case("msl", AArch64_AM::MSL)
2412           .Case("uxtb", AArch64_AM::UXTB)
2413           .Case("uxth", AArch64_AM::UXTH)
2414           .Case("uxtw", AArch64_AM::UXTW)
2415           .Case("uxtx", AArch64_AM::UXTX)
2416           .Case("sxtb", AArch64_AM::SXTB)
2417           .Case("sxth", AArch64_AM::SXTH)
2418           .Case("sxtw", AArch64_AM::SXTW)
2419           .Case("sxtx", AArch64_AM::SXTX)
2420           .Default(AArch64_AM::InvalidShiftExtend);
2421 
2422   if (ShOp == AArch64_AM::InvalidShiftExtend)
2423     return MatchOperand_NoMatch;
2424 
2425   SMLoc S = Tok.getLoc();
2426   Parser.Lex();
2427 
2428   bool Hash = getLexer().is(AsmToken::Hash);
2429   if (!Hash && getLexer().isNot(AsmToken::Integer)) {
2430     if (ShOp == AArch64_AM::LSL || ShOp == AArch64_AM::LSR ||
2431         ShOp == AArch64_AM::ASR || ShOp == AArch64_AM::ROR ||
2432         ShOp == AArch64_AM::MSL) {
2433       // We expect a number here.
2434       TokError("expected #imm after shift specifier");
2435       return MatchOperand_ParseFail;
2436     }
2437 
2438     // "extend" type operatoins don't need an immediate, #0 is implicit.
2439     SMLoc E = SMLoc::getFromPointer(getLoc().getPointer() - 1);
2440     Operands.push_back(
2441         AArch64Operand::CreateShiftExtend(ShOp, 0, false, S, E, getContext()));
2442     return MatchOperand_Success;
2443   }
2444 
2445   if (Hash)
2446     Parser.Lex(); // Eat the '#'.
2447 
2448   // Make sure we do actually have a number or a parenthesized expression.
2449   SMLoc E = Parser.getTok().getLoc();
2450   if (!Parser.getTok().is(AsmToken::Integer) &&
2451       !Parser.getTok().is(AsmToken::LParen)) {
2452     Error(E, "expected integer shift amount");
2453     return MatchOperand_ParseFail;
2454   }
2455 
2456   const MCExpr *ImmVal;
2457   if (getParser().parseExpression(ImmVal))
2458     return MatchOperand_ParseFail;
2459 
2460   const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(ImmVal);
2461   if (!MCE) {
2462     Error(E, "expected constant '#imm' after shift specifier");
2463     return MatchOperand_ParseFail;
2464   }
2465 
2466   E = SMLoc::getFromPointer(getLoc().getPointer() - 1);
2467   Operands.push_back(AArch64Operand::CreateShiftExtend(
2468       ShOp, MCE->getValue(), true, S, E, getContext()));
2469   return MatchOperand_Success;
2470 }
2471 
2472 /// parseSysAlias - The IC, DC, AT, and TLBI instructions are simple aliases for
2473 /// the SYS instruction. Parse them specially so that we create a SYS MCInst.
parseSysAlias(StringRef Name,SMLoc NameLoc,OperandVector & Operands)2474 bool AArch64AsmParser::parseSysAlias(StringRef Name, SMLoc NameLoc,
2475                                    OperandVector &Operands) {
2476   if (Name.find('.') != StringRef::npos)
2477     return TokError("invalid operand");
2478 
2479   Mnemonic = Name;
2480   Operands.push_back(
2481       AArch64Operand::CreateToken("sys", false, NameLoc, getContext()));
2482 
2483   MCAsmParser &Parser = getParser();
2484   const AsmToken &Tok = Parser.getTok();
2485   StringRef Op = Tok.getString();
2486   SMLoc S = Tok.getLoc();
2487 
2488   const MCExpr *Expr = nullptr;
2489 
2490 #define SYS_ALIAS(op1, Cn, Cm, op2)                                            \
2491   do {                                                                         \
2492     Expr = MCConstantExpr::create(op1, getContext());                          \
2493     Operands.push_back(                                                        \
2494         AArch64Operand::CreateImm(Expr, S, getLoc(), getContext()));           \
2495     Operands.push_back(                                                        \
2496         AArch64Operand::CreateSysCR(Cn, S, getLoc(), getContext()));           \
2497     Operands.push_back(                                                        \
2498         AArch64Operand::CreateSysCR(Cm, S, getLoc(), getContext()));           \
2499     Expr = MCConstantExpr::create(op2, getContext());                          \
2500     Operands.push_back(                                                        \
2501         AArch64Operand::CreateImm(Expr, S, getLoc(), getContext()));           \
2502   } while (0)
2503 
2504   if (Mnemonic == "ic") {
2505     if (!Op.compare_lower("ialluis")) {
2506       // SYS #0, C7, C1, #0
2507       SYS_ALIAS(0, 7, 1, 0);
2508     } else if (!Op.compare_lower("iallu")) {
2509       // SYS #0, C7, C5, #0
2510       SYS_ALIAS(0, 7, 5, 0);
2511     } else if (!Op.compare_lower("ivau")) {
2512       // SYS #3, C7, C5, #1
2513       SYS_ALIAS(3, 7, 5, 1);
2514     } else {
2515       return TokError("invalid operand for IC instruction");
2516     }
2517   } else if (Mnemonic == "dc") {
2518     if (!Op.compare_lower("zva")) {
2519       // SYS #3, C7, C4, #1
2520       SYS_ALIAS(3, 7, 4, 1);
2521     } else if (!Op.compare_lower("ivac")) {
2522       // SYS #3, C7, C6, #1
2523       SYS_ALIAS(0, 7, 6, 1);
2524     } else if (!Op.compare_lower("isw")) {
2525       // SYS #0, C7, C6, #2
2526       SYS_ALIAS(0, 7, 6, 2);
2527     } else if (!Op.compare_lower("cvac")) {
2528       // SYS #3, C7, C10, #1
2529       SYS_ALIAS(3, 7, 10, 1);
2530     } else if (!Op.compare_lower("csw")) {
2531       // SYS #0, C7, C10, #2
2532       SYS_ALIAS(0, 7, 10, 2);
2533     } else if (!Op.compare_lower("cvau")) {
2534       // SYS #3, C7, C11, #1
2535       SYS_ALIAS(3, 7, 11, 1);
2536     } else if (!Op.compare_lower("civac")) {
2537       // SYS #3, C7, C14, #1
2538       SYS_ALIAS(3, 7, 14, 1);
2539     } else if (!Op.compare_lower("cisw")) {
2540       // SYS #0, C7, C14, #2
2541       SYS_ALIAS(0, 7, 14, 2);
2542     } else if (!Op.compare_lower("cvap")) {
2543       if (getSTI().getFeatureBits()[AArch64::HasV8_2aOps]) {
2544         // SYS #3, C7, C12, #1
2545         SYS_ALIAS(3, 7, 12, 1);
2546       } else {
2547         return TokError("DC CVAP requires ARMv8.2a");
2548       }
2549     } else {
2550       return TokError("invalid operand for DC instruction");
2551     }
2552   } else if (Mnemonic == "at") {
2553     if (!Op.compare_lower("s1e1r")) {
2554       // SYS #0, C7, C8, #0
2555       SYS_ALIAS(0, 7, 8, 0);
2556     } else if (!Op.compare_lower("s1e2r")) {
2557       // SYS #4, C7, C8, #0
2558       SYS_ALIAS(4, 7, 8, 0);
2559     } else if (!Op.compare_lower("s1e3r")) {
2560       // SYS #6, C7, C8, #0
2561       SYS_ALIAS(6, 7, 8, 0);
2562     } else if (!Op.compare_lower("s1e1w")) {
2563       // SYS #0, C7, C8, #1
2564       SYS_ALIAS(0, 7, 8, 1);
2565     } else if (!Op.compare_lower("s1e2w")) {
2566       // SYS #4, C7, C8, #1
2567       SYS_ALIAS(4, 7, 8, 1);
2568     } else if (!Op.compare_lower("s1e3w")) {
2569       // SYS #6, C7, C8, #1
2570       SYS_ALIAS(6, 7, 8, 1);
2571     } else if (!Op.compare_lower("s1e0r")) {
2572       // SYS #0, C7, C8, #3
2573       SYS_ALIAS(0, 7, 8, 2);
2574     } else if (!Op.compare_lower("s1e0w")) {
2575       // SYS #0, C7, C8, #3
2576       SYS_ALIAS(0, 7, 8, 3);
2577     } else if (!Op.compare_lower("s12e1r")) {
2578       // SYS #4, C7, C8, #4
2579       SYS_ALIAS(4, 7, 8, 4);
2580     } else if (!Op.compare_lower("s12e1w")) {
2581       // SYS #4, C7, C8, #5
2582       SYS_ALIAS(4, 7, 8, 5);
2583     } else if (!Op.compare_lower("s12e0r")) {
2584       // SYS #4, C7, C8, #6
2585       SYS_ALIAS(4, 7, 8, 6);
2586     } else if (!Op.compare_lower("s12e0w")) {
2587       // SYS #4, C7, C8, #7
2588       SYS_ALIAS(4, 7, 8, 7);
2589     } else if (!Op.compare_lower("s1e1rp")) {
2590       if (getSTI().getFeatureBits()[AArch64::HasV8_2aOps]) {
2591         // SYS #0, C7, C9, #0
2592         SYS_ALIAS(0, 7, 9, 0);
2593       } else {
2594         return TokError("AT S1E1RP requires ARMv8.2a");
2595       }
2596     } else if (!Op.compare_lower("s1e1wp")) {
2597       if (getSTI().getFeatureBits()[AArch64::HasV8_2aOps]) {
2598         // SYS #0, C7, C9, #1
2599         SYS_ALIAS(0, 7, 9, 1);
2600       } else {
2601         return TokError("AT S1E1WP requires ARMv8.2a");
2602       }
2603     } else {
2604       return TokError("invalid operand for AT instruction");
2605     }
2606   } else if (Mnemonic == "tlbi") {
2607     if (!Op.compare_lower("vmalle1is")) {
2608       // SYS #0, C8, C3, #0
2609       SYS_ALIAS(0, 8, 3, 0);
2610     } else if (!Op.compare_lower("alle2is")) {
2611       // SYS #4, C8, C3, #0
2612       SYS_ALIAS(4, 8, 3, 0);
2613     } else if (!Op.compare_lower("alle3is")) {
2614       // SYS #6, C8, C3, #0
2615       SYS_ALIAS(6, 8, 3, 0);
2616     } else if (!Op.compare_lower("vae1is")) {
2617       // SYS #0, C8, C3, #1
2618       SYS_ALIAS(0, 8, 3, 1);
2619     } else if (!Op.compare_lower("vae2is")) {
2620       // SYS #4, C8, C3, #1
2621       SYS_ALIAS(4, 8, 3, 1);
2622     } else if (!Op.compare_lower("vae3is")) {
2623       // SYS #6, C8, C3, #1
2624       SYS_ALIAS(6, 8, 3, 1);
2625     } else if (!Op.compare_lower("aside1is")) {
2626       // SYS #0, C8, C3, #2
2627       SYS_ALIAS(0, 8, 3, 2);
2628     } else if (!Op.compare_lower("vaae1is")) {
2629       // SYS #0, C8, C3, #3
2630       SYS_ALIAS(0, 8, 3, 3);
2631     } else if (!Op.compare_lower("alle1is")) {
2632       // SYS #4, C8, C3, #4
2633       SYS_ALIAS(4, 8, 3, 4);
2634     } else if (!Op.compare_lower("vale1is")) {
2635       // SYS #0, C8, C3, #5
2636       SYS_ALIAS(0, 8, 3, 5);
2637     } else if (!Op.compare_lower("vaale1is")) {
2638       // SYS #0, C8, C3, #7
2639       SYS_ALIAS(0, 8, 3, 7);
2640     } else if (!Op.compare_lower("vmalle1")) {
2641       // SYS #0, C8, C7, #0
2642       SYS_ALIAS(0, 8, 7, 0);
2643     } else if (!Op.compare_lower("alle2")) {
2644       // SYS #4, C8, C7, #0
2645       SYS_ALIAS(4, 8, 7, 0);
2646     } else if (!Op.compare_lower("vale2is")) {
2647       // SYS #4, C8, C3, #5
2648       SYS_ALIAS(4, 8, 3, 5);
2649     } else if (!Op.compare_lower("vale3is")) {
2650       // SYS #6, C8, C3, #5
2651       SYS_ALIAS(6, 8, 3, 5);
2652     } else if (!Op.compare_lower("alle3")) {
2653       // SYS #6, C8, C7, #0
2654       SYS_ALIAS(6, 8, 7, 0);
2655     } else if (!Op.compare_lower("vae1")) {
2656       // SYS #0, C8, C7, #1
2657       SYS_ALIAS(0, 8, 7, 1);
2658     } else if (!Op.compare_lower("vae2")) {
2659       // SYS #4, C8, C7, #1
2660       SYS_ALIAS(4, 8, 7, 1);
2661     } else if (!Op.compare_lower("vae3")) {
2662       // SYS #6, C8, C7, #1
2663       SYS_ALIAS(6, 8, 7, 1);
2664     } else if (!Op.compare_lower("aside1")) {
2665       // SYS #0, C8, C7, #2
2666       SYS_ALIAS(0, 8, 7, 2);
2667     } else if (!Op.compare_lower("vaae1")) {
2668       // SYS #0, C8, C7, #3
2669       SYS_ALIAS(0, 8, 7, 3);
2670     } else if (!Op.compare_lower("alle1")) {
2671       // SYS #4, C8, C7, #4
2672       SYS_ALIAS(4, 8, 7, 4);
2673     } else if (!Op.compare_lower("vale1")) {
2674       // SYS #0, C8, C7, #5
2675       SYS_ALIAS(0, 8, 7, 5);
2676     } else if (!Op.compare_lower("vale2")) {
2677       // SYS #4, C8, C7, #5
2678       SYS_ALIAS(4, 8, 7, 5);
2679     } else if (!Op.compare_lower("vale3")) {
2680       // SYS #6, C8, C7, #5
2681       SYS_ALIAS(6, 8, 7, 5);
2682     } else if (!Op.compare_lower("vaale1")) {
2683       // SYS #0, C8, C7, #7
2684       SYS_ALIAS(0, 8, 7, 7);
2685     } else if (!Op.compare_lower("ipas2e1")) {
2686       // SYS #4, C8, C4, #1
2687       SYS_ALIAS(4, 8, 4, 1);
2688     } else if (!Op.compare_lower("ipas2le1")) {
2689       // SYS #4, C8, C4, #5
2690       SYS_ALIAS(4, 8, 4, 5);
2691     } else if (!Op.compare_lower("ipas2e1is")) {
2692       // SYS #4, C8, C4, #1
2693       SYS_ALIAS(4, 8, 0, 1);
2694     } else if (!Op.compare_lower("ipas2le1is")) {
2695       // SYS #4, C8, C4, #5
2696       SYS_ALIAS(4, 8, 0, 5);
2697     } else if (!Op.compare_lower("vmalls12e1")) {
2698       // SYS #4, C8, C7, #6
2699       SYS_ALIAS(4, 8, 7, 6);
2700     } else if (!Op.compare_lower("vmalls12e1is")) {
2701       // SYS #4, C8, C3, #6
2702       SYS_ALIAS(4, 8, 3, 6);
2703     } else {
2704       return TokError("invalid operand for TLBI instruction");
2705     }
2706   }
2707 
2708 #undef SYS_ALIAS
2709 
2710   Parser.Lex(); // Eat operand.
2711 
2712   bool ExpectRegister = (Op.lower().find("all") == StringRef::npos);
2713   bool HasRegister = false;
2714 
2715   // Check for the optional register operand.
2716   if (getLexer().is(AsmToken::Comma)) {
2717     Parser.Lex(); // Eat comma.
2718 
2719     if (Tok.isNot(AsmToken::Identifier) || parseRegister(Operands))
2720       return TokError("expected register operand");
2721 
2722     HasRegister = true;
2723   }
2724 
2725   if (getLexer().isNot(AsmToken::EndOfStatement)) {
2726     Parser.eatToEndOfStatement();
2727     return TokError("unexpected token in argument list");
2728   }
2729 
2730   if (ExpectRegister && !HasRegister) {
2731     return TokError("specified " + Mnemonic + " op requires a register");
2732   }
2733   else if (!ExpectRegister && HasRegister) {
2734     return TokError("specified " + Mnemonic + " op does not use a register");
2735   }
2736 
2737   Parser.Lex(); // Consume the EndOfStatement
2738   return false;
2739 }
2740 
2741 AArch64AsmParser::OperandMatchResultTy
tryParseBarrierOperand(OperandVector & Operands)2742 AArch64AsmParser::tryParseBarrierOperand(OperandVector &Operands) {
2743   MCAsmParser &Parser = getParser();
2744   const AsmToken &Tok = Parser.getTok();
2745 
2746   // Can be either a #imm style literal or an option name
2747   bool Hash = Tok.is(AsmToken::Hash);
2748   if (Hash || Tok.is(AsmToken::Integer)) {
2749     // Immediate operand.
2750     if (Hash)
2751       Parser.Lex(); // Eat the '#'
2752     const MCExpr *ImmVal;
2753     SMLoc ExprLoc = getLoc();
2754     if (getParser().parseExpression(ImmVal))
2755       return MatchOperand_ParseFail;
2756     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(ImmVal);
2757     if (!MCE) {
2758       Error(ExprLoc, "immediate value expected for barrier operand");
2759       return MatchOperand_ParseFail;
2760     }
2761     if (MCE->getValue() < 0 || MCE->getValue() > 15) {
2762       Error(ExprLoc, "barrier operand out of range");
2763       return MatchOperand_ParseFail;
2764     }
2765     bool Valid;
2766     auto Mapper = AArch64DB::DBarrierMapper();
2767     StringRef Name =
2768         Mapper.toString(MCE->getValue(), getSTI().getFeatureBits(), Valid);
2769     Operands.push_back( AArch64Operand::CreateBarrier(MCE->getValue(), Name,
2770                                                       ExprLoc, getContext()));
2771     return MatchOperand_Success;
2772   }
2773 
2774   if (Tok.isNot(AsmToken::Identifier)) {
2775     TokError("invalid operand for instruction");
2776     return MatchOperand_ParseFail;
2777   }
2778 
2779   bool Valid;
2780   auto Mapper = AArch64DB::DBarrierMapper();
2781   unsigned Opt =
2782       Mapper.fromString(Tok.getString(), getSTI().getFeatureBits(), Valid);
2783   if (!Valid) {
2784     TokError("invalid barrier option name");
2785     return MatchOperand_ParseFail;
2786   }
2787 
2788   // The only valid named option for ISB is 'sy'
2789   if (Mnemonic == "isb" && Opt != AArch64DB::SY) {
2790     TokError("'sy' or #imm operand expected");
2791     return MatchOperand_ParseFail;
2792   }
2793 
2794   Operands.push_back( AArch64Operand::CreateBarrier(Opt, Tok.getString(),
2795                                                     getLoc(), getContext()));
2796   Parser.Lex(); // Consume the option
2797 
2798   return MatchOperand_Success;
2799 }
2800 
2801 AArch64AsmParser::OperandMatchResultTy
tryParseSysReg(OperandVector & Operands)2802 AArch64AsmParser::tryParseSysReg(OperandVector &Operands) {
2803   MCAsmParser &Parser = getParser();
2804   const AsmToken &Tok = Parser.getTok();
2805 
2806   if (Tok.isNot(AsmToken::Identifier))
2807     return MatchOperand_NoMatch;
2808 
2809   bool IsKnown;
2810   auto MRSMapper = AArch64SysReg::MRSMapper();
2811   uint32_t MRSReg = MRSMapper.fromString(Tok.getString(),
2812                                          getSTI().getFeatureBits(), IsKnown);
2813   assert(IsKnown == (MRSReg != -1U) &&
2814          "register should be -1 if and only if it's unknown");
2815 
2816   auto MSRMapper = AArch64SysReg::MSRMapper();
2817   uint32_t MSRReg = MSRMapper.fromString(Tok.getString(),
2818                                          getSTI().getFeatureBits(), IsKnown);
2819   assert(IsKnown == (MSRReg != -1U) &&
2820          "register should be -1 if and only if it's unknown");
2821 
2822   auto PStateMapper = AArch64PState::PStateMapper();
2823   uint32_t PStateField =
2824       PStateMapper.fromString(Tok.getString(),
2825                               getSTI().getFeatureBits(), IsKnown);
2826   assert(IsKnown == (PStateField != -1U) &&
2827          "register should be -1 if and only if it's unknown");
2828 
2829   Operands.push_back(AArch64Operand::CreateSysReg(
2830       Tok.getString(), getLoc(), MRSReg, MSRReg, PStateField, getContext()));
2831   Parser.Lex(); // Eat identifier
2832 
2833   return MatchOperand_Success;
2834 }
2835 
2836 /// tryParseVectorRegister - Parse a vector register operand.
tryParseVectorRegister(OperandVector & Operands)2837 bool AArch64AsmParser::tryParseVectorRegister(OperandVector &Operands) {
2838   MCAsmParser &Parser = getParser();
2839   if (Parser.getTok().isNot(AsmToken::Identifier))
2840     return true;
2841 
2842   SMLoc S = getLoc();
2843   // Check for a vector register specifier first.
2844   StringRef Kind;
2845   int64_t Reg = tryMatchVectorRegister(Kind, false);
2846   if (Reg == -1)
2847     return true;
2848   Operands.push_back(
2849       AArch64Operand::CreateReg(Reg, true, S, getLoc(), getContext()));
2850   // If there was an explicit qualifier, that goes on as a literal text
2851   // operand.
2852   if (!Kind.empty())
2853     Operands.push_back(
2854         AArch64Operand::CreateToken(Kind, false, S, getContext()));
2855 
2856   // If there is an index specifier following the register, parse that too.
2857   if (Parser.getTok().is(AsmToken::LBrac)) {
2858     SMLoc SIdx = getLoc();
2859     Parser.Lex(); // Eat left bracket token.
2860 
2861     const MCExpr *ImmVal;
2862     if (getParser().parseExpression(ImmVal))
2863       return false;
2864     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(ImmVal);
2865     if (!MCE) {
2866       TokError("immediate value expected for vector index");
2867       return false;
2868     }
2869 
2870     SMLoc E = getLoc();
2871     if (Parser.getTok().isNot(AsmToken::RBrac)) {
2872       Error(E, "']' expected");
2873       return false;
2874     }
2875 
2876     Parser.Lex(); // Eat right bracket token.
2877 
2878     Operands.push_back(AArch64Operand::CreateVectorIndex(MCE->getValue(), SIdx,
2879                                                          E, getContext()));
2880   }
2881 
2882   return false;
2883 }
2884 
2885 /// parseRegister - Parse a non-vector register operand.
parseRegister(OperandVector & Operands)2886 bool AArch64AsmParser::parseRegister(OperandVector &Operands) {
2887   MCAsmParser &Parser = getParser();
2888   SMLoc S = getLoc();
2889   // Try for a vector register.
2890   if (!tryParseVectorRegister(Operands))
2891     return false;
2892 
2893   // Try for a scalar register.
2894   int64_t Reg = tryParseRegister();
2895   if (Reg == -1)
2896     return true;
2897   Operands.push_back(
2898       AArch64Operand::CreateReg(Reg, false, S, getLoc(), getContext()));
2899 
2900   // A small number of instructions (FMOVXDhighr, for example) have "[1]"
2901   // as a string token in the instruction itself.
2902   if (getLexer().getKind() == AsmToken::LBrac) {
2903     SMLoc LBracS = getLoc();
2904     Parser.Lex();
2905     const AsmToken &Tok = Parser.getTok();
2906     if (Tok.is(AsmToken::Integer)) {
2907       SMLoc IntS = getLoc();
2908       int64_t Val = Tok.getIntVal();
2909       if (Val == 1) {
2910         Parser.Lex();
2911         if (getLexer().getKind() == AsmToken::RBrac) {
2912           SMLoc RBracS = getLoc();
2913           Parser.Lex();
2914           Operands.push_back(
2915               AArch64Operand::CreateToken("[", false, LBracS, getContext()));
2916           Operands.push_back(
2917               AArch64Operand::CreateToken("1", false, IntS, getContext()));
2918           Operands.push_back(
2919               AArch64Operand::CreateToken("]", false, RBracS, getContext()));
2920           return false;
2921         }
2922       }
2923     }
2924   }
2925 
2926   return false;
2927 }
2928 
parseSymbolicImmVal(const MCExpr * & ImmVal)2929 bool AArch64AsmParser::parseSymbolicImmVal(const MCExpr *&ImmVal) {
2930   MCAsmParser &Parser = getParser();
2931   bool HasELFModifier = false;
2932   AArch64MCExpr::VariantKind RefKind;
2933 
2934   if (Parser.getTok().is(AsmToken::Colon)) {
2935     Parser.Lex(); // Eat ':"
2936     HasELFModifier = true;
2937 
2938     if (Parser.getTok().isNot(AsmToken::Identifier)) {
2939       Error(Parser.getTok().getLoc(),
2940             "expect relocation specifier in operand after ':'");
2941       return true;
2942     }
2943 
2944     std::string LowerCase = Parser.getTok().getIdentifier().lower();
2945     RefKind = StringSwitch<AArch64MCExpr::VariantKind>(LowerCase)
2946                   .Case("lo12", AArch64MCExpr::VK_LO12)
2947                   .Case("abs_g3", AArch64MCExpr::VK_ABS_G3)
2948                   .Case("abs_g2", AArch64MCExpr::VK_ABS_G2)
2949                   .Case("abs_g2_s", AArch64MCExpr::VK_ABS_G2_S)
2950                   .Case("abs_g2_nc", AArch64MCExpr::VK_ABS_G2_NC)
2951                   .Case("abs_g1", AArch64MCExpr::VK_ABS_G1)
2952                   .Case("abs_g1_s", AArch64MCExpr::VK_ABS_G1_S)
2953                   .Case("abs_g1_nc", AArch64MCExpr::VK_ABS_G1_NC)
2954                   .Case("abs_g0", AArch64MCExpr::VK_ABS_G0)
2955                   .Case("abs_g0_s", AArch64MCExpr::VK_ABS_G0_S)
2956                   .Case("abs_g0_nc", AArch64MCExpr::VK_ABS_G0_NC)
2957                   .Case("dtprel_g2", AArch64MCExpr::VK_DTPREL_G2)
2958                   .Case("dtprel_g1", AArch64MCExpr::VK_DTPREL_G1)
2959                   .Case("dtprel_g1_nc", AArch64MCExpr::VK_DTPREL_G1_NC)
2960                   .Case("dtprel_g0", AArch64MCExpr::VK_DTPREL_G0)
2961                   .Case("dtprel_g0_nc", AArch64MCExpr::VK_DTPREL_G0_NC)
2962                   .Case("dtprel_hi12", AArch64MCExpr::VK_DTPREL_HI12)
2963                   .Case("dtprel_lo12", AArch64MCExpr::VK_DTPREL_LO12)
2964                   .Case("dtprel_lo12_nc", AArch64MCExpr::VK_DTPREL_LO12_NC)
2965                   .Case("tprel_g2", AArch64MCExpr::VK_TPREL_G2)
2966                   .Case("tprel_g1", AArch64MCExpr::VK_TPREL_G1)
2967                   .Case("tprel_g1_nc", AArch64MCExpr::VK_TPREL_G1_NC)
2968                   .Case("tprel_g0", AArch64MCExpr::VK_TPREL_G0)
2969                   .Case("tprel_g0_nc", AArch64MCExpr::VK_TPREL_G0_NC)
2970                   .Case("tprel_hi12", AArch64MCExpr::VK_TPREL_HI12)
2971                   .Case("tprel_lo12", AArch64MCExpr::VK_TPREL_LO12)
2972                   .Case("tprel_lo12_nc", AArch64MCExpr::VK_TPREL_LO12_NC)
2973                   .Case("tlsdesc_lo12", AArch64MCExpr::VK_TLSDESC_LO12)
2974                   .Case("got", AArch64MCExpr::VK_GOT_PAGE)
2975                   .Case("got_lo12", AArch64MCExpr::VK_GOT_LO12)
2976                   .Case("gottprel", AArch64MCExpr::VK_GOTTPREL_PAGE)
2977                   .Case("gottprel_lo12", AArch64MCExpr::VK_GOTTPREL_LO12_NC)
2978                   .Case("gottprel_g1", AArch64MCExpr::VK_GOTTPREL_G1)
2979                   .Case("gottprel_g0_nc", AArch64MCExpr::VK_GOTTPREL_G0_NC)
2980                   .Case("tlsdesc", AArch64MCExpr::VK_TLSDESC_PAGE)
2981                   .Default(AArch64MCExpr::VK_INVALID);
2982 
2983     if (RefKind == AArch64MCExpr::VK_INVALID) {
2984       Error(Parser.getTok().getLoc(),
2985             "expect relocation specifier in operand after ':'");
2986       return true;
2987     }
2988 
2989     Parser.Lex(); // Eat identifier
2990 
2991     if (Parser.getTok().isNot(AsmToken::Colon)) {
2992       Error(Parser.getTok().getLoc(), "expect ':' after relocation specifier");
2993       return true;
2994     }
2995     Parser.Lex(); // Eat ':'
2996   }
2997 
2998   if (getParser().parseExpression(ImmVal))
2999     return true;
3000 
3001   if (HasELFModifier)
3002     ImmVal = AArch64MCExpr::create(ImmVal, RefKind, getContext());
3003 
3004   return false;
3005 }
3006 
3007 /// parseVectorList - Parse a vector list operand for AdvSIMD instructions.
parseVectorList(OperandVector & Operands)3008 bool AArch64AsmParser::parseVectorList(OperandVector &Operands) {
3009   MCAsmParser &Parser = getParser();
3010   assert(Parser.getTok().is(AsmToken::LCurly) && "Token is not a Left Bracket");
3011   SMLoc S = getLoc();
3012   Parser.Lex(); // Eat left bracket token.
3013   StringRef Kind;
3014   int64_t FirstReg = tryMatchVectorRegister(Kind, true);
3015   if (FirstReg == -1)
3016     return true;
3017   int64_t PrevReg = FirstReg;
3018   unsigned Count = 1;
3019 
3020   if (Parser.getTok().is(AsmToken::Minus)) {
3021     Parser.Lex(); // Eat the minus.
3022 
3023     SMLoc Loc = getLoc();
3024     StringRef NextKind;
3025     int64_t Reg = tryMatchVectorRegister(NextKind, true);
3026     if (Reg == -1)
3027       return true;
3028     // Any Kind suffices must match on all regs in the list.
3029     if (Kind != NextKind)
3030       return Error(Loc, "mismatched register size suffix");
3031 
3032     unsigned Space = (PrevReg < Reg) ? (Reg - PrevReg) : (Reg + 32 - PrevReg);
3033 
3034     if (Space == 0 || Space > 3) {
3035       return Error(Loc, "invalid number of vectors");
3036     }
3037 
3038     Count += Space;
3039   }
3040   else {
3041     while (Parser.getTok().is(AsmToken::Comma)) {
3042       Parser.Lex(); // Eat the comma token.
3043 
3044       SMLoc Loc = getLoc();
3045       StringRef NextKind;
3046       int64_t Reg = tryMatchVectorRegister(NextKind, true);
3047       if (Reg == -1)
3048         return true;
3049       // Any Kind suffices must match on all regs in the list.
3050       if (Kind != NextKind)
3051         return Error(Loc, "mismatched register size suffix");
3052 
3053       // Registers must be incremental (with wraparound at 31)
3054       if (getContext().getRegisterInfo()->getEncodingValue(Reg) !=
3055           (getContext().getRegisterInfo()->getEncodingValue(PrevReg) + 1) % 32)
3056        return Error(Loc, "registers must be sequential");
3057 
3058       PrevReg = Reg;
3059       ++Count;
3060     }
3061   }
3062 
3063   if (Parser.getTok().isNot(AsmToken::RCurly))
3064     return Error(getLoc(), "'}' expected");
3065   Parser.Lex(); // Eat the '}' token.
3066 
3067   if (Count > 4)
3068     return Error(S, "invalid number of vectors");
3069 
3070   unsigned NumElements = 0;
3071   char ElementKind = 0;
3072   if (!Kind.empty())
3073     parseValidVectorKind(Kind, NumElements, ElementKind);
3074 
3075   Operands.push_back(AArch64Operand::CreateVectorList(
3076       FirstReg, Count, NumElements, ElementKind, S, getLoc(), getContext()));
3077 
3078   // If there is an index specifier following the list, parse that too.
3079   if (Parser.getTok().is(AsmToken::LBrac)) {
3080     SMLoc SIdx = getLoc();
3081     Parser.Lex(); // Eat left bracket token.
3082 
3083     const MCExpr *ImmVal;
3084     if (getParser().parseExpression(ImmVal))
3085       return false;
3086     const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(ImmVal);
3087     if (!MCE) {
3088       TokError("immediate value expected for vector index");
3089       return false;
3090     }
3091 
3092     SMLoc E = getLoc();
3093     if (Parser.getTok().isNot(AsmToken::RBrac)) {
3094       Error(E, "']' expected");
3095       return false;
3096     }
3097 
3098     Parser.Lex(); // Eat right bracket token.
3099 
3100     Operands.push_back(AArch64Operand::CreateVectorIndex(MCE->getValue(), SIdx,
3101                                                          E, getContext()));
3102   }
3103   return false;
3104 }
3105 
3106 AArch64AsmParser::OperandMatchResultTy
tryParseGPR64sp0Operand(OperandVector & Operands)3107 AArch64AsmParser::tryParseGPR64sp0Operand(OperandVector &Operands) {
3108   MCAsmParser &Parser = getParser();
3109   const AsmToken &Tok = Parser.getTok();
3110   if (!Tok.is(AsmToken::Identifier))
3111     return MatchOperand_NoMatch;
3112 
3113   unsigned RegNum = matchRegisterNameAlias(Tok.getString().lower(), false);
3114 
3115   MCContext &Ctx = getContext();
3116   const MCRegisterInfo *RI = Ctx.getRegisterInfo();
3117   if (!RI->getRegClass(AArch64::GPR64spRegClassID).contains(RegNum))
3118     return MatchOperand_NoMatch;
3119 
3120   SMLoc S = getLoc();
3121   Parser.Lex(); // Eat register
3122 
3123   if (Parser.getTok().isNot(AsmToken::Comma)) {
3124     Operands.push_back(
3125         AArch64Operand::CreateReg(RegNum, false, S, getLoc(), Ctx));
3126     return MatchOperand_Success;
3127   }
3128   Parser.Lex(); // Eat comma.
3129 
3130   if (Parser.getTok().is(AsmToken::Hash))
3131     Parser.Lex(); // Eat hash
3132 
3133   if (Parser.getTok().isNot(AsmToken::Integer)) {
3134     Error(getLoc(), "index must be absent or #0");
3135     return MatchOperand_ParseFail;
3136   }
3137 
3138   const MCExpr *ImmVal;
3139   if (Parser.parseExpression(ImmVal) || !isa<MCConstantExpr>(ImmVal) ||
3140       cast<MCConstantExpr>(ImmVal)->getValue() != 0) {
3141     Error(getLoc(), "index must be absent or #0");
3142     return MatchOperand_ParseFail;
3143   }
3144 
3145   Operands.push_back(
3146       AArch64Operand::CreateReg(RegNum, false, S, getLoc(), Ctx));
3147   return MatchOperand_Success;
3148 }
3149 
3150 /// parseOperand - Parse a arm instruction operand.  For now this parses the
3151 /// operand regardless of the mnemonic.
parseOperand(OperandVector & Operands,bool isCondCode,bool invertCondCode)3152 bool AArch64AsmParser::parseOperand(OperandVector &Operands, bool isCondCode,
3153                                   bool invertCondCode) {
3154   MCAsmParser &Parser = getParser();
3155   // Check if the current operand has a custom associated parser, if so, try to
3156   // custom parse the operand, or fallback to the general approach.
3157   OperandMatchResultTy ResTy = MatchOperandParserImpl(Operands, Mnemonic);
3158   if (ResTy == MatchOperand_Success)
3159     return false;
3160   // If there wasn't a custom match, try the generic matcher below. Otherwise,
3161   // there was a match, but an error occurred, in which case, just return that
3162   // the operand parsing failed.
3163   if (ResTy == MatchOperand_ParseFail)
3164     return true;
3165 
3166   // Nothing custom, so do general case parsing.
3167   SMLoc S, E;
3168   switch (getLexer().getKind()) {
3169   default: {
3170     SMLoc S = getLoc();
3171     const MCExpr *Expr;
3172     if (parseSymbolicImmVal(Expr))
3173       return Error(S, "invalid operand");
3174 
3175     SMLoc E = SMLoc::getFromPointer(getLoc().getPointer() - 1);
3176     Operands.push_back(AArch64Operand::CreateImm(Expr, S, E, getContext()));
3177     return false;
3178   }
3179   case AsmToken::LBrac: {
3180     SMLoc Loc = Parser.getTok().getLoc();
3181     Operands.push_back(AArch64Operand::CreateToken("[", false, Loc,
3182                                                    getContext()));
3183     Parser.Lex(); // Eat '['
3184 
3185     // There's no comma after a '[', so we can parse the next operand
3186     // immediately.
3187     return parseOperand(Operands, false, false);
3188   }
3189   case AsmToken::LCurly:
3190     return parseVectorList(Operands);
3191   case AsmToken::Identifier: {
3192     // If we're expecting a Condition Code operand, then just parse that.
3193     if (isCondCode)
3194       return parseCondCode(Operands, invertCondCode);
3195 
3196     // If it's a register name, parse it.
3197     if (!parseRegister(Operands))
3198       return false;
3199 
3200     // This could be an optional "shift" or "extend" operand.
3201     OperandMatchResultTy GotShift = tryParseOptionalShiftExtend(Operands);
3202     // We can only continue if no tokens were eaten.
3203     if (GotShift != MatchOperand_NoMatch)
3204       return GotShift;
3205 
3206     // This was not a register so parse other operands that start with an
3207     // identifier (like labels) as expressions and create them as immediates.
3208     const MCExpr *IdVal;
3209     S = getLoc();
3210     if (getParser().parseExpression(IdVal))
3211       return true;
3212 
3213     E = SMLoc::getFromPointer(getLoc().getPointer() - 1);
3214     Operands.push_back(AArch64Operand::CreateImm(IdVal, S, E, getContext()));
3215     return false;
3216   }
3217   case AsmToken::Integer:
3218   case AsmToken::Real:
3219   case AsmToken::Hash: {
3220     // #42 -> immediate.
3221     S = getLoc();
3222     if (getLexer().is(AsmToken::Hash))
3223       Parser.Lex();
3224 
3225     // Parse a negative sign
3226     bool isNegative = false;
3227     if (Parser.getTok().is(AsmToken::Minus)) {
3228       isNegative = true;
3229       // We need to consume this token only when we have a Real, otherwise
3230       // we let parseSymbolicImmVal take care of it
3231       if (Parser.getLexer().peekTok().is(AsmToken::Real))
3232         Parser.Lex();
3233     }
3234 
3235     // The only Real that should come through here is a literal #0.0 for
3236     // the fcmp[e] r, #0.0 instructions. They expect raw token operands,
3237     // so convert the value.
3238     const AsmToken &Tok = Parser.getTok();
3239     if (Tok.is(AsmToken::Real)) {
3240       APFloat RealVal(APFloat::IEEEdouble, Tok.getString());
3241       uint64_t IntVal = RealVal.bitcastToAPInt().getZExtValue();
3242       if (Mnemonic != "fcmp" && Mnemonic != "fcmpe" && Mnemonic != "fcmeq" &&
3243           Mnemonic != "fcmge" && Mnemonic != "fcmgt" && Mnemonic != "fcmle" &&
3244           Mnemonic != "fcmlt")
3245         return TokError("unexpected floating point literal");
3246       else if (IntVal != 0 || isNegative)
3247         return TokError("expected floating-point constant #0.0");
3248       Parser.Lex(); // Eat the token.
3249 
3250       Operands.push_back(
3251           AArch64Operand::CreateToken("#0", false, S, getContext()));
3252       Operands.push_back(
3253           AArch64Operand::CreateToken(".0", false, S, getContext()));
3254       return false;
3255     }
3256 
3257     const MCExpr *ImmVal;
3258     if (parseSymbolicImmVal(ImmVal))
3259       return true;
3260 
3261     E = SMLoc::getFromPointer(getLoc().getPointer() - 1);
3262     Operands.push_back(AArch64Operand::CreateImm(ImmVal, S, E, getContext()));
3263     return false;
3264   }
3265   case AsmToken::Equal: {
3266     SMLoc Loc = Parser.getTok().getLoc();
3267     if (Mnemonic != "ldr") // only parse for ldr pseudo (e.g. ldr r0, =val)
3268       return Error(Loc, "unexpected token in operand");
3269     Parser.Lex(); // Eat '='
3270     const MCExpr *SubExprVal;
3271     if (getParser().parseExpression(SubExprVal))
3272       return true;
3273 
3274     if (Operands.size() < 2 ||
3275         !static_cast<AArch64Operand &>(*Operands[1]).isReg())
3276       return Error(Loc, "Only valid when first operand is register");
3277 
3278     bool IsXReg =
3279         AArch64MCRegisterClasses[AArch64::GPR64allRegClassID].contains(
3280             Operands[1]->getReg());
3281 
3282     MCContext& Ctx = getContext();
3283     E = SMLoc::getFromPointer(Loc.getPointer() - 1);
3284     // If the op is an imm and can be fit into a mov, then replace ldr with mov.
3285     if (isa<MCConstantExpr>(SubExprVal)) {
3286       uint64_t Imm = (cast<MCConstantExpr>(SubExprVal))->getValue();
3287       uint32_t ShiftAmt = 0, MaxShiftAmt = IsXReg ? 48 : 16;
3288       while(Imm > 0xFFFF && countTrailingZeros(Imm) >= 16) {
3289         ShiftAmt += 16;
3290         Imm >>= 16;
3291       }
3292       if (ShiftAmt <= MaxShiftAmt && Imm <= 0xFFFF) {
3293           Operands[0] = AArch64Operand::CreateToken("movz", false, Loc, Ctx);
3294           Operands.push_back(AArch64Operand::CreateImm(
3295                      MCConstantExpr::create(Imm, Ctx), S, E, Ctx));
3296         if (ShiftAmt)
3297           Operands.push_back(AArch64Operand::CreateShiftExtend(AArch64_AM::LSL,
3298                      ShiftAmt, true, S, E, Ctx));
3299         return false;
3300       }
3301       APInt Simm = APInt(64, Imm << ShiftAmt);
3302       // check if the immediate is an unsigned or signed 32-bit int for W regs
3303       if (!IsXReg && !(Simm.isIntN(32) || Simm.isSignedIntN(32)))
3304         return Error(Loc, "Immediate too large for register");
3305     }
3306     // If it is a label or an imm that cannot fit in a movz, put it into CP.
3307     const MCExpr *CPLoc =
3308         getTargetStreamer().addConstantPoolEntry(SubExprVal, IsXReg ? 8 : 4, Loc);
3309     Operands.push_back(AArch64Operand::CreateImm(CPLoc, S, E, Ctx));
3310     return false;
3311   }
3312   }
3313 }
3314 
3315 /// ParseInstruction - Parse an AArch64 instruction mnemonic followed by its
3316 /// operands.
ParseInstruction(ParseInstructionInfo & Info,StringRef Name,SMLoc NameLoc,OperandVector & Operands)3317 bool AArch64AsmParser::ParseInstruction(ParseInstructionInfo &Info,
3318                                         StringRef Name, SMLoc NameLoc,
3319                                         OperandVector &Operands) {
3320   MCAsmParser &Parser = getParser();
3321   Name = StringSwitch<StringRef>(Name.lower())
3322              .Case("beq", "b.eq")
3323              .Case("bne", "b.ne")
3324              .Case("bhs", "b.hs")
3325              .Case("bcs", "b.cs")
3326              .Case("blo", "b.lo")
3327              .Case("bcc", "b.cc")
3328              .Case("bmi", "b.mi")
3329              .Case("bpl", "b.pl")
3330              .Case("bvs", "b.vs")
3331              .Case("bvc", "b.vc")
3332              .Case("bhi", "b.hi")
3333              .Case("bls", "b.ls")
3334              .Case("bge", "b.ge")
3335              .Case("blt", "b.lt")
3336              .Case("bgt", "b.gt")
3337              .Case("ble", "b.le")
3338              .Case("bal", "b.al")
3339              .Case("bnv", "b.nv")
3340              .Default(Name);
3341 
3342   // First check for the AArch64-specific .req directive.
3343   if (Parser.getTok().is(AsmToken::Identifier) &&
3344       Parser.getTok().getIdentifier() == ".req") {
3345     parseDirectiveReq(Name, NameLoc);
3346     // We always return 'error' for this, as we're done with this
3347     // statement and don't need to match the 'instruction."
3348     return true;
3349   }
3350 
3351   // Create the leading tokens for the mnemonic, split by '.' characters.
3352   size_t Start = 0, Next = Name.find('.');
3353   StringRef Head = Name.slice(Start, Next);
3354 
3355   // IC, DC, AT, and TLBI instructions are aliases for the SYS instruction.
3356   if (Head == "ic" || Head == "dc" || Head == "at" || Head == "tlbi") {
3357     bool IsError = parseSysAlias(Head, NameLoc, Operands);
3358     if (IsError && getLexer().isNot(AsmToken::EndOfStatement))
3359       Parser.eatToEndOfStatement();
3360     return IsError;
3361   }
3362 
3363   Operands.push_back(
3364       AArch64Operand::CreateToken(Head, false, NameLoc, getContext()));
3365   Mnemonic = Head;
3366 
3367   // Handle condition codes for a branch mnemonic
3368   if (Head == "b" && Next != StringRef::npos) {
3369     Start = Next;
3370     Next = Name.find('.', Start + 1);
3371     Head = Name.slice(Start + 1, Next);
3372 
3373     SMLoc SuffixLoc = SMLoc::getFromPointer(NameLoc.getPointer() +
3374                                             (Head.data() - Name.data()));
3375     AArch64CC::CondCode CC = parseCondCodeString(Head);
3376     if (CC == AArch64CC::Invalid)
3377       return Error(SuffixLoc, "invalid condition code");
3378     Operands.push_back(
3379         AArch64Operand::CreateToken(".", true, SuffixLoc, getContext()));
3380     Operands.push_back(
3381         AArch64Operand::CreateCondCode(CC, NameLoc, NameLoc, getContext()));
3382   }
3383 
3384   // Add the remaining tokens in the mnemonic.
3385   while (Next != StringRef::npos) {
3386     Start = Next;
3387     Next = Name.find('.', Start + 1);
3388     Head = Name.slice(Start, Next);
3389     SMLoc SuffixLoc = SMLoc::getFromPointer(NameLoc.getPointer() +
3390                                             (Head.data() - Name.data()) + 1);
3391     Operands.push_back(
3392         AArch64Operand::CreateToken(Head, true, SuffixLoc, getContext()));
3393   }
3394 
3395   // Conditional compare instructions have a Condition Code operand, which needs
3396   // to be parsed and an immediate operand created.
3397   bool condCodeFourthOperand =
3398       (Head == "ccmp" || Head == "ccmn" || Head == "fccmp" ||
3399        Head == "fccmpe" || Head == "fcsel" || Head == "csel" ||
3400        Head == "csinc" || Head == "csinv" || Head == "csneg");
3401 
3402   // These instructions are aliases to some of the conditional select
3403   // instructions. However, the condition code is inverted in the aliased
3404   // instruction.
3405   //
3406   // FIXME: Is this the correct way to handle these? Or should the parser
3407   //        generate the aliased instructions directly?
3408   bool condCodeSecondOperand = (Head == "cset" || Head == "csetm");
3409   bool condCodeThirdOperand =
3410       (Head == "cinc" || Head == "cinv" || Head == "cneg");
3411 
3412   // Read the remaining operands.
3413   if (getLexer().isNot(AsmToken::EndOfStatement)) {
3414     // Read the first operand.
3415     if (parseOperand(Operands, false, false)) {
3416       Parser.eatToEndOfStatement();
3417       return true;
3418     }
3419 
3420     unsigned N = 2;
3421     while (getLexer().is(AsmToken::Comma)) {
3422       Parser.Lex(); // Eat the comma.
3423 
3424       // Parse and remember the operand.
3425       if (parseOperand(Operands, (N == 4 && condCodeFourthOperand) ||
3426                                      (N == 3 && condCodeThirdOperand) ||
3427                                      (N == 2 && condCodeSecondOperand),
3428                        condCodeSecondOperand || condCodeThirdOperand)) {
3429         Parser.eatToEndOfStatement();
3430         return true;
3431       }
3432 
3433       // After successfully parsing some operands there are two special cases to
3434       // consider (i.e. notional operands not separated by commas). Both are due
3435       // to memory specifiers:
3436       //  + An RBrac will end an address for load/store/prefetch
3437       //  + An '!' will indicate a pre-indexed operation.
3438       //
3439       // It's someone else's responsibility to make sure these tokens are sane
3440       // in the given context!
3441       if (Parser.getTok().is(AsmToken::RBrac)) {
3442         SMLoc Loc = Parser.getTok().getLoc();
3443         Operands.push_back(AArch64Operand::CreateToken("]", false, Loc,
3444                                                        getContext()));
3445         Parser.Lex();
3446       }
3447 
3448       if (Parser.getTok().is(AsmToken::Exclaim)) {
3449         SMLoc Loc = Parser.getTok().getLoc();
3450         Operands.push_back(AArch64Operand::CreateToken("!", false, Loc,
3451                                                        getContext()));
3452         Parser.Lex();
3453       }
3454 
3455       ++N;
3456     }
3457   }
3458 
3459   if (getLexer().isNot(AsmToken::EndOfStatement)) {
3460     SMLoc Loc = Parser.getTok().getLoc();
3461     Parser.eatToEndOfStatement();
3462     return Error(Loc, "unexpected token in argument list");
3463   }
3464 
3465   Parser.Lex(); // Consume the EndOfStatement
3466   return false;
3467 }
3468 
3469 // FIXME: This entire function is a giant hack to provide us with decent
3470 // operand range validation/diagnostics until TableGen/MC can be extended
3471 // to support autogeneration of this kind of validation.
validateInstruction(MCInst & Inst,SmallVectorImpl<SMLoc> & Loc)3472 bool AArch64AsmParser::validateInstruction(MCInst &Inst,
3473                                          SmallVectorImpl<SMLoc> &Loc) {
3474   const MCRegisterInfo *RI = getContext().getRegisterInfo();
3475   // Check for indexed addressing modes w/ the base register being the
3476   // same as a destination/source register or pair load where
3477   // the Rt == Rt2. All of those are undefined behaviour.
3478   switch (Inst.getOpcode()) {
3479   case AArch64::LDPSWpre:
3480   case AArch64::LDPWpost:
3481   case AArch64::LDPWpre:
3482   case AArch64::LDPXpost:
3483   case AArch64::LDPXpre: {
3484     unsigned Rt = Inst.getOperand(1).getReg();
3485     unsigned Rt2 = Inst.getOperand(2).getReg();
3486     unsigned Rn = Inst.getOperand(3).getReg();
3487     if (RI->isSubRegisterEq(Rn, Rt))
3488       return Error(Loc[0], "unpredictable LDP instruction, writeback base "
3489                            "is also a destination");
3490     if (RI->isSubRegisterEq(Rn, Rt2))
3491       return Error(Loc[1], "unpredictable LDP instruction, writeback base "
3492                            "is also a destination");
3493     // FALLTHROUGH
3494   }
3495   case AArch64::LDPDi:
3496   case AArch64::LDPQi:
3497   case AArch64::LDPSi:
3498   case AArch64::LDPSWi:
3499   case AArch64::LDPWi:
3500   case AArch64::LDPXi: {
3501     unsigned Rt = Inst.getOperand(0).getReg();
3502     unsigned Rt2 = Inst.getOperand(1).getReg();
3503     if (Rt == Rt2)
3504       return Error(Loc[1], "unpredictable LDP instruction, Rt2==Rt");
3505     break;
3506   }
3507   case AArch64::LDPDpost:
3508   case AArch64::LDPDpre:
3509   case AArch64::LDPQpost:
3510   case AArch64::LDPQpre:
3511   case AArch64::LDPSpost:
3512   case AArch64::LDPSpre:
3513   case AArch64::LDPSWpost: {
3514     unsigned Rt = Inst.getOperand(1).getReg();
3515     unsigned Rt2 = Inst.getOperand(2).getReg();
3516     if (Rt == Rt2)
3517       return Error(Loc[1], "unpredictable LDP instruction, Rt2==Rt");
3518     break;
3519   }
3520   case AArch64::STPDpost:
3521   case AArch64::STPDpre:
3522   case AArch64::STPQpost:
3523   case AArch64::STPQpre:
3524   case AArch64::STPSpost:
3525   case AArch64::STPSpre:
3526   case AArch64::STPWpost:
3527   case AArch64::STPWpre:
3528   case AArch64::STPXpost:
3529   case AArch64::STPXpre: {
3530     unsigned Rt = Inst.getOperand(1).getReg();
3531     unsigned Rt2 = Inst.getOperand(2).getReg();
3532     unsigned Rn = Inst.getOperand(3).getReg();
3533     if (RI->isSubRegisterEq(Rn, Rt))
3534       return Error(Loc[0], "unpredictable STP instruction, writeback base "
3535                            "is also a source");
3536     if (RI->isSubRegisterEq(Rn, Rt2))
3537       return Error(Loc[1], "unpredictable STP instruction, writeback base "
3538                            "is also a source");
3539     break;
3540   }
3541   case AArch64::LDRBBpre:
3542   case AArch64::LDRBpre:
3543   case AArch64::LDRHHpre:
3544   case AArch64::LDRHpre:
3545   case AArch64::LDRSBWpre:
3546   case AArch64::LDRSBXpre:
3547   case AArch64::LDRSHWpre:
3548   case AArch64::LDRSHXpre:
3549   case AArch64::LDRSWpre:
3550   case AArch64::LDRWpre:
3551   case AArch64::LDRXpre:
3552   case AArch64::LDRBBpost:
3553   case AArch64::LDRBpost:
3554   case AArch64::LDRHHpost:
3555   case AArch64::LDRHpost:
3556   case AArch64::LDRSBWpost:
3557   case AArch64::LDRSBXpost:
3558   case AArch64::LDRSHWpost:
3559   case AArch64::LDRSHXpost:
3560   case AArch64::LDRSWpost:
3561   case AArch64::LDRWpost:
3562   case AArch64::LDRXpost: {
3563     unsigned Rt = Inst.getOperand(1).getReg();
3564     unsigned Rn = Inst.getOperand(2).getReg();
3565     if (RI->isSubRegisterEq(Rn, Rt))
3566       return Error(Loc[0], "unpredictable LDR instruction, writeback base "
3567                            "is also a source");
3568     break;
3569   }
3570   case AArch64::STRBBpost:
3571   case AArch64::STRBpost:
3572   case AArch64::STRHHpost:
3573   case AArch64::STRHpost:
3574   case AArch64::STRWpost:
3575   case AArch64::STRXpost:
3576   case AArch64::STRBBpre:
3577   case AArch64::STRBpre:
3578   case AArch64::STRHHpre:
3579   case AArch64::STRHpre:
3580   case AArch64::STRWpre:
3581   case AArch64::STRXpre: {
3582     unsigned Rt = Inst.getOperand(1).getReg();
3583     unsigned Rn = Inst.getOperand(2).getReg();
3584     if (RI->isSubRegisterEq(Rn, Rt))
3585       return Error(Loc[0], "unpredictable STR instruction, writeback base "
3586                            "is also a source");
3587     break;
3588   }
3589   }
3590 
3591   // Now check immediate ranges. Separate from the above as there is overlap
3592   // in the instructions being checked and this keeps the nested conditionals
3593   // to a minimum.
3594   switch (Inst.getOpcode()) {
3595   case AArch64::ADDSWri:
3596   case AArch64::ADDSXri:
3597   case AArch64::ADDWri:
3598   case AArch64::ADDXri:
3599   case AArch64::SUBSWri:
3600   case AArch64::SUBSXri:
3601   case AArch64::SUBWri:
3602   case AArch64::SUBXri: {
3603     // Annoyingly we can't do this in the isAddSubImm predicate, so there is
3604     // some slight duplication here.
3605     if (Inst.getOperand(2).isExpr()) {
3606       const MCExpr *Expr = Inst.getOperand(2).getExpr();
3607       AArch64MCExpr::VariantKind ELFRefKind;
3608       MCSymbolRefExpr::VariantKind DarwinRefKind;
3609       int64_t Addend;
3610       if (!classifySymbolRef(Expr, ELFRefKind, DarwinRefKind, Addend)) {
3611         return Error(Loc[2], "invalid immediate expression");
3612       }
3613 
3614       // Only allow these with ADDXri.
3615       if ((DarwinRefKind == MCSymbolRefExpr::VK_PAGEOFF ||
3616           DarwinRefKind == MCSymbolRefExpr::VK_TLVPPAGEOFF) &&
3617           Inst.getOpcode() == AArch64::ADDXri)
3618         return false;
3619 
3620       // Only allow these with ADDXri/ADDWri
3621       if ((ELFRefKind == AArch64MCExpr::VK_LO12 ||
3622           ELFRefKind == AArch64MCExpr::VK_DTPREL_HI12 ||
3623           ELFRefKind == AArch64MCExpr::VK_DTPREL_LO12 ||
3624           ELFRefKind == AArch64MCExpr::VK_DTPREL_LO12_NC ||
3625           ELFRefKind == AArch64MCExpr::VK_TPREL_HI12 ||
3626           ELFRefKind == AArch64MCExpr::VK_TPREL_LO12 ||
3627           ELFRefKind == AArch64MCExpr::VK_TPREL_LO12_NC ||
3628           ELFRefKind == AArch64MCExpr::VK_TLSDESC_LO12) &&
3629           (Inst.getOpcode() == AArch64::ADDXri ||
3630           Inst.getOpcode() == AArch64::ADDWri))
3631         return false;
3632 
3633       // Don't allow expressions in the immediate field otherwise
3634       return Error(Loc[2], "invalid immediate expression");
3635     }
3636     return false;
3637   }
3638   default:
3639     return false;
3640   }
3641 }
3642 
showMatchError(SMLoc Loc,unsigned ErrCode)3643 bool AArch64AsmParser::showMatchError(SMLoc Loc, unsigned ErrCode) {
3644   switch (ErrCode) {
3645   case Match_MissingFeature:
3646     return Error(Loc,
3647                  "instruction requires a CPU feature not currently enabled");
3648   case Match_InvalidOperand:
3649     return Error(Loc, "invalid operand for instruction");
3650   case Match_InvalidSuffix:
3651     return Error(Loc, "invalid type suffix for instruction");
3652   case Match_InvalidCondCode:
3653     return Error(Loc, "expected AArch64 condition code");
3654   case Match_AddSubRegExtendSmall:
3655     return Error(Loc,
3656       "expected '[su]xt[bhw]' or 'lsl' with optional integer in range [0, 4]");
3657   case Match_AddSubRegExtendLarge:
3658     return Error(Loc,
3659       "expected 'sxtx' 'uxtx' or 'lsl' with optional integer in range [0, 4]");
3660   case Match_AddSubSecondSource:
3661     return Error(Loc,
3662       "expected compatible register, symbol or integer in range [0, 4095]");
3663   case Match_LogicalSecondSource:
3664     return Error(Loc, "expected compatible register or logical immediate");
3665   case Match_InvalidMovImm32Shift:
3666     return Error(Loc, "expected 'lsl' with optional integer 0 or 16");
3667   case Match_InvalidMovImm64Shift:
3668     return Error(Loc, "expected 'lsl' with optional integer 0, 16, 32 or 48");
3669   case Match_AddSubRegShift32:
3670     return Error(Loc,
3671        "expected 'lsl', 'lsr' or 'asr' with optional integer in range [0, 31]");
3672   case Match_AddSubRegShift64:
3673     return Error(Loc,
3674        "expected 'lsl', 'lsr' or 'asr' with optional integer in range [0, 63]");
3675   case Match_InvalidFPImm:
3676     return Error(Loc,
3677                  "expected compatible register or floating-point constant");
3678   case Match_InvalidMemoryIndexedSImm9:
3679     return Error(Loc, "index must be an integer in range [-256, 255].");
3680   case Match_InvalidMemoryIndexed4SImm7:
3681     return Error(Loc, "index must be a multiple of 4 in range [-256, 252].");
3682   case Match_InvalidMemoryIndexed8SImm7:
3683     return Error(Loc, "index must be a multiple of 8 in range [-512, 504].");
3684   case Match_InvalidMemoryIndexed16SImm7:
3685     return Error(Loc, "index must be a multiple of 16 in range [-1024, 1008].");
3686   case Match_InvalidMemoryWExtend8:
3687     return Error(Loc,
3688                  "expected 'uxtw' or 'sxtw' with optional shift of #0");
3689   case Match_InvalidMemoryWExtend16:
3690     return Error(Loc,
3691                  "expected 'uxtw' or 'sxtw' with optional shift of #0 or #1");
3692   case Match_InvalidMemoryWExtend32:
3693     return Error(Loc,
3694                  "expected 'uxtw' or 'sxtw' with optional shift of #0 or #2");
3695   case Match_InvalidMemoryWExtend64:
3696     return Error(Loc,
3697                  "expected 'uxtw' or 'sxtw' with optional shift of #0 or #3");
3698   case Match_InvalidMemoryWExtend128:
3699     return Error(Loc,
3700                  "expected 'uxtw' or 'sxtw' with optional shift of #0 or #4");
3701   case Match_InvalidMemoryXExtend8:
3702     return Error(Loc,
3703                  "expected 'lsl' or 'sxtx' with optional shift of #0");
3704   case Match_InvalidMemoryXExtend16:
3705     return Error(Loc,
3706                  "expected 'lsl' or 'sxtx' with optional shift of #0 or #1");
3707   case Match_InvalidMemoryXExtend32:
3708     return Error(Loc,
3709                  "expected 'lsl' or 'sxtx' with optional shift of #0 or #2");
3710   case Match_InvalidMemoryXExtend64:
3711     return Error(Loc,
3712                  "expected 'lsl' or 'sxtx' with optional shift of #0 or #3");
3713   case Match_InvalidMemoryXExtend128:
3714     return Error(Loc,
3715                  "expected 'lsl' or 'sxtx' with optional shift of #0 or #4");
3716   case Match_InvalidMemoryIndexed1:
3717     return Error(Loc, "index must be an integer in range [0, 4095].");
3718   case Match_InvalidMemoryIndexed2:
3719     return Error(Loc, "index must be a multiple of 2 in range [0, 8190].");
3720   case Match_InvalidMemoryIndexed4:
3721     return Error(Loc, "index must be a multiple of 4 in range [0, 16380].");
3722   case Match_InvalidMemoryIndexed8:
3723     return Error(Loc, "index must be a multiple of 8 in range [0, 32760].");
3724   case Match_InvalidMemoryIndexed16:
3725     return Error(Loc, "index must be a multiple of 16 in range [0, 65520].");
3726   case Match_InvalidImm0_1:
3727     return Error(Loc, "immediate must be an integer in range [0, 1].");
3728   case Match_InvalidImm0_7:
3729     return Error(Loc, "immediate must be an integer in range [0, 7].");
3730   case Match_InvalidImm0_15:
3731     return Error(Loc, "immediate must be an integer in range [0, 15].");
3732   case Match_InvalidImm0_31:
3733     return Error(Loc, "immediate must be an integer in range [0, 31].");
3734   case Match_InvalidImm0_63:
3735     return Error(Loc, "immediate must be an integer in range [0, 63].");
3736   case Match_InvalidImm0_127:
3737     return Error(Loc, "immediate must be an integer in range [0, 127].");
3738   case Match_InvalidImm0_65535:
3739     return Error(Loc, "immediate must be an integer in range [0, 65535].");
3740   case Match_InvalidImm1_8:
3741     return Error(Loc, "immediate must be an integer in range [1, 8].");
3742   case Match_InvalidImm1_16:
3743     return Error(Loc, "immediate must be an integer in range [1, 16].");
3744   case Match_InvalidImm1_32:
3745     return Error(Loc, "immediate must be an integer in range [1, 32].");
3746   case Match_InvalidImm1_64:
3747     return Error(Loc, "immediate must be an integer in range [1, 64].");
3748   case Match_InvalidIndex1:
3749     return Error(Loc, "expected lane specifier '[1]'");
3750   case Match_InvalidIndexB:
3751     return Error(Loc, "vector lane must be an integer in range [0, 15].");
3752   case Match_InvalidIndexH:
3753     return Error(Loc, "vector lane must be an integer in range [0, 7].");
3754   case Match_InvalidIndexS:
3755     return Error(Loc, "vector lane must be an integer in range [0, 3].");
3756   case Match_InvalidIndexD:
3757     return Error(Loc, "vector lane must be an integer in range [0, 1].");
3758   case Match_InvalidLabel:
3759     return Error(Loc, "expected label or encodable integer pc offset");
3760   case Match_MRS:
3761     return Error(Loc, "expected readable system register");
3762   case Match_MSR:
3763     return Error(Loc, "expected writable system register or pstate");
3764   case Match_MnemonicFail:
3765     return Error(Loc, "unrecognized instruction mnemonic");
3766   default:
3767     llvm_unreachable("unexpected error code!");
3768   }
3769 }
3770 
3771 static const char *getSubtargetFeatureName(uint64_t Val);
3772 
MatchAndEmitInstruction(SMLoc IDLoc,unsigned & Opcode,OperandVector & Operands,MCStreamer & Out,uint64_t & ErrorInfo,bool MatchingInlineAsm)3773 bool AArch64AsmParser::MatchAndEmitInstruction(SMLoc IDLoc, unsigned &Opcode,
3774                                                OperandVector &Operands,
3775                                                MCStreamer &Out,
3776                                                uint64_t &ErrorInfo,
3777                                                bool MatchingInlineAsm) {
3778   assert(!Operands.empty() && "Unexpect empty operand list!");
3779   AArch64Operand &Op = static_cast<AArch64Operand &>(*Operands[0]);
3780   assert(Op.isToken() && "Leading operand should always be a mnemonic!");
3781 
3782   StringRef Tok = Op.getToken();
3783   unsigned NumOperands = Operands.size();
3784 
3785   if (NumOperands == 4 && Tok == "lsl") {
3786     AArch64Operand &Op2 = static_cast<AArch64Operand &>(*Operands[2]);
3787     AArch64Operand &Op3 = static_cast<AArch64Operand &>(*Operands[3]);
3788     if (Op2.isReg() && Op3.isImm()) {
3789       const MCConstantExpr *Op3CE = dyn_cast<MCConstantExpr>(Op3.getImm());
3790       if (Op3CE) {
3791         uint64_t Op3Val = Op3CE->getValue();
3792         uint64_t NewOp3Val = 0;
3793         uint64_t NewOp4Val = 0;
3794         if (AArch64MCRegisterClasses[AArch64::GPR32allRegClassID].contains(
3795                 Op2.getReg())) {
3796           NewOp3Val = (32 - Op3Val) & 0x1f;
3797           NewOp4Val = 31 - Op3Val;
3798         } else {
3799           NewOp3Val = (64 - Op3Val) & 0x3f;
3800           NewOp4Val = 63 - Op3Val;
3801         }
3802 
3803         const MCExpr *NewOp3 = MCConstantExpr::create(NewOp3Val, getContext());
3804         const MCExpr *NewOp4 = MCConstantExpr::create(NewOp4Val, getContext());
3805 
3806         Operands[0] = AArch64Operand::CreateToken(
3807             "ubfm", false, Op.getStartLoc(), getContext());
3808         Operands.push_back(AArch64Operand::CreateImm(
3809             NewOp4, Op3.getStartLoc(), Op3.getEndLoc(), getContext()));
3810         Operands[3] = AArch64Operand::CreateImm(NewOp3, Op3.getStartLoc(),
3811                                                 Op3.getEndLoc(), getContext());
3812       }
3813     }
3814   } else if (NumOperands == 4 && Tok == "bfc") {
3815     // FIXME: Horrible hack to handle BFC->BFM alias.
3816     AArch64Operand &Op1 = static_cast<AArch64Operand &>(*Operands[1]);
3817     AArch64Operand LSBOp = static_cast<AArch64Operand &>(*Operands[2]);
3818     AArch64Operand WidthOp = static_cast<AArch64Operand &>(*Operands[3]);
3819 
3820     if (Op1.isReg() && LSBOp.isImm() && WidthOp.isImm()) {
3821       const MCConstantExpr *LSBCE = dyn_cast<MCConstantExpr>(LSBOp.getImm());
3822       const MCConstantExpr *WidthCE = dyn_cast<MCConstantExpr>(WidthOp.getImm());
3823 
3824       if (LSBCE && WidthCE) {
3825         uint64_t LSB = LSBCE->getValue();
3826         uint64_t Width = WidthCE->getValue();
3827 
3828         uint64_t RegWidth = 0;
3829         if (AArch64MCRegisterClasses[AArch64::GPR64allRegClassID].contains(
3830                 Op1.getReg()))
3831           RegWidth = 64;
3832         else
3833           RegWidth = 32;
3834 
3835         if (LSB >= RegWidth)
3836           return Error(LSBOp.getStartLoc(),
3837                        "expected integer in range [0, 31]");
3838         if (Width < 1 || Width > RegWidth)
3839           return Error(WidthOp.getStartLoc(),
3840                        "expected integer in range [1, 32]");
3841 
3842         uint64_t ImmR = 0;
3843         if (RegWidth == 32)
3844           ImmR = (32 - LSB) & 0x1f;
3845         else
3846           ImmR = (64 - LSB) & 0x3f;
3847 
3848         uint64_t ImmS = Width - 1;
3849 
3850         if (ImmR != 0 && ImmS >= ImmR)
3851           return Error(WidthOp.getStartLoc(),
3852                        "requested insert overflows register");
3853 
3854         const MCExpr *ImmRExpr = MCConstantExpr::create(ImmR, getContext());
3855         const MCExpr *ImmSExpr = MCConstantExpr::create(ImmS, getContext());
3856         Operands[0] = AArch64Operand::CreateToken(
3857               "bfm", false, Op.getStartLoc(), getContext());
3858         Operands[2] = AArch64Operand::CreateReg(
3859             RegWidth == 32 ? AArch64::WZR : AArch64::XZR, false, SMLoc(),
3860             SMLoc(), getContext());
3861         Operands[3] = AArch64Operand::CreateImm(
3862             ImmRExpr, LSBOp.getStartLoc(), LSBOp.getEndLoc(), getContext());
3863         Operands.emplace_back(
3864             AArch64Operand::CreateImm(ImmSExpr, WidthOp.getStartLoc(),
3865                                       WidthOp.getEndLoc(), getContext()));
3866       }
3867     }
3868   } else if (NumOperands == 5) {
3869     // FIXME: Horrible hack to handle the BFI -> BFM, SBFIZ->SBFM, and
3870     // UBFIZ -> UBFM aliases.
3871     if (Tok == "bfi" || Tok == "sbfiz" || Tok == "ubfiz") {
3872       AArch64Operand &Op1 = static_cast<AArch64Operand &>(*Operands[1]);
3873       AArch64Operand &Op3 = static_cast<AArch64Operand &>(*Operands[3]);
3874       AArch64Operand &Op4 = static_cast<AArch64Operand &>(*Operands[4]);
3875 
3876       if (Op1.isReg() && Op3.isImm() && Op4.isImm()) {
3877         const MCConstantExpr *Op3CE = dyn_cast<MCConstantExpr>(Op3.getImm());
3878         const MCConstantExpr *Op4CE = dyn_cast<MCConstantExpr>(Op4.getImm());
3879 
3880         if (Op3CE && Op4CE) {
3881           uint64_t Op3Val = Op3CE->getValue();
3882           uint64_t Op4Val = Op4CE->getValue();
3883 
3884           uint64_t RegWidth = 0;
3885           if (AArch64MCRegisterClasses[AArch64::GPR64allRegClassID].contains(
3886                   Op1.getReg()))
3887             RegWidth = 64;
3888           else
3889             RegWidth = 32;
3890 
3891           if (Op3Val >= RegWidth)
3892             return Error(Op3.getStartLoc(),
3893                          "expected integer in range [0, 31]");
3894           if (Op4Val < 1 || Op4Val > RegWidth)
3895             return Error(Op4.getStartLoc(),
3896                          "expected integer in range [1, 32]");
3897 
3898           uint64_t NewOp3Val = 0;
3899           if (RegWidth == 32)
3900             NewOp3Val = (32 - Op3Val) & 0x1f;
3901           else
3902             NewOp3Val = (64 - Op3Val) & 0x3f;
3903 
3904           uint64_t NewOp4Val = Op4Val - 1;
3905 
3906           if (NewOp3Val != 0 && NewOp4Val >= NewOp3Val)
3907             return Error(Op4.getStartLoc(),
3908                          "requested insert overflows register");
3909 
3910           const MCExpr *NewOp3 =
3911               MCConstantExpr::create(NewOp3Val, getContext());
3912           const MCExpr *NewOp4 =
3913               MCConstantExpr::create(NewOp4Val, getContext());
3914           Operands[3] = AArch64Operand::CreateImm(
3915               NewOp3, Op3.getStartLoc(), Op3.getEndLoc(), getContext());
3916           Operands[4] = AArch64Operand::CreateImm(
3917               NewOp4, Op4.getStartLoc(), Op4.getEndLoc(), getContext());
3918           if (Tok == "bfi")
3919             Operands[0] = AArch64Operand::CreateToken(
3920                 "bfm", false, Op.getStartLoc(), getContext());
3921           else if (Tok == "sbfiz")
3922             Operands[0] = AArch64Operand::CreateToken(
3923                 "sbfm", false, Op.getStartLoc(), getContext());
3924           else if (Tok == "ubfiz")
3925             Operands[0] = AArch64Operand::CreateToken(
3926                 "ubfm", false, Op.getStartLoc(), getContext());
3927           else
3928             llvm_unreachable("No valid mnemonic for alias?");
3929         }
3930       }
3931 
3932       // FIXME: Horrible hack to handle the BFXIL->BFM, SBFX->SBFM, and
3933       // UBFX -> UBFM aliases.
3934     } else if (NumOperands == 5 &&
3935                (Tok == "bfxil" || Tok == "sbfx" || Tok == "ubfx")) {
3936       AArch64Operand &Op1 = static_cast<AArch64Operand &>(*Operands[1]);
3937       AArch64Operand &Op3 = static_cast<AArch64Operand &>(*Operands[3]);
3938       AArch64Operand &Op4 = static_cast<AArch64Operand &>(*Operands[4]);
3939 
3940       if (Op1.isReg() && Op3.isImm() && Op4.isImm()) {
3941         const MCConstantExpr *Op3CE = dyn_cast<MCConstantExpr>(Op3.getImm());
3942         const MCConstantExpr *Op4CE = dyn_cast<MCConstantExpr>(Op4.getImm());
3943 
3944         if (Op3CE && Op4CE) {
3945           uint64_t Op3Val = Op3CE->getValue();
3946           uint64_t Op4Val = Op4CE->getValue();
3947 
3948           uint64_t RegWidth = 0;
3949           if (AArch64MCRegisterClasses[AArch64::GPR64allRegClassID].contains(
3950                   Op1.getReg()))
3951             RegWidth = 64;
3952           else
3953             RegWidth = 32;
3954 
3955           if (Op3Val >= RegWidth)
3956             return Error(Op3.getStartLoc(),
3957                          "expected integer in range [0, 31]");
3958           if (Op4Val < 1 || Op4Val > RegWidth)
3959             return Error(Op4.getStartLoc(),
3960                          "expected integer in range [1, 32]");
3961 
3962           uint64_t NewOp4Val = Op3Val + Op4Val - 1;
3963 
3964           if (NewOp4Val >= RegWidth || NewOp4Val < Op3Val)
3965             return Error(Op4.getStartLoc(),
3966                          "requested extract overflows register");
3967 
3968           const MCExpr *NewOp4 =
3969               MCConstantExpr::create(NewOp4Val, getContext());
3970           Operands[4] = AArch64Operand::CreateImm(
3971               NewOp4, Op4.getStartLoc(), Op4.getEndLoc(), getContext());
3972           if (Tok == "bfxil")
3973             Operands[0] = AArch64Operand::CreateToken(
3974                 "bfm", false, Op.getStartLoc(), getContext());
3975           else if (Tok == "sbfx")
3976             Operands[0] = AArch64Operand::CreateToken(
3977                 "sbfm", false, Op.getStartLoc(), getContext());
3978           else if (Tok == "ubfx")
3979             Operands[0] = AArch64Operand::CreateToken(
3980                 "ubfm", false, Op.getStartLoc(), getContext());
3981           else
3982             llvm_unreachable("No valid mnemonic for alias?");
3983         }
3984       }
3985     }
3986   }
3987   // FIXME: Horrible hack for sxtw and uxtw with Wn src and Xd dst operands.
3988   //        InstAlias can't quite handle this since the reg classes aren't
3989   //        subclasses.
3990   if (NumOperands == 3 && (Tok == "sxtw" || Tok == "uxtw")) {
3991     // The source register can be Wn here, but the matcher expects a
3992     // GPR64. Twiddle it here if necessary.
3993     AArch64Operand &Op = static_cast<AArch64Operand &>(*Operands[2]);
3994     if (Op.isReg()) {
3995       unsigned Reg = getXRegFromWReg(Op.getReg());
3996       Operands[2] = AArch64Operand::CreateReg(Reg, false, Op.getStartLoc(),
3997                                               Op.getEndLoc(), getContext());
3998     }
3999   }
4000   // FIXME: Likewise for sxt[bh] with a Xd dst operand
4001   else if (NumOperands == 3 && (Tok == "sxtb" || Tok == "sxth")) {
4002     AArch64Operand &Op = static_cast<AArch64Operand &>(*Operands[1]);
4003     if (Op.isReg() &&
4004         AArch64MCRegisterClasses[AArch64::GPR64allRegClassID].contains(
4005             Op.getReg())) {
4006       // The source register can be Wn here, but the matcher expects a
4007       // GPR64. Twiddle it here if necessary.
4008       AArch64Operand &Op = static_cast<AArch64Operand &>(*Operands[2]);
4009       if (Op.isReg()) {
4010         unsigned Reg = getXRegFromWReg(Op.getReg());
4011         Operands[2] = AArch64Operand::CreateReg(Reg, false, Op.getStartLoc(),
4012                                                 Op.getEndLoc(), getContext());
4013       }
4014     }
4015   }
4016   // FIXME: Likewise for uxt[bh] with a Xd dst operand
4017   else if (NumOperands == 3 && (Tok == "uxtb" || Tok == "uxth")) {
4018     AArch64Operand &Op = static_cast<AArch64Operand &>(*Operands[1]);
4019     if (Op.isReg() &&
4020         AArch64MCRegisterClasses[AArch64::GPR64allRegClassID].contains(
4021             Op.getReg())) {
4022       // The source register can be Wn here, but the matcher expects a
4023       // GPR32. Twiddle it here if necessary.
4024       AArch64Operand &Op = static_cast<AArch64Operand &>(*Operands[1]);
4025       if (Op.isReg()) {
4026         unsigned Reg = getWRegFromXReg(Op.getReg());
4027         Operands[1] = AArch64Operand::CreateReg(Reg, false, Op.getStartLoc(),
4028                                                 Op.getEndLoc(), getContext());
4029       }
4030     }
4031   }
4032 
4033   // Yet another horrible hack to handle FMOV Rd, #0.0 using [WX]ZR.
4034   if (NumOperands == 3 && Tok == "fmov") {
4035     AArch64Operand &RegOp = static_cast<AArch64Operand &>(*Operands[1]);
4036     AArch64Operand &ImmOp = static_cast<AArch64Operand &>(*Operands[2]);
4037     if (RegOp.isReg() && ImmOp.isFPImm() && ImmOp.getFPImm() == (unsigned)-1) {
4038       unsigned zreg =
4039           !AArch64MCRegisterClasses[AArch64::FPR64RegClassID].contains(
4040               RegOp.getReg())
4041               ? AArch64::WZR
4042               : AArch64::XZR;
4043       Operands[2] = AArch64Operand::CreateReg(zreg, false, Op.getStartLoc(),
4044                                               Op.getEndLoc(), getContext());
4045     }
4046   }
4047 
4048   MCInst Inst;
4049   // First try to match against the secondary set of tables containing the
4050   // short-form NEON instructions (e.g. "fadd.2s v0, v1, v2").
4051   unsigned MatchResult =
4052       MatchInstructionImpl(Operands, Inst, ErrorInfo, MatchingInlineAsm, 1);
4053 
4054   // If that fails, try against the alternate table containing long-form NEON:
4055   // "fadd v0.2s, v1.2s, v2.2s"
4056   if (MatchResult != Match_Success) {
4057     // But first, save the short-form match result: we can use it in case the
4058     // long-form match also fails.
4059     auto ShortFormNEONErrorInfo = ErrorInfo;
4060     auto ShortFormNEONMatchResult = MatchResult;
4061 
4062     MatchResult =
4063         MatchInstructionImpl(Operands, Inst, ErrorInfo, MatchingInlineAsm, 0);
4064 
4065     // Now, both matches failed, and the long-form match failed on the mnemonic
4066     // suffix token operand.  The short-form match failure is probably more
4067     // relevant: use it instead.
4068     if (MatchResult == Match_InvalidOperand && ErrorInfo == 1 &&
4069         Operands.size() > 1 && ((AArch64Operand &)*Operands[1]).isToken() &&
4070         ((AArch64Operand &)*Operands[1]).isTokenSuffix()) {
4071       MatchResult = ShortFormNEONMatchResult;
4072       ErrorInfo = ShortFormNEONErrorInfo;
4073     }
4074   }
4075 
4076 
4077   switch (MatchResult) {
4078   case Match_Success: {
4079     // Perform range checking and other semantic validations
4080     SmallVector<SMLoc, 8> OperandLocs;
4081     NumOperands = Operands.size();
4082     for (unsigned i = 1; i < NumOperands; ++i)
4083       OperandLocs.push_back(Operands[i]->getStartLoc());
4084     if (validateInstruction(Inst, OperandLocs))
4085       return true;
4086 
4087     Inst.setLoc(IDLoc);
4088     Out.EmitInstruction(Inst, getSTI());
4089     return false;
4090   }
4091   case Match_MissingFeature: {
4092     assert(ErrorInfo && "Unknown missing feature!");
4093     // Special case the error message for the very common case where only
4094     // a single subtarget feature is missing (neon, e.g.).
4095     std::string Msg = "instruction requires:";
4096     uint64_t Mask = 1;
4097     for (unsigned i = 0; i < (sizeof(ErrorInfo)*8-1); ++i) {
4098       if (ErrorInfo & Mask) {
4099         Msg += " ";
4100         Msg += getSubtargetFeatureName(ErrorInfo & Mask);
4101       }
4102       Mask <<= 1;
4103     }
4104     return Error(IDLoc, Msg);
4105   }
4106   case Match_MnemonicFail:
4107     return showMatchError(IDLoc, MatchResult);
4108   case Match_InvalidOperand: {
4109     SMLoc ErrorLoc = IDLoc;
4110 
4111     if (ErrorInfo != ~0ULL) {
4112       if (ErrorInfo >= Operands.size())
4113         return Error(IDLoc, "too few operands for instruction");
4114 
4115       ErrorLoc = ((AArch64Operand &)*Operands[ErrorInfo]).getStartLoc();
4116       if (ErrorLoc == SMLoc())
4117         ErrorLoc = IDLoc;
4118     }
4119     // If the match failed on a suffix token operand, tweak the diagnostic
4120     // accordingly.
4121     if (((AArch64Operand &)*Operands[ErrorInfo]).isToken() &&
4122         ((AArch64Operand &)*Operands[ErrorInfo]).isTokenSuffix())
4123       MatchResult = Match_InvalidSuffix;
4124 
4125     return showMatchError(ErrorLoc, MatchResult);
4126   }
4127   case Match_InvalidMemoryIndexed1:
4128   case Match_InvalidMemoryIndexed2:
4129   case Match_InvalidMemoryIndexed4:
4130   case Match_InvalidMemoryIndexed8:
4131   case Match_InvalidMemoryIndexed16:
4132   case Match_InvalidCondCode:
4133   case Match_AddSubRegExtendSmall:
4134   case Match_AddSubRegExtendLarge:
4135   case Match_AddSubSecondSource:
4136   case Match_LogicalSecondSource:
4137   case Match_AddSubRegShift32:
4138   case Match_AddSubRegShift64:
4139   case Match_InvalidMovImm32Shift:
4140   case Match_InvalidMovImm64Shift:
4141   case Match_InvalidFPImm:
4142   case Match_InvalidMemoryWExtend8:
4143   case Match_InvalidMemoryWExtend16:
4144   case Match_InvalidMemoryWExtend32:
4145   case Match_InvalidMemoryWExtend64:
4146   case Match_InvalidMemoryWExtend128:
4147   case Match_InvalidMemoryXExtend8:
4148   case Match_InvalidMemoryXExtend16:
4149   case Match_InvalidMemoryXExtend32:
4150   case Match_InvalidMemoryXExtend64:
4151   case Match_InvalidMemoryXExtend128:
4152   case Match_InvalidMemoryIndexed4SImm7:
4153   case Match_InvalidMemoryIndexed8SImm7:
4154   case Match_InvalidMemoryIndexed16SImm7:
4155   case Match_InvalidMemoryIndexedSImm9:
4156   case Match_InvalidImm0_1:
4157   case Match_InvalidImm0_7:
4158   case Match_InvalidImm0_15:
4159   case Match_InvalidImm0_31:
4160   case Match_InvalidImm0_63:
4161   case Match_InvalidImm0_127:
4162   case Match_InvalidImm0_65535:
4163   case Match_InvalidImm1_8:
4164   case Match_InvalidImm1_16:
4165   case Match_InvalidImm1_32:
4166   case Match_InvalidImm1_64:
4167   case Match_InvalidIndex1:
4168   case Match_InvalidIndexB:
4169   case Match_InvalidIndexH:
4170   case Match_InvalidIndexS:
4171   case Match_InvalidIndexD:
4172   case Match_InvalidLabel:
4173   case Match_MSR:
4174   case Match_MRS: {
4175     if (ErrorInfo >= Operands.size())
4176       return Error(IDLoc, "too few operands for instruction");
4177     // Any time we get here, there's nothing fancy to do. Just get the
4178     // operand SMLoc and display the diagnostic.
4179     SMLoc ErrorLoc = ((AArch64Operand &)*Operands[ErrorInfo]).getStartLoc();
4180     if (ErrorLoc == SMLoc())
4181       ErrorLoc = IDLoc;
4182     return showMatchError(ErrorLoc, MatchResult);
4183   }
4184   }
4185 
4186   llvm_unreachable("Implement any new match types added!");
4187 }
4188 
4189 /// ParseDirective parses the arm specific directives
ParseDirective(AsmToken DirectiveID)4190 bool AArch64AsmParser::ParseDirective(AsmToken DirectiveID) {
4191   const MCObjectFileInfo::Environment Format =
4192     getContext().getObjectFileInfo()->getObjectFileType();
4193   bool IsMachO = Format == MCObjectFileInfo::IsMachO;
4194   bool IsCOFF = Format == MCObjectFileInfo::IsCOFF;
4195 
4196   StringRef IDVal = DirectiveID.getIdentifier();
4197   SMLoc Loc = DirectiveID.getLoc();
4198   if (IDVal == ".hword")
4199     return parseDirectiveWord(2, Loc);
4200   if (IDVal == ".word")
4201     return parseDirectiveWord(4, Loc);
4202   if (IDVal == ".xword")
4203     return parseDirectiveWord(8, Loc);
4204   if (IDVal == ".tlsdesccall")
4205     return parseDirectiveTLSDescCall(Loc);
4206   if (IDVal == ".ltorg" || IDVal == ".pool")
4207     return parseDirectiveLtorg(Loc);
4208   if (IDVal == ".unreq")
4209     return parseDirectiveUnreq(Loc);
4210 
4211   if (!IsMachO && !IsCOFF) {
4212     if (IDVal == ".inst")
4213       return parseDirectiveInst(Loc);
4214   }
4215 
4216   return parseDirectiveLOH(IDVal, Loc);
4217 }
4218 
4219 /// parseDirectiveWord
4220 ///  ::= .word [ expression (, expression)* ]
parseDirectiveWord(unsigned Size,SMLoc L)4221 bool AArch64AsmParser::parseDirectiveWord(unsigned Size, SMLoc L) {
4222   MCAsmParser &Parser = getParser();
4223   if (getLexer().isNot(AsmToken::EndOfStatement)) {
4224     for (;;) {
4225       const MCExpr *Value;
4226       if (getParser().parseExpression(Value))
4227         return true;
4228 
4229       getParser().getStreamer().EmitValue(Value, Size, L);
4230 
4231       if (getLexer().is(AsmToken::EndOfStatement))
4232         break;
4233 
4234       // FIXME: Improve diagnostic.
4235       if (getLexer().isNot(AsmToken::Comma))
4236         return Error(L, "unexpected token in directive");
4237       Parser.Lex();
4238     }
4239   }
4240 
4241   Parser.Lex();
4242   return false;
4243 }
4244 
4245 /// parseDirectiveInst
4246 ///  ::= .inst opcode [, ...]
parseDirectiveInst(SMLoc Loc)4247 bool AArch64AsmParser::parseDirectiveInst(SMLoc Loc) {
4248   MCAsmParser &Parser = getParser();
4249   if (getLexer().is(AsmToken::EndOfStatement)) {
4250     Parser.eatToEndOfStatement();
4251     Error(Loc, "expected expression following directive");
4252     return false;
4253   }
4254 
4255   for (;;) {
4256     const MCExpr *Expr;
4257 
4258     if (getParser().parseExpression(Expr)) {
4259       Error(Loc, "expected expression");
4260       return false;
4261     }
4262 
4263     const MCConstantExpr *Value = dyn_cast_or_null<MCConstantExpr>(Expr);
4264     if (!Value) {
4265       Error(Loc, "expected constant expression");
4266       return false;
4267     }
4268 
4269     getTargetStreamer().emitInst(Value->getValue());
4270 
4271     if (getLexer().is(AsmToken::EndOfStatement))
4272       break;
4273 
4274     if (getLexer().isNot(AsmToken::Comma)) {
4275       Error(Loc, "unexpected token in directive");
4276       return false;
4277     }
4278 
4279     Parser.Lex(); // Eat comma.
4280   }
4281 
4282   Parser.Lex();
4283   return false;
4284 }
4285 
4286 // parseDirectiveTLSDescCall:
4287 //   ::= .tlsdesccall symbol
parseDirectiveTLSDescCall(SMLoc L)4288 bool AArch64AsmParser::parseDirectiveTLSDescCall(SMLoc L) {
4289   StringRef Name;
4290   if (getParser().parseIdentifier(Name))
4291     return Error(L, "expected symbol after directive");
4292 
4293   MCSymbol *Sym = getContext().getOrCreateSymbol(Name);
4294   const MCExpr *Expr = MCSymbolRefExpr::create(Sym, getContext());
4295   Expr = AArch64MCExpr::create(Expr, AArch64MCExpr::VK_TLSDESC, getContext());
4296 
4297   MCInst Inst;
4298   Inst.setOpcode(AArch64::TLSDESCCALL);
4299   Inst.addOperand(MCOperand::createExpr(Expr));
4300 
4301   getParser().getStreamer().EmitInstruction(Inst, getSTI());
4302   return false;
4303 }
4304 
4305 /// ::= .loh <lohName | lohId> label1, ..., labelN
4306 /// The number of arguments depends on the loh identifier.
parseDirectiveLOH(StringRef IDVal,SMLoc Loc)4307 bool AArch64AsmParser::parseDirectiveLOH(StringRef IDVal, SMLoc Loc) {
4308   if (IDVal != MCLOHDirectiveName())
4309     return true;
4310   MCLOHType Kind;
4311   if (getParser().getTok().isNot(AsmToken::Identifier)) {
4312     if (getParser().getTok().isNot(AsmToken::Integer))
4313       return TokError("expected an identifier or a number in directive");
4314     // We successfully get a numeric value for the identifier.
4315     // Check if it is valid.
4316     int64_t Id = getParser().getTok().getIntVal();
4317     if (Id <= -1U && !isValidMCLOHType(Id))
4318       return TokError("invalid numeric identifier in directive");
4319     Kind = (MCLOHType)Id;
4320   } else {
4321     StringRef Name = getTok().getIdentifier();
4322     // We successfully parse an identifier.
4323     // Check if it is a recognized one.
4324     int Id = MCLOHNameToId(Name);
4325 
4326     if (Id == -1)
4327       return TokError("invalid identifier in directive");
4328     Kind = (MCLOHType)Id;
4329   }
4330   // Consume the identifier.
4331   Lex();
4332   // Get the number of arguments of this LOH.
4333   int NbArgs = MCLOHIdToNbArgs(Kind);
4334 
4335   assert(NbArgs != -1 && "Invalid number of arguments");
4336 
4337   SmallVector<MCSymbol *, 3> Args;
4338   for (int Idx = 0; Idx < NbArgs; ++Idx) {
4339     StringRef Name;
4340     if (getParser().parseIdentifier(Name))
4341       return TokError("expected identifier in directive");
4342     Args.push_back(getContext().getOrCreateSymbol(Name));
4343 
4344     if (Idx + 1 == NbArgs)
4345       break;
4346     if (getLexer().isNot(AsmToken::Comma))
4347       return TokError("unexpected token in '" + Twine(IDVal) + "' directive");
4348     Lex();
4349   }
4350   if (getLexer().isNot(AsmToken::EndOfStatement))
4351     return TokError("unexpected token in '" + Twine(IDVal) + "' directive");
4352 
4353   getStreamer().EmitLOHDirective((MCLOHType)Kind, Args);
4354   return false;
4355 }
4356 
4357 /// parseDirectiveLtorg
4358 ///  ::= .ltorg | .pool
parseDirectiveLtorg(SMLoc L)4359 bool AArch64AsmParser::parseDirectiveLtorg(SMLoc L) {
4360   getTargetStreamer().emitCurrentConstantPool();
4361   return false;
4362 }
4363 
4364 /// parseDirectiveReq
4365 ///  ::= name .req registername
parseDirectiveReq(StringRef Name,SMLoc L)4366 bool AArch64AsmParser::parseDirectiveReq(StringRef Name, SMLoc L) {
4367   MCAsmParser &Parser = getParser();
4368   Parser.Lex(); // Eat the '.req' token.
4369   SMLoc SRegLoc = getLoc();
4370   unsigned RegNum = tryParseRegister();
4371   bool IsVector = false;
4372 
4373   if (RegNum == static_cast<unsigned>(-1)) {
4374     StringRef Kind;
4375     RegNum = tryMatchVectorRegister(Kind, false);
4376     if (!Kind.empty()) {
4377       Error(SRegLoc, "vector register without type specifier expected");
4378       return false;
4379     }
4380     IsVector = true;
4381   }
4382 
4383   if (RegNum == static_cast<unsigned>(-1)) {
4384     Parser.eatToEndOfStatement();
4385     Error(SRegLoc, "register name or alias expected");
4386     return false;
4387   }
4388 
4389   // Shouldn't be anything else.
4390   if (Parser.getTok().isNot(AsmToken::EndOfStatement)) {
4391     Error(Parser.getTok().getLoc(), "unexpected input in .req directive");
4392     Parser.eatToEndOfStatement();
4393     return false;
4394   }
4395 
4396   Parser.Lex(); // Consume the EndOfStatement
4397 
4398   auto pair = std::make_pair(IsVector, RegNum);
4399   if (RegisterReqs.insert(std::make_pair(Name, pair)).first->second != pair)
4400     Warning(L, "ignoring redefinition of register alias '" + Name + "'");
4401 
4402   return true;
4403 }
4404 
4405 /// parseDirectiveUneq
4406 ///  ::= .unreq registername
parseDirectiveUnreq(SMLoc L)4407 bool AArch64AsmParser::parseDirectiveUnreq(SMLoc L) {
4408   MCAsmParser &Parser = getParser();
4409   if (Parser.getTok().isNot(AsmToken::Identifier)) {
4410     Error(Parser.getTok().getLoc(), "unexpected input in .unreq directive.");
4411     Parser.eatToEndOfStatement();
4412     return false;
4413   }
4414   RegisterReqs.erase(Parser.getTok().getIdentifier().lower());
4415   Parser.Lex(); // Eat the identifier.
4416   return false;
4417 }
4418 
4419 bool
classifySymbolRef(const MCExpr * Expr,AArch64MCExpr::VariantKind & ELFRefKind,MCSymbolRefExpr::VariantKind & DarwinRefKind,int64_t & Addend)4420 AArch64AsmParser::classifySymbolRef(const MCExpr *Expr,
4421                                     AArch64MCExpr::VariantKind &ELFRefKind,
4422                                     MCSymbolRefExpr::VariantKind &DarwinRefKind,
4423                                     int64_t &Addend) {
4424   ELFRefKind = AArch64MCExpr::VK_INVALID;
4425   DarwinRefKind = MCSymbolRefExpr::VK_None;
4426   Addend = 0;
4427 
4428   if (const AArch64MCExpr *AE = dyn_cast<AArch64MCExpr>(Expr)) {
4429     ELFRefKind = AE->getKind();
4430     Expr = AE->getSubExpr();
4431   }
4432 
4433   const MCSymbolRefExpr *SE = dyn_cast<MCSymbolRefExpr>(Expr);
4434   if (SE) {
4435     // It's a simple symbol reference with no addend.
4436     DarwinRefKind = SE->getKind();
4437     return true;
4438   }
4439 
4440   const MCBinaryExpr *BE = dyn_cast<MCBinaryExpr>(Expr);
4441   if (!BE)
4442     return false;
4443 
4444   SE = dyn_cast<MCSymbolRefExpr>(BE->getLHS());
4445   if (!SE)
4446     return false;
4447   DarwinRefKind = SE->getKind();
4448 
4449   if (BE->getOpcode() != MCBinaryExpr::Add &&
4450       BE->getOpcode() != MCBinaryExpr::Sub)
4451     return false;
4452 
4453   // See if the addend is is a constant, otherwise there's more going
4454   // on here than we can deal with.
4455   auto AddendExpr = dyn_cast<MCConstantExpr>(BE->getRHS());
4456   if (!AddendExpr)
4457     return false;
4458 
4459   Addend = AddendExpr->getValue();
4460   if (BE->getOpcode() == MCBinaryExpr::Sub)
4461     Addend = -Addend;
4462 
4463   // It's some symbol reference + a constant addend, but really
4464   // shouldn't use both Darwin and ELF syntax.
4465   return ELFRefKind == AArch64MCExpr::VK_INVALID ||
4466          DarwinRefKind == MCSymbolRefExpr::VK_None;
4467 }
4468 
4469 /// Force static initialization.
LLVMInitializeAArch64AsmParser()4470 extern "C" void LLVMInitializeAArch64AsmParser() {
4471   RegisterMCAsmParser<AArch64AsmParser> X(TheAArch64leTarget);
4472   RegisterMCAsmParser<AArch64AsmParser> Y(TheAArch64beTarget);
4473   RegisterMCAsmParser<AArch64AsmParser> Z(TheARM64Target);
4474 }
4475 
4476 #define GET_REGISTER_MATCHER
4477 #define GET_SUBTARGET_FEATURE_NAME
4478 #define GET_MATCHER_IMPLEMENTATION
4479 #include "AArch64GenAsmMatcher.inc"
4480 
4481 // Define this matcher function after the auto-generated include so we
4482 // have the match class enum definitions.
validateTargetOperandClass(MCParsedAsmOperand & AsmOp,unsigned Kind)4483 unsigned AArch64AsmParser::validateTargetOperandClass(MCParsedAsmOperand &AsmOp,
4484                                                       unsigned Kind) {
4485   AArch64Operand &Op = static_cast<AArch64Operand &>(AsmOp);
4486   // If the kind is a token for a literal immediate, check if our asm
4487   // operand matches. This is for InstAliases which have a fixed-value
4488   // immediate in the syntax.
4489   int64_t ExpectedVal;
4490   switch (Kind) {
4491   default:
4492     return Match_InvalidOperand;
4493   case MCK__35_0:
4494     ExpectedVal = 0;
4495     break;
4496   case MCK__35_1:
4497     ExpectedVal = 1;
4498     break;
4499   case MCK__35_12:
4500     ExpectedVal = 12;
4501     break;
4502   case MCK__35_16:
4503     ExpectedVal = 16;
4504     break;
4505   case MCK__35_2:
4506     ExpectedVal = 2;
4507     break;
4508   case MCK__35_24:
4509     ExpectedVal = 24;
4510     break;
4511   case MCK__35_3:
4512     ExpectedVal = 3;
4513     break;
4514   case MCK__35_32:
4515     ExpectedVal = 32;
4516     break;
4517   case MCK__35_4:
4518     ExpectedVal = 4;
4519     break;
4520   case MCK__35_48:
4521     ExpectedVal = 48;
4522     break;
4523   case MCK__35_6:
4524     ExpectedVal = 6;
4525     break;
4526   case MCK__35_64:
4527     ExpectedVal = 64;
4528     break;
4529   case MCK__35_8:
4530     ExpectedVal = 8;
4531     break;
4532   }
4533   if (!Op.isImm())
4534     return Match_InvalidOperand;
4535   const MCConstantExpr *CE = dyn_cast<MCConstantExpr>(Op.getImm());
4536   if (!CE)
4537     return Match_InvalidOperand;
4538   if (CE->getValue() == ExpectedVal)
4539     return Match_Success;
4540   return Match_InvalidOperand;
4541 }
4542 
4543 
4544 AArch64AsmParser::OperandMatchResultTy
tryParseGPRSeqPair(OperandVector & Operands)4545 AArch64AsmParser::tryParseGPRSeqPair(OperandVector &Operands) {
4546 
4547   SMLoc S = getLoc();
4548 
4549   if (getParser().getTok().isNot(AsmToken::Identifier)) {
4550     Error(S, "expected register");
4551     return MatchOperand_ParseFail;
4552   }
4553 
4554   int FirstReg = tryParseRegister();
4555   if (FirstReg == -1) {
4556     return MatchOperand_ParseFail;
4557   }
4558   const MCRegisterClass &WRegClass =
4559       AArch64MCRegisterClasses[AArch64::GPR32RegClassID];
4560   const MCRegisterClass &XRegClass =
4561       AArch64MCRegisterClasses[AArch64::GPR64RegClassID];
4562 
4563   bool isXReg = XRegClass.contains(FirstReg),
4564        isWReg = WRegClass.contains(FirstReg);
4565   if (!isXReg && !isWReg) {
4566     Error(S, "expected first even register of a "
4567              "consecutive same-size even/odd register pair");
4568     return MatchOperand_ParseFail;
4569   }
4570 
4571   const MCRegisterInfo *RI = getContext().getRegisterInfo();
4572   unsigned FirstEncoding = RI->getEncodingValue(FirstReg);
4573 
4574   if (FirstEncoding & 0x1) {
4575     Error(S, "expected first even register of a "
4576              "consecutive same-size even/odd register pair");
4577     return MatchOperand_ParseFail;
4578   }
4579 
4580   SMLoc M = getLoc();
4581   if (getParser().getTok().isNot(AsmToken::Comma)) {
4582     Error(M, "expected comma");
4583     return MatchOperand_ParseFail;
4584   }
4585   // Eat the comma
4586   getParser().Lex();
4587 
4588   SMLoc E = getLoc();
4589   int SecondReg = tryParseRegister();
4590   if (SecondReg ==-1) {
4591     return MatchOperand_ParseFail;
4592   }
4593 
4594  if (RI->getEncodingValue(SecondReg) != FirstEncoding + 1 ||
4595       (isXReg && !XRegClass.contains(SecondReg)) ||
4596       (isWReg && !WRegClass.contains(SecondReg))) {
4597     Error(E,"expected second odd register of a "
4598              "consecutive same-size even/odd register pair");
4599     return MatchOperand_ParseFail;
4600   }
4601 
4602   unsigned Pair = 0;
4603   if(isXReg) {
4604     Pair = RI->getMatchingSuperReg(FirstReg, AArch64::sube64,
4605            &AArch64MCRegisterClasses[AArch64::XSeqPairsClassRegClassID]);
4606   } else {
4607     Pair = RI->getMatchingSuperReg(FirstReg, AArch64::sube32,
4608            &AArch64MCRegisterClasses[AArch64::WSeqPairsClassRegClassID]);
4609   }
4610 
4611   Operands.push_back(AArch64Operand::CreateReg(Pair, false, S, getLoc(),
4612       getContext()));
4613 
4614   return MatchOperand_Success;
4615 }
4616