This patch introduces changes to MipsAsmParser register parsing routines. The code now follows more deterministic path and makes the code more efficient and easier to maintain.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@188264 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Vladimir Medic 2013-08-13 13:07:09 +00:00
parent d89266dd82
commit bd980e5569

View File

@ -63,6 +63,7 @@ class MipsAsmParser : public MCTargetAsmParser {
MCSubtargetInfo &STI; MCSubtargetInfo &STI;
MCAsmParser &Parser; MCAsmParser &Parser;
MipsAssemblerOptions Options; MipsAssemblerOptions Options;
bool hasConsumedDollar;
#define GET_ASSEMBLER_HEADER #define GET_ASSEMBLER_HEADER
#include "MipsGenAsmMatcher.inc" #include "MipsGenAsmMatcher.inc"
@ -175,15 +176,13 @@ class MipsAsmParser : public MCTargetAsmParser {
int matchRegisterByNumber(unsigned RegNum, unsigned RegClass); int matchRegisterByNumber(unsigned RegNum, unsigned RegClass);
int matchFPURegisterName(StringRef Name, FpFormatTy Format); int matchFPURegisterName(StringRef Name);
void setFpFormat(FpFormatTy Format) { int matchFCCRegisterName(StringRef Name);
FpFormat = Format;
}
void setDefaultFpFormat(); int matchACRegisterName(StringRef Name);
void setFpFormat(StringRef Format); int regKindToRegClass(int RegKind);
FpFormatTy getFpFormat() {return FpFormat;} FpFormatTy getFpFormat() {return FpFormat;}
@ -195,7 +194,7 @@ class MipsAsmParser : public MCTargetAsmParser {
SmallVectorImpl<MCInst> &Instructions); SmallVectorImpl<MCInst> &Instructions);
public: public:
MipsAsmParser(MCSubtargetInfo &sti, MCAsmParser &parser) MipsAsmParser(MCSubtargetInfo &sti, MCAsmParser &parser)
: MCTargetAsmParser(), STI(sti), Parser(parser) { : MCTargetAsmParser(), STI(sti), Parser(parser), hasConsumedDollar(false) {
// Initialize the set of available features. // Initialize the set of available features.
setAvailableFeatures(ComputeAvailableFeatures(STI.getFeatureBits())); setAvailableFeatures(ComputeAvailableFeatures(STI.getFeatureBits()));
} }
@ -816,60 +815,76 @@ int MipsAsmParser::matchCPURegisterName(StringRef Name) {
return CC; return CC;
} }
int MipsAsmParser::matchFPURegisterName(StringRef Name, FpFormatTy Format) { int MipsAsmParser::matchFPURegisterName(StringRef Name) {
if (Name[0] == 'f') { if (Name[0] == 'f') {
StringRef NumString = Name.substr(1); StringRef NumString = Name.substr(1);
unsigned IntVal; unsigned IntVal;
if (NumString.getAsInteger(10, IntVal)) if (NumString.getAsInteger(10, IntVal))
return -1; // This is not an integer. return -1; // This is not an integer.
if (IntVal > 31) if (IntVal > 31) // Maximum index for fpu register.
return -1; return -1;
return IntVal;
}
return -1;
}
if (Format == FP_FORMAT_S || Format == FP_FORMAT_W) int MipsAsmParser::matchFCCRegisterName(StringRef Name) {
return getReg(Mips::FGR32RegClassID, IntVal);
if (Format == FP_FORMAT_D) { if (Name.startswith("fcc")) {
if (isFP64()) { StringRef NumString = Name.substr(3);
return getReg(Mips::FGR64RegClassID, IntVal); unsigned IntVal;
} if (NumString.getAsInteger(10, IntVal))
// Only even numbers available as register pairs. return -1; // This is not an integer.
if ((IntVal > 31) || (IntVal % 2 != 0)) if (IntVal > 7) // There are only 8 fcc registers.
return -1; return -1;
return getReg(Mips::AFGR64RegClassID, IntVal / 2); return IntVal;
} }
return -1;
}
int MipsAsmParser::matchACRegisterName(StringRef Name) {
if (Name.startswith("acc")) {
StringRef NumString = Name.substr(3);
unsigned IntVal;
if (NumString.getAsInteger(10, IntVal))
return -1; // This is not an integer.
if (IntVal > 3) // There are only 3 acc registers.
return -1;
return IntVal;
} }
return -1; return -1;
} }
int MipsAsmParser::matchRegisterName(StringRef Name, bool is64BitReg) { int MipsAsmParser::matchRegisterName(StringRef Name, bool is64BitReg) {
if (Name.equals("fcc0"))
return Mips::FCC0;
int CC; int CC;
CC = matchCPURegisterName(Name); CC = matchCPURegisterName(Name);
if (CC != -1) if (CC != -1)
return matchRegisterByNumber(CC, is64BitReg ? Mips::GPR64RegClassID return matchRegisterByNumber(CC, is64BitReg ? Mips::GPR64RegClassID
: Mips::GPR32RegClassID); : Mips::GPR32RegClassID);
return matchFPURegisterName(Name, getFpFormat()); CC= matchFPURegisterName(Name);
//TODO: decide about fpu register class
return matchRegisterByNumber(CC, isFP64() ? Mips::FGR64RegClassID
: Mips::FGR32RegClassID);
} }
void MipsAsmParser::setDefaultFpFormat() { int MipsAsmParser::regKindToRegClass(int RegKind) {
if (isMips64() || isFP64()) switch (RegKind) {
FpFormat = FP_FORMAT_D; case MipsOperand::Kind_GPR32: return Mips::GPR32RegClassID;
else case MipsOperand::Kind_GPR64: return Mips::GPR64RegClassID;
FpFormat = FP_FORMAT_S; case MipsOperand::Kind_HWRegs: return Mips::HWRegsRegClassID;
} case MipsOperand::Kind_FGR32Regs: return Mips::FGR32RegClassID;
case MipsOperand::Kind_FGR64Regs: return Mips::FGR64RegClassID;
case MipsOperand::Kind_AFGR64Regs: return Mips::AFGR64RegClassID;
case MipsOperand::Kind_CCRRegs: return Mips::CCRRegClassID;
case MipsOperand::Kind_ACC64DSP: return Mips::ACC64DSPRegClassID;
case MipsOperand::Kind_FCCRegs: return Mips::FCCRegClassID;
default :return -1;
}
void MipsAsmParser::setFpFormat(StringRef Format) {
FpFormat = StringSwitch<FpFormatTy>(Format.lower())
.Case(".s", FP_FORMAT_S)
.Case(".d", FP_FORMAT_D)
.Case(".l", FP_FORMAT_L)
.Case(".w", FP_FORMAT_W)
.Default(FP_FORMAT_NONE);
} }
bool MipsAssemblerOptions::setATReg(unsigned Reg) { bool MipsAssemblerOptions::setATReg(unsigned Reg) {
@ -889,8 +904,8 @@ unsigned MipsAsmParser::getReg(int RC, int RegNo) {
} }
int MipsAsmParser::matchRegisterByNumber(unsigned RegNum, unsigned RegClass) { int MipsAsmParser::matchRegisterByNumber(unsigned RegNum, unsigned RegClass) {
if (RegNum >
if (RegNum > 31) getContext().getRegisterInfo()->getRegClass(RegClass).getNumRegs())
return -1; return -1;
return getReg(RegClass, RegNum); return getReg(RegClass, RegNum);
@ -1173,6 +1188,7 @@ MipsAsmParser::OperandMatchResultTy MipsAsmParser::parseMemOperand(
const MCExpr *IdVal = 0; const MCExpr *IdVal = 0;
SMLoc S; SMLoc S;
bool isParenExpr = false; bool isParenExpr = false;
MipsAsmParser::OperandMatchResultTy Res = MatchOperand_NoMatch;
// First operand is the offset. // First operand is the offset.
S = Parser.getTok().getLoc(); S = Parser.getTok().getLoc();
@ -1211,21 +1227,12 @@ MipsAsmParser::OperandMatchResultTy MipsAsmParser::parseMemOperand(
Parser.Lex(); // Eat the '(' token. Parser.Lex(); // Eat the '(' token.
} }
const AsmToken &Tok1 = Parser.getTok(); // Get next token Res = parseRegs(Operands, isMips64()? (int) MipsOperand::Kind_GPR64:
if (Tok1.is(AsmToken::Dollar)) { (int) MipsOperand::Kind_GPR32);
Parser.Lex(); // Eat the '$' token. if (Res != MatchOperand_Success)
if (tryParseRegisterOperand(Operands, isMips64())) { return Res;
Error(Parser.getTok().getLoc(), "unexpected token in operand");
return MatchOperand_ParseFail;
}
} else { if (Parser.getTok().isNot(AsmToken::RParen)) {
Error(Parser.getTok().getLoc(), "unexpected token in operand");
return MatchOperand_ParseFail;
}
const AsmToken &Tok2 = Parser.getTok(); // Get next token.
if (Tok2.isNot(AsmToken::RParen)) {
Error(Parser.getTok().getLoc(), "')' expected"); Error(Parser.getTok().getLoc(), "')' expected");
return MatchOperand_ParseFail; return MatchOperand_ParseFail;
} }
@ -1261,22 +1268,88 @@ MipsAsmParser::OperandMatchResultTy
MipsAsmParser::parseRegs(SmallVectorImpl<MCParsedAsmOperand*> &Operands, MipsAsmParser::parseRegs(SmallVectorImpl<MCParsedAsmOperand*> &Operands,
int RegKind) { int RegKind) {
MipsOperand::RegisterKind Kind = (MipsOperand::RegisterKind)RegKind; MipsOperand::RegisterKind Kind = (MipsOperand::RegisterKind)RegKind;
if (getLexer().getKind() == AsmToken::Identifier) { if (getLexer().getKind() == AsmToken::Identifier
&& !hasConsumedDollar) {
if (searchSymbolAlias(Operands, Kind)) if (searchSymbolAlias(Operands, Kind))
return MatchOperand_Success; return MatchOperand_Success;
return MatchOperand_NoMatch; return MatchOperand_NoMatch;
} }
SMLoc S = Parser.getTok().getLoc();
// If the first token is not '$', we have an error. // If the first token is not '$', we have an error.
if (Parser.getTok().isNot(AsmToken::Dollar)) if (Parser.getTok().isNot(AsmToken::Dollar) && !hasConsumedDollar)
return MatchOperand_NoMatch; return MatchOperand_NoMatch;
if (!hasConsumedDollar) {
Parser.Lex(); // Eat the '$'
hasConsumedDollar = true;
}
if (getLexer().getKind() == AsmToken::Identifier) {
int RegNum = -1;
std::string RegName = Parser.getTok().getString().lower();
// Match register by name
switch (RegKind) {
case MipsOperand::Kind_GPR32:
case MipsOperand::Kind_GPR64:
RegNum = matchCPURegisterName(RegName);
break;
case MipsOperand::Kind_AFGR64Regs:
case MipsOperand::Kind_FGR64Regs:
case MipsOperand::Kind_FGR32Regs:
RegNum = matchFPURegisterName(RegName);
if (RegKind == MipsOperand::Kind_AFGR64Regs)
RegNum /= 2;
break;
case MipsOperand::Kind_FCCRegs:
RegNum = matchFCCRegisterName(RegName);
break;
case MipsOperand::Kind_ACC64DSP:
RegNum = matchACRegisterName(RegName);
break;
default: break; // No match, value is set to -1.
}
// No match found, return _NoMatch to give a chance to other round.
if (RegNum < 0)
return MatchOperand_NoMatch;
Parser.Lex(); // Eat $ int RegVal = getReg(regKindToRegClass(Kind), RegNum);
if (!tryParseRegisterOperand(Operands, if (RegVal == -1)
RegKind == MipsOperand::Kind_GPR64)) { return MatchOperand_NoMatch;
// Set the proper register kind.
MipsOperand* op = static_cast<MipsOperand*>(Operands.back()); MipsOperand *Op = MipsOperand::CreateReg(RegVal, S,
op->setRegKind(Kind); Parser.getTok().getLoc());
if ((Kind == MipsOperand::Kind_GPR32) Op->setRegKind(Kind);
Operands.push_back(Op);
hasConsumedDollar = false;
Parser.Lex(); // Eat the register name.
if ((RegKind == MipsOperand::Kind_GPR32)
&& (getLexer().is(AsmToken::LParen))) {
// Check if it is indexed addressing operand.
Operands.push_back(MipsOperand::CreateToken("(", getLexer().getLoc()));
Parser.Lex(); // Eat the parenthesis.
if (parseRegs(Operands,RegKind) != MatchOperand_Success)
return MatchOperand_NoMatch;
if (getLexer().isNot(AsmToken::RParen))
return MatchOperand_NoMatch;
Operands.push_back(MipsOperand::CreateToken(")", getLexer().getLoc()));
Parser.Lex();
}
return MatchOperand_Success;
} else if (getLexer().getKind() == AsmToken::Integer) {
unsigned RegNum = Parser.getTok().getIntVal();
if (Kind == MipsOperand::Kind_HWRegs) {
if (RegNum != 29)
return MatchOperand_NoMatch;
// Only hwreg 29 is supported, found at index 0.
RegNum = 0;
}
int Reg = matchRegisterByNumber(RegNum, regKindToRegClass(Kind));
if (Reg == -1)
return MatchOperand_NoMatch;
MipsOperand *Op = MipsOperand::CreateReg(Reg, S, Parser.getTok().getLoc());
Op->setRegKind(Kind);
Operands.push_back(Op);
hasConsumedDollar = false;
Parser.Lex(); // Eat the register number.
if ((RegKind == MipsOperand::Kind_GPR32)
&& (getLexer().is(AsmToken::LParen))) { && (getLexer().is(AsmToken::LParen))) {
// Check if it is indexed addressing operand. // Check if it is indexed addressing operand.
Operands.push_back(MipsOperand::CreateToken("(", getLexer().getLoc())); Operands.push_back(MipsOperand::CreateToken("(", getLexer().getLoc()));
@ -1311,9 +1384,6 @@ MipsAsmParser::parseAFGR64Regs(SmallVectorImpl<MCParsedAsmOperand*> &Operands) {
if (isFP64()) if (isFP64())
return MatchOperand_NoMatch; return MatchOperand_NoMatch;
// Double operand is expected, set appropriate format
setFpFormat(FP_FORMAT_D);
return parseRegs(Operands, (int) MipsOperand::Kind_AFGR64Regs); return parseRegs(Operands, (int) MipsOperand::Kind_AFGR64Regs);
} }
@ -1321,83 +1391,22 @@ MipsAsmParser::OperandMatchResultTy
MipsAsmParser::parseFGR64Regs(SmallVectorImpl<MCParsedAsmOperand*> &Operands) { MipsAsmParser::parseFGR64Regs(SmallVectorImpl<MCParsedAsmOperand*> &Operands) {
if (!isFP64()) if (!isFP64())
return MatchOperand_NoMatch; return MatchOperand_NoMatch;
// Double operand is expected, set appropriate format
setFpFormat(FP_FORMAT_D);
return parseRegs(Operands, (int) MipsOperand::Kind_FGR64Regs); return parseRegs(Operands, (int) MipsOperand::Kind_FGR64Regs);
} }
MipsAsmParser::OperandMatchResultTy MipsAsmParser::OperandMatchResultTy
MipsAsmParser::parseFGR32Regs(SmallVectorImpl<MCParsedAsmOperand*> &Operands) { MipsAsmParser::parseFGR32Regs(SmallVectorImpl<MCParsedAsmOperand*> &Operands) {
// Single operand is expected, set appropriate format
setFpFormat(FP_FORMAT_S);
return parseRegs(Operands, (int) MipsOperand::Kind_FGR32Regs); return parseRegs(Operands, (int) MipsOperand::Kind_FGR32Regs);
} }
MipsAsmParser::OperandMatchResultTy MipsAsmParser::OperandMatchResultTy
MipsAsmParser::parseFCCRegs(SmallVectorImpl<MCParsedAsmOperand*> &Operands) { MipsAsmParser::parseFCCRegs(SmallVectorImpl<MCParsedAsmOperand*> &Operands) {
// If the first token is not '$' we have an error. return parseRegs(Operands, (int) MipsOperand::Kind_FCCRegs);
if (Parser.getTok().isNot(AsmToken::Dollar))
return MatchOperand_NoMatch;
SMLoc S = Parser.getTok().getLoc();
Parser.Lex(); // Eat the '$'
const AsmToken &Tok = Parser.getTok(); // Get next token.
if (Tok.isNot(AsmToken::Identifier))
return MatchOperand_NoMatch;
if (!Tok.getIdentifier().startswith("fcc"))
return MatchOperand_NoMatch;
StringRef NumString = Tok.getIdentifier().substr(3);
unsigned IntVal;
if (NumString.getAsInteger(10, IntVal))
return MatchOperand_NoMatch;
unsigned Reg = matchRegisterByNumber(IntVal, Mips::FCCRegClassID);
MipsOperand *Op = MipsOperand::CreateReg(Reg, S, Parser.getTok().getLoc());
Op->setRegKind(MipsOperand::Kind_FCCRegs);
Operands.push_back(Op);
Parser.Lex(); // Eat the register number.
return MatchOperand_Success;
} }
MipsAsmParser::OperandMatchResultTy MipsAsmParser::OperandMatchResultTy
MipsAsmParser::parseACC64DSP(SmallVectorImpl<MCParsedAsmOperand*> &Operands) { MipsAsmParser::parseACC64DSP(SmallVectorImpl<MCParsedAsmOperand*> &Operands) {
// If the first token is not '$' we have an error. return parseRegs(Operands, (int) MipsOperand::Kind_ACC64DSP);
if (Parser.getTok().isNot(AsmToken::Dollar))
return MatchOperand_NoMatch;
SMLoc S = Parser.getTok().getLoc();
Parser.Lex(); // Eat the '$'
const AsmToken &Tok = Parser.getTok(); // Get next token.
if (Tok.isNot(AsmToken::Identifier))
return MatchOperand_NoMatch;
if (!Tok.getIdentifier().startswith("acc"))
return MatchOperand_NoMatch;
StringRef NumString = Tok.getIdentifier().substr(3);
unsigned IntVal;
if (NumString.getAsInteger(10, IntVal))
return MatchOperand_NoMatch;
unsigned Reg = matchRegisterByNumber(IntVal, Mips::ACC64DSPRegClassID);
MipsOperand *Op = MipsOperand::CreateReg(Reg, S, Parser.getTok().getLoc());
Op->setRegKind(MipsOperand::Kind_ACC64DSP);
Operands.push_back(Op);
Parser.Lex(); // Eat the register number.
return MatchOperand_Success;
} }
bool MipsAsmParser::searchSymbolAlias( bool MipsAsmParser::searchSymbolAlias(
@ -1428,17 +1437,19 @@ bool MipsAsmParser::searchSymbolAlias(
switch (Kind) { switch (Kind) {
case MipsOperand::Kind_AFGR64Regs: case MipsOperand::Kind_AFGR64Regs:
case MipsOperand::Kind_FGR64Regs: case MipsOperand::Kind_FGR64Regs:
RegNum = matchFPURegisterName(DefSymbol.substr(1), FP_FORMAT_D); RegNum = matchFPURegisterName(DefSymbol.substr(1));
break; break;
case MipsOperand::Kind_FGR32Regs: case MipsOperand::Kind_FGR32Regs:
RegNum = matchFPURegisterName(DefSymbol.substr(1), FP_FORMAT_S); RegNum = matchFPURegisterName(DefSymbol.substr(1));
break; break;
case MipsOperand::Kind_GPR64: case MipsOperand::Kind_GPR64:
case MipsOperand::Kind_GPR32: case MipsOperand::Kind_GPR32:
default: default:
RegNum = matchRegisterName(DefSymbol.substr(1), isMips64()); RegNum = matchCPURegisterName(DefSymbol.substr(1));
break; break;
} }
if (RegNum > -1)
RegNum = getReg(regKindToRegClass(Kind), RegNum);
} }
if (RegNum > -1) { if (RegNum > -1) {
Parser.Lex(); Parser.Lex();
@ -1463,53 +1474,12 @@ bool MipsAsmParser::searchSymbolAlias(
MipsAsmParser::OperandMatchResultTy MipsAsmParser::OperandMatchResultTy
MipsAsmParser::parseHWRegs(SmallVectorImpl<MCParsedAsmOperand*> &Operands) { MipsAsmParser::parseHWRegs(SmallVectorImpl<MCParsedAsmOperand*> &Operands) {
return parseRegs(Operands, (int) MipsOperand::Kind_HWRegs);
// If the first token is not '$' we have error.
if (Parser.getTok().isNot(AsmToken::Dollar))
return MatchOperand_NoMatch;
SMLoc S = Parser.getTok().getLoc();
Parser.Lex(); // Eat the '$'.
const AsmToken &Tok = Parser.getTok(); // Get the next token.
if (Tok.isNot(AsmToken::Integer))
return MatchOperand_NoMatch;
unsigned RegNum = Tok.getIntVal();
// At the moment only hwreg29 is supported.
if (RegNum != 29)
return MatchOperand_ParseFail;
MipsOperand *op = MipsOperand::CreateReg(Mips::HWR29, S,
Parser.getTok().getLoc());
op->setRegKind(MipsOperand::Kind_HWRegs);
Operands.push_back(op);
Parser.Lex(); // Eat the register number.
return MatchOperand_Success;
} }
MipsAsmParser::OperandMatchResultTy MipsAsmParser::OperandMatchResultTy
MipsAsmParser::parseCCRRegs(SmallVectorImpl<MCParsedAsmOperand*> &Operands) { MipsAsmParser::parseCCRRegs(SmallVectorImpl<MCParsedAsmOperand*> &Operands) {
// If the first token is not '$' we have an error. return parseRegs(Operands, (int) MipsOperand::Kind_CCRRegs);
if (Parser.getTok().isNot(AsmToken::Dollar))
return MatchOperand_NoMatch;
SMLoc S = Parser.getTok().getLoc();
Parser.Lex(); // Eat the '$'
const AsmToken &Tok = Parser.getTok(); // Get next token.
if (Tok.isNot(AsmToken::Integer))
return MatchOperand_NoMatch;
unsigned Reg = matchRegisterByNumber(Tok.getIntVal(), Mips::CCRRegClassID);
MipsOperand *Op = MipsOperand::CreateReg(Reg, S, Parser.getTok().getLoc());
Op->setRegKind(MipsOperand::Kind_CCRRegs);
Operands.push_back(Op);
Parser.Lex(); // Eat the register number.
return MatchOperand_Success;
} }
MCSymbolRefExpr::VariantKind MipsAsmParser::getVariantKind(StringRef Symbol) { MCSymbolRefExpr::VariantKind MipsAsmParser::getVariantKind(StringRef Symbol) {