xref: /freebsd/contrib/llvm-project/llvm/lib/Target/M68k/AsmParser/M68kAsmParser.cpp (revision 357378bbdedf24ce2b90e9bd831af4a9db3ec70a)
1 //===-- M68kAsmParser.cpp - Parse M68k assembly to MCInst instructions ----===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #include "M68kInstrInfo.h"
10 #include "M68kRegisterInfo.h"
11 #include "TargetInfo/M68kTargetInfo.h"
12 
13 #include "llvm/MC/MCContext.h"
14 #include "llvm/MC/MCParser/MCAsmLexer.h"
15 #include "llvm/MC/MCParser/MCParsedAsmOperand.h"
16 #include "llvm/MC/MCParser/MCTargetAsmParser.h"
17 #include "llvm/MC/MCStreamer.h"
18 #include "llvm/MC/TargetRegistry.h"
19 
20 #include <sstream>
21 
22 #define DEBUG_TYPE "m68k-asm-parser"
23 
24 using namespace llvm;
25 
26 static cl::opt<bool> RegisterPrefixOptional(
27     "m68k-register-prefix-optional", cl::Hidden,
28     cl::desc("Enable specifying registers without the % prefix"),
29     cl::init(false));
30 
31 namespace {
32 /// Parses M68k assembly from a stream.
33 class M68kAsmParser : public MCTargetAsmParser {
34   const MCSubtargetInfo &STI;
35   MCAsmParser &Parser;
36   const MCRegisterInfo *MRI;
37 
38 #define GET_ASSEMBLER_HEADER
39 #include "M68kGenAsmMatcher.inc"
40 
41   // Helpers for Match&Emit.
42   bool invalidOperand(const SMLoc &Loc, const OperandVector &Operands,
43                       const uint64_t &ErrorInfo);
44   bool missingFeature(const SMLoc &Loc, const uint64_t &ErrorInfo);
45   bool emit(MCInst &Inst, SMLoc const &Loc, MCStreamer &Out) const;
46   bool parseRegisterName(MCRegister &RegNo, SMLoc Loc, StringRef RegisterName);
47   ParseStatus parseRegister(MCRegister &RegNo);
48 
49   // Parser functions.
50   void eatComma();
51 
52   bool isExpr();
53   ParseStatus parseImm(OperandVector &Operands);
54   ParseStatus parseMemOp(OperandVector &Operands);
55   ParseStatus parseRegOrMoveMask(OperandVector &Operands);
56 
57 public:
58   M68kAsmParser(const MCSubtargetInfo &STI, MCAsmParser &Parser,
59                 const MCInstrInfo &MII, const MCTargetOptions &Options)
60       : MCTargetAsmParser(Options, STI, MII), STI(STI), Parser(Parser) {
61     MCAsmParserExtension::Initialize(Parser);
62     MRI = getContext().getRegisterInfo();
63 
64     setAvailableFeatures(ComputeAvailableFeatures(STI.getFeatureBits()));
65   }
66 
67   unsigned validateTargetOperandClass(MCParsedAsmOperand &Op,
68                                       unsigned Kind) override;
69   bool parseRegister(MCRegister &Reg, SMLoc &StartLoc, SMLoc &EndLoc) override;
70   ParseStatus tryParseRegister(MCRegister &Reg, SMLoc &StartLoc,
71                                SMLoc &EndLoc) override;
72   bool ParseInstruction(ParseInstructionInfo &Info, StringRef Name,
73                         SMLoc NameLoc, OperandVector &Operands) override;
74   bool MatchAndEmitInstruction(SMLoc IDLoc, unsigned &Opcode,
75                                OperandVector &Operands, MCStreamer &Out,
76                                uint64_t &ErrorInfo,
77                                bool MatchingInlineAsm) override;
78 };
79 
80 struct M68kMemOp {
81   enum class Kind {
82     Addr,
83     RegMask,
84     Reg,
85     RegIndirect,
86     RegPostIncrement,
87     RegPreDecrement,
88     RegIndirectDisplacement,
89     RegIndirectDisplacementIndex,
90   };
91 
92   // These variables are used for the following forms:
93   // Addr: (OuterDisp)
94   // RegMask: RegMask (as register mask)
95   // Reg: %OuterReg
96   // RegIndirect: (%OuterReg)
97   // RegPostIncrement: (%OuterReg)+
98   // RegPreDecrement: -(%OuterReg)
99   // RegIndirectDisplacement: OuterDisp(%OuterReg)
100   // RegIndirectDisplacementIndex:
101   //   OuterDisp(%OuterReg, %InnerReg.Size * Scale, InnerDisp)
102 
103   Kind Op;
104   MCRegister OuterReg;
105   MCRegister InnerReg;
106   const MCExpr *OuterDisp;
107   const MCExpr *InnerDisp;
108   uint8_t Size : 4;
109   uint8_t Scale : 4;
110   const MCExpr *Expr;
111   uint16_t RegMask;
112 
113   M68kMemOp() {}
114   M68kMemOp(Kind Op) : Op(Op) {}
115 
116   void print(raw_ostream &OS) const;
117 };
118 
119 /// An parsed M68k assembly operand.
120 class M68kOperand : public MCParsedAsmOperand {
121   typedef MCParsedAsmOperand Base;
122 
123   enum class KindTy {
124     Invalid,
125     Token,
126     Imm,
127     MemOp,
128   };
129 
130   KindTy Kind;
131   SMLoc Start, End;
132   union {
133     StringRef Token;
134     const MCExpr *Expr;
135     M68kMemOp MemOp;
136   };
137 
138   template <unsigned N> bool isAddrN() const;
139 
140 public:
141   M68kOperand(KindTy Kind, SMLoc Start, SMLoc End)
142       : Base(), Kind(Kind), Start(Start), End(End) {}
143 
144   SMLoc getStartLoc() const override { return Start; }
145   SMLoc getEndLoc() const override { return End; }
146 
147   void print(raw_ostream &OS) const override;
148 
149   bool isMem() const override { return false; }
150   bool isMemOp() const { return Kind == KindTy::MemOp; }
151 
152   static void addExpr(MCInst &Inst, const MCExpr *Expr);
153 
154   // Reg
155   bool isReg() const override;
156   bool isAReg() const;
157   bool isDReg() const;
158   bool isFPDReg() const;
159   bool isFPCReg() const;
160   unsigned getReg() const override;
161   void addRegOperands(MCInst &Inst, unsigned N) const;
162 
163   static std::unique_ptr<M68kOperand> createMemOp(M68kMemOp MemOp, SMLoc Start,
164                                                   SMLoc End);
165 
166   // Token
167   bool isToken() const override;
168   StringRef getToken() const;
169   static std::unique_ptr<M68kOperand> createToken(StringRef Token, SMLoc Start,
170                                                   SMLoc End);
171 
172   // Imm
173   bool isImm() const override;
174   void addImmOperands(MCInst &Inst, unsigned N) const;
175 
176   static std::unique_ptr<M68kOperand> createImm(const MCExpr *Expr, SMLoc Start,
177                                                 SMLoc End);
178 
179   // Imm for TRAP instruction
180   bool isTrapImm() const;
181   // Imm for BKPT instruction
182   bool isBkptImm() const;
183 
184   // MoveMask
185   bool isMoveMask() const;
186   void addMoveMaskOperands(MCInst &Inst, unsigned N) const;
187 
188   // Addr
189   bool isAddr() const;
190   bool isAddr8() const { return isAddrN<8>(); }
191   bool isAddr16() const { return isAddrN<16>(); }
192   bool isAddr32() const { return isAddrN<32>(); }
193   void addAddrOperands(MCInst &Inst, unsigned N) const;
194 
195   // ARI
196   bool isARI() const;
197   void addARIOperands(MCInst &Inst, unsigned N) const;
198 
199   // ARID
200   bool isARID() const;
201   void addARIDOperands(MCInst &Inst, unsigned N) const;
202 
203   // ARII
204   bool isARII() const;
205   void addARIIOperands(MCInst &Inst, unsigned N) const;
206 
207   // ARIPD
208   bool isARIPD() const;
209   void addARIPDOperands(MCInst &Inst, unsigned N) const;
210 
211   // ARIPI
212   bool isARIPI() const;
213   void addARIPIOperands(MCInst &Inst, unsigned N) const;
214 
215   // PCD
216   bool isPCD() const;
217   void addPCDOperands(MCInst &Inst, unsigned N) const;
218 
219   // PCI
220   bool isPCI() const;
221   void addPCIOperands(MCInst &Inst, unsigned N) const;
222 };
223 
224 } // end anonymous namespace.
225 
226 extern "C" LLVM_EXTERNAL_VISIBILITY void LLVMInitializeM68kAsmParser() {
227   RegisterMCAsmParser<M68kAsmParser> X(getTheM68kTarget());
228 }
229 
230 #define GET_REGISTER_MATCHER
231 #define GET_MATCHER_IMPLEMENTATION
232 #include "M68kGenAsmMatcher.inc"
233 
234 static inline unsigned getRegisterByIndex(unsigned RegisterIndex) {
235   static unsigned RegistersByIndex[] = {
236       M68k::D0,  M68k::D1,  M68k::D2,  M68k::D3,  M68k::D4,  M68k::D5,
237       M68k::D6,  M68k::D7,  M68k::A0,  M68k::A1,  M68k::A2,  M68k::A3,
238       M68k::A4,  M68k::A5,  M68k::A6,  M68k::SP,  M68k::FP0, M68k::FP1,
239       M68k::FP2, M68k::FP3, M68k::FP4, M68k::FP5, M68k::FP6, M68k::FP7};
240   assert(RegisterIndex <=
241          sizeof(RegistersByIndex) / sizeof(RegistersByIndex[0]));
242   return RegistersByIndex[RegisterIndex];
243 }
244 
245 static inline unsigned getRegisterIndex(unsigned Register) {
246   if (Register >= M68k::D0 && Register <= M68k::D7)
247     return Register - M68k::D0;
248   if (Register >= M68k::A0 && Register <= M68k::A6)
249     return Register - M68k::A0 + 8;
250   if (Register >= M68k::FP0 && Register <= M68k::FP7)
251     return Register - M68k::FP0 + 16;
252 
253   switch (Register) {
254   case M68k::SP:
255     // SP is sadly not contiguous with the rest of the An registers
256     return 15;
257 
258   // We don't care about the indices of these registers.
259   case M68k::PC:
260   case M68k::CCR:
261   case M68k::FPC:
262   case M68k::FPS:
263   case M68k::FPIAR:
264     return UINT_MAX;
265 
266   default:
267     llvm_unreachable("unexpected register number");
268   }
269 }
270 
271 void M68kMemOp::print(raw_ostream &OS) const {
272   switch (Op) {
273   case Kind::Addr:
274     OS << OuterDisp;
275     break;
276   case Kind::RegMask:
277     OS << "RegMask(" << format("%04x", RegMask) << ")";
278     break;
279   case Kind::Reg:
280     OS << '%' << OuterReg;
281     break;
282   case Kind::RegIndirect:
283     OS << "(%" << OuterReg << ')';
284     break;
285   case Kind::RegPostIncrement:
286     OS << "(%" << OuterReg << ")+";
287     break;
288   case Kind::RegPreDecrement:
289     OS << "-(%" << OuterReg << ")";
290     break;
291   case Kind::RegIndirectDisplacement:
292     OS << OuterDisp << "(%" << OuterReg << ")";
293     break;
294   case Kind::RegIndirectDisplacementIndex:
295     OS << OuterDisp << "(%" << OuterReg << ", " << InnerReg << "." << Size
296        << ", " << InnerDisp << ")";
297     break;
298   }
299 }
300 
301 void M68kOperand::addExpr(MCInst &Inst, const MCExpr *Expr) {
302   if (auto Const = dyn_cast<MCConstantExpr>(Expr)) {
303     Inst.addOperand(MCOperand::createImm(Const->getValue()));
304     return;
305   }
306 
307   Inst.addOperand(MCOperand::createExpr(Expr));
308 }
309 
310 // Reg
311 bool M68kOperand::isReg() const {
312   return Kind == KindTy::MemOp && MemOp.Op == M68kMemOp::Kind::Reg;
313 }
314 
315 unsigned M68kOperand::getReg() const {
316   assert(isReg());
317   return MemOp.OuterReg;
318 }
319 
320 void M68kOperand::addRegOperands(MCInst &Inst, unsigned N) const {
321   assert(isReg() && "wrong operand kind");
322   assert((N == 1) && "can only handle one register operand");
323 
324   Inst.addOperand(MCOperand::createReg(getReg()));
325 }
326 
327 std::unique_ptr<M68kOperand> M68kOperand::createMemOp(M68kMemOp MemOp,
328                                                       SMLoc Start, SMLoc End) {
329   auto Op = std::make_unique<M68kOperand>(KindTy::MemOp, Start, End);
330   Op->MemOp = MemOp;
331   return Op;
332 }
333 
334 // Token
335 bool M68kOperand::isToken() const { return Kind == KindTy::Token; }
336 StringRef M68kOperand::getToken() const {
337   assert(isToken());
338   return Token;
339 }
340 
341 std::unique_ptr<M68kOperand> M68kOperand::createToken(StringRef Token,
342                                                       SMLoc Start, SMLoc End) {
343   auto Op = std::make_unique<M68kOperand>(KindTy::Token, Start, End);
344   Op->Token = Token;
345   return Op;
346 }
347 
348 // Imm
349 bool M68kOperand::isImm() const { return Kind == KindTy::Imm; }
350 void M68kOperand::addImmOperands(MCInst &Inst, unsigned N) const {
351   assert(isImm() && "wrong operand kind");
352   assert((N == 1) && "can only handle one register operand");
353 
354   M68kOperand::addExpr(Inst, Expr);
355 }
356 
357 std::unique_ptr<M68kOperand> M68kOperand::createImm(const MCExpr *Expr,
358                                                     SMLoc Start, SMLoc End) {
359   auto Op = std::make_unique<M68kOperand>(KindTy::Imm, Start, End);
360   Op->Expr = Expr;
361   return Op;
362 }
363 
364 bool M68kOperand::isTrapImm() const {
365   int64_t Value;
366   if (!isImm() || !Expr->evaluateAsAbsolute(Value))
367     return false;
368 
369   return isUInt<4>(Value);
370 }
371 
372 bool M68kOperand::isBkptImm() const {
373   int64_t Value;
374   if (!isImm() || !Expr->evaluateAsAbsolute(Value))
375     return false;
376 
377   return isUInt<3>(Value);
378 }
379 
380 // MoveMask
381 bool M68kOperand::isMoveMask() const {
382   if (!isMemOp())
383     return false;
384 
385   if (MemOp.Op == M68kMemOp::Kind::RegMask)
386     return true;
387 
388   if (MemOp.Op != M68kMemOp::Kind::Reg)
389     return false;
390 
391   // Only regular address / data registers are allowed to be used
392   // in register masks.
393   return getRegisterIndex(MemOp.OuterReg) < 16;
394 }
395 
396 void M68kOperand::addMoveMaskOperands(MCInst &Inst, unsigned N) const {
397   assert(isMoveMask() && "wrong operand kind");
398   assert((N == 1) && "can only handle one immediate operand");
399 
400   uint16_t MoveMask = MemOp.RegMask;
401   if (MemOp.Op == M68kMemOp::Kind::Reg)
402     MoveMask = 1 << getRegisterIndex(MemOp.OuterReg);
403 
404   Inst.addOperand(MCOperand::createImm(MoveMask));
405 }
406 
407 // Addr
408 bool M68kOperand::isAddr() const {
409   return isMemOp() && MemOp.Op == M68kMemOp::Kind::Addr;
410 }
411 // TODO: Maybe we can also store the size of OuterDisp
412 // in Size?
413 template <unsigned N> bool M68kOperand::isAddrN() const {
414   if (isAddr()) {
415     int64_t Res;
416     if (MemOp.OuterDisp->evaluateAsAbsolute(Res))
417       return isInt<N>(Res);
418     return true;
419   }
420   return false;
421 }
422 void M68kOperand::addAddrOperands(MCInst &Inst, unsigned N) const {
423   M68kOperand::addExpr(Inst, MemOp.OuterDisp);
424 }
425 
426 // ARI
427 bool M68kOperand::isARI() const {
428   return isMemOp() && MemOp.Op == M68kMemOp::Kind::RegIndirect &&
429          M68k::AR32RegClass.contains(MemOp.OuterReg);
430 }
431 void M68kOperand::addARIOperands(MCInst &Inst, unsigned N) const {
432   Inst.addOperand(MCOperand::createReg(MemOp.OuterReg));
433 }
434 
435 // ARID
436 bool M68kOperand::isARID() const {
437   return isMemOp() && MemOp.Op == M68kMemOp::Kind::RegIndirectDisplacement &&
438          M68k::AR32RegClass.contains(MemOp.OuterReg);
439 }
440 void M68kOperand::addARIDOperands(MCInst &Inst, unsigned N) const {
441   M68kOperand::addExpr(Inst, MemOp.OuterDisp);
442   Inst.addOperand(MCOperand::createReg(MemOp.OuterReg));
443 }
444 
445 // ARII
446 bool M68kOperand::isARII() const {
447   return isMemOp() &&
448          MemOp.Op == M68kMemOp::Kind::RegIndirectDisplacementIndex &&
449          M68k::AR32RegClass.contains(MemOp.OuterReg);
450 }
451 void M68kOperand::addARIIOperands(MCInst &Inst, unsigned N) const {
452   M68kOperand::addExpr(Inst, MemOp.OuterDisp);
453   Inst.addOperand(MCOperand::createReg(MemOp.OuterReg));
454   Inst.addOperand(MCOperand::createReg(MemOp.InnerReg));
455 }
456 
457 // ARIPD
458 bool M68kOperand::isARIPD() const {
459   return isMemOp() && MemOp.Op == M68kMemOp::Kind::RegPreDecrement &&
460          M68k::AR32RegClass.contains(MemOp.OuterReg);
461 }
462 void M68kOperand::addARIPDOperands(MCInst &Inst, unsigned N) const {
463   Inst.addOperand(MCOperand::createReg(MemOp.OuterReg));
464 }
465 
466 // ARIPI
467 bool M68kOperand::isARIPI() const {
468   return isMemOp() && MemOp.Op == M68kMemOp::Kind::RegPostIncrement &&
469          M68k::AR32RegClass.contains(MemOp.OuterReg);
470 }
471 void M68kOperand::addARIPIOperands(MCInst &Inst, unsigned N) const {
472   Inst.addOperand(MCOperand::createReg(MemOp.OuterReg));
473 }
474 
475 // PCD
476 bool M68kOperand::isPCD() const {
477   return isMemOp() && MemOp.Op == M68kMemOp::Kind::RegIndirectDisplacement &&
478          MemOp.OuterReg == M68k::PC;
479 }
480 void M68kOperand::addPCDOperands(MCInst &Inst, unsigned N) const {
481   M68kOperand::addExpr(Inst, MemOp.OuterDisp);
482 }
483 
484 // PCI
485 bool M68kOperand::isPCI() const {
486   return isMemOp() &&
487          MemOp.Op == M68kMemOp::Kind::RegIndirectDisplacementIndex &&
488          MemOp.OuterReg == M68k::PC;
489 }
490 void M68kOperand::addPCIOperands(MCInst &Inst, unsigned N) const {
491   M68kOperand::addExpr(Inst, MemOp.OuterDisp);
492   Inst.addOperand(MCOperand::createReg(MemOp.InnerReg));
493 }
494 
495 static inline bool checkRegisterClass(unsigned RegNo, bool Data, bool Address,
496                                       bool SP, bool FPDR = false,
497                                       bool FPCR = false) {
498   switch (RegNo) {
499   case M68k::A0:
500   case M68k::A1:
501   case M68k::A2:
502   case M68k::A3:
503   case M68k::A4:
504   case M68k::A5:
505   case M68k::A6:
506     return Address;
507 
508   case M68k::SP:
509     return SP;
510 
511   case M68k::D0:
512   case M68k::D1:
513   case M68k::D2:
514   case M68k::D3:
515   case M68k::D4:
516   case M68k::D5:
517   case M68k::D6:
518   case M68k::D7:
519     return Data;
520 
521   case M68k::SR:
522   case M68k::CCR:
523     return false;
524 
525   case M68k::FP0:
526   case M68k::FP1:
527   case M68k::FP2:
528   case M68k::FP3:
529   case M68k::FP4:
530   case M68k::FP5:
531   case M68k::FP6:
532   case M68k::FP7:
533     return FPDR;
534 
535   case M68k::FPC:
536   case M68k::FPS:
537   case M68k::FPIAR:
538     return FPCR;
539 
540   default:
541     llvm_unreachable("unexpected register type");
542     return false;
543   }
544 }
545 
546 bool M68kOperand::isAReg() const {
547   return isReg() && checkRegisterClass(getReg(),
548                                        /*Data=*/false,
549                                        /*Address=*/true, /*SP=*/true);
550 }
551 
552 bool M68kOperand::isDReg() const {
553   return isReg() && checkRegisterClass(getReg(),
554                                        /*Data=*/true,
555                                        /*Address=*/false, /*SP=*/false);
556 }
557 
558 bool M68kOperand::isFPDReg() const {
559   return isReg() && checkRegisterClass(getReg(),
560                                        /*Data=*/false,
561                                        /*Address=*/false, /*SP=*/false,
562                                        /*FPDR=*/true);
563 }
564 
565 bool M68kOperand::isFPCReg() const {
566   return isReg() && checkRegisterClass(getReg(),
567                                        /*Data=*/false,
568                                        /*Address=*/false, /*SP=*/false,
569                                        /*FPDR=*/false, /*FPCR=*/true);
570 }
571 
572 unsigned M68kAsmParser::validateTargetOperandClass(MCParsedAsmOperand &Op,
573                                                    unsigned Kind) {
574   M68kOperand &Operand = (M68kOperand &)Op;
575 
576   switch (Kind) {
577   case MCK_XR16:
578   case MCK_SPILL:
579     if (Operand.isReg() &&
580         checkRegisterClass(Operand.getReg(), true, true, true)) {
581       return Match_Success;
582     }
583     break;
584 
585   case MCK_AR16:
586   case MCK_AR32:
587     if (Operand.isReg() &&
588         checkRegisterClass(Operand.getReg(), false, true, true)) {
589       return Match_Success;
590     }
591     break;
592 
593   case MCK_AR32_NOSP:
594     if (Operand.isReg() &&
595         checkRegisterClass(Operand.getReg(), false, true, false)) {
596       return Match_Success;
597     }
598     break;
599 
600   case MCK_DR8:
601   case MCK_DR16:
602   case MCK_DR32:
603     if (Operand.isReg() &&
604         checkRegisterClass(Operand.getReg(), true, false, false)) {
605       return Match_Success;
606     }
607     break;
608 
609   case MCK_AR16_TC:
610     if (Operand.isReg() &&
611         ((Operand.getReg() == M68k::A0) || (Operand.getReg() == M68k::A1))) {
612       return Match_Success;
613     }
614     break;
615 
616   case MCK_DR16_TC:
617     if (Operand.isReg() &&
618         ((Operand.getReg() == M68k::D0) || (Operand.getReg() == M68k::D1))) {
619       return Match_Success;
620     }
621     break;
622 
623   case MCK_XR16_TC:
624     if (Operand.isReg() &&
625         ((Operand.getReg() == M68k::D0) || (Operand.getReg() == M68k::D1) ||
626          (Operand.getReg() == M68k::A0) || (Operand.getReg() == M68k::A1))) {
627       return Match_Success;
628     }
629     break;
630   }
631 
632   return Match_InvalidOperand;
633 }
634 
635 bool M68kAsmParser::parseRegisterName(MCRegister &RegNo, SMLoc Loc,
636                                       StringRef RegisterName) {
637   auto RegisterNameLower = RegisterName.lower();
638 
639   // CCR register
640   if (RegisterNameLower == "ccr") {
641     RegNo = M68k::CCR;
642     return true;
643   }
644 
645   // Parse simple general-purpose registers.
646   if (RegisterNameLower.size() == 2) {
647 
648     switch (RegisterNameLower[0]) {
649     case 'd':
650     case 'a': {
651       if (isdigit(RegisterNameLower[1])) {
652         unsigned IndexOffset = (RegisterNameLower[0] == 'a') ? 8 : 0;
653         unsigned RegIndex = (unsigned)(RegisterNameLower[1] - '0');
654         if (RegIndex < 8) {
655           RegNo = getRegisterByIndex(IndexOffset + RegIndex);
656           return true;
657         }
658       }
659       break;
660     }
661 
662     case 's':
663       if (RegisterNameLower[1] == 'p') {
664         RegNo = M68k::SP;
665         return true;
666       } else if (RegisterNameLower[1] == 'r') {
667         RegNo = M68k::SR;
668         return true;
669       }
670       break;
671 
672     case 'p':
673       if (RegisterNameLower[1] == 'c') {
674         RegNo = M68k::PC;
675         return true;
676       }
677       break;
678     }
679   } else if (StringRef(RegisterNameLower).starts_with("fp") &&
680              RegisterNameLower.size() > 2) {
681     auto RegIndex = unsigned(RegisterNameLower[2] - '0');
682     if (RegIndex < 8 && RegisterNameLower.size() == 3) {
683       // Floating point data register.
684       RegNo = getRegisterByIndex(16 + RegIndex);
685       return true;
686     } else {
687       // Floating point control register.
688       RegNo = StringSwitch<unsigned>(RegisterNameLower)
689                   .Cases("fpc", "fpcr", M68k::FPC)
690                   .Cases("fps", "fpsr", M68k::FPS)
691                   .Cases("fpi", "fpiar", M68k::FPIAR)
692                   .Default(M68k::NoRegister);
693       assert(RegNo != M68k::NoRegister &&
694              "Unrecognized FP control register name");
695       return true;
696     }
697   }
698 
699   return false;
700 }
701 
702 ParseStatus M68kAsmParser::parseRegister(MCRegister &RegNo) {
703   bool HasPercent = false;
704   AsmToken PercentToken;
705 
706   LLVM_DEBUG(dbgs() << "parseRegister "; getTok().dump(dbgs()); dbgs() << "\n");
707 
708   if (getTok().is(AsmToken::Percent)) {
709     HasPercent = true;
710     PercentToken = Lex();
711   } else if (!RegisterPrefixOptional.getValue()) {
712     return ParseStatus::NoMatch;
713   }
714 
715   if (!Parser.getTok().is(AsmToken::Identifier)) {
716     if (HasPercent) {
717       getLexer().UnLex(PercentToken);
718     }
719     return ParseStatus::NoMatch;
720   }
721 
722   auto RegisterName = Parser.getTok().getString();
723   if (!parseRegisterName(RegNo, Parser.getLexer().getLoc(), RegisterName)) {
724     if (HasPercent) {
725       getLexer().UnLex(PercentToken);
726     }
727     return ParseStatus::NoMatch;
728   }
729 
730   Parser.Lex();
731   return ParseStatus::Success;
732 }
733 
734 bool M68kAsmParser::parseRegister(MCRegister &Reg, SMLoc &StartLoc,
735                                   SMLoc &EndLoc) {
736   ParseStatus Result = tryParseRegister(Reg, StartLoc, EndLoc);
737   if (!Result.isSuccess())
738     return Error(StartLoc, "expected register");
739 
740   return false;
741 }
742 
743 ParseStatus M68kAsmParser::tryParseRegister(MCRegister &Reg, SMLoc &StartLoc,
744                                             SMLoc &EndLoc) {
745   StartLoc = getLexer().getLoc();
746   ParseStatus Result = parseRegister(Reg);
747   EndLoc = getLexer().getLoc();
748   return Result;
749 }
750 
751 bool M68kAsmParser::isExpr() {
752   switch (Parser.getTok().getKind()) {
753   case AsmToken::Identifier:
754   case AsmToken::Integer:
755     return true;
756   case AsmToken::Minus:
757     return getLexer().peekTok().getKind() == AsmToken::Integer;
758 
759   default:
760     return false;
761   }
762 }
763 
764 ParseStatus M68kAsmParser::parseImm(OperandVector &Operands) {
765   if (getLexer().isNot(AsmToken::Hash))
766     return ParseStatus::NoMatch;
767   SMLoc Start = getLexer().getLoc();
768   Parser.Lex();
769 
770   SMLoc End;
771   const MCExpr *Expr;
772 
773   if (getParser().parseExpression(Expr, End))
774     return ParseStatus::Failure;
775 
776   Operands.push_back(M68kOperand::createImm(Expr, Start, End));
777   return ParseStatus::Success;
778 }
779 
780 ParseStatus M68kAsmParser::parseMemOp(OperandVector &Operands) {
781   SMLoc Start = getLexer().getLoc();
782   bool IsPD = false;
783   M68kMemOp MemOp;
784 
785   // Check for a plain register or register mask.
786   ParseStatus Result = parseRegOrMoveMask(Operands);
787   if (!Result.isNoMatch())
788     return Result;
789 
790   // Check for pre-decrement & outer displacement.
791   bool HasDisplacement = false;
792   if (getLexer().is(AsmToken::Minus)) {
793     IsPD = true;
794     Parser.Lex();
795   } else if (isExpr()) {
796     if (Parser.parseExpression(MemOp.OuterDisp))
797       return ParseStatus::Failure;
798     HasDisplacement = true;
799   }
800 
801   if (getLexer().isNot(AsmToken::LParen)) {
802     if (HasDisplacement) {
803       MemOp.Op = M68kMemOp::Kind::Addr;
804       Operands.push_back(
805           M68kOperand::createMemOp(MemOp, Start, getLexer().getLoc()));
806       return ParseStatus::Success;
807     }
808     if (IsPD)
809       return Error(getLexer().getLoc(), "expected (");
810 
811     return ParseStatus::NoMatch;
812   }
813   Parser.Lex();
814 
815   // Check for constant dereference & MIT-style displacement
816   if (!HasDisplacement && isExpr()) {
817     if (Parser.parseExpression(MemOp.OuterDisp))
818       return ParseStatus::Failure;
819     HasDisplacement = true;
820 
821     // If we're not followed by a comma, we're a constant dereference.
822     if (getLexer().isNot(AsmToken::Comma)) {
823       MemOp.Op = M68kMemOp::Kind::Addr;
824       Operands.push_back(
825           M68kOperand::createMemOp(MemOp, Start, getLexer().getLoc()));
826       return ParseStatus::Success;
827     }
828 
829     Parser.Lex();
830   }
831 
832   Result = parseRegister(MemOp.OuterReg);
833   if (Result.isFailure())
834     return ParseStatus::Failure;
835 
836   if (!Result.isSuccess())
837     return Error(getLexer().getLoc(), "expected register");
838 
839   // Check for Index.
840   bool HasIndex = false;
841   if (Parser.getTok().is(AsmToken::Comma)) {
842     Parser.Lex();
843 
844     Result = parseRegister(MemOp.InnerReg);
845     if (Result.isFailure())
846       return Result;
847 
848     if (Result.isNoMatch())
849       return Error(getLexer().getLoc(), "expected register");
850 
851     // TODO: parse size, scale and inner displacement.
852     MemOp.Size = 4;
853     MemOp.Scale = 1;
854     MemOp.InnerDisp = MCConstantExpr::create(0, Parser.getContext(), true, 4);
855     HasIndex = true;
856   }
857 
858   if (Parser.getTok().isNot(AsmToken::RParen))
859     return Error(getLexer().getLoc(), "expected )");
860   Parser.Lex();
861 
862   bool IsPI = false;
863   if (!IsPD && Parser.getTok().is(AsmToken::Plus)) {
864     Parser.Lex();
865     IsPI = true;
866   }
867 
868   SMLoc End = getLexer().getLoc();
869 
870   unsigned OpCount = IsPD + IsPI + (HasIndex || HasDisplacement);
871   if (OpCount > 1)
872     return Error(Start, "only one of post-increment, pre-decrement or "
873                         "displacement can be used");
874 
875   if (IsPD) {
876     MemOp.Op = M68kMemOp::Kind::RegPreDecrement;
877   } else if (IsPI) {
878     MemOp.Op = M68kMemOp::Kind::RegPostIncrement;
879   } else if (HasIndex) {
880     MemOp.Op = M68kMemOp::Kind::RegIndirectDisplacementIndex;
881   } else if (HasDisplacement) {
882     MemOp.Op = M68kMemOp::Kind::RegIndirectDisplacement;
883   } else {
884     MemOp.Op = M68kMemOp::Kind::RegIndirect;
885   }
886 
887   Operands.push_back(M68kOperand::createMemOp(MemOp, Start, End));
888   return ParseStatus::Success;
889 }
890 
891 ParseStatus M68kAsmParser::parseRegOrMoveMask(OperandVector &Operands) {
892   SMLoc Start = getLexer().getLoc();
893   M68kMemOp MemOp(M68kMemOp::Kind::RegMask);
894   MemOp.RegMask = 0;
895 
896   for (;;) {
897     bool IsFirstRegister =
898         (MemOp.Op == M68kMemOp::Kind::RegMask) && (MemOp.RegMask == 0);
899 
900     MCRegister FirstRegister;
901     ParseStatus Result = parseRegister(FirstRegister);
902     if (IsFirstRegister && Result.isNoMatch())
903       return ParseStatus::NoMatch;
904     if (!Result.isSuccess())
905       return Error(getLexer().getLoc(), "expected start register");
906 
907     MCRegister LastRegister = FirstRegister;
908     if (parseOptionalToken(AsmToken::Minus)) {
909       Result = parseRegister(LastRegister);
910       if (!Result.isSuccess())
911         return Error(getLexer().getLoc(), "expected end register");
912     }
913 
914     unsigned FirstRegisterIndex = getRegisterIndex(FirstRegister);
915     unsigned LastRegisterIndex = getRegisterIndex(LastRegister);
916 
917     uint16_t NumNewBits = LastRegisterIndex - FirstRegisterIndex + 1;
918     uint16_t NewMaskBits = ((1 << NumNewBits) - 1) << FirstRegisterIndex;
919 
920     if (IsFirstRegister && (FirstRegister == LastRegister)) {
921       // First register range is a single register, simplify to just Reg
922       // so that it matches more operands.
923       MemOp.Op = M68kMemOp::Kind::Reg;
924       MemOp.OuterReg = FirstRegister;
925     } else {
926       if (MemOp.Op == M68kMemOp::Kind::Reg) {
927         // This is the second register being specified - expand the Reg operand
928         // into a mask first.
929         MemOp.Op = M68kMemOp::Kind::RegMask;
930         MemOp.RegMask = 1 << getRegisterIndex(MemOp.OuterReg);
931 
932         if (MemOp.RegMask == 0)
933           return Error(getLexer().getLoc(),
934                        "special registers cannot be used in register masks");
935       }
936 
937       if ((FirstRegisterIndex >= 16) || (LastRegisterIndex >= 16))
938         return Error(getLexer().getLoc(),
939                      "special registers cannot be used in register masks");
940 
941       if (NewMaskBits & MemOp.RegMask)
942         return Error(getLexer().getLoc(), "conflicting masked registers");
943 
944       MemOp.RegMask |= NewMaskBits;
945     }
946 
947     if (!parseOptionalToken(AsmToken::Slash))
948       break;
949   }
950 
951   Operands.push_back(
952       M68kOperand::createMemOp(MemOp, Start, getLexer().getLoc()));
953   return ParseStatus::Success;
954 }
955 
956 void M68kAsmParser::eatComma() {
957   if (Parser.getTok().is(AsmToken::Comma)) {
958     Parser.Lex();
959   }
960 }
961 
962 bool M68kAsmParser::ParseInstruction(ParseInstructionInfo &Info, StringRef Name,
963                                      SMLoc NameLoc, OperandVector &Operands) {
964   SMLoc Start = getLexer().getLoc();
965   Operands.push_back(M68kOperand::createToken(Name, Start, Start));
966 
967   bool First = true;
968   while (Parser.getTok().isNot(AsmToken::EndOfStatement)) {
969     if (!First) {
970       eatComma();
971     } else {
972       First = false;
973     }
974 
975     ParseStatus MatchResult = MatchOperandParserImpl(Operands, Name);
976     if (MatchResult.isSuccess())
977       continue;
978 
979     // Add custom operand formats here...
980     SMLoc Loc = getLexer().getLoc();
981     Parser.eatToEndOfStatement();
982     return Error(Loc, "unexpected token parsing operands");
983   }
984 
985   // Eat EndOfStatement.
986   Parser.Lex();
987   return false;
988 }
989 
990 bool M68kAsmParser::invalidOperand(SMLoc const &Loc,
991                                    OperandVector const &Operands,
992                                    uint64_t const &ErrorInfo) {
993   SMLoc ErrorLoc = Loc;
994   char const *Diag = 0;
995 
996   if (ErrorInfo != ~0U) {
997     if (ErrorInfo >= Operands.size()) {
998       Diag = "too few operands for instruction.";
999     } else {
1000       auto const &Op = (M68kOperand const &)*Operands[ErrorInfo];
1001       if (Op.getStartLoc() != SMLoc()) {
1002         ErrorLoc = Op.getStartLoc();
1003       }
1004     }
1005   }
1006 
1007   if (!Diag) {
1008     Diag = "invalid operand for instruction";
1009   }
1010 
1011   return Error(ErrorLoc, Diag);
1012 }
1013 
1014 bool M68kAsmParser::missingFeature(llvm::SMLoc const &Loc,
1015                                    uint64_t const &ErrorInfo) {
1016   return Error(Loc, "instruction requires a CPU feature not currently enabled");
1017 }
1018 
1019 bool M68kAsmParser::emit(MCInst &Inst, SMLoc const &Loc,
1020                          MCStreamer &Out) const {
1021   Inst.setLoc(Loc);
1022   Out.emitInstruction(Inst, STI);
1023 
1024   return false;
1025 }
1026 
1027 bool M68kAsmParser::MatchAndEmitInstruction(SMLoc Loc, unsigned &Opcode,
1028                                             OperandVector &Operands,
1029                                             MCStreamer &Out,
1030                                             uint64_t &ErrorInfo,
1031                                             bool MatchingInlineAsm) {
1032   MCInst Inst;
1033   unsigned MatchResult =
1034       MatchInstructionImpl(Operands, Inst, ErrorInfo, MatchingInlineAsm);
1035 
1036   switch (MatchResult) {
1037   case Match_Success:
1038     return emit(Inst, Loc, Out);
1039   case Match_MissingFeature:
1040     return missingFeature(Loc, ErrorInfo);
1041   case Match_InvalidOperand:
1042     return invalidOperand(Loc, Operands, ErrorInfo);
1043   case Match_MnemonicFail:
1044     return Error(Loc, "invalid instruction");
1045   default:
1046     return true;
1047   }
1048 }
1049 
1050 void M68kOperand::print(raw_ostream &OS) const {
1051   switch (Kind) {
1052   case KindTy::Invalid:
1053     OS << "invalid";
1054     break;
1055 
1056   case KindTy::Token:
1057     OS << "token '" << Token << "'";
1058     break;
1059 
1060   case KindTy::Imm: {
1061     int64_t Value;
1062     Expr->evaluateAsAbsolute(Value);
1063     OS << "immediate " << Value;
1064     break;
1065   }
1066 
1067   case KindTy::MemOp:
1068     MemOp.print(OS);
1069     break;
1070   }
1071 }
1072