Second attempt at Removing special form of AddRegFrm used by FP instructions. These instructions can be handled by MRMXr instead.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@198276 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Craig Topper 2014-01-01 14:22:37 +00:00
parent 0cc6c9b3d9
commit 979b2cd2bc
7 changed files with 90 additions and 131 deletions

View File

@ -685,8 +685,8 @@ static bool translateOperand(MCInst &mcInst, const OperandSpecifier &operand,
case ENCODING_RO: case ENCODING_RO:
translateRegister(mcInst, insn.opcodeRegister); translateRegister(mcInst, insn.opcodeRegister);
return false; return false;
case ENCODING_I: case ENCODING_FP:
return translateFPRegister(mcInst, insn.opcodeModifier); return translateFPRegister(mcInst, insn.modRM & 7);
case ENCODING_Rv: case ENCODING_Rv:
translateRegister(mcInst, insn.opcodeRegister); translateRegister(mcInst, insn.opcodeRegister);
return false; return false;

View File

@ -1526,9 +1526,6 @@ static int readOpcodeModifier(struct InternalInstruction* insn) {
case MODIFIER_OPCODE: case MODIFIER_OPCODE:
insn->opcodeModifier = insn->opcode - insn->spec->modifierBase; insn->opcodeModifier = insn->opcode - insn->spec->modifierBase;
return 0; return 0;
case MODIFIER_MODRM:
insn->opcodeModifier = insn->modRM - insn->spec->modifierBase;
return 0;
} }
} }
@ -1783,9 +1780,7 @@ static int readOperands(struct InternalInstruction* insn) {
if (readOpcodeRegister(insn, 0)) if (readOpcodeRegister(insn, 0))
return -1; return -1;
break; break;
case ENCODING_I: case ENCODING_FP:
if (readOpcodeModifier(insn))
return -1;
break; break;
case ENCODING_VVVV: case ENCODING_VVVV:
needVVVV = 0; /* Mark that we have found a VVVV operand. */ needVVVV = 0; /* Mark that we have found a VVVV operand. */

View File

@ -401,8 +401,8 @@ struct ContextDecision {
ENUM_ENTRY(ENCODING_RW, "(AX..DI, R8W..R15W)") \ ENUM_ENTRY(ENCODING_RW, "(AX..DI, R8W..R15W)") \
ENUM_ENTRY(ENCODING_RD, "(EAX..EDI, R8D..R15D)") \ ENUM_ENTRY(ENCODING_RD, "(EAX..EDI, R8D..R15D)") \
ENUM_ENTRY(ENCODING_RO, "(RAX..RDI, R8..R15)") \ ENUM_ENTRY(ENCODING_RO, "(RAX..RDI, R8..R15)") \
ENUM_ENTRY(ENCODING_I, "Position on floating-point stack added to the " \ ENUM_ENTRY(ENCODING_FP, "Position on floating-point stack in ModR/M " \
"opcode byte") \ "byte.") \
\ \
ENUM_ENTRY(ENCODING_Iv, "Immediate of operand size") \ ENUM_ENTRY(ENCODING_Iv, "Immediate of operand size") \
ENUM_ENTRY(ENCODING_Ia, "Immediate of address size") \ ENUM_ENTRY(ENCODING_Ia, "Immediate of address size") \
@ -526,8 +526,7 @@ struct OperandSpecifier {
#define MODIFIER_TYPES \ #define MODIFIER_TYPES \
ENUM_ENTRY(MODIFIER_NONE) \ ENUM_ENTRY(MODIFIER_NONE) \
ENUM_ENTRY(MODIFIER_OPCODE) \ ENUM_ENTRY(MODIFIER_OPCODE)
ENUM_ENTRY(MODIFIER_MODRM)
#define ENUM_ENTRY(n) n, #define ENUM_ENTRY(n) n,
typedef enum { typedef enum {

View File

@ -218,37 +218,37 @@ defm DIV : FPBinary<fdiv, MRM6m, "div">;
defm DIVR: FPBinary<fdiv, MRM7m, "divr">; defm DIVR: FPBinary<fdiv, MRM7m, "divr">;
} }
class FPST0rInst<bits<8> o, string asm> class FPST0rInst<Format fp, string asm>
: FPI<o, AddRegFrm, (outs), (ins RST:$op), asm>, D8; : FPI<0xD8, fp, (outs), (ins RST:$op), asm>;
class FPrST0Inst<bits<8> o, string asm> class FPrST0Inst<Format fp, string asm>
: FPI<o, AddRegFrm, (outs), (ins RST:$op), asm>, DC; : FPI<0xDC, fp, (outs), (ins RST:$op), asm>;
class FPrST0PInst<bits<8> o, string asm> class FPrST0PInst<Format fp, string asm>
: FPI<o, AddRegFrm, (outs), (ins RST:$op), asm>, DE; : FPI<0xDE, fp, (outs), (ins RST:$op), asm>;
// NOTE: GAS and apparently all other AT&T style assemblers have a broken notion // NOTE: GAS and apparently all other AT&T style assemblers have a broken notion
// of some of the 'reverse' forms of the fsub and fdiv instructions. As such, // of some of the 'reverse' forms of the fsub and fdiv instructions. As such,
// we have to put some 'r's in and take them out of weird places. // we have to put some 'r's in and take them out of weird places.
def ADD_FST0r : FPST0rInst <0xC0, "fadd\t$op">; def ADD_FST0r : FPST0rInst <MRM0r, "fadd\t$op">;
def ADD_FrST0 : FPrST0Inst <0xC0, "fadd\t{%st(0), $op|$op, st(0)}">; def ADD_FrST0 : FPrST0Inst <MRM0r, "fadd\t{%st(0), $op|$op, st(0)}">;
def ADD_FPrST0 : FPrST0PInst<0xC0, "faddp\t$op">; def ADD_FPrST0 : FPrST0PInst<MRM0r, "faddp\t$op">;
def SUBR_FST0r : FPST0rInst <0xE8, "fsubr\t$op">; def SUBR_FST0r : FPST0rInst <MRM5r, "fsubr\t$op">;
def SUB_FrST0 : FPrST0Inst <0xE8, "fsub{r}\t{%st(0), $op|$op, st(0)}">; def SUB_FrST0 : FPrST0Inst <MRM5r, "fsub{r}\t{%st(0), $op|$op, st(0)}">;
def SUB_FPrST0 : FPrST0PInst<0xE8, "fsub{r}p\t$op">; def SUB_FPrST0 : FPrST0PInst<MRM5r, "fsub{r}p\t$op">;
def SUB_FST0r : FPST0rInst <0xE0, "fsub\t$op">; def SUB_FST0r : FPST0rInst <MRM4r, "fsub\t$op">;
def SUBR_FrST0 : FPrST0Inst <0xE0, "fsub{|r}\t{%st(0), $op|$op, st(0)}">; def SUBR_FrST0 : FPrST0Inst <MRM4r, "fsub{|r}\t{%st(0), $op|$op, st(0)}">;
def SUBR_FPrST0 : FPrST0PInst<0xE0, "fsub{|r}p\t$op">; def SUBR_FPrST0 : FPrST0PInst<MRM4r, "fsub{|r}p\t$op">;
def MUL_FST0r : FPST0rInst <0xC8, "fmul\t$op">; def MUL_FST0r : FPST0rInst <MRM1r, "fmul\t$op">;
def MUL_FrST0 : FPrST0Inst <0xC8, "fmul\t{%st(0), $op|$op, st(0)}">; def MUL_FrST0 : FPrST0Inst <MRM1r, "fmul\t{%st(0), $op|$op, st(0)}">;
def MUL_FPrST0 : FPrST0PInst<0xC8, "fmulp\t$op">; def MUL_FPrST0 : FPrST0PInst<MRM1r, "fmulp\t$op">;
def DIVR_FST0r : FPST0rInst <0xF8, "fdivr\t$op">; def DIVR_FST0r : FPST0rInst <MRM7r, "fdivr\t$op">;
def DIV_FrST0 : FPrST0Inst <0xF8, "fdiv{r}\t{%st(0), $op|$op, st(0)}">; def DIV_FrST0 : FPrST0Inst <MRM7r, "fdiv{r}\t{%st(0), $op|$op, st(0)}">;
def DIV_FPrST0 : FPrST0PInst<0xF8, "fdiv{r}p\t$op">; def DIV_FPrST0 : FPrST0PInst<MRM7r, "fdiv{r}p\t$op">;
def DIV_FST0r : FPST0rInst <0xF0, "fdiv\t$op">; def DIV_FST0r : FPST0rInst <MRM6r, "fdiv\t$op">;
def DIVR_FrST0 : FPrST0Inst <0xF0, "fdiv{|r}\t{%st(0), $op|$op, st(0)}">; def DIVR_FrST0 : FPrST0Inst <MRM6r, "fdiv{|r}\t{%st(0), $op|$op, st(0)}">;
def DIVR_FPrST0 : FPrST0PInst<0xF0, "fdiv{|r}p\t$op">; def DIVR_FPrST0 : FPrST0PInst<MRM6r, "fdiv{|r}p\t$op">;
def COM_FST0r : FPST0rInst <0xD0, "fcom\t$op">; def COM_FST0r : FPST0rInst <MRM2r, "fcom\t$op">;
def COMP_FST0r : FPST0rInst <0xD8, "fcomp\t$op">; def COMP_FST0r : FPST0rInst <MRM3r, "fcomp\t$op">;
// Unary operations. // Unary operations.
multiclass FPUnary<SDNode OpNode, bits<8> opcode, string asmstring> { multiclass FPUnary<SDNode OpNode, bits<8> opcode, string asmstring> {
@ -336,22 +336,22 @@ defm CMOVNP : FPCMov<X86_COND_NP>;
let Predicates = [HasCMov] in { let Predicates = [HasCMov] in {
// These are not factored because there's no clean way to pass DA/DB. // These are not factored because there's no clean way to pass DA/DB.
def CMOVB_F : FPI<0xC0, AddRegFrm, (outs RST:$op), (ins), def CMOVB_F : FPI<0xDA, MRM0r, (outs RST:$op), (ins),
"fcmovb\t{$op, %st(0)|st(0), $op}">, DA; "fcmovb\t{$op, %st(0)|st(0), $op}">;
def CMOVBE_F : FPI<0xD0, AddRegFrm, (outs RST:$op), (ins), def CMOVBE_F : FPI<0xDA, MRM2r, (outs RST:$op), (ins),
"fcmovbe\t{$op, %st(0)|st(0), $op}">, DA; "fcmovbe\t{$op, %st(0)|st(0), $op}">;
def CMOVE_F : FPI<0xC8, AddRegFrm, (outs RST:$op), (ins), def CMOVE_F : FPI<0xDA, MRM1r, (outs RST:$op), (ins),
"fcmove\t{$op, %st(0)|st(0), $op}">, DA; "fcmove\t{$op, %st(0)|st(0), $op}">;
def CMOVP_F : FPI<0xD8, AddRegFrm, (outs RST:$op), (ins), def CMOVP_F : FPI<0xDA, MRM3r, (outs RST:$op), (ins),
"fcmovu\t{$op, %st(0)|st(0), $op}">, DA; "fcmovu\t{$op, %st(0)|st(0), $op}">;
def CMOVNB_F : FPI<0xC0, AddRegFrm, (outs RST:$op), (ins), def CMOVNB_F : FPI<0xDB, MRM0r, (outs RST:$op), (ins),
"fcmovnb\t{$op, %st(0)|st(0), $op}">, DB; "fcmovnb\t{$op, %st(0)|st(0), $op}">;
def CMOVNBE_F: FPI<0xD0, AddRegFrm, (outs RST:$op), (ins), def CMOVNBE_F: FPI<0xDB, MRM2r, (outs RST:$op), (ins),
"fcmovnbe\t{$op, %st(0)|st(0), $op}">, DB; "fcmovnbe\t{$op, %st(0)|st(0), $op}">;
def CMOVNE_F : FPI<0xC8, AddRegFrm, (outs RST:$op), (ins), def CMOVNE_F : FPI<0xDB, MRM1r, (outs RST:$op), (ins),
"fcmovne\t{$op, %st(0)|st(0), $op}">, DB; "fcmovne\t{$op, %st(0)|st(0), $op}">;
def CMOVNP_F : FPI<0xD8, AddRegFrm, (outs RST:$op), (ins), def CMOVNP_F : FPI<0xDB, MRM3r, (outs RST:$op), (ins),
"fcmovnu\t{$op, %st(0)|st(0), $op}">, DB; "fcmovnu\t{$op, %st(0)|st(0), $op}">;
} // Predicates = [HasCMov] } // Predicates = [HasCMov]
// Floating point loads & stores. // Floating point loads & stores.
@ -492,14 +492,10 @@ def ISTT_FP64m : FPI<0xDD, MRM1m, (outs), (ins i64mem:$dst),
// FP Stack manipulation instructions. // FP Stack manipulation instructions.
let SchedRW = [WriteMove] in { let SchedRW = [WriteMove] in {
def LD_Frr : FPI<0xC0, AddRegFrm, (outs), (ins RST:$op), "fld\t$op", def LD_Frr : FPI<0xD9, MRM0r, (outs), (ins RST:$op), "fld\t$op", IIC_FLD>;
IIC_FLD>, D9; def ST_Frr : FPI<0xDD, MRM2r, (outs), (ins RST:$op), "fst\t$op", IIC_FST>;
def ST_Frr : FPI<0xD0, AddRegFrm, (outs), (ins RST:$op), "fst\t$op", def ST_FPrr : FPI<0xDD, MRM3r, (outs), (ins RST:$op), "fstp\t$op", IIC_FST>;
IIC_FST>, DD; def XCH_F : FPI<0xD9, MRM1r, (outs), (ins RST:$op), "fxch\t$op", IIC_FXCH>;
def ST_FPrr : FPI<0xD8, AddRegFrm, (outs), (ins RST:$op), "fstp\t$op",
IIC_FST>, DD;
def XCH_F : FPI<0xC8, AddRegFrm, (outs), (ins RST:$op), "fxch\t$op",
IIC_FXCH>, D9;
} }
// Floating point constant loads. // Floating point constant loads.
@ -546,31 +542,26 @@ def UCOM_FpIr80: FpI_<(outs), (ins RFP80:$lhs, RFP80:$rhs), CompareFP,
} }
let Defs = [FPSW], Uses = [ST0] in { let Defs = [FPSW], Uses = [ST0] in {
def UCOM_Fr : FPI<0xE0, AddRegFrm, // FPSW = cmp ST(0) with ST(i) def UCOM_Fr : FPI<0xDD, MRM4r, // FPSW = cmp ST(0) with ST(i)
(outs), (ins RST:$reg), (outs), (ins RST:$reg), "fucom\t$reg", IIC_FUCOM>;
"fucom\t$reg", IIC_FUCOM>, DD; def UCOM_FPr : FPI<0xDD, MRM5r, // FPSW = cmp ST(0) with ST(i), pop
def UCOM_FPr : FPI<0xE8, AddRegFrm, // FPSW = cmp ST(0) with ST(i), pop (outs), (ins RST:$reg), "fucomp\t$reg", IIC_FUCOM>;
(outs), (ins RST:$reg),
"fucomp\t$reg", IIC_FUCOM>, DD;
def UCOM_FPPr : FPI<0xE9, RawFrm, // cmp ST(0) with ST(1), pop, pop def UCOM_FPPr : FPI<0xE9, RawFrm, // cmp ST(0) with ST(1), pop, pop
(outs), (ins), (outs), (ins), "fucompp", IIC_FUCOM>, DA;
"fucompp", IIC_FUCOM>, DA;
} }
let Defs = [EFLAGS, FPSW], Uses = [ST0] in { let Defs = [EFLAGS, FPSW], Uses = [ST0] in {
def UCOM_FIr : FPI<0xE8, AddRegFrm, // CC = cmp ST(0) with ST(i) def UCOM_FIr : FPI<0xDB, MRM5r, // CC = cmp ST(0) with ST(i)
(outs), (ins RST:$reg), (outs), (ins RST:$reg), "fucomi\t$reg", IIC_FUCOMI>;
"fucomi\t$reg", IIC_FUCOMI>, DB; def UCOM_FIPr : FPI<0xDF, MRM5r, // CC = cmp ST(0) with ST(i), pop
def UCOM_FIPr : FPI<0xE8, AddRegFrm, // CC = cmp ST(0) with ST(i), pop (outs), (ins RST:$reg), "fucompi\t$reg", IIC_FUCOMI>;
(outs), (ins RST:$reg),
"fucompi\t$reg", IIC_FUCOMI>, DF;
} }
let Defs = [EFLAGS, FPSW] in { let Defs = [EFLAGS, FPSW] in {
def COM_FIr : FPI<0xF0, AddRegFrm, (outs), (ins RST:$reg), def COM_FIr : FPI<0xDB, MRM6r, (outs), (ins RST:$reg),
"fcomi\t$reg", IIC_FCOMI>, DB; "fcomi\t$reg", IIC_FCOMI>;
def COM_FIPr : FPI<0xF0, AddRegFrm, (outs), (ins RST:$reg), def COM_FIPr : FPI<0xDF, MRM6r, (outs), (ins RST:$reg),
"fcompi\t$reg", IIC_FCOMI>, DF; "fcompi\t$reg", IIC_FCOMI>;
} }
} // SchedRW } // SchedRW
@ -594,8 +585,8 @@ def FLDCW16m : I<0xD9, MRM5m, // X87 control world = [mem16]
let SchedRW = [WriteMicrocoded] in { let SchedRW = [WriteMicrocoded] in {
let Defs = [FPSW] in let Defs = [FPSW] in
def FNINIT : I<0xE3, RawFrm, (outs), (ins), "fninit", [], IIC_FNINIT>, DB; def FNINIT : I<0xE3, RawFrm, (outs), (ins), "fninit", [], IIC_FNINIT>, DB;
def FFREE : FPI<0xC0, AddRegFrm, (outs), (ins RST:$reg), def FFREE : FPI<0xDD, MRM0r, (outs), (ins RST:$reg),
"ffree\t$reg", IIC_FFREE>, DD; "ffree\t$reg", IIC_FFREE>;
// Clear exceptions // Clear exceptions
let Defs = [FPSW] in let Defs = [FPSW] in

View File

@ -17,8 +17,6 @@ void DumbFilter::anchor() { }
void ModFilter::anchor() { } void ModFilter::anchor() { }
void AddRegEscapeFilter::anchor() { }
void ExtendedFilter::anchor() { } void ExtendedFilter::anchor() { }
void ExactFilter::anchor() { } void ExactFilter::anchor() { }

View File

@ -84,26 +84,6 @@ public:
} }
}; };
/// AddRegEscapeFilter - Some escape opcodes have one of the register operands
/// added to the ModR/M byte, meaning that a range of eight ModR/M values
/// maps to a single instruction. Such instructions require the ModR/M byte
/// to fall between 0xc0 and 0xff.
class AddRegEscapeFilter : public ModRMFilter {
virtual void anchor();
uint8_t ModRM;
public:
/// Constructor
///
/// \param modRM The value of the ModR/M byte when the register operand
/// refers to the first register in the register set.
AddRegEscapeFilter(uint8_t modRM) : ModRM(modRM) {
}
bool accepts(uint8_t modRM) const {
return (modRM >= ModRM && modRM < ModRM + 8);
}
};
/// ExtendedFilter - Extended opcodes are classified based on the value of the /// ExtendedFilter - Extended opcodes are classified based on the value of the
/// mod field [bits 7-6] and the value of the nnn field [bits 5-3]. /// mod field [bits 7-6] and the value of the nnn field [bits 5-3].
class ExtendedFilter : public ModRMFilter { class ExtendedFilter : public ModRMFilter {

View File

@ -1075,14 +1075,9 @@ void RecognizableInstr::emitDecodePath(DisassemblerTables &tables) const {
case X86Local::DE: case X86Local::DE:
case X86Local::DF: case X86Local::DF:
assert(Opcode >= 0xc0 && "Unexpected opcode for an escape opcode"); assert(Opcode >= 0xc0 && "Unexpected opcode for an escape opcode");
assert(Form == X86Local::RawFrm);
opcodeType = ONEBYTE; opcodeType = ONEBYTE;
if (Form == X86Local::AddRegFrm) { filter = new ExactFilter(Opcode);
Spec->modifierType = MODIFIER_MODRM;
Spec->modifierBase = Opcode;
filter = new AddRegEscapeFilter(Opcode);
} else {
filter = new ExactFilter(Opcode);
}
opcodeToSet = 0xd8 + (Prefix - X86Local::D8); opcodeToSet = 0xd8 + (Prefix - X86Local::D8);
break; break;
case X86Local::REP: case X86Local::REP:
@ -1130,6 +1125,16 @@ void RecognizableInstr::emitDecodePath(DisassemblerTables &tables) const {
switch (Form) { switch (Form) {
default: default:
llvm_unreachable("Unhandled escape opcode form"); llvm_unreachable("Unhandled escape opcode form");
case X86Local::MRM0r:
case X86Local::MRM1r:
case X86Local::MRM2r:
case X86Local::MRM3r:
case X86Local::MRM4r:
case X86Local::MRM5r:
case X86Local::MRM6r:
case X86Local::MRM7r:
filter = new ExtendedFilter(true, Form - X86Local::MRM0r);
break;
case X86Local::MRM0m: case X86Local::MRM0m:
case X86Local::MRM1m: case X86Local::MRM1m:
case X86Local::MRM2m: case X86Local::MRM2m:
@ -1157,31 +1162,22 @@ void RecognizableInstr::emitDecodePath(DisassemblerTables &tables) const {
assert(filter && "Filter not set"); assert(filter && "Filter not set");
if (Form == X86Local::AddRegFrm) { if (Form == X86Local::AddRegFrm) {
if(Spec->modifierType != MODIFIER_MODRM) { assert(opcodeToSet < 0xf9 &&
assert(opcodeToSet < 0xf9 && "Not enough room for all ADDREG_FRM operands");
"Not enough room for all ADDREG_FRM operands");
uint8_t currentOpcode; uint8_t currentOpcode;
for (currentOpcode = opcodeToSet; for (currentOpcode = opcodeToSet;
currentOpcode < opcodeToSet + 8; currentOpcode < opcodeToSet + 8;
++currentOpcode) ++currentOpcode)
tables.setTableFields(opcodeType,
insnContext(),
currentOpcode,
*filter,
UID, Is32Bit, IgnoresVEX_L);
Spec->modifierType = MODIFIER_OPCODE;
Spec->modifierBase = opcodeToSet;
} else {
// modifierBase was set where MODIFIER_MODRM was set
tables.setTableFields(opcodeType, tables.setTableFields(opcodeType,
insnContext(), insnContext(),
opcodeToSet, currentOpcode,
*filter, *filter,
UID, Is32Bit, IgnoresVEX_L); UID, Is32Bit, IgnoresVEX_L);
}
Spec->modifierType = MODIFIER_OPCODE;
Spec->modifierBase = opcodeToSet;
} else { } else {
tables.setTableFields(opcodeType, tables.setTableFields(opcodeType,
insnContext(), insnContext(),
@ -1341,6 +1337,7 @@ OperandEncoding RecognizableInstr::immediateEncodingFromString
OperandEncoding RecognizableInstr::rmRegisterEncodingFromString OperandEncoding RecognizableInstr::rmRegisterEncodingFromString
(const std::string &s, (const std::string &s,
bool hasOpSizePrefix) { bool hasOpSizePrefix) {
ENCODING("RST", ENCODING_FP)
ENCODING("GR16", ENCODING_RM) ENCODING("GR16", ENCODING_RM)
ENCODING("GR32", ENCODING_RM) ENCODING("GR32", ENCODING_RM)
ENCODING("GR32orGR64", ENCODING_RM) ENCODING("GR32orGR64", ENCODING_RM)
@ -1493,7 +1490,6 @@ OperandEncoding RecognizableInstr::relocationEncodingFromString
OperandEncoding RecognizableInstr::opcodeModifierEncodingFromString OperandEncoding RecognizableInstr::opcodeModifierEncodingFromString
(const std::string &s, (const std::string &s,
bool hasOpSizePrefix) { bool hasOpSizePrefix) {
ENCODING("RST", ENCODING_I)
ENCODING("GR32", ENCODING_Rv) ENCODING("GR32", ENCODING_Rv)
ENCODING("GR64", ENCODING_RO) ENCODING("GR64", ENCODING_RO)
ENCODING("GR16", ENCODING_Rv) ENCODING("GR16", ENCODING_Rv)