196 lines
4.6 KiB
Plaintext
196 lines
4.6 KiB
Plaintext
macro tzcntflags(input, output) {
|
|
ZF = (output == 0);
|
|
CF = (input == 0);
|
|
# OF, SF, PF, AF are undefined
|
|
}
|
|
|
|
|
|
####
|
|
#### BMI1 instructions
|
|
####
|
|
|
|
# TODO remove ANDN from ia.sinc ?????
|
|
:ANDN Reg32, vexVVVV_r32, rm32 is $(VEX_NDS) & $(VEX_LZ) & $(VEX_PRE_NONE) & $(VEX_0F38) & $(VEX_W0) & vexVVVV_r32; byte=0xf2; Reg32 ... & check_Reg32_dest ... &rm32
|
|
{
|
|
Reg32 = ~(vexVVVV_r32) & rm32;
|
|
resultflags(Reg32);
|
|
OF = 0;
|
|
CF = 0;
|
|
build check_Reg32_dest;
|
|
}
|
|
|
|
@ifdef IA64
|
|
# TODO remove ANDN from ia.sinc ?????
|
|
:ANDN Reg64, vexVVVV_r64, rm64 is $(VEX_NDS) & $(VEX_LZ) & $(VEX_PRE_NONE) & $(VEX_0F38) & $(VEX_W1) & vexVVVV_r64; byte=0xf2; Reg64 ... & rm64
|
|
{
|
|
Reg64 = ~(vexVVVV_r64) & rm64;
|
|
resultflags(Reg64);
|
|
OF = 0;
|
|
CF = 0;
|
|
}
|
|
@endif
|
|
|
|
|
|
:BEXTR Reg32, rm32, vexVVVV_r32 is $(VEX_NDS) & $(VEX_LZ) & $(VEX_PRE_NONE) & $(VEX_0F38) & $(VEX_W0) & vexVVVV_r32; byte=0xf7; Reg32 ... & check_Reg32_dest ... & rm32
|
|
{
|
|
sourceTmp:1 = vexVVVV_r32[0,8];
|
|
lengthTmp:1 = vexVVVV_r32[8,8];
|
|
|
|
Reg32 = (rm32 >> sourceTmp) & ((1 << lengthTmp) - 1);
|
|
build check_Reg32_dest;
|
|
|
|
ZF = (Reg32 == 0);
|
|
OF = 0;
|
|
CF = 0;
|
|
# AF, SF, and PF are undefined
|
|
}
|
|
|
|
@ifdef IA64
|
|
:BEXTR Reg64, rm64, vexVVVV_r64 is $(VEX_NDS) & $(VEX_LZ) & $(VEX_PRE_NONE) & $(VEX_0F38) & $(VEX_W1) & vexVVVV_r64; byte=0xf7; Reg64 ... & rm64
|
|
{
|
|
sourceTmp:1 = vexVVVV_r64[0,8];
|
|
lengthTmp:1 = vexVVVV_r64[8,8];
|
|
|
|
Reg64 = (rm64 >> sourceTmp) & ((1 << lengthTmp) - 1);
|
|
|
|
ZF = (Reg64 == 0);
|
|
OF = 0;
|
|
CF = 0;
|
|
# AF, SF, and PF are undefined
|
|
}
|
|
@endif
|
|
|
|
|
|
:BLSI vexVVVV_r32, rm32 is $(VEX_NDD) & $(VEX_LZ) & $(VEX_PRE_NONE) & $(VEX_0F38) & $(VEX_W0) & vexVVVV_r32; byte=0xf3; reg_opcode=3 ... & check_vexVVVV_r32_dest ... & rm32
|
|
{
|
|
vexVVVV_r32 = -rm32 & rm32;
|
|
build check_vexVVVV_r32_dest;
|
|
|
|
ZF = (vexVVVV_r32 == 0);
|
|
SF = (vexVVVV_r32 s< 0);
|
|
CF = (rm32 != 0);
|
|
OF = 0;
|
|
# AF and PF are undefined
|
|
}
|
|
|
|
@ifdef IA64
|
|
:BLSI vexVVVV_r64, rm64 is $(VEX_NDD) & $(VEX_LZ) & $(VEX_PRE_NONE) & $(VEX_0F38) & $(VEX_W1) & vexVVVV_r64; byte=0xf3; reg_opcode=3 ... & rm64
|
|
{
|
|
vexVVVV_r64 = -rm64 & rm64;
|
|
|
|
ZF = (vexVVVV_r64 == 0);
|
|
SF = (vexVVVV_r64 s< 0);
|
|
CF = (rm64 != 0);
|
|
OF = 0;
|
|
# AF and PF are undefined
|
|
}
|
|
@endif
|
|
|
|
|
|
:BLSMSK vexVVVV_r32, rm32 is $(VEX_NDD) & $(VEX_LZ) & $(VEX_PRE_NONE) & $(VEX_0F38) & $(VEX_W0) & vexVVVV_r32; byte=0xf3; reg_opcode=2 ... & check_vexVVVV_r32_dest ... &rm32
|
|
{
|
|
CF = (rm32 == 0);
|
|
vexVVVV_r32 = (rm32 - 1) ^ rm32;
|
|
|
|
SF = (vexVVVV_r32 s< 0);
|
|
build check_vexVVVV_r32_dest;
|
|
ZF = 0;
|
|
OF = 0;
|
|
# AF and PF are undefined
|
|
}
|
|
|
|
@ifdef IA64
|
|
:BLSMSK vexVVVV_r64, rm64 is $(VEX_NDD) & $(VEX_LZ) & $(VEX_PRE_NONE) & $(VEX_0F38) & $(VEX_W1) & vexVVVV_r64; byte=0xf3; reg_opcode=2 ... & rm64
|
|
{
|
|
CF = (rm64 == 0);
|
|
vexVVVV_r64 = (rm64 - 1) ^ rm64;
|
|
|
|
SF = (vexVVVV_r64 s< 0);
|
|
ZF = 0;
|
|
OF = 0;
|
|
# AF and PF are undefined
|
|
}
|
|
@endif
|
|
|
|
|
|
:BLSR vexVVVV_r32, rm32 is $(VEX_NDD) & $(VEX_LZ) & $(VEX_PRE_NONE) & $(VEX_0F38) & $(VEX_W0) & vexVVVV_r32; byte=0xf3; reg_opcode=1 ... & check_vexVVVV_r32_dest ... &rm32
|
|
{
|
|
CF = (rm32 == 0);
|
|
vexVVVV_r32 = (rm32 - 1) & rm32;
|
|
build check_vexVVVV_r32_dest;
|
|
|
|
ZF = (vexVVVV_r32 == 0);
|
|
SF = (vexVVVV_r32 s< 0);
|
|
OF = 0;
|
|
# AF and PF are undefined
|
|
}
|
|
|
|
@ifdef IA64
|
|
:BLSR vexVVVV_r64, rm64 is $(VEX_NDD) & $(VEX_LZ) & $(VEX_PRE_NONE) & $(VEX_0F38) & $(VEX_W1) & vexVVVV_r64; byte=0xf3; reg_opcode=1 ... & rm64
|
|
{
|
|
CF = (rm64 == 0);
|
|
vexVVVV_r64 = (rm64 - 1) & rm64;
|
|
|
|
ZF = (vexVVVV_r64 == 0);
|
|
SF = (vexVVVV_r64 s< 0);
|
|
OF = 0;
|
|
# AF and PF are undefined
|
|
}
|
|
@endif
|
|
|
|
# not as documented in manual; requires PRE_66 prefix to get 16-bit operation
|
|
:TZCNT Reg16, rm16 is vexMode=0 & opsize=0 & $(PRE_66) & $(PRE_F3) & byte=0x0F; byte=0xBC; Reg16 ... & rm16 {
|
|
|
|
countTmp:2 = 0;
|
|
inputTmp:2 = rm16;
|
|
|
|
<loopbegin>
|
|
if ((inputTmp & 1) != 0) goto <loopend>;
|
|
|
|
countTmp = countTmp + 1;
|
|
inputTmp = (inputTmp >> 1) | 0x8000;
|
|
goto <loopbegin>;
|
|
|
|
<loopend>
|
|
tzcntflags(rm16, countTmp);
|
|
Reg16 = countTmp;
|
|
|
|
}
|
|
|
|
:TZCNT Reg32, rm32 is vexMode=0 & opsize=1 & $(PRE_F3) & byte=0x0F; byte=0xBC; Reg32 ... & check_Reg32_dest ... & rm32 {
|
|
|
|
countTmp:4 = 0;
|
|
inputTmp:4 = rm32;
|
|
|
|
<loopbegin>
|
|
if ((inputTmp & 1) != 0) goto <loopend>;
|
|
|
|
countTmp = countTmp + 1;
|
|
inputTmp = (inputTmp >> 1) | 0x80000000;
|
|
goto <loopbegin>;
|
|
|
|
<loopend>
|
|
tzcntflags(rm32, countTmp);
|
|
Reg32 = countTmp;
|
|
build check_Reg32_dest;
|
|
}
|
|
|
|
@ifdef IA64
|
|
:TZCNT Reg64, rm64 is vexMode=0 & opsize=2 & $(PRE_F3) & $(REX_W) & byte=0x0F; byte=0xBC; Reg64 ... & rm64 {
|
|
|
|
countTmp:8 = 0;
|
|
inputTmp:8 = rm64;
|
|
|
|
<loopbegin>
|
|
if ((inputTmp & 1) != 0) goto <loopend>;
|
|
|
|
countTmp = countTmp + 1;
|
|
inputTmp = (inputTmp >> 1) | 0x8000000000000000;
|
|
goto <loopbegin>;
|
|
|
|
<loopend>
|
|
tzcntflags(rm64, countTmp);
|
|
Reg64 = countTmp;
|
|
}
|
|
@endif
|