mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-23 11:13:28 +01:00
Revert r198238 and add FP disassembler tests. It didn't work and I didn't realized we had no FP disassembler test cases.
llvm-svn: 198265
This commit is contained in:
parent
21846ff5d8
commit
d0374cb642
@ -1526,6 +1526,9 @@ static int readOpcodeModifier(struct InternalInstruction* insn) {
|
||||
case MODIFIER_OPCODE:
|
||||
insn->opcodeModifier = insn->opcode - insn->spec->modifierBase;
|
||||
return 0;
|
||||
case MODIFIER_MODRM:
|
||||
insn->opcodeModifier = insn->modRM - insn->spec->modifierBase;
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -526,7 +526,8 @@ struct OperandSpecifier {
|
||||
|
||||
#define MODIFIER_TYPES \
|
||||
ENUM_ENTRY(MODIFIER_NONE) \
|
||||
ENUM_ENTRY(MODIFIER_OPCODE)
|
||||
ENUM_ENTRY(MODIFIER_OPCODE) \
|
||||
ENUM_ENTRY(MODIFIER_MODRM)
|
||||
|
||||
#define ENUM_ENTRY(n) n,
|
||||
typedef enum {
|
||||
|
@ -218,37 +218,37 @@ defm DIV : FPBinary<fdiv, MRM6m, "div">;
|
||||
defm DIVR: FPBinary<fdiv, MRM7m, "divr">;
|
||||
}
|
||||
|
||||
class FPST0rInst<Format fp, string asm>
|
||||
: FPI<0xD8, fp, (outs), (ins RST:$op), asm>;
|
||||
class FPrST0Inst<Format fp, string asm>
|
||||
: FPI<0xDC, fp, (outs), (ins RST:$op), asm>;
|
||||
class FPrST0PInst<Format fp, string asm>
|
||||
: FPI<0xDE, fp, (outs), (ins RST:$op), asm>;
|
||||
class FPST0rInst<bits<8> o, string asm>
|
||||
: FPI<o, AddRegFrm, (outs), (ins RST:$op), asm>, D8;
|
||||
class FPrST0Inst<bits<8> o, string asm>
|
||||
: FPI<o, AddRegFrm, (outs), (ins RST:$op), asm>, DC;
|
||||
class FPrST0PInst<bits<8> o, string asm>
|
||||
: FPI<o, AddRegFrm, (outs), (ins RST:$op), asm>, DE;
|
||||
|
||||
// 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,
|
||||
// we have to put some 'r's in and take them out of weird places.
|
||||
def ADD_FST0r : FPST0rInst <MRM0r, "fadd\t$op">;
|
||||
def ADD_FrST0 : FPrST0Inst <MRM0r, "fadd\t{%st(0), $op|$op, st(0)}">;
|
||||
def ADD_FPrST0 : FPrST0PInst<MRM0r, "faddp\t$op">;
|
||||
def SUBR_FST0r : FPST0rInst <MRM5r, "fsubr\t$op">;
|
||||
def SUB_FrST0 : FPrST0Inst <MRM5r, "fsub{r}\t{%st(0), $op|$op, st(0)}">;
|
||||
def SUB_FPrST0 : FPrST0PInst<MRM5r, "fsub{r}p\t$op">;
|
||||
def SUB_FST0r : FPST0rInst <MRM4r, "fsub\t$op">;
|
||||
def SUBR_FrST0 : FPrST0Inst <MRM4r, "fsub{|r}\t{%st(0), $op|$op, st(0)}">;
|
||||
def SUBR_FPrST0 : FPrST0PInst<MRM4r, "fsub{|r}p\t$op">;
|
||||
def MUL_FST0r : FPST0rInst <MRM1r, "fmul\t$op">;
|
||||
def MUL_FrST0 : FPrST0Inst <MRM1r, "fmul\t{%st(0), $op|$op, st(0)}">;
|
||||
def MUL_FPrST0 : FPrST0PInst<MRM1r, "fmulp\t$op">;
|
||||
def DIVR_FST0r : FPST0rInst <MRM7r, "fdivr\t$op">;
|
||||
def DIV_FrST0 : FPrST0Inst <MRM7r, "fdiv{r}\t{%st(0), $op|$op, st(0)}">;
|
||||
def DIV_FPrST0 : FPrST0PInst<MRM7r, "fdiv{r}p\t$op">;
|
||||
def DIV_FST0r : FPST0rInst <MRM6r, "fdiv\t$op">;
|
||||
def DIVR_FrST0 : FPrST0Inst <MRM6r, "fdiv{|r}\t{%st(0), $op|$op, st(0)}">;
|
||||
def DIVR_FPrST0 : FPrST0PInst<MRM6r, "fdiv{|r}p\t$op">;
|
||||
def ADD_FST0r : FPST0rInst <0xC0, "fadd\t$op">;
|
||||
def ADD_FrST0 : FPrST0Inst <0xC0, "fadd\t{%st(0), $op|$op, st(0)}">;
|
||||
def ADD_FPrST0 : FPrST0PInst<0xC0, "faddp\t$op">;
|
||||
def SUBR_FST0r : FPST0rInst <0xE8, "fsubr\t$op">;
|
||||
def SUB_FrST0 : FPrST0Inst <0xE8, "fsub{r}\t{%st(0), $op|$op, st(0)}">;
|
||||
def SUB_FPrST0 : FPrST0PInst<0xE8, "fsub{r}p\t$op">;
|
||||
def SUB_FST0r : FPST0rInst <0xE0, "fsub\t$op">;
|
||||
def SUBR_FrST0 : FPrST0Inst <0xE0, "fsub{|r}\t{%st(0), $op|$op, st(0)}">;
|
||||
def SUBR_FPrST0 : FPrST0PInst<0xE0, "fsub{|r}p\t$op">;
|
||||
def MUL_FST0r : FPST0rInst <0xC8, "fmul\t$op">;
|
||||
def MUL_FrST0 : FPrST0Inst <0xC8, "fmul\t{%st(0), $op|$op, st(0)}">;
|
||||
def MUL_FPrST0 : FPrST0PInst<0xC8, "fmulp\t$op">;
|
||||
def DIVR_FST0r : FPST0rInst <0xF8, "fdivr\t$op">;
|
||||
def DIV_FrST0 : FPrST0Inst <0xF8, "fdiv{r}\t{%st(0), $op|$op, st(0)}">;
|
||||
def DIV_FPrST0 : FPrST0PInst<0xF8, "fdiv{r}p\t$op">;
|
||||
def DIV_FST0r : FPST0rInst <0xF0, "fdiv\t$op">;
|
||||
def DIVR_FrST0 : FPrST0Inst <0xF0, "fdiv{|r}\t{%st(0), $op|$op, st(0)}">;
|
||||
def DIVR_FPrST0 : FPrST0PInst<0xF0, "fdiv{|r}p\t$op">;
|
||||
|
||||
def COM_FST0r : FPST0rInst <MRM2r, "fcom\t$op">;
|
||||
def COMP_FST0r : FPST0rInst <MRM3r, "fcomp\t$op">;
|
||||
def COM_FST0r : FPST0rInst <0xD0, "fcom\t$op">;
|
||||
def COMP_FST0r : FPST0rInst <0xD8, "fcomp\t$op">;
|
||||
|
||||
// Unary operations.
|
||||
multiclass FPUnary<SDNode OpNode, bits<8> opcode, string asmstring> {
|
||||
@ -336,22 +336,22 @@ defm CMOVNP : FPCMov<X86_COND_NP>;
|
||||
|
||||
let Predicates = [HasCMov] in {
|
||||
// These are not factored because there's no clean way to pass DA/DB.
|
||||
def CMOVB_F : FPI<0xDA, MRM0r, (outs RST:$op), (ins),
|
||||
"fcmovb\t{$op, %st(0)|st(0), $op}">;
|
||||
def CMOVBE_F : FPI<0xDA, MRM2r, (outs RST:$op), (ins),
|
||||
"fcmovbe\t{$op, %st(0)|st(0), $op}">;
|
||||
def CMOVE_F : FPI<0xDA, MRM1r, (outs RST:$op), (ins),
|
||||
"fcmove\t{$op, %st(0)|st(0), $op}">;
|
||||
def CMOVP_F : FPI<0xDA, MRM3r, (outs RST:$op), (ins),
|
||||
"fcmovu\t{$op, %st(0)|st(0), $op}">;
|
||||
def CMOVNB_F : FPI<0xDB, MRM0r, (outs RST:$op), (ins),
|
||||
"fcmovnb\t{$op, %st(0)|st(0), $op}">;
|
||||
def CMOVNBE_F: FPI<0xDB, MRM2r, (outs RST:$op), (ins),
|
||||
"fcmovnbe\t{$op, %st(0)|st(0), $op}">;
|
||||
def CMOVNE_F : FPI<0xDB, MRM1r, (outs RST:$op), (ins),
|
||||
"fcmovne\t{$op, %st(0)|st(0), $op}">;
|
||||
def CMOVNP_F : FPI<0xDB, MRM3r, (outs RST:$op), (ins),
|
||||
"fcmovnu\t{$op, %st(0)|st(0), $op}">;
|
||||
def CMOVB_F : FPI<0xC0, AddRegFrm, (outs RST:$op), (ins),
|
||||
"fcmovb\t{$op, %st(0)|st(0), $op}">, DA;
|
||||
def CMOVBE_F : FPI<0xD0, AddRegFrm, (outs RST:$op), (ins),
|
||||
"fcmovbe\t{$op, %st(0)|st(0), $op}">, DA;
|
||||
def CMOVE_F : FPI<0xC8, AddRegFrm, (outs RST:$op), (ins),
|
||||
"fcmove\t{$op, %st(0)|st(0), $op}">, DA;
|
||||
def CMOVP_F : FPI<0xD8, AddRegFrm, (outs RST:$op), (ins),
|
||||
"fcmovu\t{$op, %st(0)|st(0), $op}">, DA;
|
||||
def CMOVNB_F : FPI<0xC0, AddRegFrm, (outs RST:$op), (ins),
|
||||
"fcmovnb\t{$op, %st(0)|st(0), $op}">, DB;
|
||||
def CMOVNBE_F: FPI<0xD0, AddRegFrm, (outs RST:$op), (ins),
|
||||
"fcmovnbe\t{$op, %st(0)|st(0), $op}">, DB;
|
||||
def CMOVNE_F : FPI<0xC8, AddRegFrm, (outs RST:$op), (ins),
|
||||
"fcmovne\t{$op, %st(0)|st(0), $op}">, DB;
|
||||
def CMOVNP_F : FPI<0xD8, AddRegFrm, (outs RST:$op), (ins),
|
||||
"fcmovnu\t{$op, %st(0)|st(0), $op}">, DB;
|
||||
} // Predicates = [HasCMov]
|
||||
|
||||
// Floating point loads & stores.
|
||||
@ -492,10 +492,14 @@ def ISTT_FP64m : FPI<0xDD, MRM1m, (outs), (ins i64mem:$dst),
|
||||
|
||||
// FP Stack manipulation instructions.
|
||||
let SchedRW = [WriteMove] in {
|
||||
def LD_Frr : FPI<0xD9, MRM0r, (outs), (ins RST:$op), "fld\t$op", IIC_FLD>;
|
||||
def ST_Frr : FPI<0xDD, MRM2r, (outs), (ins RST:$op), "fst\t$op", IIC_FST>;
|
||||
def ST_FPrr : FPI<0xDD, MRM3r, (outs), (ins RST:$op), "fstp\t$op", IIC_FST>;
|
||||
def XCH_F : FPI<0xD9, MRM1r, (outs), (ins RST:$op), "fxch\t$op", IIC_FXCH>;
|
||||
def LD_Frr : FPI<0xC0, AddRegFrm, (outs), (ins RST:$op), "fld\t$op",
|
||||
IIC_FLD>, D9;
|
||||
def ST_Frr : FPI<0xD0, AddRegFrm, (outs), (ins RST:$op), "fst\t$op",
|
||||
IIC_FST>, DD;
|
||||
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.
|
||||
@ -542,26 +546,31 @@ def UCOM_FpIr80: FpI_<(outs), (ins RFP80:$lhs, RFP80:$rhs), CompareFP,
|
||||
}
|
||||
|
||||
let Defs = [FPSW], Uses = [ST0] in {
|
||||
def UCOM_Fr : FPI<0xDD, MRM4r, // FPSW = cmp ST(0) with ST(i)
|
||||
(outs), (ins RST:$reg), "fucom\t$reg", IIC_FUCOM>;
|
||||
def UCOM_FPr : FPI<0xDD, MRM5r, // FPSW = cmp ST(0) with ST(i), pop
|
||||
(outs), (ins RST:$reg), "fucomp\t$reg", IIC_FUCOM>;
|
||||
def UCOM_Fr : FPI<0xE0, AddRegFrm, // FPSW = cmp ST(0) with ST(i)
|
||||
(outs), (ins RST:$reg),
|
||||
"fucom\t$reg", IIC_FUCOM>, DD;
|
||||
def UCOM_FPr : FPI<0xE8, AddRegFrm, // FPSW = cmp ST(0) with ST(i), pop
|
||||
(outs), (ins RST:$reg),
|
||||
"fucomp\t$reg", IIC_FUCOM>, DD;
|
||||
def UCOM_FPPr : FPI<0xE9, RawFrm, // cmp ST(0) with ST(1), pop, pop
|
||||
(outs), (ins), "fucompp", IIC_FUCOM>, DA;
|
||||
(outs), (ins),
|
||||
"fucompp", IIC_FUCOM>, DA;
|
||||
}
|
||||
|
||||
let Defs = [EFLAGS, FPSW], Uses = [ST0] in {
|
||||
def UCOM_FIr : FPI<0xDB, MRM5r, // CC = cmp ST(0) with ST(i)
|
||||
(outs), (ins RST:$reg), "fucomi\t$reg", IIC_FUCOMI>;
|
||||
def UCOM_FIPr : FPI<0xDF, MRM5r, // CC = cmp ST(0) with ST(i), pop
|
||||
(outs), (ins RST:$reg), "fucompi\t$reg", IIC_FUCOMI>;
|
||||
def UCOM_FIr : FPI<0xE8, AddRegFrm, // CC = cmp ST(0) with ST(i)
|
||||
(outs), (ins RST:$reg),
|
||||
"fucomi\t$reg", IIC_FUCOMI>, DB;
|
||||
def UCOM_FIPr : FPI<0xE8, AddRegFrm, // CC = cmp ST(0) with ST(i), pop
|
||||
(outs), (ins RST:$reg),
|
||||
"fucompi\t$reg", IIC_FUCOMI>, DF;
|
||||
}
|
||||
|
||||
let Defs = [EFLAGS, FPSW] in {
|
||||
def COM_FIr : FPI<0xDB, MRM6r, (outs), (ins RST:$reg),
|
||||
"fcomi\t$reg", IIC_FCOMI>;
|
||||
def COM_FIPr : FPI<0xDF, MRM6r, (outs), (ins RST:$reg),
|
||||
"fcompi\t$reg", IIC_FCOMI>;
|
||||
def COM_FIr : FPI<0xF0, AddRegFrm, (outs), (ins RST:$reg),
|
||||
"fcomi\t$reg", IIC_FCOMI>, DB;
|
||||
def COM_FIPr : FPI<0xF0, AddRegFrm, (outs), (ins RST:$reg),
|
||||
"fcompi\t$reg", IIC_FCOMI>, DF;
|
||||
}
|
||||
} // SchedRW
|
||||
|
||||
@ -585,8 +594,8 @@ def FLDCW16m : I<0xD9, MRM5m, // X87 control world = [mem16]
|
||||
let SchedRW = [WriteMicrocoded] in {
|
||||
let Defs = [FPSW] in
|
||||
def FNINIT : I<0xE3, RawFrm, (outs), (ins), "fninit", [], IIC_FNINIT>, DB;
|
||||
def FFREE : FPI<0xDD, MRM0r, (outs), (ins RST:$reg),
|
||||
"ffree\t$reg", IIC_FFREE>;
|
||||
def FFREE : FPI<0xC0, AddRegFrm, (outs), (ins RST:$reg),
|
||||
"ffree\t$reg", IIC_FFREE>, DD;
|
||||
// Clear exceptions
|
||||
|
||||
let Defs = [FPSW] in
|
||||
|
@ -241,3 +241,327 @@
|
||||
|
||||
# CHECK: pextrw $3, %xmm3, (%rax)
|
||||
0x66 0x0f 0x3a 0x15 0x18 0x03
|
||||
|
||||
# CHECK: fadd %st(0)
|
||||
0xd8,0xc0
|
||||
|
||||
# CHECK: fadd %st(1)
|
||||
0xd8,0xc1
|
||||
|
||||
# CHECK: fadd %st(2)
|
||||
0xd8,0xc2
|
||||
|
||||
# CHECK: fadd %st(3)
|
||||
0xd8,0xc3
|
||||
|
||||
# CHECK: fadd %st(4)
|
||||
0xd8,0xc4
|
||||
|
||||
# CHECK: fadd %st(5)
|
||||
0xd8,0xc5
|
||||
|
||||
# CHECK: fadd %st(6)
|
||||
0xd8,0xc6
|
||||
|
||||
# CHECK: fadd %st(7)
|
||||
0xd8,0xc7
|
||||
|
||||
# CHECK: fmul %st(0)
|
||||
0xd8,0xc8
|
||||
|
||||
# CHECK: fmul %st(1)
|
||||
0xd8,0xc9
|
||||
|
||||
# CHECK: fmul %st(2)
|
||||
0xd8,0xca
|
||||
|
||||
# CHECK: fmul %st(3)
|
||||
0xd8,0xcb
|
||||
|
||||
# CHECK: fmul %st(4)
|
||||
0xd8,0xcc
|
||||
|
||||
# CHECK: fmul %st(5)
|
||||
0xd8,0xcd
|
||||
|
||||
# CHECK: fmul %st(6)
|
||||
0xd8,0xce
|
||||
|
||||
# CHECK: fmul %st(7)
|
||||
0xd8,0xcf
|
||||
|
||||
# CHECK: fcom %st(0)
|
||||
0xd8,0xd0
|
||||
|
||||
# CHECK: fcom %st(1)
|
||||
0xd8,0xd1
|
||||
|
||||
# CHECK: fcom %st(2)
|
||||
0xd8,0xd2
|
||||
|
||||
# CHECK: fcom %st(3)
|
||||
0xd8,0xd3
|
||||
|
||||
# CHECK: fcom %st(4)
|
||||
0xd8,0xd4
|
||||
|
||||
# CHECK: fcom %st(5)
|
||||
0xd8,0xd5
|
||||
|
||||
# CHECK: fcom %st(6)
|
||||
0xd8,0xd6
|
||||
|
||||
# CHECK: fcom %st(7)
|
||||
0xd8,0xd7
|
||||
|
||||
# CHECK: fcomp %st(0)
|
||||
0xd8,0xd8
|
||||
|
||||
# CHECK: fcomp %st(1)
|
||||
0xd8,0xd9
|
||||
|
||||
# CHECK: fcomp %st(2)
|
||||
0xd8,0xda
|
||||
|
||||
# CHECK: fcomp %st(3)
|
||||
0xd8,0xdb
|
||||
|
||||
# CHECK: fcomp %st(4)
|
||||
0xd8,0xdc
|
||||
|
||||
# CHECK: fcomp %st(5)
|
||||
0xd8,0xdd
|
||||
|
||||
# CHECK: fcomp %st(6)
|
||||
0xd8,0xde
|
||||
|
||||
# CHECK: fcomp %st(7)
|
||||
0xd8,0xdf
|
||||
|
||||
# CHECK: fsub %st(0)
|
||||
0xd8,0xe0
|
||||
|
||||
# CHECK: fsub %st(1)
|
||||
0xd8,0xe1
|
||||
|
||||
# CHECK: fsub %st(2)
|
||||
0xd8,0xe2
|
||||
|
||||
# CHECK: fsub %st(3)
|
||||
0xd8,0xe3
|
||||
|
||||
# CHECK: fsub %st(4)
|
||||
0xd8,0xe4
|
||||
|
||||
# CHECK: fsub %st(5)
|
||||
0xd8,0xe5
|
||||
|
||||
# CHECK: fsub %st(6)
|
||||
0xd8,0xe6
|
||||
|
||||
# CHECK: fsub %st(7)
|
||||
0xd8,0xe7
|
||||
|
||||
# CHECK: fsubr %st(0)
|
||||
0xd8,0xe8
|
||||
|
||||
# CHECK: fsubr %st(1)
|
||||
0xd8,0xe9
|
||||
|
||||
# CHECK: fsubr %st(2)
|
||||
0xd8,0xea
|
||||
|
||||
# CHECK: fsubr %st(3)
|
||||
0xd8,0xeb
|
||||
|
||||
# CHECK: fsubr %st(4)
|
||||
0xd8,0xec
|
||||
|
||||
# CHECK: fsubr %st(5)
|
||||
0xd8,0xed
|
||||
|
||||
# CHECK: fsubr %st(6)
|
||||
0xd8,0xee
|
||||
|
||||
# CHECK: fsubr %st(7)
|
||||
0xd8,0xef
|
||||
|
||||
# CHECK: fdiv %st(0)
|
||||
0xd8,0xf0
|
||||
|
||||
# CHECK: fdiv %st(1)
|
||||
0xd8,0xf1
|
||||
|
||||
# CHECK: fdiv %st(2)
|
||||
0xd8,0xf2
|
||||
|
||||
# CHECK: fdiv %st(3)
|
||||
0xd8,0xf3
|
||||
|
||||
# CHECK: fdiv %st(4)
|
||||
0xd8,0xf4
|
||||
|
||||
# CHECK: fdiv %st(5)
|
||||
0xd8,0xf5
|
||||
|
||||
# CHECK: fdiv %st(6)
|
||||
0xd8,0xf6
|
||||
|
||||
# CHECK: fdiv %st(7)
|
||||
0xd8,0xf7
|
||||
|
||||
# CHECK: fdivr %st(0)
|
||||
0xd8,0xf8
|
||||
|
||||
# CHECK: fdivr %st(1)
|
||||
0xd8,0xf9
|
||||
|
||||
# CHECK: fdivr %st(2)
|
||||
0xd8,0xfa
|
||||
|
||||
# CHECK: fdivr %st(3)
|
||||
0xd8,0xfb
|
||||
|
||||
# CHECK: fdivr %st(4)
|
||||
0xd8,0xfc
|
||||
|
||||
# CHECK: fdivr %st(5)
|
||||
0xd8,0xfd
|
||||
|
||||
# CHECK: fdivr %st(6)
|
||||
0xd8,0xfe
|
||||
|
||||
# CHECK: fdivr %st(7)
|
||||
0xd8,0xff
|
||||
|
||||
# CHECK: fld %st(0)
|
||||
0xd9,0xc0
|
||||
|
||||
# CHECK: fld %st(1)
|
||||
0xd9,0xc1
|
||||
|
||||
# CHECK: fld %st(2)
|
||||
0xd9,0xc2
|
||||
|
||||
# CHECK: fld %st(3)
|
||||
0xd9,0xc3
|
||||
|
||||
# CHECK: fld %st(4)
|
||||
0xd9,0xc4
|
||||
|
||||
# CHECK: fld %st(5)
|
||||
0xd9,0xc5
|
||||
|
||||
# CHECK: fld %st(6)
|
||||
0xd9,0xc6
|
||||
|
||||
# CHECK: fld %st(7)
|
||||
0xd9,0xc7
|
||||
|
||||
# CHECK: fxch %st(0)
|
||||
0xd9,0xc8
|
||||
|
||||
# CHECK: fxch %st(1)
|
||||
0xd9,0xc9
|
||||
|
||||
# CHECK: fxch %st(2)
|
||||
0xd9,0xca
|
||||
|
||||
# CHECK: fxch %st(3)
|
||||
0xd9,0xcb
|
||||
|
||||
# CHECK: fxch %st(4)
|
||||
0xd9,0xcc
|
||||
|
||||
# CHECK: fxch %st(5)
|
||||
0xd9,0xcd
|
||||
|
||||
# CHECK: fxch %st(6)
|
||||
0xd9,0xce
|
||||
|
||||
# CHECK: fxch %st(7)
|
||||
0xd9,0xcf
|
||||
|
||||
# CHECK: fnop
|
||||
0xd9,0xd0
|
||||
|
||||
# CHECK: fchs
|
||||
0xd9,0xe0
|
||||
|
||||
# CHECK: fabs
|
||||
0xd9,0xe1
|
||||
|
||||
# CHECK: ftst
|
||||
0xd9,0xe4
|
||||
|
||||
# CHECK: fxam
|
||||
0xd9,0xe5
|
||||
|
||||
# CHECK: fld1
|
||||
0xd9,0xe8
|
||||
|
||||
# CHECK: fldl2t
|
||||
0xd9,0xe9
|
||||
|
||||
# CHECK: fldl2e
|
||||
0xd9,0xea
|
||||
|
||||
# CHECK: fldpi
|
||||
0xd9,0xeb
|
||||
|
||||
# CHECK: fldlg2
|
||||
0xd9,0xec
|
||||
|
||||
# CHECK: fldln2
|
||||
0xd9,0xed
|
||||
|
||||
# CHECK: fldz
|
||||
0xd9,0xee
|
||||
|
||||
# CHECK: f2xm1
|
||||
0xd9,0xf0
|
||||
|
||||
# CHECK: fyl2x
|
||||
0xd9,0xf1
|
||||
|
||||
# CHECK: fptan
|
||||
0xd9,0xf2
|
||||
|
||||
# CHECK: fpatan
|
||||
0xd9,0xf3
|
||||
|
||||
# CHECK: fxtract
|
||||
0xd9,0xf4
|
||||
|
||||
# CHECK: fprem1
|
||||
0xd9,0xf5
|
||||
|
||||
# CHECK: fdecstp
|
||||
0xd9,0xf6
|
||||
|
||||
# CHECK: fincstp
|
||||
0xd9,0xf7
|
||||
|
||||
# CHECK: fprem
|
||||
0xd9,0xf8
|
||||
|
||||
# CHECK: fyl2xp1
|
||||
0xd9,0xf9
|
||||
|
||||
# CHECK: fsqrt
|
||||
0xd9,0xfa
|
||||
|
||||
# CHECK: fsincos
|
||||
0xd9,0xfb
|
||||
|
||||
# CHECK: frndint
|
||||
0xd9,0xfc
|
||||
|
||||
# CHECK: fscale
|
||||
0xd9,0xfd
|
||||
|
||||
# CHECK: fsin
|
||||
0xd9,0xfe
|
||||
|
||||
# CHECK: fcos
|
||||
0xd9,0xff
|
||||
|
@ -17,6 +17,8 @@ void DumbFilter::anchor() { }
|
||||
|
||||
void ModFilter::anchor() { }
|
||||
|
||||
void AddRegEscapeFilter::anchor() { }
|
||||
|
||||
void ExtendedFilter::anchor() { }
|
||||
|
||||
void ExactFilter::anchor() { }
|
||||
|
@ -84,6 +84,26 @@ 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
|
||||
/// mod field [bits 7-6] and the value of the nnn field [bits 5-3].
|
||||
class ExtendedFilter : public ModRMFilter {
|
||||
|
@ -1075,9 +1075,14 @@ void RecognizableInstr::emitDecodePath(DisassemblerTables &tables) const {
|
||||
case X86Local::DE:
|
||||
case X86Local::DF:
|
||||
assert(Opcode >= 0xc0 && "Unexpected opcode for an escape opcode");
|
||||
assert(Form == X86Local::RawFrm);
|
||||
opcodeType = ONEBYTE;
|
||||
filter = new ExactFilter(Opcode);
|
||||
if (Form == X86Local::AddRegFrm) {
|
||||
Spec->modifierType = MODIFIER_MODRM;
|
||||
Spec->modifierBase = Opcode;
|
||||
filter = new AddRegEscapeFilter(Opcode);
|
||||
} else {
|
||||
filter = new ExactFilter(Opcode);
|
||||
}
|
||||
opcodeToSet = 0xd8 + (Prefix - X86Local::D8);
|
||||
break;
|
||||
case X86Local::REP:
|
||||
@ -1125,16 +1130,6 @@ void RecognizableInstr::emitDecodePath(DisassemblerTables &tables) const {
|
||||
switch (Form) {
|
||||
default:
|
||||
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::MRM1m:
|
||||
case X86Local::MRM2m:
|
||||
@ -1162,22 +1157,31 @@ void RecognizableInstr::emitDecodePath(DisassemblerTables &tables) const {
|
||||
assert(filter && "Filter not set");
|
||||
|
||||
if (Form == X86Local::AddRegFrm) {
|
||||
assert(opcodeToSet < 0xf9 &&
|
||||
"Not enough room for all ADDREG_FRM operands");
|
||||
if(Spec->modifierType != MODIFIER_MODRM) {
|
||||
assert(opcodeToSet < 0xf9 &&
|
||||
"Not enough room for all ADDREG_FRM operands");
|
||||
|
||||
uint8_t currentOpcode;
|
||||
uint8_t currentOpcode;
|
||||
|
||||
for (currentOpcode = opcodeToSet;
|
||||
currentOpcode < opcodeToSet + 8;
|
||||
++currentOpcode)
|
||||
for (currentOpcode = opcodeToSet;
|
||||
currentOpcode < opcodeToSet + 8;
|
||||
++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,
|
||||
insnContext(),
|
||||
currentOpcode,
|
||||
opcodeToSet,
|
||||
*filter,
|
||||
UID, Is32Bit, IgnoresVEX_L);
|
||||
|
||||
Spec->modifierType = MODIFIER_OPCODE;
|
||||
Spec->modifierBase = opcodeToSet;
|
||||
}
|
||||
} else {
|
||||
tables.setTableFields(opcodeType,
|
||||
insnContext(),
|
||||
@ -1337,7 +1341,6 @@ OperandEncoding RecognizableInstr::immediateEncodingFromString
|
||||
OperandEncoding RecognizableInstr::rmRegisterEncodingFromString
|
||||
(const std::string &s,
|
||||
bool hasOpSizePrefix) {
|
||||
ENCODING("RST", ENCODING_I)
|
||||
ENCODING("GR16", ENCODING_RM)
|
||||
ENCODING("GR32", ENCODING_RM)
|
||||
ENCODING("GR32orGR64", ENCODING_RM)
|
||||
@ -1490,6 +1493,7 @@ OperandEncoding RecognizableInstr::relocationEncodingFromString
|
||||
OperandEncoding RecognizableInstr::opcodeModifierEncodingFromString
|
||||
(const std::string &s,
|
||||
bool hasOpSizePrefix) {
|
||||
ENCODING("RST", ENCODING_I)
|
||||
ENCODING("GR32", ENCODING_Rv)
|
||||
ENCODING("GR64", ENCODING_RO)
|
||||
ENCODING("GR16", ENCODING_Rv)
|
||||
|
Loading…
Reference in New Issue
Block a user