mirror of
https://github.com/bitdefender/bddisasm.git
synced 2025-01-18 11:00:57 +00:00
Optimized ror/rol/rcr/rcl instruction emulation - don't use slow loops anymore.
This commit is contained in:
parent
d16f1d8ba3
commit
f293c936ee
@ -2145,93 +2145,107 @@ ShemuEmulate(
|
|||||||
case ND_INS_ROL:
|
case ND_INS_ROL:
|
||||||
case ND_INS_ROR:
|
case ND_INS_ROR:
|
||||||
{
|
{
|
||||||
ND_UINT32 cnt, tempcnt, cntmask;
|
ND_UINT32 cnt, tempcnt, cntmask, bitwidth;
|
||||||
ND_UINT8 tempCF = 0;
|
ND_UINT8 tempCF = 0;
|
||||||
|
|
||||||
GET_OP(Context, 0, &dst);
|
GET_OP(Context, 0, &dst);
|
||||||
GET_OP(Context, 1, &src);
|
GET_OP(Context, 1, &src);
|
||||||
|
|
||||||
cnt = (ND_UINT32)src.Value.Qwords[0];
|
cnt = (ND_UINT32)src.Value.Qwords[0];
|
||||||
tempcnt = 0;
|
|
||||||
cntmask = ((dst.Size == 8) ? 0x3F : 0x1F);
|
cntmask = ((dst.Size == 8) ? 0x3F : 0x1F);
|
||||||
|
tempcnt = (cnt & cntmask);
|
||||||
|
bitwidth = (ND_UINT32)dst.Size * 8;
|
||||||
|
|
||||||
if (ND_INS_RCL == Context->Instruction.Instruction ||
|
if (ND_INS_RCL == Context->Instruction.Instruction ||
|
||||||
ND_INS_RCR == Context->Instruction.Instruction)
|
ND_INS_RCR == Context->Instruction.Instruction)
|
||||||
{
|
{
|
||||||
switch (dst.Size)
|
if (dst.Size == 1)
|
||||||
{
|
{
|
||||||
case 1: tempcnt = (cnt & 0x1F) % 9; break;
|
tempcnt %= 9;
|
||||||
case 2: tempcnt = (cnt & 0x1F) % 17; break;
|
}
|
||||||
case 4: tempcnt = (cnt & 0x1F); break;
|
else if (dst.Size == 2)
|
||||||
case 8: tempcnt = (cnt & 0x3F); break;
|
{
|
||||||
default: break;
|
tempcnt %= 17;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
tempcnt = (cnt & cntmask) % (dst.Size * 8);
|
tempcnt %= (dst.Size * 8);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ND_INS_RCL == Context->Instruction.Instruction)
|
if (ND_INS_RCL == Context->Instruction.Instruction)
|
||||||
{
|
{
|
||||||
while (tempcnt != 0)
|
tempCF = GET_FLAG(Context, NDR_RFLAG_CF);
|
||||||
|
|
||||||
|
if (tempcnt != 0)
|
||||||
{
|
{
|
||||||
tempCF = ND_MSB(dst.Size, dst.Value.Qwords[0]);
|
// tempcnt is in range [1, dst bit width].
|
||||||
dst.Value.Qwords[0] = (dst.Value.Qwords[0] << 1) + GET_FLAG(Context, NDR_RFLAG_CF);
|
ND_UINT64 left = (tempcnt == bitwidth) ? 0 : (dst.Value.Qwords[0] << tempcnt);
|
||||||
SET_FLAG(Context, NDR_RFLAG_CF, tempCF);
|
ND_UINT64 right = (tempcnt == 1) ? 0 : (dst.Value.Qwords[0] >> (bitwidth - tempcnt + 1));
|
||||||
tempcnt--;
|
|
||||||
|
SET_FLAG(Context, NDR_RFLAG_CF, ND_GET_BIT(bitwidth - tempcnt, dst.Value.Qwords[0]));
|
||||||
|
|
||||||
|
dst.Value.Qwords[0] = left | ((ND_UINT64)tempCF << (tempcnt - 1)) | right;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((cnt & cntmask) == 1)
|
if ((cnt & cntmask) == 1)
|
||||||
{
|
{
|
||||||
SET_FLAG(Context, NDR_RFLAG_OF, ND_MSB(dst.Size, dst.Value.Qwords[0]) ^
|
SET_FLAG(Context, NDR_RFLAG_OF, ND_MSB(dst.Size, dst.Value.Qwords[0]) ^
|
||||||
GET_FLAG(Context, NDR_RFLAG_CF));
|
GET_FLAG(Context, NDR_RFLAG_CF));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (ND_INS_RCR == Context->Instruction.Instruction)
|
else if (ND_INS_RCR == Context->Instruction.Instruction)
|
||||||
{
|
{
|
||||||
|
tempCF = GET_FLAG(Context, NDR_RFLAG_CF);
|
||||||
|
|
||||||
if ((cnt & cntmask) == 1)
|
if ((cnt & cntmask) == 1)
|
||||||
{
|
{
|
||||||
SET_FLAG(Context, NDR_RFLAG_OF, ND_MSB(dst.Size, dst.Value.Qwords[0]) ^
|
SET_FLAG(Context, NDR_RFLAG_OF, ND_MSB(dst.Size, dst.Value.Qwords[0]) ^
|
||||||
GET_FLAG(Context, NDR_RFLAG_CF));
|
GET_FLAG(Context, NDR_RFLAG_CF));
|
||||||
}
|
}
|
||||||
|
|
||||||
while (tempcnt != 0)
|
if (tempcnt != 0)
|
||||||
{
|
{
|
||||||
tempCF = ND_LSB(dst.Size, dst.Value.Qwords[0]);
|
// tempcnt is in range [1, dst bit width].
|
||||||
dst.Value.Qwords[0] = (dst.Value.Qwords[0] >> 1) +
|
ND_UINT64 left = (tempcnt == bitwidth) ? 0 : (dst.Value.Qwords[0] >> tempcnt);
|
||||||
((ND_UINT64)GET_FLAG(Context, NDR_RFLAG_CF) << (dst.Size * 8 - 1));
|
ND_UINT64 right = (tempcnt == 1) ? 0 : (dst.Value.Qwords[0] << (bitwidth - tempcnt + 1));
|
||||||
SET_FLAG(Context, NDR_RFLAG_CF, tempCF);
|
|
||||||
tempcnt--;
|
SET_FLAG(Context, NDR_RFLAG_CF, ND_GET_BIT(tempcnt - 1, dst.Value.Qwords[0]));
|
||||||
|
|
||||||
|
dst.Value.Qwords[0] = left | ((ND_UINT64)tempCF << (bitwidth - tempcnt)) | right;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (ND_INS_ROL == Context->Instruction.Instruction)
|
else if (ND_INS_ROL == Context->Instruction.Instruction)
|
||||||
{
|
{
|
||||||
while (tempcnt != 0)
|
if (tempcnt != 0)
|
||||||
{
|
{
|
||||||
tempCF = ND_MSB(dst.Size, dst.Value.Qwords[0]);
|
// tempcnt is in range [1, dst bit width - 1].
|
||||||
dst.Value.Qwords[0] = (dst.Value.Qwords[0] << 1) + tempCF;
|
ND_UINT64 left = dst.Value.Qwords[0] << tempcnt;
|
||||||
tempcnt--;
|
ND_UINT64 right = dst.Value.Qwords[0] >> (bitwidth - tempcnt);
|
||||||
|
|
||||||
|
dst.Value.Qwords[0] = left | right;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((cnt & cntmask) != 0)
|
if ((cnt & cntmask) != 0)
|
||||||
{
|
{
|
||||||
SET_FLAG(Context, NDR_RFLAG_CF, dst.Value.Qwords[0] & 1);
|
SET_FLAG(Context, NDR_RFLAG_CF, ND_LSB(dst.Size, dst.Value.Qwords[0]));
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((cnt & cntmask) == 1)
|
if ((cnt & cntmask) == 1)
|
||||||
{
|
{
|
||||||
SET_FLAG(Context, NDR_RFLAG_OF, ND_MSB(dst.Size, dst.Value.Qwords[0]) ^
|
SET_FLAG(Context, NDR_RFLAG_OF, ND_MSB(dst.Size, dst.Value.Qwords[0]) ^
|
||||||
GET_FLAG(Context, NDR_RFLAG_CF));
|
GET_FLAG(Context, NDR_RFLAG_CF));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else // ND_INS_ROR
|
else // ND_INS_ROR
|
||||||
{
|
{
|
||||||
while (tempcnt != 0)
|
if (tempcnt != 0)
|
||||||
{
|
{
|
||||||
tempCF = ND_LSB(dst.Size, dst.Value.Qwords[0]);
|
// tempcnt is in range [1, dst bit width - 1].
|
||||||
dst.Value.Qwords[0] = (dst.Value.Qwords[0] >> 1) + ((ND_UINT64)tempCF << (dst.Size * 8 - 1));
|
ND_UINT64 left = (dst.Value.Qwords[0] >> tempcnt);
|
||||||
tempcnt--;
|
ND_UINT64 right = (dst.Value.Qwords[0] << (bitwidth - tempcnt));
|
||||||
|
|
||||||
|
dst.Value.Qwords[0] = left | right;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((cnt & cntmask) != 0)
|
if ((cnt & cntmask) != 0)
|
||||||
@ -2241,7 +2255,8 @@ ShemuEmulate(
|
|||||||
|
|
||||||
if ((cnt & cntmask) == 1)
|
if ((cnt & cntmask) == 1)
|
||||||
{
|
{
|
||||||
SET_FLAG(Context, NDR_RFLAG_OF, ND_MSB(dst.Size, dst.Value.Qwords[0]) ^ ND_GET_BIT(dst.Size * 8ULL - 2, dst.Value.Qwords[0]));
|
SET_FLAG(Context, NDR_RFLAG_OF, ND_MSB(dst.Size, dst.Value.Qwords[0]) ^
|
||||||
|
ND_GET_BIT(dst.Size * 8ULL - 2, dst.Value.Qwords[0]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user