windows-nt/Source/XPSP1/NT/base/wow64/mscpu/fraglib/shr1632a.c
2020-09-26 16:20:57 +08:00

306 lines
7.2 KiB
C

/*++
Copyright (c) 1995 Microsoft Corporation
Module Name:
shr1632a.c
Abstract:
Instruction fragments with common (shared) WORD, and DWORD flavors
(but not BYTE).
Compiled twice per flavor, once with UNALIGNED and once with ALIGNED
pointers.
Author:
05-Nov-1995 BarryBo
Revision History:
--*/
#include <nt.h>
#include <ntrtl.h>
#include <nturtl.h>
#include <windows.h>
#include <stdio.h>
#include "shr1632a.h"
FRAGCOMMON2(BtMemFrag)
{
UTYPE bit = 1<<(op2&LMB);
op2 /= LMB+1; // compute offset of the correct WORD/DWORD
SET_CFLAG_IND(GET_VAL(pop1+op2) & bit);
}
FRAGCOMMON2(BtsMemFrag)
{
DWORD NewCFlag;
UTYPE bit = 1<<(op2&LMB);
UTYPE op1;
op2 /= LMB+1; // compute offset of the correct WORD/DWORD
pop1 += op2;
op1 = GET_VAL(pop1);
NewCFlag = op1 & bit;
// Intel docs indicate that we can safely overwrite all 2/4 bytes
// when writing the value back out. (pg. 26/43)
PUT_VAL(pop1, (op1|bit));
SET_CFLAG_IND(NewCFlag);
}
FRAGCOMMON2(BtcMemFrag)
{
DWORD NewCFlag;
UTYPE bit = 1<<(op2&LMB);
UTYPE op1;
op2 /= LMB+1; // compute offset of the correct WORD/DWORD
pop1 += op2;
op1 = GET_VAL(pop1);
NewCFlag = op1 & bit;
// Intel docs indicate that we can safely overwrite all 2/4 bytes
// when writing the value back out. (pg. 26/43)
PUT_VAL(pop1, op1 ^ bit);
SET_CFLAG_IND(NewCFlag);
}
FRAGCOMMON2(BtrMemFrag)
{
DWORD NewCFlag;
UTYPE bit = 1<<(op2&LMB);
UTYPE op1;
op2 /= LMB+1; // compute offset of the correct WORD/DWORD
pop1 += op2;
op1 = GET_VAL(pop1);
NewCFlag = op1 & bit;
// Intel docs indicate that we can safely overwrite all 2/4 bytes
// when writing the value back out. (pg. 26/43)
PUT_VAL(pop1, (op1&(~bit)));
SET_CFLAG_IND(NewCFlag);
}
FRAGCOMMON2(BtRegFrag)
{
UTYPE bit = 1<<(op2&LMB);
SET_CFLAG_IND(GET_VAL(pop1) & bit);
}
FRAGCOMMON2(BtsRegFrag)
{
DWORD NewCFlag;
UTYPE bit = 1<<(op2&LMB);
UTYPE op1;
op1 = GET_VAL(pop1);
NewCFlag = op1 & bit;
// Intel docs indicate that we can safely overwrite all 2/4 bytes
// when writing the value back out. (pg. 26/43)
PUT_VAL(pop1, (op1|bit));
SET_CFLAG_IND(NewCFlag);
}
FRAGCOMMON2(BtcRegFrag)
{
DWORD NewCFlag;
UTYPE bit = 1<<(op2&LMB);
UTYPE op1;
op1 = GET_VAL(pop1);
NewCFlag = op1 & bit;
// Intel docs indicate that we can safely overwrite all 2/4 bytes
// when writing the value back out. (pg. 26/43)
PUT_VAL(pop1, (op1 ^ bit));
SET_CFLAG_IND(NewCFlag);
}
FRAGCOMMON2(BtrRegFrag)
{
DWORD NewCFlag;
UTYPE bit = 1<<(op2&LMB);
UTYPE op1;
op1 = GET_VAL(pop1);
NewCFlag = op1 & bit;
// Intel docs indicate that we can safely overwrite all 2/4 bytes
// when writing the value back out. (pg. 26/43)
PUT_VAL(pop1, (op1&(~bit)));
SET_CFLAG_IND(NewCFlag);
}
FRAGCOMMON3(ShldFrag)
{
// pop1 = Base -- ptr to dest reg/mem
// op2 = inBits -- value of register containing bits to shift into Base
// op3 = count -- number of bits to shift
DWORD NewCFlag;
UTYPE Base;
if (op3 == 0) {
return; // nothing to do - nop with all flags preserved
}
op3 &= 0x1f; // make the count MOD 32 (now op3 = ShiftAmt)
#if MSB == 0x8000
if (op3 > 16) {
// Bad parameters - *pop1 UNDEFINED!
// - CF,OF,SF,ZF,AF,PF UNDEFINED!
return;
}
#endif
Base = GET_VAL(pop1);
NewCFlag = Base & (1<<(LMB+1-op3)); // Get the new CF value
Base <<= op3; // shift Base left
op2 >>= LMB+1-op3; // shift the top op3 bits of op2 right
Base |= op2; // merge the two together
PUT_VAL(pop1, Base);
SET_CFLAG_IND(NewCFlag);
SET_ZFLAG(Base);
SET_PFLAG(Base);
SET_SFLAG(Base << (31-LMB));
}
FRAGCOMMON3(ShldNoFlagsFrag)
{
// pop1 = Base -- ptr to dest reg/mem
// op2 = inBits -- value of register containing bits to shift into Base
// op3 = count -- number of bits to shift
UTYPE Base;
if (op3 == 0) {
return; // nothing to do - nop with all flags preserved
}
op3 &= 0x1f; // make the count MOD 32 (now op3 = ShiftAmt)
#if MSB == 0x8000
if (op3 > 16) {
// Bad parameters - *pop1 UNDEFINED!
// - CF,OF,SF,ZF,AF,PF UNDEFINED!
return;
}
#endif
Base = GET_VAL(pop1);
Base <<= op3; // shift Base left
op2 >>= LMB+1-op3; // shift the top op3 bits of op2 right
Base |= op2; // merge the two together
PUT_VAL(pop1, Base);
}
FRAGCOMMON3(ShrdFrag)
{
// pop1 = Base -- ptr to dest reg/mem
// op2 = inBits -- value of register containing bits to shift into Base
// op3 = count -- number of bits to shift
DWORD NewCFlag;
UTYPE Base;
int i;
if (op3 == 0) {
return; // nothing to do - nop with all flags preserved
}
op3 &= 0x1f; // make the count MOD 32 (now op3 = ShiftAmt)
#if MSB == 0x8000
if (op3 > 16) {
// Bad parameters - *pop1 UNDEFINED!
// - CF,OF,SF,ZF,AF,PF UNDEFINED!
return;
}
#endif
Base = GET_VAL(pop1);
NewCFlag = Base & (1<<(op3-1)); // Get the new CF value
Base >>= op3; // shift Base right
op2 <<= LMB+1-op3; // shift the low op3 bits of op2
Base |= op2; // merge the two together
PUT_VAL(pop1, Base);
SET_CFLAG_IND(NewCFlag);
SET_ZFLAG(Base);
SET_PFLAG(Base);
SET_SFLAG(Base << (31-LMB));
}
FRAGCOMMON3(ShrdNoFlagsFrag)
{
// pop1 = Base -- ptr to dest reg/mem
// op2 = inBits -- value of register containing bits to shift into Base
// op3 = count -- number of bits to shift
UTYPE Base;
int i;
if (op3 == 0) {
return; // nothing to do - nop with all flags preserved
}
op3 &= 0x1f; // make the count MOD 32 (now op3 = ShiftAmt)
#if MSB == 0x8000
if (op3 > 16) {
// Bad parameters - *pop1 UNDEFINED!
// - CF,OF,SF,ZF,AF,PF UNDEFINED!
return;
}
#endif
Base = GET_VAL(pop1);
Base >>= op3; // shift Base right
op2 <<= LMB+1-op3; // shift the low op3 bits of op2
Base |= op2; // merge the two together
PUT_VAL(pop1, Base);
}
FRAGCOMMON2(BsfFrag)
{
int i;
if (op2 == 0) {
// value is 0 - set ZFLAG and return
SET_ZFLAG(0);
// *pop1 = UNDEFINED
return;
}
// scan from bit 0 forward, looking for the index of '1' bit
for (i=0; (op2 & 1) == 0; ++i) {
op2 >>= 1;
}
// write the index of the '1' bit and clear the ZFLAG
PUT_VAL(pop1, i);
SET_ZFLAG(op2);
}
FRAGCOMMON2(BsrFrag)
{
int i;
if (op2 == 0) {
// value is 0 - set ZFLAG and return
SET_ZFLAG(0);
// *pop1 = UNDEFINED
return;
}
// scan from bit 31/15 downward, looking for the index of '1' bit
for (i=LMB; (op2 & MSB) == 0; --i) {
op2 <<= 1;
}
// write the index of the '1' bit and clear the ZFLAG
PUT_VAL(pop1, i);
SET_ZFLAG(op2);
}
FRAGCOMMON1(PopFrag)
{
POP_VAL(GET_VAL(pop1));
}