U0 ICAddEct(CIntermediateCode *tmpi,CICType t1,I64 r1,I64 d1, CICType t2,I64 r2,I64 d2,CICType t3,I64 r3,I64 d3,I64 op,I64 rip) { I64 i,tmp,res_reg=REG_RAX; Bool swap=FALSE; if (r3!=res_reg) { swap^=TRUE; SwapI64(&t2,&t3); SwapI64(&r2,&r3); SwapI64(&d2,&d3); } if (t2.raw_type>=RT_I64 && r2!=res_reg && t2&MDG_REG_DISP_SIB_RIP) { if (t1&MDF_REG && !(r2==r1 && t2&MDG_REG_DISP_SIB)) res_reg=r1; ICMov(tmpi,MDF_REG+RT_I64,res_reg,0,t3,r3,d3,rip); i=ICModr1(res_reg,t2,r2,d2); if (tmpi->ic_flags&ICF_LOCK) ICU8(tmpi,OC_LOCK_PREFIX); ICRex(tmpi,i.u8[1]); ICU16(tmpi,i.u8[2]<<8+op); ICModr2(tmpi,i,,d2,rip); } else { if (t2&MDF_REG) tmp=r2; else tmp=REG_RCX; if (t1&MDF_REG) res_reg=r1; if (tmp==res_reg) res_reg=REG_RDX; if (swap) { if (r3==tmp && t3&MDG_REG_DISP_SIB) tmp=REG_RCX; ICMov(tmpi,MDF_REG+RT_I64,tmp,0,t2,r2,d2,rip); ICMov(tmpi,MDF_REG+RT_I64,res_reg,0,t3,r3,d3,rip); } else { if (r2==res_reg && t2&MDG_REG_DISP_SIB) res_reg=REG_RDX; ICMov(tmpi,MDF_REG+RT_I64,res_reg,0,t3,r3,d3,rip); ICMov(tmpi,MDF_REG+RT_I64,tmp,0,t2,r2,d2,rip); } i=0x48; if (res_reg>7) i+=4; if (tmp>7) i++; if (tmpi->ic_flags&ICF_LOCK) ICU8(tmpi,OC_LOCK_PREFIX); ICU24(tmpi,0xC00000+i+(tmp&7)<<16+(res_reg&7)<<19+op<<8); } ICMov(tmpi,t1,r1,d1,MDF_REG+RT_I64,res_reg,0,rip); } U0 ICAddSubEctImm(CIntermediateCode *tmpi,CICType t1,I64 r1,I64 d1, CICType t2,I64 r2,I64 d2,I64 d,I64 op,I64 rip) { I64 i; if (op.u8[0]==0x2B) { op=0x0003; d=-d; } if (t1&MDF_REG) { if (!(t2&MDF_REG)) { ICMov(tmpi,t1,r1,d1,t2,r2,d2,rip); t2=t1; r2=r1; d2=d1; } if (r1==r2) { if (r1>7) i=0x49; else i=0x48; if (!d && (op.u8[0]==0x03||op.u8[0]==0x2B||op.u8[0]==0x33||op.u8[0]==0x0B)) return; else if (d==1 && op.u8[0]==0x03) { ICU24(tmpi,0xC0FF00+op.u8[1]<<19+i+(r1&7)<<16); return; } else if (d==-1 && op.u8[0]==0x03) { ICU24(tmpi,0xC8FF00+i+(r1&7)<<16); return; } else if (I8_MIN<=d<=I8_MAX) { ICU24(tmpi,0xC08300+op.u8[1]<<19+i+(r1&7)<<16); ICU8(tmpi,d); return; } else if (I32_MIN<=d<=I32_MAX) { ICU24(tmpi,0xC08100+op.u8[1]<<19+i+(r1&7)<<16); ICU32(tmpi,d); return; } } if (op.u8[0]==0x03 && I32_MIN<=d<=I32_MAX && !Bt(&cmp.non_ptr_vars_mask,r2)) { i=ICModr1(r1,MDF_DISP+RT_I64,r2,d); i.u8[1]|=0x48; ICU24(tmpi,i.u8[2]<<16+0x8D00+i.u8[1]); ICModr2(tmpi,i,,d,rip); return; } } switch (Bsr(t1)) { case MDf_REG: case MDf_DISP: case MDf_SIB: case MDf_RIP_DISP32: if (t1!=t2 || r1!=r2 || d1!=d2) { ICMov(tmpi,t1,r1,d1,t2,r2,d2,rip); t2=t1; r2=r1; d2=d1; } if (!d &&(op.u8[0]==0x03||op.u8[0]==0x2B||op.u8[0]==0x33||op.u8[0]==0x0B)) return; if (op.u8[0]==0x03 && d==-1) //add -1 op.u8[1]=1; //Decrement slash val if (op.u8[0]==0x03 && (d==1 || d==-1)) //Add { i=ICModr1(op.u8[1],t1,r1,d1); if (!(t1&MDF_REG) && tmpi->ic_flags&ICF_LOCK) ICU8(tmpi,OC_LOCK_PREFIX); switch (t1.raw_type) { case RT_I8: case RT_U8: ICRex(tmpi,i.u8[1]); ICU16(tmpi,i.u8[2]<<8+0xFE); break; case RT_I16: case RT_U16: ICOpSizeRex(tmpi,i.u8[1]); ICU16(tmpi,i.u8[2]<<8+0xFF); break; default: ICRex(tmpi,i.u8[1]); ICU16(tmpi,i.u8[2]<<8+0xFF); } ICModr2(tmpi,i,,d1,rip); return; } if (I8_MIN<=d<=I8_MAX || t1&(RTG_MASK-RTF_UNSIGNED)==RT_I8) { ICSlashOp(tmpi,t1,r1,d1,SLASH_OP_IMM_U8+op.u8[1],rip+1); ICU8(tmpi,d); return; } if (I32_MIN<=d<=I32_MAX || t1.raw_type<RT_I64) { ICSlashOp(tmpi,t1,r1,d1,SLASH_OP_IMM_U32+op.u8[1],rip); if (t1&(RTG_MASK-RTF_UNSIGNED)==RT_I16) ICU16(tmpi,d); else ICU32(tmpi,d); return; } break; case MDf_STK: ICAddSubEctImm(tmpi,MDF_REG+RT_I64,REG_RAX,0,t2,r2,d2,d,op,rip); ICPushRegs(tmpi,1<<REG_RAX); return; } ICAddEct(tmpi,t1,r1,d1,MDF_IMM+RT_I64,0,d,t2,r2,d2,op.u8[0],rip); } U0 ICSub(CIntermediateCode *tmpi,CICType t1,I64 r1,I64 d1, CICType t2,I64 r2,I64 d2,CICType t3,I64 r3,I64 d3,I64 rip) { I64 i=0x48,op=0x2B; Bool swap=FALSE; if (r3!=REG_RAX) { swap=TRUE; SwapI64(&t2,&t3); SwapI64(&r2,&r3); SwapI64(&d2,&d3); } if (t2.raw_type>=RT_I64 && r2.u8[0]!=REG_RAX && (!(t2&MDF_SIB) || r2.u8[1]&15!=REG_RAX) && t2&MDG_REG_DISP_SIB_RIP) { ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0,t3,r3,d3,rip); if (!swap) { op=0x03; ICU24(tmpi,0xD8F748); } i=ICModr1(REG_RAX,t2,r2,d2); if (tmpi->ic_flags&ICF_LOCK) ICU8(tmpi,OC_LOCK_PREFIX); ICRex(tmpi,i.u8[1]); ICU16(tmpi,i.u8[2]<<8+op); ICModr2(tmpi,i,,d2,rip); ICMov(tmpi,t1,r1,d1,MDF_REG+RT_I64,REG_RAX,0,rip); } else { if (!(t3&MDF_REG) || t3.raw_type<RT_I64) { if (swap) { swap=FALSE; ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0,t3,r3,d3,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0,t2,r2,d2,rip); r2=REG_RAX; r3=REG_RCX; } else { ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0,t3,r3,d3,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0,t2,r2,d2,rip); r3=REG_RAX; r2=REG_RCX; } } else { ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0,t2,r2,d2,rip); r2=REG_RCX; } if (swap) { op=0x03; ICU24(tmpi,0xD9F748); } if (r3>7) i++; if (r2>7) i+=4; if (tmpi->ic_flags&ICF_LOCK) ICU8(tmpi,OC_LOCK_PREFIX); ICU24(tmpi,0xC00000+i+(r3&7)<<16+(r2&7)<<19+op<<8); ICMov(tmpi,t1,r1,d1,MDF_REG+RT_I64,r2,0,rip); } } U0 ICMul(CIntermediateCode *tmpi,I64 rip) { I64 i,r2,r=REG_RAX,j; CICArg *arg1,*arg2; Bool alt; if (tmpi->arg1.type&MDF_IMM) { arg1=&tmpi->arg2; arg2=&tmpi->arg1; alt=TRUE; } else { arg1=&tmpi->arg1; arg2=&tmpi->arg2; alt=FALSE; } i=arg2->disp; if (!(tmpi->ic_class->raw_type&RTF_UNSIGNED) && arg2->type&MDF_IMM && I32_MIN<=i<=I32_MAX) { if (tmpi->res.type==MDF_REG+RT_I64) { ICMov(tmpi,tmpi->res.type,tmpi->res.reg,tmpi->res.disp, arg1->type,arg1->reg,arg1->disp,rip); r=tmpi->res.reg; } else ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0,arg1->type,arg1->reg,arg1->disp,rip); if (r>7) j=0xC0004D; else j=0xC00048; if (I8_MIN<=i<=I8_MAX) ICU32(tmpi,i<<24+0x6B00+j+(r&7)<<16+(r&7)<<19); else { ICU24(tmpi,0x6900+j+(r&7)<<16+(r&7)<<19); ICU32(tmpi,i); } } else { if (tmpi->ic_class->raw_type&RTF_UNSIGNED) i=0xE0F748; else i=0xE8F748; if (alt) { ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0,arg1->type,arg1->reg,arg1->disp,rip); r2=REG_RCX; ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0,arg2->type,arg2->reg,arg2->disp,rip); } else { ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0,arg2->type,arg2->reg,arg2->disp,rip); if (!(arg1->type&MDF_REG) || arg1->type.raw_type<RT_I64) { ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0, arg1->type,arg1->reg,arg1->disp,rip); r2=REG_RCX; } else r2=arg1->reg; } if (r2>7) { i++; r2&=7; } ICU24(tmpi,i+r2<<16); } ICMov(tmpi,tmpi->res.type,tmpi->res.reg,tmpi->res.disp, MDF_REG+RT_I64,r,0,rip); } U0 ICMulEqu(CIntermediateCode *tmpi,I64 rip) { I64 i=tmpi->arg2.disp,r=REG_RAX,j; if (!(tmpi->ic_class->raw_type&RTF_UNSIGNED) && tmpi->arg2.type&MDF_IMM && I32_MIN<=i<=I32_MAX) { if (tmpi->ic_flags & ICF_BY_VAL) { if (tmpi->arg1.type==MDF_REG+RT_I64) r=tmpi->arg1.reg; else ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0, tmpi->arg1.type&MDG_MASK+tmpi->arg1_type_pointed_to, tmpi->arg1.reg,tmpi->arg1.disp,rip); if (r>7) j=0xC0004D; else j=0xC00048; if (I8_MIN<=i<=I8_MAX) ICU32(tmpi,i<<24+0x6B00+j+(r&7)<<16+(r&7)<<19); else { ICU24(tmpi,0x6900+j+(r&7)<<16+(r&7)<<19); ICU32(tmpi,i); } ICMov(tmpi,tmpi->arg1.type&MDG_MASK+tmpi->arg1_type_pointed_to, tmpi->arg1.reg,tmpi->arg1.disp,MDF_REG+RT_I64,r,0,rip); } else { ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0, tmpi->arg1.type,tmpi->arg1.reg,tmpi->arg1.disp,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RBX,0, MDF_DISP+tmpi->arg1_type_pointed_to,REG_RCX,0,rip); r=REG_RBX; if (I8_MIN<=i<=I8_MAX) ICU32(tmpi,i<<24+0xDB6B48); else { ICU24(tmpi,0xDB6948); ICU32(tmpi,i); } ICMov(tmpi,MDF_DISP+tmpi->arg1_type_pointed_to,REG_RCX,0, MDF_REG+RT_I64,REG_RBX,0,rip); } } else { if (tmpi->ic_class->raw_type&RTF_UNSIGNED) i=0xE3F748; else i=0xEBF748; if (tmpi->ic_flags & ICF_BY_VAL) { ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0, tmpi->arg2.type,tmpi->arg2.reg,tmpi->arg2.disp,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RBX,0, tmpi->arg1.type&MDG_MASK+tmpi->arg1_type_pointed_to, tmpi->arg1.reg,tmpi->arg1.disp,rip); ICU24(tmpi,i); ICMov(tmpi,tmpi->arg1.type&MDG_MASK+tmpi->arg1_type_pointed_to, tmpi->arg1.reg,tmpi->arg1.disp,MDF_REG+RT_I64,REG_RAX,0,rip); } else { ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0, tmpi->arg2.type,tmpi->arg2.reg,tmpi->arg2.disp,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0, tmpi->arg1.type,tmpi->arg1.reg,tmpi->arg1.disp,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RBX,0, MDF_DISP+tmpi->arg1_type_pointed_to,REG_RCX,0,rip); ICU24(tmpi,i); ICMov(tmpi,MDF_DISP+tmpi->arg1_type_pointed_to,REG_RCX,0, MDF_REG+RT_I64,REG_RAX,0,rip); } } if (tmpi->res.type.mode) ICMov(tmpi,tmpi->res.type,tmpi->res.reg,tmpi->res.disp, MDF_REG+RT_I64,r,0,rip); } U0 ICDiv(CIntermediateCode *tmpi,I64 rip) { ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0, tmpi->arg2.type,tmpi->arg2.reg,tmpi->arg2.disp,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0, tmpi->arg1.type,tmpi->arg1.reg,tmpi->arg1.disp,rip); if (tmpi->ic_class->raw_type&RTF_UNSIGNED) { ICZero(tmpi,REG_RDX); ICU24(tmpi,0xF1F748); } else { ICU16(tmpi,0x9948); ICU24(tmpi,0xF9F748); } ICMov(tmpi,tmpi->res.type,tmpi->res.reg,tmpi->res.disp, MDF_REG+RT_I64,REG_RAX,0,rip); } U0 ICDivEqu(CIntermediateCode *tmpi,Bool is_mod,I64 rip) { if (tmpi->ic_flags & ICF_BY_VAL) { ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0, tmpi->arg2.type,tmpi->arg2.reg,tmpi->arg2.disp,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0, tmpi->arg1.type&MDG_MASK+tmpi->arg1_type_pointed_to, tmpi->arg1.reg,tmpi->arg1.disp,rip); if (tmpi->ic_class->raw_type&RTF_UNSIGNED) { ICZero(tmpi,REG_RDX); ICU24(tmpi,0xF1F748); } else { ICU16(tmpi,0x9948); ICU24(tmpi,0xF9F748); } if (is_mod) ICMov(tmpi,tmpi->arg1.type&MDG_MASK+tmpi->arg1_type_pointed_to, tmpi->arg1.reg,tmpi->arg1.disp,MDF_REG+RT_I64,REG_RDX,0,rip); else ICMov(tmpi,tmpi->arg1.type&MDG_MASK+tmpi->arg1_type_pointed_to, tmpi->arg1.reg,tmpi->arg1.disp,MDF_REG+RT_I64,REG_RAX,0,rip); } else { ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0, tmpi->arg2.type,tmpi->arg2.reg,tmpi->arg2.disp,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RBX,0, tmpi->arg1.type,tmpi->arg1.reg,tmpi->arg1.disp,rip); //dangerous might clobber RBX in Mov, but it doesn't ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0,MDF_DISP+tmpi->arg1_type_pointed_to, REG_RBX,0,rip); if (tmpi->ic_class->raw_type&RTF_UNSIGNED) { ICZero(tmpi,REG_RDX); ICU24(tmpi,0xF1F748); } else { ICU16(tmpi,0x9948); ICU24(tmpi,0xF9F748); } if (is_mod) ICMov(tmpi,MDF_DISP+tmpi->arg1_type_pointed_to,REG_RBX,0, MDF_REG+RT_I64,REG_RDX,0,rip); else ICMov(tmpi,MDF_DISP+tmpi->arg1_type_pointed_to,REG_RBX,0, MDF_REG+RT_I64,REG_RAX,0,rip); } if (tmpi->res.type.mode) { if (is_mod) ICMov(tmpi,tmpi->res.type,tmpi->res.reg,tmpi->res.disp, MDF_REG+RT_I64,REG_RDX,0,rip); else ICMov(tmpi,tmpi->res.type,tmpi->res.reg,tmpi->res.disp, MDF_REG+RT_I64,REG_RAX,0,rip); } } U0 ICMod(CIntermediateCode *tmpi,I64 rip) { ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0, tmpi->arg2.type,tmpi->arg2.reg,tmpi->arg2.disp,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0, tmpi->arg1.type,tmpi->arg1.reg,tmpi->arg1.disp,rip); if (tmpi->ic_class->raw_type&RTF_UNSIGNED) { ICZero(tmpi,REG_RDX); ICU24(tmpi,0xF1F748); } else { ICU16(tmpi,0x9948); ICU24(tmpi,0xF9F748); } ICMov(tmpi,tmpi->res.type,tmpi->res.reg,tmpi->res.disp, MDF_REG+RT_I64,REG_RDX,0,rip); } U0 ICAddSubEctEqu(CIntermediateCode *tmpi,U8 type_pointed_to, CICType t1,I64 r1,I64 d1,CICType t2,I64 r2,I64 d2, CICType t3,I64 r3,I64 d3,I64 op,I64 rip) { Bool done; I64 res_reg,tmp,i; if (tmpi->ic_flags & ICF_BY_VAL) { if (t3&MDF_IMM) { ICAddSubEctImm(tmpi,t2&MDG_MASK+type_pointed_to,r2,d2,t2&MDG_MASK+ type_pointed_to,r2,d2,d3,op,rip); if (t1.mode) ICMov(tmpi,t1,r1,d1,t2&MDG_MASK+type_pointed_to,r2,d2,rip); return; } else { done=FALSE; if (type_pointed_to>=RT_I64) { if (!t1.mode && t2&MDG_REG_DISP_SIB_RIP) { if (t3&MDF_REG) tmp=r3; else { tmp=REG_RCX; ICMov(tmpi,MDF_REG+RT_I64,tmp,0,t3,r3,d3,rip); } i=ICModr1(tmp,t2&MDG_MASK+type_pointed_to,r2,d2); if (tmpi->ic_flags&ICF_LOCK) ICU8(tmpi,OC_LOCK_PREFIX); ICRex(tmpi,i.u8[1]); ICU16(tmpi,i.u8[2]<<8+op.u8[5]); ICModr2(tmpi,i,,d2,rip); return; } if (t3.raw_type>=RT_I64 && t3&MDG_REG_DISP_SIB_RIP) { if (t2&MDF_REG) res_reg=r2; else { res_reg=REG_RCX; ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0,t2,r2,d2,rip); } i=ICModr1(res_reg,t3&MDG_MASK+type_pointed_to,r3,d3); if (tmpi->ic_flags&ICF_LOCK) ICU8(tmpi,OC_LOCK_PREFIX); ICRex(tmpi,i.u8[1]); ICU16(tmpi,i.u8[2]<<8+op.u8[0]); ICModr2(tmpi,i,,d3,rip); ICMov(tmpi,t2&MDG_MASK+type_pointed_to,r2,d2, MDF_REG+RT_I64,res_reg,0,rip); done=TRUE; } } if (!done) { ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0,t3,r3,d3,rip); if (t2&MDF_REG && r2!=REG_RAX) res_reg=r2; else { res_reg=REG_RCX; ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0, t2&MDG_MASK+type_pointed_to,r2,d2,rip); } if (tmpi->ic_flags&ICF_LOCK) ICU8(tmpi,OC_LOCK_PREFIX); if (res_reg>7) ICU8(tmpi,0x4C); else ICU8(tmpi,0x48); ICU16(tmpi,0xC000+op.u8[0]+(res_reg&7)<<11); ICMov(tmpi,t2&MDG_MASK+type_pointed_to,r2,d2, MDF_REG+RT_I64,res_reg,0,rip); } } } else { done=FALSE; if (t3&MDF_IMM && op.u8[2]) { if (!d3.u32[1]) { if (tmpi->ic_flags&ICF_RES_NOT_USED && t2&MDF_REG && d3(U64)<=I8_MAX) { ICSlashOp(tmpi,MDF_DISP+type_pointed_to,r2,0,0x838000+op.u8[4],rip); ICU8(tmpi,d3); done=TRUE; } else if (op.u8[2]==0x24) //AND { ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0,t2,r2,d2,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0, MDF_DISP+type_pointed_to,REG_RCX,0,rip); res_reg=REG_RAX; if (tmpi->ic_flags&ICF_LOCK) ICU8(tmpi,OC_LOCK_PREFIX); ICU16(tmpi,op.u8[3]<<8+0x40); ICU32(tmpi,d3); ICMov(tmpi,MDF_DISP+type_pointed_to,REG_RCX,0, MDF_REG+RT_I64,res_reg,0,rip); done=TRUE; } else if (type_pointed_to<RT_I64) //OR/XOR { ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0,t2,r2,d2,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0, MDF_DISP+type_pointed_to,REG_RCX,0,rip); res_reg=REG_RAX; if (tmpi->ic_flags&ICF_LOCK) ICU8(tmpi,OC_LOCK_PREFIX); if (d3.u16[1]) { ICU16(tmpi,op.u8[3]<<8+0x40); ICU32(tmpi,d3); } else if (d3.u8[1]) { ICU24(tmpi,op.u8[3]<<16+0x4000+OC_OP_SIZE_PREFIX); ICU16(tmpi,d3); } else { ICU16(tmpi,op.u8[2]<<8+0x40); ICU8(tmpi,d3); } ICMov(tmpi,MDF_DISP+type_pointed_to,REG_RCX,0, MDF_REG+RT_I64,res_reg,0,rip); done=TRUE; } } } if (!done) { ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0,t3,r3,d3,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0,t2,r2,d2,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RBX,0, MDF_DISP+type_pointed_to,REG_RCX,0,rip); res_reg=REG_RBX; if (tmpi->ic_flags&ICF_LOCK) ICU8(tmpi,OC_LOCK_PREFIX); ICU8(tmpi,0x48); ICU16(tmpi,0xC000+op.u8[0]+(res_reg&7)<<11); ICMov(tmpi,MDF_DISP+type_pointed_to,REG_RCX,0, MDF_REG+RT_I64,res_reg,0,rip); } } if (t1.mode) ICMov(tmpi,t1,r1,d1,MDF_REG+RT_I64,res_reg,0,rip); } U0 ICShift(CIntermediateCode *tmpi,CICType t1,I64 r1,I64 d1, CICType t2,I64 r2,I64 d2,CICType t3,I64 r3,I64 d3, I64 us,I64 is,I64 rip) { I64 i=0x48,res_reg; if (tmpi->ic_class->raw_type&RTF_UNSIGNED || tmpi->ic_flags & ICF_USE_UNSIGNED) is=us; if (t1&MDF_REG) { res_reg=r1; if (res_reg>7) i++; } else res_reg=REG_RAX; if (t3&MDF_IMM) { ICMov(tmpi,MDF_REG+RT_I64,res_reg,0,t2,r2,d2,rip); if (d3==1) ICU24(tmpi,i+is.u16[2]<<8+(res_reg&7)<<16); else { ICU24(tmpi,i+is.u16[0]<<8+(res_reg&7)<<16); ICU8(tmpi,d3); } } else { ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0,t3,r3,d3,rip); ICMov(tmpi,MDF_REG+RT_I64,res_reg,0,t2,r2,d2,rip); ICU24(tmpi,i+is.u16[1]<<8+(res_reg&7)<<16); } ICMov(tmpi,t1,r1,d1,MDF_REG+RT_I64,res_reg,0,rip); } U0 ICShiftEqu(CIntermediateCode *tmpi,U8 type_pointed_to, CICType t1,I64 r1,I64 d1, CICType t2,I64 r2,I64 d2, CICType t3,I64 r3,I64 d3,I64 us,I64 is,I64 rip) { I64 res_reg; if (tmpi->ic_class->raw_type&RTF_UNSIGNED || tmpi->ic_flags & ICF_USE_UNSIGNED) is=us; if (tmpi->ic_flags & ICF_BY_VAL) { if (!(t3&MDF_IMM)) ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0,t3,r3,d3,rip); if (t2&MDF_REG) res_reg=r2; else { res_reg=REG_RAX; ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0, t2&MDG_MASK+type_pointed_to,r2,d2,rip); } if (res_reg>7) ICU8(tmpi,0x49); else ICU8(tmpi,0x48); if (t3&MDF_IMM) { if (d3==1) ICU16(tmpi,is.u16[2]+(res_reg&7)<<8); else { ICU16(tmpi,is.u16[0]+(res_reg&7)<<8); ICU8(tmpi,d3); } } else ICU16(tmpi,is.u16[1]+(res_reg&7)<<8); ICMov(tmpi,t2&MDG_MASK+type_pointed_to,r2,d2, MDF_REG+RT_I64,res_reg,0,rip); } else { if (!(t3&MDF_IMM)) ICMov(tmpi,MDF_REG+RT_I64,REG_RCX,0,t3,r3,d3,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RDX,0,t2,r2,d2,rip); ICMov(tmpi,MDF_REG+RT_I64,REG_RAX,0,MDF_DISP+type_pointed_to,REG_RDX,0,rip); res_reg=REG_RAX; ICU8(tmpi,0x48); if (t3&MDF_IMM) { if (d3==1) ICU16(tmpi,is.u16[2]+(res_reg&7)<<8); else { ICU16(tmpi,is.u16[0]+(res_reg&7)<<8); ICU8(tmpi,d3); } } else ICU16(tmpi,is.u16[1]+(res_reg&7)<<8); ICMov(tmpi, MDF_DISP+type_pointed_to,REG_RDX,0,MDF_REG+RT_I64,res_reg,0,rip); } if (t1.mode) ICMov(tmpi,t1,r1,d1,MDF_REG+RT_I64,res_reg,0,rip); }