[M68k][AsmParser] Support parsing register masks & fix printing them

Fixes PR51580.

Register masks will now be printed as 'movem.l (%sp), %a0-%a5/%d5'
for example and can now be parsed in the same format.

Previously the printed syntax was 'movem.l (%sp), %a0-%a5,%d', which
didn't match prior art and was too ambiguous to easily parse.

Differential Revision: https://reviews.llvm.org/D108597
This commit is contained in:
Ricky Taylor 2021-08-23 23:49:52 +01:00
parent 992e21eeee
commit 47f52f989b
5 changed files with 167 additions and 23 deletions

View File

@ -52,6 +52,7 @@ class M68kAsmParser : public MCTargetAsmParser {
bool isExpr(); bool isExpr();
OperandMatchResultTy parseImm(OperandVector &Operands); OperandMatchResultTy parseImm(OperandVector &Operands);
OperandMatchResultTy parseMemOp(OperandVector &Operands); OperandMatchResultTy parseMemOp(OperandVector &Operands);
OperandMatchResultTy parseRegOrMoveMask(OperandVector &Operands);
public: public:
M68kAsmParser(const MCSubtargetInfo &STI, MCAsmParser &Parser, M68kAsmParser(const MCSubtargetInfo &STI, MCAsmParser &Parser,
@ -80,6 +81,7 @@ public:
struct M68kMemOp { struct M68kMemOp {
enum class Kind { enum class Kind {
Addr, Addr,
RegMask,
Reg, Reg,
RegIndirect, RegIndirect,
RegPostIncrement, RegPostIncrement,
@ -90,6 +92,7 @@ struct M68kMemOp {
// These variables are used for the following forms: // These variables are used for the following forms:
// Addr: (OuterDisp) // Addr: (OuterDisp)
// RegMask: RegMask (as register mask)
// Reg: %OuterReg // Reg: %OuterReg
// RegIndirect: (%OuterReg) // RegIndirect: (%OuterReg)
// RegPostIncrement: (%OuterReg)+ // RegPostIncrement: (%OuterReg)+
@ -106,6 +109,7 @@ struct M68kMemOp {
uint8_t Size : 4; uint8_t Size : 4;
uint8_t Scale : 4; uint8_t Scale : 4;
const MCExpr *Expr; const MCExpr *Expr;
uint16_t RegMask;
M68kMemOp() {} M68kMemOp() {}
M68kMemOp(Kind Op) : Op(Op) {} M68kMemOp(Kind Op) : Op(Op) {}
@ -172,6 +176,10 @@ public:
static std::unique_ptr<M68kOperand> createImm(const MCExpr *Expr, SMLoc Start, static std::unique_ptr<M68kOperand> createImm(const MCExpr *Expr, SMLoc Start,
SMLoc End); SMLoc End);
// MoveMask
bool isMoveMask() const;
void addMoveMaskOperands(MCInst &Inst, unsigned N) const;
// Addr // Addr
bool isAddr() const; bool isAddr() const;
bool isAddr8() const { return isAddrN<8>(); } bool isAddr8() const { return isAddrN<8>(); }
@ -217,11 +225,45 @@ extern "C" LLVM_EXTERNAL_VISIBILITY void LLVMInitializeM68kAsmParser() {
#define GET_MATCHER_IMPLEMENTATION #define GET_MATCHER_IMPLEMENTATION
#include "M68kGenAsmMatcher.inc" #include "M68kGenAsmMatcher.inc"
static inline unsigned getRegisterByIndex(unsigned RegisterIndex) {
static unsigned RegistersByIndex[] = {
M68k::D0, M68k::D1, M68k::D2, M68k::D3, M68k::D4, M68k::D5,
M68k::D6, M68k::D7, M68k::A0, M68k::A1, M68k::A2, M68k::A3,
M68k::A4, M68k::A5, M68k::A6, M68k::SP,
};
assert(RegisterIndex <=
sizeof(RegistersByIndex) / sizeof(RegistersByIndex[0]));
return RegistersByIndex[RegisterIndex];
}
static inline unsigned getRegisterIndex(unsigned Register) {
if (Register >= M68k::D0 && Register <= M68k::D7)
return Register - M68k::D0;
if (Register >= M68k::A0 && Register <= M68k::A6)
return Register - M68k::A0 + 8;
switch (Register) {
case M68k::SP:
// SP is sadly not contiguous with the rest of the An registers
return 15;
case M68k::PC:
case M68k::CCR:
return 16;
default:
llvm_unreachable("unexpected register number");
}
}
void M68kMemOp::print(raw_ostream &OS) const { void M68kMemOp::print(raw_ostream &OS) const {
switch (Op) { switch (Op) {
case Kind::Addr: case Kind::Addr:
OS << OuterDisp; OS << OuterDisp;
break; break;
case Kind::RegMask:
OS << "RegMask(" << format("%04x", RegMask) << ")";
break;
case Kind::Reg: case Kind::Reg:
OS << '%' << OuterReg; OS << '%' << OuterReg;
break; break;
@ -294,7 +336,7 @@ std::unique_ptr<M68kOperand> M68kOperand::createToken(StringRef Token,
// Imm // Imm
bool M68kOperand::isImm() const { return Kind == Kind::Imm; } bool M68kOperand::isImm() const { return Kind == Kind::Imm; }
void M68kOperand::addImmOperands(MCInst &Inst, unsigned N) const { void M68kOperand::addImmOperands(MCInst &Inst, unsigned N) const {
assert(isImm() && "wrong oeprand kind"); assert(isImm() && "wrong operand kind");
assert((N == 1) && "can only handle one register operand"); assert((N == 1) && "can only handle one register operand");
M68kOperand::addExpr(Inst, Expr); M68kOperand::addExpr(Inst, Expr);
@ -307,6 +349,33 @@ std::unique_ptr<M68kOperand> M68kOperand::createImm(const MCExpr *Expr,
return Op; return Op;
} }
// MoveMask
bool M68kOperand::isMoveMask() const {
if (!isMemOp())
return false;
if (MemOp.Op == M68kMemOp::Kind::RegMask)
return true;
if (MemOp.Op != M68kMemOp::Kind::Reg)
return false;
// Only regular address / data registers are allowed to be used
// in register masks.
return getRegisterIndex(MemOp.OuterReg) < 16;
}
void M68kOperand::addMoveMaskOperands(MCInst &Inst, unsigned N) const {
assert(isMoveMask() && "wrong operand kind");
assert((N == 1) && "can only handle one immediate operand");
uint16_t MoveMask = MemOp.RegMask;
if (MemOp.Op == M68kMemOp::Kind::Reg)
MoveMask = 1 << getRegisterIndex(MemOp.OuterReg);
Inst.addOperand(MCOperand::createImm(MoveMask));
}
// Addr // Addr
bool M68kOperand::isAddr() const { bool M68kOperand::isAddr() const {
return isMemOp() && MemOp.Op == M68kMemOp::Kind::Addr; return isMemOp() && MemOp.Op == M68kMemOp::Kind::Addr;
@ -517,11 +586,6 @@ bool M68kAsmParser::parseRegisterName(unsigned &RegNo, SMLoc Loc,
// Parse simple general-purpose registers. // Parse simple general-purpose registers.
if (RegisterNameLower.size() == 2) { if (RegisterNameLower.size() == 2) {
static unsigned RegistersByIndex[] = {
M68k::D0, M68k::D1, M68k::D2, M68k::D3, M68k::D4, M68k::D5,
M68k::D6, M68k::D7, M68k::A0, M68k::A1, M68k::A2, M68k::A3,
M68k::A4, M68k::A5, M68k::A6, M68k::SP,
};
switch (RegisterNameLower[0]) { switch (RegisterNameLower[0]) {
case 'd': case 'd':
@ -530,7 +594,7 @@ bool M68kAsmParser::parseRegisterName(unsigned &RegNo, SMLoc Loc,
unsigned IndexOffset = (RegisterNameLower[0] == 'a') ? 8 : 0; unsigned IndexOffset = (RegisterNameLower[0] == 'a') ? 8 : 0;
unsigned RegIndex = (unsigned)(RegisterNameLower[1] - '0'); unsigned RegIndex = (unsigned)(RegisterNameLower[1] - '0');
if (RegIndex < 8) { if (RegIndex < 8) {
RegNo = RegistersByIndex[IndexOffset + RegIndex]; RegNo = getRegisterByIndex(IndexOffset + RegIndex);
return true; return true;
} }
} }
@ -646,16 +710,9 @@ OperandMatchResultTy M68kAsmParser::parseMemOp(OperandVector &Operands) {
bool IsPD = false; bool IsPD = false;
M68kMemOp MemOp; M68kMemOp MemOp;
// Check for a plain register. // Check for a plain register or register mask.
auto Result = parseRegister(MemOp.OuterReg); auto Result = parseRegOrMoveMask(Operands);
if (Result == MatchOperand_Success) { if (Result != llvm::MatchOperand_NoMatch) {
MemOp.Op = M68kMemOp::Kind::Reg;
Operands.push_back(
M68kOperand::createMemOp(MemOp, Start, getLexer().getLoc()));
return MatchOperand_Success;
}
if (Result == MatchOperand_ParseFail) {
return Result; return Result;
} }
@ -773,6 +830,87 @@ OperandMatchResultTy M68kAsmParser::parseMemOp(OperandVector &Operands) {
return MatchOperand_Success; return MatchOperand_Success;
} }
OperandMatchResultTy
M68kAsmParser::parseRegOrMoveMask(OperandVector &Operands) {
SMLoc Start = getLexer().getLoc();
M68kMemOp MemOp(M68kMemOp::Kind::RegMask);
MemOp.RegMask = 0;
for (;;) {
bool IsFirstRegister =
(MemOp.Op == M68kMemOp::Kind::RegMask) && (MemOp.RegMask == 0);
unsigned FirstRegister;
auto Result = parseRegister(FirstRegister);
if (IsFirstRegister && (Result == llvm::MatchOperand_NoMatch)) {
return MatchOperand_NoMatch;
}
if (Result != llvm::MatchOperand_Success) {
Error(getLexer().getLoc(), "expected start register");
return MatchOperand_ParseFail;
}
unsigned LastRegister = FirstRegister;
if (getLexer().is(AsmToken::Minus)) {
getLexer().Lex();
Result = parseRegister(LastRegister);
if (Result != llvm::MatchOperand_Success) {
Error(getLexer().getLoc(), "expected end register");
return MatchOperand_ParseFail;
}
}
unsigned FirstRegisterIndex = getRegisterIndex(FirstRegister);
unsigned LastRegisterIndex = getRegisterIndex(LastRegister);
uint16_t NumNewBits = LastRegisterIndex - FirstRegisterIndex + 1;
uint16_t NewMaskBits = ((1 << NumNewBits) - 1) << FirstRegisterIndex;
if (IsFirstRegister && (FirstRegister == LastRegister)) {
// First register range is a single register, simplify to just Reg
// so that it matches more operands.
MemOp.Op = M68kMemOp::Kind::Reg;
MemOp.OuterReg = FirstRegister;
} else {
if (MemOp.Op == M68kMemOp::Kind::Reg) {
// This is the second register being specified - expand the Reg operand
// into a mask first.
MemOp.Op = M68kMemOp::Kind::RegMask;
MemOp.RegMask = 1 << getRegisterIndex(MemOp.OuterReg);
if (MemOp.RegMask == 0) {
Error(getLexer().getLoc(),
"special registers cannot be used in register masks");
return MatchOperand_ParseFail;
}
}
if ((FirstRegisterIndex >= 16) || (LastRegisterIndex >= 16)) {
Error(getLexer().getLoc(),
"special registers cannot be used in register masks");
return MatchOperand_ParseFail;
}
if (NewMaskBits & MemOp.RegMask) {
Error(getLexer().getLoc(), "conflicting masked registers");
return MatchOperand_ParseFail;
}
MemOp.RegMask |= NewMaskBits;
}
if (getLexer().isNot(AsmToken::Slash)) {
break;
}
getLexer().Lex();
}
Operands.push_back(
M68kOperand::createMemOp(MemOp, Start, getLexer().getLoc()));
return MatchOperand_Success;
}
void M68kAsmParser::eatComma() { void M68kAsmParser::eatComma() {
if (Parser.getTok().is(AsmToken::Comma)) { if (Parser.getTok().is(AsmToken::Comma)) {
Parser.Lex(); Parser.Lex();

View File

@ -404,9 +404,11 @@ def MxBrTarget16 : MxBrTargetOperand<16>;
def MxBrTarget32 : MxBrTargetOperand<32>; def MxBrTarget32 : MxBrTargetOperand<32>;
// Used with MOVEM // Used with MOVEM
def MxMoveMaskClass : MxOpClass<"MoveMask">;
def MxMoveMask : MxOp<i16, MxSize16, "m"> { def MxMoveMask : MxOp<i16, MxSize16, "m"> {
let OperandType = "OPERAND_IMMEDIATE"; let OperandType = "OPERAND_IMMEDIATE";
let PrintMethod = "printMoveMask"; let PrintMethod = "printMoveMask";
let ParserMatchClass = MxMoveMaskClass;
} }
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//

View File

@ -109,7 +109,7 @@ void M68kInstPrinter::printMoveMask(const MCInst *MI, unsigned opNum,
// Print separation comma only if // Print separation comma only if
// both data & register parts have bit(s) set // both data & register parts have bit(s) set
if (s != 0 && (Mask & 0xFF) && HalfMask) if (s != 0 && (Mask & 0xFF) && HalfMask)
O << ','; O << '/';
for (int i = 0; HalfMask; ++i) { for (int i = 0; HalfMask; ++i) {
if ((HalfMask >> i) & 0b1) { if ((HalfMask >> i) & 0b1) {
@ -130,7 +130,7 @@ void M68kInstPrinter::printMoveMask(const MCInst *MI, unsigned opNum,
i = j; i = j;
if (HalfMask) if (HalfMask)
O << ','; O << '/';
} }
} }
} }

View File

@ -10,7 +10,7 @@
--- # CollapseMOVEM_RM --- # CollapseMOVEM_RM
# #
# CHECK-LABEL: CollapseMOVEM_RM # CHECK-LABEL: CollapseMOVEM_RM
# CHECK: movem.l (0,%sp), %d0-%d2,%d7,%a1-%a3,%a5 # CHECK: movem.l (0,%sp), %d0-%d2/%d7/%a1-%a3/%a5
name: CollapseMOVEM_RM name: CollapseMOVEM_RM
body: | body: |
bb.0: bb.0:
@ -26,7 +26,7 @@ body: |
... ...
# #
# CHECK-LABEL: CollapseMOVEM_RM_Reversed # CHECK-LABEL: CollapseMOVEM_RM_Reversed
# CHECK: movem.l (0,%sp), %d0-%d2,%d7,%a1-%a3,%a5 # CHECK: movem.l (0,%sp), %d0-%d2/%d7/%a1-%a3/%a5
name: CollapseMOVEM_RM_Reversed name: CollapseMOVEM_RM_Reversed
body: | body: |
bb.0: bb.0:
@ -66,7 +66,7 @@ body: |
--- # CollapseMOVEM_MR --- # CollapseMOVEM_MR
# #
# CHECK-LABEL: CollapseMOVEM_MR # CHECK-LABEL: CollapseMOVEM_MR
# CHECK: movem.l %d0-%d2,%d7,%a1-%a3,%a5, (0,%sp) # CHECK: movem.l %d0-%d2/%d7/%a1-%a3/%a5, (0,%sp)
name: CollapseMOVEM_MR name: CollapseMOVEM_MR
body: | body: |
bb.0: bb.0:
@ -84,7 +84,7 @@ body: |
# #
# CHECK-LABEL: CollapseMOVEM_Mixed # CHECK-LABEL: CollapseMOVEM_Mixed
# CHECK: movem.l %d0-%d1, (0,%sp) # CHECK: movem.l %d0-%d1, (0,%sp)
# CHECK: movem.l (8,%sp), %d2,%d7 # CHECK: movem.l (8,%sp), %d2/%d7
# CHECK: movem.l %a1-%a2, (16,%sp) # CHECK: movem.l %a1-%a2, (16,%sp)
# CHECK: movem.l (24,%sp), %a3 # CHECK: movem.l (24,%sp), %a3
# CHECK: movem.l %a5, (28,%sp) # CHECK: movem.l %a5, (28,%sp)

View File

@ -46,3 +46,7 @@ ror.l #8, %d1
nop nop
; CHECK: rts ; CHECK: rts
rts rts
; CHECK: movem.l %d0-%d6/%a0, (%sp)
movem.l %d0-%d6/%a0, (%sp)
; CHECK: movem.l (10,%sp), %d0-%d6/%a0
movem.l (10,%sp), %d0-%d6/%a0