Bochs/bochs/cpu/mmx.cc

2814 lines
76 KiB
C++
Raw Normal View History

2002-10-16 21:37:35 +04:00
/////////////////////////////////////////////////////////////////////////
2010-09-27 00:20:27 +04:00
// $Id: mmx.cc,v 1.96 2010-09-26 20:20:27 sshwarts Exp $
/////////////////////////////////////////////////////////////////////////
2002-10-16 21:37:35 +04:00
//
2009-10-15 00:45:29 +04:00
// Copyright (c) 2002-2009 Stanislav Shwartsman
2007-03-24 00:27:13 +03:00
// Written by Stanislav Shwartsman [sshwarts at sourceforge net]
2002-10-16 21:37:35 +04:00
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License as published by the Free Software Foundation; either
// version 2 of the License, or (at your option) any later version.
//
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
2009-01-16 21:18:59 +03:00
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA B 02110-1301 USA
2002-10-16 21:37:35 +04:00
//
/////////////////////////////////////////////////////////////////////////
2002-10-16 21:37:35 +04:00
#define NEED_CPU_REG_SHORTCUTS 1
#include "bochs.h"
#include "cpu.h"
#define LOG_THIS BX_CPU_THIS_PTR
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2003-08-01 13:32:33 +04:00
void BX_CPU_C::print_state_MMX(void)
{
for(int i=0;i<8;i++) {
BxPackedMmxRegister mm = BX_READ_MMX_REG(i);
BX_DEBUG(("MM%d: %08x%08x\n", i, MMXUD1(mm), MMXUD0(mm)));
}
}
2003-05-15 20:41:17 +04:00
void BX_CPU_C::prepareFPU2MMX(void)
{
2003-08-01 13:32:33 +04:00
FPU_TAG_WORD = 0;
FPU_TOS = 0; /* reset FPU Top-Of-Stack */
}
#endif
/* 0F 38 00 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSHUFB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2008-02-03 00:46:54 +03:00
for(unsigned j=0; j<8; j++)
{
unsigned mask = op2.mmxubyte(j);
if (mask & 0x80)
result.mmxubyte(j) = 0;
else
result.mmxubyte(j) = op1.mmxubyte(mask & 0x7);
}
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 38 01 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PHADDW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUW0(result) = MMXUW0(op1) + MMXUW1(op1);
MMXUW1(result) = MMXUW2(op1) + MMXUW3(op1);
MMXUW2(result) = MMXUW0(op2) + MMXUW1(op2);
MMXUW3(result) = MMXUW2(op2) + MMXUW3(op2);
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 38 02 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PHADDD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUD0(result) = MMXUD0(op1) + MMXUD1(op1);
MMXUD1(result) = MMXUD0(op2) + MMXUD1(op2);
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 38 03 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PHADDSW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXSW0(result) = SaturateDwordSToWordS(Bit32s(MMXSW0(op1)) + Bit32s(MMXSW1(op1)));
MMXSW1(result) = SaturateDwordSToWordS(Bit32s(MMXSW2(op1)) + Bit32s(MMXSW3(op1)));
MMXSW2(result) = SaturateDwordSToWordS(Bit32s(MMXSW0(op2)) + Bit32s(MMXSW1(op2)));
MMXSW3(result) = SaturateDwordSToWordS(Bit32s(MMXSW2(op2)) + Bit32s(MMXSW3(op2)));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 38 04 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMADDUBSW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
for(unsigned j=0; j<4; j++)
{
Bit32s temp = Bit32s(op1.mmxubyte(j*2+0))*Bit32s(op2.mmxsbyte(j*2+0)) +
Bit32s(op1.mmxubyte(j*2+1))*Bit32s(op2.mmxsbyte(j*2+1));
result.mmx16s(j) = SaturateDwordSToWordS(temp);
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 38 05 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PHSUBSW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXSW0(result) = SaturateDwordSToWordS(Bit32s(MMXSW0(op1)) - Bit32s(MMXSW1(op1)));
MMXSW1(result) = SaturateDwordSToWordS(Bit32s(MMXSW2(op1)) - Bit32s(MMXSW3(op1)));
MMXSW2(result) = SaturateDwordSToWordS(Bit32s(MMXSW0(op2)) - Bit32s(MMXSW1(op2)));
MMXSW3(result) = SaturateDwordSToWordS(Bit32s(MMXSW2(op2)) - Bit32s(MMXSW3(op2)));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 38 05 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PHSUBW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUW0(result) = MMXUW0(op1) - MMXUW1(op1);
MMXUW1(result) = MMXUW2(op1) - MMXUW3(op1);
MMXUW2(result) = MMXUW0(op2) - MMXUW1(op2);
MMXUW3(result) = MMXUW2(op2) - MMXUW3(op2);
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 38 06 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PHSUBD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUD0(result) = MMXUD0(op1) - MMXUD1(op1);
MMXUD1(result) = MMXUD0(op2) - MMXUD1(op2);
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 38 08 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSIGNB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
for(unsigned j=0; j<8; j++) {
int sign = (op2.mmxsbyte(j) > 0) - (op2.mmxsbyte(j) < 0);
op1.mmxsbyte(j) *= sign;
}
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F 38 09 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSIGNW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
for(unsigned j=0; j<4; j++) {
int sign = (op2.mmx16s(j) > 0) - (op2.mmx16s(j) < 0);
op1.mmx16s(j) *= sign;
}
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F 38 0A */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSIGND_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
int sign;
2008-02-03 00:46:54 +03:00
sign = (MMXSD0(op2) > 0) - (MMXSD0(op2) < 0);
MMXSD0(op1) *= sign;
sign = (MMXSD1(op2) > 0) - (MMXSD1(op2) < 0);
MMXSD1(op1) *= sign;
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F 38 0B */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMULHRSW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
for(unsigned j=0; j<4; j++) {
Bit32s temp = Bit32s(op1.mmx16s(j)) * Bit32s(op2.mmx16s(j));
result.mmx16u(j) = ((temp >> 14) + 1) >> 1;
}
MMXUW0(result) = (((MMXSW0(op1) * MMXSW0(op2)) >> 14) + 1) >> 1;
MMXUW1(result) = (((MMXSW1(op1) * MMXSW1(op2)) >> 14) + 1) >> 1;
MMXUW2(result) = (((MMXSW2(op1) * MMXSW2(op2)) >> 14) + 1) >> 1;
MMXUW3(result) = (((MMXSW3(op1) * MMXSW3(op2)) >> 14) + 1) >> 1;
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 38 1C */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PABSB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op;
if (i->modC0()) {
op = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if (MMXSB0(op) < 0) MMXUB0(op) = -MMXSB0(op);
if (MMXSB1(op) < 0) MMXUB1(op) = -MMXSB1(op);
if (MMXSB2(op) < 0) MMXUB2(op) = -MMXSB2(op);
if (MMXSB3(op) < 0) MMXUB3(op) = -MMXSB3(op);
if (MMXSB4(op) < 0) MMXUB4(op) = -MMXSB4(op);
if (MMXSB5(op) < 0) MMXUB5(op) = -MMXSB5(op);
if (MMXSB6(op) < 0) MMXUB6(op) = -MMXSB6(op);
if (MMXSB7(op) < 0) MMXUB7(op) = -MMXSB7(op);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op);
#endif
}
/* 0F 38 1D */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PABSW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op;
if (i->modC0()) {
op = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if (MMXSW0(op) < 0) MMXUW0(op) = -MMXSW0(op);
if (MMXSW1(op) < 0) MMXUW1(op) = -MMXSW1(op);
if (MMXSW2(op) < 0) MMXUW2(op) = -MMXSW2(op);
if (MMXSW3(op) < 0) MMXUW3(op) = -MMXSW3(op);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op);
#endif
}
/* 0F 38 1E */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PABSD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op;
if (i->modC0()) {
op = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if (MMXSD0(op) < 0) MMXUD0(op) = -MMXSD0(op);
if (MMXSD1(op) < 0) MMXUD1(op) = -MMXSD1(op);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op);
#endif
}
/* 0F 3A 0F */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PALIGNR_PqQqIb(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2006-05-16 20:20:26 +04:00
unsigned shift = i->Ib() * 8;
if(shift == 0)
MMXUQ(result) = MMXUQ(op2);
else if(shift < 64)
MMXUQ(result) = (MMXUQ(op2) >> shift) | (MMXUQ(op1) << (64-shift));
else if(shift < 128)
MMXUQ(result) = MMXUQ(op1) >> (shift-64);
else
MMXUQ(result) = 0;
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 60 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PUNPCKLBW_PqQd(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
2008-02-03 00:46:54 +03:00
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUB7(result) = MMXUB3(op2);
MMXUB6(result) = MMXUB3(op1);
MMXUB5(result) = MMXUB2(op2);
MMXUB4(result) = MMXUB2(op1);
MMXUB3(result) = MMXUB1(op2);
MMXUB2(result) = MMXUB1(op1);
MMXUB1(result) = MMXUB0(op2);
MMXUB0(result) = MMXUB0(op1);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 61 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PUNPCKLWD_PqQd(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
2008-02-03 00:46:54 +03:00
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUW3(result) = MMXUW1(op2);
MMXUW2(result) = MMXUW1(op1);
MMXUW1(result) = MMXUW0(op2);
MMXUW0(result) = MMXUW0(op1);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 62 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PUNPCKLDQ_PqQd(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
2008-02-03 00:46:54 +03:00
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUD1(op1) = MMXUD0(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F 63 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PACKSSWB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXSB0(result) = SaturateWordSToByteS(MMXSW0(op1));
MMXSB1(result) = SaturateWordSToByteS(MMXSW1(op1));
MMXSB2(result) = SaturateWordSToByteS(MMXSW2(op1));
MMXSB3(result) = SaturateWordSToByteS(MMXSW3(op1));
MMXSB4(result) = SaturateWordSToByteS(MMXSW0(op2));
MMXSB5(result) = SaturateWordSToByteS(MMXSW1(op2));
MMXSB6(result) = SaturateWordSToByteS(MMXSW2(op2));
MMXSB7(result) = SaturateWordSToByteS(MMXSW3(op2));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 64 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PCMPGTB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2003-05-15 20:41:17 +04:00
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2003-05-15 20:41:17 +04:00
MMXUB0(op1) = (MMXSB0(op1) > MMXSB0(op2)) ? 0xff : 0;
MMXUB1(op1) = (MMXSB1(op1) > MMXSB1(op2)) ? 0xff : 0;
MMXUB2(op1) = (MMXSB2(op1) > MMXSB2(op2)) ? 0xff : 0;
MMXUB3(op1) = (MMXSB3(op1) > MMXSB3(op2)) ? 0xff : 0;
MMXUB4(op1) = (MMXSB4(op1) > MMXSB4(op2)) ? 0xff : 0;
MMXUB5(op1) = (MMXSB5(op1) > MMXSB5(op2)) ? 0xff : 0;
MMXUB6(op1) = (MMXSB6(op1) > MMXSB6(op2)) ? 0xff : 0;
MMXUB7(op1) = (MMXSB7(op1) > MMXSB7(op2)) ? 0xff : 0;
/* now write result back to destination */
2003-05-15 20:41:17 +04:00
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F 65 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PCMPGTW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2003-05-15 20:41:17 +04:00
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2003-05-15 20:41:17 +04:00
MMXUW0(op1) = (MMXSW0(op1) > MMXSW0(op2)) ? 0xffff : 0;
MMXUW1(op1) = (MMXSW1(op1) > MMXSW1(op2)) ? 0xffff : 0;
MMXUW2(op1) = (MMXSW2(op1) > MMXSW2(op2)) ? 0xffff : 0;
MMXUW3(op1) = (MMXSW3(op1) > MMXSW3(op2)) ? 0xffff : 0;
/* now write result back to destination */
2003-05-15 20:41:17 +04:00
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F 66 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PCMPGTD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2003-05-15 20:41:17 +04:00
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2003-05-15 20:41:17 +04:00
MMXUD0(op1) = (MMXSD0(op1) > MMXSD0(op2)) ? 0xffffffff : 0;
MMXUD1(op1) = (MMXSD1(op1) > MMXSD1(op2)) ? 0xffffffff : 0;
/* now write result back to destination */
2003-05-15 20:41:17 +04:00
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F 67 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PACKUSWB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUB0(result) = SaturateWordSToByteU(MMXSW0(op1));
MMXUB1(result) = SaturateWordSToByteU(MMXSW1(op1));
MMXUB2(result) = SaturateWordSToByteU(MMXSW2(op1));
MMXUB3(result) = SaturateWordSToByteU(MMXSW3(op1));
MMXUB4(result) = SaturateWordSToByteU(MMXSW0(op2));
MMXUB5(result) = SaturateWordSToByteU(MMXSW1(op2));
MMXUB6(result) = SaturateWordSToByteU(MMXSW2(op2));
MMXUB7(result) = SaturateWordSToByteU(MMXSW3(op2));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 68 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PUNPCKHBW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUB7(result) = MMXUB7(op2);
MMXUB6(result) = MMXUB7(op1);
MMXUB5(result) = MMXUB6(op2);
MMXUB4(result) = MMXUB6(op1);
MMXUB3(result) = MMXUB5(op2);
MMXUB2(result) = MMXUB5(op1);
MMXUB1(result) = MMXUB4(op2);
MMXUB0(result) = MMXUB4(op1);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 69 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PUNPCKHWD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUW3(result) = MMXUW3(op2);
MMXUW2(result) = MMXUW3(op1);
MMXUW1(result) = MMXUW2(op2);
MMXUW0(result) = MMXUW2(op1);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 6A */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PUNPCKHDQ_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUD1(result) = MMXUD1(op2);
MMXUD0(result) = MMXUD1(op1);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 6B */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PACKSSDW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXSW0(result) = SaturateDwordSToWordS(MMXSD0(op1));
MMXSW1(result) = SaturateDwordSToWordS(MMXSD1(op1));
MMXSW2(result) = SaturateDwordSToWordS(MMXSD0(op2));
MMXSW3(result) = SaturateDwordSToWordS(MMXSD1(op2));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 6E */
2010-02-10 20:21:15 +03:00
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOVD_PqEdR(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2010-02-10 20:21:15 +03:00
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
BxPackedMmxRegister op;
2010-02-10 20:21:15 +03:00
MMXUD0(op) = BX_READ_32BIT_REG(i->rm());
2007-11-28 01:12:45 +03:00
MMXUD1(op) = 0;
2003-01-08 23:33:28 +03:00
2010-02-10 20:21:15 +03:00
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op);
#endif
2010-02-10 20:21:15 +03:00
}
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOVD_PqEdM(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2010-02-10 20:21:15 +03:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op;
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
MMXUD0(op) = read_virtual_dword(i->seg(), eaddr);
MMXUD1(op) = 0;
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op);
#endif
}
2007-11-28 01:12:45 +03:00
/* 0F 6E */
#if BX_SUPPORT_X86_64
2009-10-24 15:17:51 +04:00
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOVQ_PqEqR(bxInstruction_c *i)
2007-11-28 01:12:45 +03:00
{
BX_CPU_THIS_PTR prepareMMX();
2009-10-24 15:17:51 +04:00
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2007-11-28 01:12:45 +03:00
BxPackedMmxRegister op;
2009-10-24 15:17:51 +04:00
MMXUQ(op) = BX_READ_64BIT_REG(i->rm());
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op);
}
2009-12-14 14:55:42 +03:00
#endif
2009-10-24 15:17:51 +04:00
2009-12-14 14:55:42 +03:00
/* 0F 6F */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOVQ_PqQqR(bxInstruction_c *i)
2009-10-24 15:17:51 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2009-10-24 15:17:51 +04:00
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2010-02-10 20:21:15 +03:00
BX_WRITE_MMX_REG(i->nnn(), BX_READ_MMX_REG(i->rm()));
#endif
2009-12-14 14:55:42 +03:00
}
2007-11-28 01:12:45 +03:00
2009-12-14 14:55:42 +03:00
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOVQ_PqQqM(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op;
2009-12-14 14:55:42 +03:00
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
MMXUQ(op) = read_virtual_qword(i->seg(), eaddr);
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F 70 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSHUFW_PqQqIb(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op, result;
Bit8u order = i->Ib();
2002-10-16 21:37:35 +04:00
/* op is a register or memory reference */
2002-10-16 21:37:35 +04:00
if (i->modC0()) {
op = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUW0(result) = op.mmx16u((order) & 0x3);
MMXUW1(result) = op.mmx16u((order>>2) & 0x3);
MMXUW2(result) = op.mmx16u((order>>4) & 0x3);
MMXUW3(result) = op.mmx16u((order>>6) & 0x3);
2002-10-16 21:37:35 +04:00
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F 74 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PCMPEQB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2003-05-15 20:41:17 +04:00
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2003-05-15 20:41:17 +04:00
MMXUB0(op1) = (MMXUB0(op1) == MMXUB0(op2)) ? 0xff : 0;
MMXUB1(op1) = (MMXUB1(op1) == MMXUB1(op2)) ? 0xff : 0;
MMXUB2(op1) = (MMXUB2(op1) == MMXUB2(op2)) ? 0xff : 0;
MMXUB3(op1) = (MMXUB3(op1) == MMXUB3(op2)) ? 0xff : 0;
MMXUB4(op1) = (MMXUB4(op1) == MMXUB4(op2)) ? 0xff : 0;
MMXUB5(op1) = (MMXUB5(op1) == MMXUB5(op2)) ? 0xff : 0;
MMXUB6(op1) = (MMXUB6(op1) == MMXUB6(op2)) ? 0xff : 0;
MMXUB7(op1) = (MMXUB7(op1) == MMXUB7(op2)) ? 0xff : 0;
/* now write result back to destination */
2003-05-15 20:41:17 +04:00
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F 75 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PCMPEQW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2003-05-15 20:41:17 +04:00
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2003-05-15 20:41:17 +04:00
MMXUW0(op1) = (MMXUW0(op1) == MMXUW0(op2)) ? 0xffff : 0;
MMXUW1(op1) = (MMXUW1(op1) == MMXUW1(op2)) ? 0xffff : 0;
MMXUW2(op1) = (MMXUW2(op1) == MMXUW2(op2)) ? 0xffff : 0;
MMXUW3(op1) = (MMXUW3(op1) == MMXUW3(op2)) ? 0xffff : 0;
/* now write result back to destination */
2003-05-15 20:41:17 +04:00
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F 76 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PCMPEQD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2003-05-15 20:41:17 +04:00
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2003-05-15 20:41:17 +04:00
MMXUD0(op1) = (MMXUD0(op1) == MMXUD0(op2)) ? 0xffffffff : 0;
MMXUD1(op1) = (MMXUD1(op1) == MMXUD1(op2)) ? 0xffffffff : 0;
/* now write result back to destination */
2003-05-15 20:41:17 +04:00
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F 77 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::EMMS(bxInstruction_c *i)
{
2010-03-02 09:49:41 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
2003-08-01 13:32:33 +04:00
FPU_TAG_WORD = 0xffff;
FPU_TOS = 0; /* reset FPU Top-Of-Stack */
#endif
}
/* 0F 7E */
2010-02-10 20:21:15 +03:00
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOVD_EdPdR(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2010-02-10 20:21:15 +03:00
BX_CPU_THIS_PTR prepareFPU2MMX();
BxPackedMmxRegister op = BX_READ_MMX_REG(i->nnn());
2010-02-10 20:21:15 +03:00
BX_WRITE_32BIT_REGZ(i->rm(), MMXUD0(op));
#endif
2010-02-10 20:21:15 +03:00
}
2010-02-10 20:21:15 +03:00
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOVD_EdPdM(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2010-02-10 20:21:15 +03:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op = BX_READ_MMX_REG(i->nnn());
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
write_virtual_dword(i->seg(), eaddr, MMXUD0(op));
// do not cause FPU2MMX transition if memory write faults
BX_CPU_THIS_PTR prepareFPU2MMX();
#endif
}
2007-11-28 01:12:45 +03:00
#if BX_SUPPORT_X86_64
/* 0F 7E */
2010-02-10 20:21:15 +03:00
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOVQ_EqPqR(bxInstruction_c *i)
2007-11-28 01:12:45 +03:00
{
BX_CPU_THIS_PTR prepareMMX();
2010-02-10 20:21:15 +03:00
BX_CPU_THIS_PTR prepareFPU2MMX();
2007-11-28 01:12:45 +03:00
BxPackedMmxRegister op = BX_READ_MMX_REG(i->nnn());
2010-02-10 20:21:15 +03:00
BX_WRITE_64BIT_REG(i->rm(), MMXUQ(op));
}
2007-11-28 01:12:45 +03:00
2010-02-10 20:21:15 +03:00
#endif
2010-02-10 20:21:15 +03:00
/* 0F 7F */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOVQ_QqPqR(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2010-02-10 20:21:15 +03:00
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX();
2007-11-28 01:12:45 +03:00
2010-02-10 20:21:15 +03:00
BX_WRITE_MMX_REG(i->rm(), BX_READ_MMX_REG(i->nnn()));
#endif
2010-02-10 20:21:15 +03:00
}
2007-11-28 01:12:45 +03:00
2010-02-10 20:21:15 +03:00
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOVQ_QqPqM(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op = BX_READ_MMX_REG(i->nnn());
2010-02-10 20:21:15 +03:00
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
write_virtual_qword(i->seg(), eaddr, MMXUQ(op));
// do not cause FPU2MMX transition if memory write faults
BX_CPU_THIS_PTR prepareFPU2MMX();
#endif
}
/* 0F C4 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PINSRW_PqEwIb(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn());
Bit16u op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_16BIT_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
op2 = read_virtual_word(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2010-09-27 00:20:27 +04:00
op1.mmx16u(i->Ib() & 0x3) = op2;
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F C5 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PEXTRW_GdPqIb(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
BxPackedMmxRegister op = BX_READ_MMX_REG(i->rm());
Bit32u result = (Bit32u) op.mmx16u(i->Ib() & 0x3);
BX_WRITE_32BIT_REGZ(i->nnn(), result);
#endif
}
/* 0F D1 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSRLW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if(MMXUQ(op2) > 15) MMXUQ(op1) = 0;
else
{
Bit8u shift = MMXUB0(op2);
2008-02-03 00:46:54 +03:00
MMXUW0(op1) >>= shift;
MMXUW1(op1) >>= shift;
MMXUW2(op1) >>= shift;
MMXUW3(op1) >>= shift;
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F D2 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSRLD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if(MMXUQ(op2) > 31) MMXUQ(op1) = 0;
else
{
Bit8u shift = MMXUB0(op2);
MMXUD0(op1) >>= shift;
MMXUD1(op1) >>= shift;
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F D3 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSRLQ_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if(MMXUQ(op2) > 63) {
MMXUQ(op1) = 0;
}
else {
MMXUQ(op1) >>= MMXUB0(op2);
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F D4 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PADDQ_PqQq(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2002-10-16 21:37:35 +04:00
MMXUQ(op1) += MMXUQ(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F D5 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMULLW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
Bit32u product1 = Bit32u(MMXUW0(op1)) * Bit32u(MMXUW0(op2));
Bit32u product2 = Bit32u(MMXUW1(op1)) * Bit32u(MMXUW1(op2));
Bit32u product3 = Bit32u(MMXUW2(op1)) * Bit32u(MMXUW2(op2));
Bit32u product4 = Bit32u(MMXUW3(op1)) * Bit32u(MMXUW3(op2));
2002-10-16 21:37:35 +04:00
MMXUW0(result) = product1 & 0xffff;
MMXUW1(result) = product2 & 0xffff;
MMXUW2(result) = product3 & 0xffff;
MMXUW3(result) = product4 & 0xffff;
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F D7 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMOVMSKB_GdPRq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
BxPackedMmxRegister op = BX_READ_MMX_REG(i->rm());
Bit32u result = 0;
2008-02-03 00:46:54 +03:00
if(MMXUB0(op) & 0x80) result |= 0x01;
if(MMXUB1(op) & 0x80) result |= 0x02;
if(MMXUB2(op) & 0x80) result |= 0x04;
if(MMXUB3(op) & 0x80) result |= 0x08;
if(MMXUB4(op) & 0x80) result |= 0x10;
if(MMXUB5(op) & 0x80) result |= 0x20;
if(MMXUB6(op) & 0x80) result |= 0x40;
if(MMXUB7(op) & 0x80) result |= 0x80;
/* now write result back to destination */
BX_WRITE_32BIT_REGZ(i->nnn(), result);
#endif
}
/* 0F D8 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSUBUSB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUQ(result) = 0;
if(MMXUB0(op1) > MMXUB0(op2)) MMXUB0(result) = MMXUB0(op1) - MMXUB0(op2);
if(MMXUB1(op1) > MMXUB1(op2)) MMXUB1(result) = MMXUB1(op1) - MMXUB1(op2);
if(MMXUB2(op1) > MMXUB2(op2)) MMXUB2(result) = MMXUB2(op1) - MMXUB2(op2);
if(MMXUB3(op1) > MMXUB3(op2)) MMXUB3(result) = MMXUB3(op1) - MMXUB3(op2);
if(MMXUB4(op1) > MMXUB4(op2)) MMXUB4(result) = MMXUB4(op1) - MMXUB4(op2);
if(MMXUB5(op1) > MMXUB5(op2)) MMXUB5(result) = MMXUB5(op1) - MMXUB5(op2);
if(MMXUB6(op1) > MMXUB6(op2)) MMXUB6(result) = MMXUB6(op1) - MMXUB6(op2);
if(MMXUB7(op1) > MMXUB7(op2)) MMXUB7(result) = MMXUB7(op1) - MMXUB7(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F D9 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSUBUSW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUQ(result) = 0;
if(MMXUW0(op1) > MMXUW0(op2)) MMXUW0(result) = MMXUW0(op1) - MMXUW0(op2);
if(MMXUW1(op1) > MMXUW1(op2)) MMXUW1(result) = MMXUW1(op1) - MMXUW1(op2);
if(MMXUW2(op1) > MMXUW2(op2)) MMXUW2(result) = MMXUW2(op1) - MMXUW2(op2);
if(MMXUW3(op1) > MMXUW3(op2)) MMXUW3(result) = MMXUW3(op1) - MMXUW3(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F DA */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMINUB_PqQq(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
2002-10-16 21:37:35 +04:00
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if(MMXUB0(op2) < MMXUB0(op1)) MMXUB0(op1) = MMXUB0(op2);
if(MMXUB1(op2) < MMXUB1(op1)) MMXUB1(op1) = MMXUB1(op2);
if(MMXUB2(op2) < MMXUB2(op1)) MMXUB2(op1) = MMXUB2(op2);
if(MMXUB3(op2) < MMXUB3(op1)) MMXUB3(op1) = MMXUB3(op2);
if(MMXUB4(op2) < MMXUB4(op1)) MMXUB4(op1) = MMXUB4(op2);
if(MMXUB5(op2) < MMXUB5(op1)) MMXUB5(op1) = MMXUB5(op2);
if(MMXUB6(op2) < MMXUB6(op1)) MMXUB6(op1) = MMXUB6(op2);
if(MMXUB7(op2) < MMXUB7(op1)) MMXUB7(op1) = MMXUB7(op2);
2002-10-16 21:37:35 +04:00
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F DB */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PAND_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUQ(op1) &= MMXUQ(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F DC */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PADDUSB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUB0(result) = SaturateWordSToByteU(Bit16s(MMXUB0(op1)) + Bit16s(MMXUB0(op2)));
MMXUB1(result) = SaturateWordSToByteU(Bit16s(MMXUB1(op1)) + Bit16s(MMXUB1(op2)));
MMXUB2(result) = SaturateWordSToByteU(Bit16s(MMXUB2(op1)) + Bit16s(MMXUB2(op2)));
MMXUB3(result) = SaturateWordSToByteU(Bit16s(MMXUB3(op1)) + Bit16s(MMXUB3(op2)));
MMXUB4(result) = SaturateWordSToByteU(Bit16s(MMXUB4(op1)) + Bit16s(MMXUB4(op2)));
MMXUB5(result) = SaturateWordSToByteU(Bit16s(MMXUB5(op1)) + Bit16s(MMXUB5(op2)));
MMXUB6(result) = SaturateWordSToByteU(Bit16s(MMXUB6(op1)) + Bit16s(MMXUB6(op2)));
MMXUB7(result) = SaturateWordSToByteU(Bit16s(MMXUB7(op1)) + Bit16s(MMXUB7(op2)));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F DD */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PADDUSW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUW0(result) = SaturateDwordSToWordU(Bit32s(MMXUW0(op1)) + Bit32s(MMXUW0(op2)));
MMXUW1(result) = SaturateDwordSToWordU(Bit32s(MMXUW1(op1)) + Bit32s(MMXUW1(op2)));
MMXUW2(result) = SaturateDwordSToWordU(Bit32s(MMXUW2(op1)) + Bit32s(MMXUW2(op2)));
MMXUW3(result) = SaturateDwordSToWordU(Bit32s(MMXUW3(op1)) + Bit32s(MMXUW3(op2)));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F DE */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMAXUB_PqQq(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
2002-10-16 21:37:35 +04:00
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if(MMXUB0(op2) > MMXUB0(op1)) MMXUB0(op1) = MMXUB0(op2);
if(MMXUB1(op2) > MMXUB1(op1)) MMXUB1(op1) = MMXUB1(op2);
if(MMXUB2(op2) > MMXUB2(op1)) MMXUB2(op1) = MMXUB2(op2);
if(MMXUB3(op2) > MMXUB3(op1)) MMXUB3(op1) = MMXUB3(op2);
if(MMXUB4(op2) > MMXUB4(op1)) MMXUB4(op1) = MMXUB4(op2);
if(MMXUB5(op2) > MMXUB5(op1)) MMXUB5(op1) = MMXUB5(op2);
if(MMXUB6(op2) > MMXUB6(op1)) MMXUB6(op1) = MMXUB6(op2);
if(MMXUB7(op2) > MMXUB7(op1)) MMXUB7(op1) = MMXUB7(op2);
2002-10-16 21:37:35 +04:00
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F DF */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PANDN_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2003-09-26 23:20:17 +04:00
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2003-09-26 23:20:17 +04:00
MMXUQ(op1) = ~(MMXUQ(op1)) & MMXUQ(op2);
/* now write result back to destination */
2003-09-26 23:20:17 +04:00
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F E0 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PAVGB_PqQq(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2003-05-15 20:41:17 +04:00
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
2002-10-16 21:37:35 +04:00
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2003-05-15 20:41:17 +04:00
MMXUB0(op1) = (MMXUB0(op1) + MMXUB0(op2) + 1) >> 1;
MMXUB1(op1) = (MMXUB1(op1) + MMXUB1(op2) + 1) >> 1;
MMXUB2(op1) = (MMXUB2(op1) + MMXUB2(op2) + 1) >> 1;
MMXUB3(op1) = (MMXUB3(op1) + MMXUB3(op2) + 1) >> 1;
MMXUB4(op1) = (MMXUB4(op1) + MMXUB4(op2) + 1) >> 1;
MMXUB5(op1) = (MMXUB5(op1) + MMXUB5(op2) + 1) >> 1;
MMXUB6(op1) = (MMXUB6(op1) + MMXUB6(op2) + 1) >> 1;
MMXUB7(op1) = (MMXUB7(op1) + MMXUB7(op2) + 1) >> 1;
2002-10-16 21:37:35 +04:00
/* now write result back to destination */
2003-05-15 20:41:17 +04:00
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F E1 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSRAW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2006-02-22 23:58:16 +03:00
if(!MMXUQ(op2)) return;
if(MMXUQ(op2) > 15) {
2002-10-16 21:37:35 +04:00
MMXUW0(result) = (MMXUW0(op1) & 0x8000) ? 0xffff : 0;
MMXUW1(result) = (MMXUW1(op1) & 0x8000) ? 0xffff : 0;
MMXUW2(result) = (MMXUW2(op1) & 0x8000) ? 0xffff : 0;
MMXUW3(result) = (MMXUW3(op1) & 0x8000) ? 0xffff : 0;
}
else {
2006-02-22 23:58:16 +03:00
Bit8u shift = MMXUB0(op2);
MMXUW0(result) = MMXUW0(op1) >> shift;
MMXUW1(result) = MMXUW1(op1) >> shift;
MMXUW2(result) = MMXUW2(op1) >> shift;
MMXUW3(result) = MMXUW3(op1) >> shift;
2002-10-16 21:37:35 +04:00
if(MMXUW0(op1) & 0x8000) MMXUW0(result) |= (0xffff << (16 - shift));
if(MMXUW1(op1) & 0x8000) MMXUW1(result) |= (0xffff << (16 - shift));
if(MMXUW2(op1) & 0x8000) MMXUW2(result) |= (0xffff << (16 - shift));
if(MMXUW3(op1) & 0x8000) MMXUW3(result) |= (0xffff << (16 - shift));
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F E2 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSRAD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2006-02-22 23:58:16 +03:00
if(!MMXUQ(op2)) return;
if(MMXUQ(op2) > 31) {
2002-10-16 21:37:35 +04:00
MMXUD0(result) = (MMXUD0(op1) & 0x80000000) ? 0xffffffff : 0;
MMXUD1(result) = (MMXUD1(op1) & 0x80000000) ? 0xffffffff : 0;
}
else {
2006-02-22 23:58:16 +03:00
Bit8u shift = MMXUB0(op2);
MMXUD0(result) = MMXUD0(op1) >> shift;
MMXUD1(result) = MMXUD1(op1) >> shift;
2008-02-03 00:46:54 +03:00
if(MMXUD0(op1) & 0x80000000)
2002-10-16 21:37:35 +04:00
MMXUD0(result) |= (0xffffffff << (32 - shift));
2008-02-03 00:46:54 +03:00
if(MMXUD1(op1) & 0x80000000)
2002-10-16 21:37:35 +04:00
MMXUD1(result) |= (0xffffffff << (32 - shift));
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F E3 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PAVGW_PqQq(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2003-05-15 20:41:17 +04:00
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
2002-10-16 21:37:35 +04:00
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2003-05-15 20:41:17 +04:00
MMXUW0(op1) = (MMXUW0(op1) + MMXUW0(op2) + 1) >> 1;
MMXUW1(op1) = (MMXUW1(op1) + MMXUW1(op2) + 1) >> 1;
MMXUW2(op1) = (MMXUW2(op1) + MMXUW2(op2) + 1) >> 1;
MMXUW3(op1) = (MMXUW3(op1) + MMXUW3(op2) + 1) >> 1;
2002-10-16 21:37:35 +04:00
/* now write result back to destination */
2003-05-15 20:41:17 +04:00
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F E4 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMULHUW_PqQq(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
Bit32u product1 = Bit32u(MMXUW0(op1)) * Bit32u(MMXUW0(op2));
Bit32u product2 = Bit32u(MMXUW1(op1)) * Bit32u(MMXUW1(op2));
Bit32u product3 = Bit32u(MMXUW2(op1)) * Bit32u(MMXUW2(op2));
Bit32u product4 = Bit32u(MMXUW3(op1)) * Bit32u(MMXUW3(op2));
2002-10-16 21:37:35 +04:00
MMXUW0(result) = (Bit16u)(product1 >> 16);
MMXUW1(result) = (Bit16u)(product2 >> 16);
MMXUW2(result) = (Bit16u)(product3 >> 16);
MMXUW3(result) = (Bit16u)(product4 >> 16);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F E5 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMULHW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
Bit32s product1 = Bit32s(MMXSW0(op1)) * Bit32s(MMXSW0(op2));
Bit32s product2 = Bit32s(MMXSW1(op1)) * Bit32s(MMXSW1(op2));
Bit32s product3 = Bit32s(MMXSW2(op1)) * Bit32s(MMXSW2(op2));
Bit32s product4 = Bit32s(MMXSW3(op1)) * Bit32s(MMXSW3(op2));
MMXUW0(result) = Bit16u(product1 >> 16);
MMXUW1(result) = Bit16u(product2 >> 16);
MMXUW2(result) = Bit16u(product3 >> 16);
MMXUW3(result) = Bit16u(product4 >> 16);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
2002-10-20 01:28:50 +04:00
/* 0F E7 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOVNTQ_MqPq(bxInstruction_c *i)
2002-10-20 01:28:50 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
BX_CPU_THIS_PTR prepareMMX();
2002-10-20 01:28:50 +04:00
BxPackedMmxRegister reg = BX_READ_MMX_REG(i->nnn());
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
write_virtual_qword(i->seg(), eaddr, MMXUQ(reg));
// do not cause FPU2MMX transition if memory write faults
BX_CPU_THIS_PTR prepareFPU2MMX();
#endif
2002-10-20 01:28:50 +04:00
}
/* 0F E8 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSUBSB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXSB0(result) = SaturateWordSToByteS(Bit16s(MMXSB0(op1)) - Bit16s(MMXSB0(op2)));
MMXSB1(result) = SaturateWordSToByteS(Bit16s(MMXSB1(op1)) - Bit16s(MMXSB1(op2)));
MMXSB2(result) = SaturateWordSToByteS(Bit16s(MMXSB2(op1)) - Bit16s(MMXSB2(op2)));
MMXSB3(result) = SaturateWordSToByteS(Bit16s(MMXSB3(op1)) - Bit16s(MMXSB3(op2)));
MMXSB4(result) = SaturateWordSToByteS(Bit16s(MMXSB4(op1)) - Bit16s(MMXSB4(op2)));
MMXSB5(result) = SaturateWordSToByteS(Bit16s(MMXSB5(op1)) - Bit16s(MMXSB5(op2)));
MMXSB6(result) = SaturateWordSToByteS(Bit16s(MMXSB6(op1)) - Bit16s(MMXSB6(op2)));
MMXSB7(result) = SaturateWordSToByteS(Bit16s(MMXSB7(op1)) - Bit16s(MMXSB7(op2)));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F E9 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSUBSW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXSW0(result) = SaturateDwordSToWordS(Bit32s(MMXSW0(op1)) - Bit32s(MMXSW0(op2)));
MMXSW1(result) = SaturateDwordSToWordS(Bit32s(MMXSW1(op1)) - Bit32s(MMXSW1(op2)));
MMXSW2(result) = SaturateDwordSToWordS(Bit32s(MMXSW2(op1)) - Bit32s(MMXSW2(op2)));
MMXSW3(result) = SaturateDwordSToWordS(Bit32s(MMXSW3(op1)) - Bit32s(MMXSW3(op2)));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F EA */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMINSW_PqQq(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
2002-10-16 21:37:35 +04:00
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if(MMXSW0(op2) < MMXSW0(op1)) MMXSW0(op1) = MMXSW0(op2);
if(MMXSW1(op2) < MMXSW1(op1)) MMXSW1(op1) = MMXSW1(op2);
if(MMXSW2(op2) < MMXSW2(op1)) MMXSW2(op1) = MMXSW2(op2);
if(MMXSW3(op2) < MMXSW3(op1)) MMXSW3(op1) = MMXSW3(op2);
2002-10-16 21:37:35 +04:00
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F EB */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::POR_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUQ(op1) |= MMXUQ(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F EC */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PADDSB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXSB0(result) = SaturateWordSToByteS(Bit16s(MMXSB0(op1)) + Bit16s(MMXSB0(op2)));
MMXSB1(result) = SaturateWordSToByteS(Bit16s(MMXSB1(op1)) + Bit16s(MMXSB1(op2)));
MMXSB2(result) = SaturateWordSToByteS(Bit16s(MMXSB2(op1)) + Bit16s(MMXSB2(op2)));
MMXSB3(result) = SaturateWordSToByteS(Bit16s(MMXSB3(op1)) + Bit16s(MMXSB3(op2)));
MMXSB4(result) = SaturateWordSToByteS(Bit16s(MMXSB4(op1)) + Bit16s(MMXSB4(op2)));
MMXSB5(result) = SaturateWordSToByteS(Bit16s(MMXSB5(op1)) + Bit16s(MMXSB5(op2)));
MMXSB6(result) = SaturateWordSToByteS(Bit16s(MMXSB6(op1)) + Bit16s(MMXSB6(op2)));
MMXSB7(result) = SaturateWordSToByteS(Bit16s(MMXSB7(op1)) + Bit16s(MMXSB7(op2)));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F ED */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PADDSW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXSW0(result) = SaturateDwordSToWordS(Bit32s(MMXSW0(op1)) + Bit32s(MMXSW0(op2)));
MMXSW1(result) = SaturateDwordSToWordS(Bit32s(MMXSW1(op1)) + Bit32s(MMXSW1(op2)));
MMXSW2(result) = SaturateDwordSToWordS(Bit32s(MMXSW2(op1)) + Bit32s(MMXSW2(op2)));
MMXSW3(result) = SaturateDwordSToWordS(Bit32s(MMXSW3(op1)) + Bit32s(MMXSW3(op2)));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F EE */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMAXSW_PqQq(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
2002-10-16 21:37:35 +04:00
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if(MMXSW0(op2) > MMXSW0(op1)) MMXSW0(op1) = MMXSW0(op2);
if(MMXSW1(op2) > MMXSW1(op1)) MMXSW1(op1) = MMXSW1(op2);
if(MMXSW2(op2) > MMXSW2(op1)) MMXSW2(op1) = MMXSW2(op2);
if(MMXSW3(op2) > MMXSW3(op1)) MMXSW3(op1) = MMXSW3(op2);
2002-10-16 21:37:35 +04:00
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F EF */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PXOR_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUQ(op1) ^= MMXUQ(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F F1 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSLLW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if(MMXUQ(op2) > 15) MMXUQ(op1) = 0;
else
{
Bit8u shift = MMXUB0(op2);
MMXUW0(op1) <<= shift;
MMXUW1(op1) <<= shift;
MMXUW2(op1) <<= shift;
MMXUW3(op1) <<= shift;
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F F2 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSLLD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if(MMXUQ(op2) > 31) MMXUQ(op1) = 0;
else
{
Bit8u shift = MMXUB0(op2);
MMXUD0(op1) <<= shift;
MMXUD1(op1) <<= shift;
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F F3 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSLLQ_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
if(MMXUQ(op2) > 63) {
MMXUQ(op1) = 0;
}
else {
MMXUQ(op1) <<= MMXUB0(op2);
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F F4 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMULUDQ_PqQq(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2002-10-16 21:37:35 +04:00
MMXUQ(result) = Bit64u(MMXUD0(op1)) * Bit64u(MMXUD0(op2));
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
}
/* 0F F5 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PMADDWD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2, result;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2002-11-15 18:51:12 +03:00
if(MMXUD0(op1) == 0x80008000 && MMXUD0(op2) == 0x80008000) {
MMXUD0(result) = 0x80000000;
}
else {
MMXUD0(result) = Bit32s(MMXSW0(op1))*Bit32s(MMXSW0(op2)) + Bit32s(MMXSW1(op1))*Bit32s(MMXSW1(op2));
}
2002-11-15 18:55:36 +03:00
if(MMXUD1(op1) == 0x80008000 && MMXUD1(op2) == 0x80008000) {
MMXUD1(result) = 0x80000000;
}
else {
MMXUD1(result) = Bit32s(MMXSW2(op1))*Bit32s(MMXSW2(op2)) + Bit32s(MMXSW3(op1))*Bit32s(MMXSW3(op2));
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), result);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F F6 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSADBW_PqQq(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
2003-09-26 23:20:17 +04:00
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
2002-10-16 21:37:35 +04:00
Bit16u temp = 0;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2002-10-16 21:37:35 +04:00
temp += abs(MMXUB0(op1) - MMXUB0(op2));
temp += abs(MMXUB1(op1) - MMXUB1(op2));
temp += abs(MMXUB2(op1) - MMXUB2(op2));
temp += abs(MMXUB3(op1) - MMXUB3(op2));
temp += abs(MMXUB4(op1) - MMXUB4(op2));
temp += abs(MMXUB5(op1) - MMXUB5(op2));
temp += abs(MMXUB6(op1) - MMXUB6(op2));
temp += abs(MMXUB7(op1) - MMXUB7(op2));
2007-12-27 02:07:44 +03:00
MMXUQ(op1) = (Bit64u) temp;
2002-10-16 21:37:35 +04:00
/* now write result back to destination */
2003-09-26 23:20:17 +04:00
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F F7 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MASKMOVQ_PqPRq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2003-04-23 22:57:57 +04:00
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2003-04-23 22:57:57 +04:00
2003-05-15 20:41:17 +04:00
bx_address rdi;
BxPackedMmxRegister op = BX_READ_MMX_REG(i->nnn()), tmp,
2003-04-23 22:57:57 +04:00
mask = BX_READ_MMX_REG(i->rm());
2003-05-15 20:41:17 +04:00
#if BX_SUPPORT_X86_64
2005-08-10 22:40:38 +04:00
if (i->as64L()) { /* 64 bit address mode */
2003-05-15 20:41:17 +04:00
rdi = RDI;
2008-02-03 00:46:54 +03:00
}
2003-05-15 20:41:17 +04:00
else
#endif
2003-04-23 22:57:57 +04:00
if (i->as32L()) {
2003-05-15 20:41:17 +04:00
rdi = EDI;
2003-04-23 22:57:57 +04:00
}
2003-05-15 20:41:17 +04:00
else { /* 16 bit address mode */
rdi = DI;
2003-04-23 22:57:57 +04:00
}
/* do read-modify-write for efficiency */
2008-04-16 09:41:43 +04:00
MMXUQ(tmp) = read_RMW_virtual_qword(i->seg(), rdi);
if (MMXUQ(mask) == 0) return;
if(MMXUB0(mask) & 0x80) MMXUB0(tmp) = MMXUB0(op);
if(MMXUB1(mask) & 0x80) MMXUB1(tmp) = MMXUB1(op);
if(MMXUB2(mask) & 0x80) MMXUB2(tmp) = MMXUB2(op);
if(MMXUB3(mask) & 0x80) MMXUB3(tmp) = MMXUB3(op);
if(MMXUB4(mask) & 0x80) MMXUB4(tmp) = MMXUB4(op);
if(MMXUB5(mask) & 0x80) MMXUB5(tmp) = MMXUB5(op);
if(MMXUB6(mask) & 0x80) MMXUB6(tmp) = MMXUB6(op);
if(MMXUB7(mask) & 0x80) MMXUB7(tmp) = MMXUB7(op);
write_RMW_virtual_qword(MMXUQ(tmp));
#endif
}
/* 0F F8 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSUBB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUB0(op1) -= MMXUB0(op2);
MMXUB1(op1) -= MMXUB1(op2);
MMXUB2(op1) -= MMXUB2(op2);
MMXUB3(op1) -= MMXUB3(op2);
MMXUB4(op1) -= MMXUB4(op2);
MMXUB5(op1) -= MMXUB5(op2);
MMXUB6(op1) -= MMXUB6(op2);
MMXUB7(op1) -= MMXUB7(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F F9 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSUBW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUW0(op1) -= MMXUW0(op2);
MMXUW1(op1) -= MMXUW1(op2);
MMXUW2(op1) -= MMXUW2(op2);
MMXUW3(op1) -= MMXUW3(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F FA */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSUBD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUD0(op1) -= MMXUD0(op2);
MMXUD1(op1) -= MMXUD1(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
2002-10-16 21:37:35 +04:00
}
/* 0F FB */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSUBQ_PqQq(bxInstruction_c *i)
2002-10-16 21:37:35 +04:00
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
2002-10-16 21:37:35 +04:00
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
2002-10-16 21:37:35 +04:00
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
2002-10-16 21:37:35 +04:00
MMXUQ(op1) -= MMXUQ(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F FC */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PADDB_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUB0(op1) += MMXUB0(op2);
MMXUB1(op1) += MMXUB1(op2);
MMXUB2(op1) += MMXUB2(op2);
MMXUB3(op1) += MMXUB3(op2);
MMXUB4(op1) += MMXUB4(op2);
MMXUB5(op1) += MMXUB5(op2);
MMXUB6(op1) += MMXUB6(op2);
MMXUB7(op1) += MMXUB7(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F FD */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PADDW_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUW0(op1) += MMXUW0(op2);
MMXUW1(op1) += MMXUW1(op2);
MMXUW2(op1) += MMXUW2(op2);
MMXUW3(op1) += MMXUW3(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F FE */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PADDD_PqQq(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BxPackedMmxRegister op1 = BX_READ_MMX_REG(i->nnn()), op2;
/* op2 is a register or memory reference */
if (i->modC0()) {
op2 = BX_READ_MMX_REG(i->rm());
}
else {
bx_address eaddr = BX_CPU_CALL_METHODR(i->ResolveModrm, (i));
/* pointer, segment address pair */
MMXUQ(op2) = read_virtual_qword(i->seg(), eaddr);
}
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
MMXUD0(op1) += MMXUD0(op2);
MMXUD1(op1) += MMXUD1(op2);
/* now write result back to destination */
BX_WRITE_MMX_REG(i->nnn(), op1);
#endif
}
/* 0F 71 GrpA 010 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSRLW_PqIb(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
BxPackedMmxRegister op = BX_READ_MMX_REG(i->rm());
Bit8u shift = i->Ib();
if(shift > 15) MMXUQ(op) = 0;
else
{
MMXUW0(op) >>= shift;
MMXUW1(op) >>= shift;
MMXUW2(op) >>= shift;
MMXUW3(op) >>= shift;
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->rm(), op);
#endif
}
/* 0F 71 GrpA 100 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSRAW_PqIb(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
BxPackedMmxRegister op = BX_READ_MMX_REG(i->rm()), result;
Bit8u shift = i->Ib();
if(shift == 0) return;
if(shift > 15) {
MMXUW0(result) = (MMXUW0(op) & 0x8000) ? 0xffff : 0;
MMXUW1(result) = (MMXUW1(op) & 0x8000) ? 0xffff : 0;
MMXUW2(result) = (MMXUW2(op) & 0x8000) ? 0xffff : 0;
MMXUW3(result) = (MMXUW3(op) & 0x8000) ? 0xffff : 0;
}
else {
MMXUW0(result) = MMXUW0(op) >> shift;
MMXUW1(result) = MMXUW1(op) >> shift;
MMXUW2(result) = MMXUW2(op) >> shift;
MMXUW3(result) = MMXUW3(op) >> shift;
if(MMXUW0(op) & 0x8000) MMXUW0(result) |= (0xffff << (16 - shift));
if(MMXUW1(op) & 0x8000) MMXUW1(result) |= (0xffff << (16 - shift));
if(MMXUW2(op) & 0x8000) MMXUW2(result) |= (0xffff << (16 - shift));
if(MMXUW3(op) & 0x8000) MMXUW3(result) |= (0xffff << (16 - shift));
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->rm(), result);
#endif
}
/* 0F 71 GrpA 110 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSLLW_PqIb(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
BxPackedMmxRegister op = BX_READ_MMX_REG(i->rm());
Bit8u shift = i->Ib();
if(shift > 15) MMXUQ(op) = 0;
else
{
MMXUW0(op) <<= shift;
MMXUW1(op) <<= shift;
MMXUW2(op) <<= shift;
MMXUW3(op) <<= shift;
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->rm(), op);
#endif
}
/* 0F 72 GrpA 010 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSRLD_PqIb(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
BxPackedMmxRegister op = BX_READ_MMX_REG(i->rm());
Bit8u shift = i->Ib();
if(shift > 31) MMXUQ(op) = 0;
else
{
MMXUD0(op) >>= shift;
MMXUD1(op) >>= shift;
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->rm(), op);
#endif
}
/* 0F 72 GrpA 100 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSRAD_PqIb(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
BxPackedMmxRegister op = BX_READ_MMX_REG(i->rm()), result;
Bit8u shift = i->Ib();
if(shift == 0) return;
if(shift > 31) {
MMXUD0(result) = (MMXUD0(op) & 0x80000000) ? 0xffffffff : 0;
MMXUD1(result) = (MMXUD1(op) & 0x80000000) ? 0xffffffff : 0;
}
else {
MMXUD0(result) = MMXUD0(op) >> shift;
MMXUD1(result) = MMXUD1(op) >> shift;
2008-02-03 00:46:54 +03:00
if(MMXUD0(op) & 0x80000000)
2002-10-16 21:37:35 +04:00
MMXUD0(result) |= (0xffffffff << (32 - shift));
2008-02-03 00:46:54 +03:00
if(MMXUD1(op) & 0x80000000)
2002-10-16 21:37:35 +04:00
MMXUD1(result) |= (0xffffffff << (32 - shift));
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->rm(), result);
#endif
}
/* 0F 72 GrpA 110 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSLLD_PqIb(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
BxPackedMmxRegister op = BX_READ_MMX_REG(i->rm());
Bit8u shift = i->Ib();
if(shift > 31) MMXUQ(op) = 0;
else
{
MMXUD0(op) <<= shift;
MMXUD1(op) <<= shift;
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->rm(), op);
#endif
}
/* 0F 73 GrpA 010 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSRLQ_PqIb(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
BxPackedMmxRegister op = BX_READ_MMX_REG(i->rm());
Bit8u shift = i->Ib();
if(shift > 63) {
MMXUQ(op) = 0;
}
else {
MMXUQ(op) >>= shift;
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->rm(), op);
#endif
}
/* 0F 73 GrpA 110 */
void BX_CPP_AttrRegparmN(1) BX_CPU_C::PSLLQ_PqIb(bxInstruction_c *i)
{
2010-03-01 21:53:53 +03:00
#if BX_CPU_LEVEL >= 5
2002-10-16 21:37:35 +04:00
BX_CPU_THIS_PTR prepareMMX();
BX_CPU_THIS_PTR prepareFPU2MMX(); /* FPU2MMX transition */
BxPackedMmxRegister op = BX_READ_MMX_REG(i->rm());
Bit8u shift = i->Ib();
if(shift > 63) {
MMXUQ(op) = 0;
}
else {
MMXUQ(op) <<= shift;
}
/* now write result back to destination */
BX_WRITE_MMX_REG(i->rm(), op);
#endif
}