forked from OSchip/llvm-project
[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:
parent
992e21eeee
commit
47f52f989b
|
@ -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();
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
|
@ -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 << '/';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in New Issue