2014-07-25 12:35:06 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
// $Id$
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2015-05-17 00:06:59 +03:00
|
|
|
// Copyright (c) 2011-2015 Stanislav Shwartsman
|
2014-07-25 12:35:06 +04:00
|
|
|
// Written by Stanislav Shwartsman [sshwarts at sourceforge net]
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA B 02110-1301 USA
|
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#define NEED_CPU_REG_SHORTCUTS 1
|
|
|
|
#include "bochs.h"
|
|
|
|
#include "cpu.h"
|
|
|
|
#define LOG_THIS BX_CPU_THIS_PTR
|
|
|
|
|
|
|
|
#if BX_SUPPORT_AVX
|
|
|
|
|
|
|
|
#include "simd_int.h"
|
|
|
|
|
|
|
|
/* VZEROUPPER: VEX.128.0F.77 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VZEROUPPER(bxInstruction_c *i)
|
|
|
|
{
|
2014-08-13 22:34:42 +04:00
|
|
|
for(unsigned index=0; index < 16; index++) // clear only 16 registers even if AVX-512 is present
|
2014-07-25 12:35:06 +04:00
|
|
|
{
|
|
|
|
if (index < 8 || long64_mode())
|
|
|
|
BX_CLEAR_AVX_HIGH128(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VZEROALL: VEX.256.0F.77 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VZEROALL(bxInstruction_c *i)
|
|
|
|
{
|
2014-08-13 22:34:42 +04:00
|
|
|
for(unsigned index=0; index < 16; index++) // clear only 16 registers even if AVX-512 is present
|
2014-07-25 12:35:06 +04:00
|
|
|
{
|
|
|
|
if (index < 8 || long64_mode())
|
2013-12-05 23:17:16 +04:00
|
|
|
BX_CLEAR_AVX_REG(index);
|
2014-07-25 12:35:06 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VMOVSS: VEX.F3.0F 10 (VEX.W ignore) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVSS_VssHpsWssR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedXmmRegister op = BX_READ_XMM_REG(i->src1());
|
|
|
|
|
|
|
|
op.xmm32u(0) = BX_READ_XMM_REG_LO_DWORD(i->src2());
|
|
|
|
|
|
|
|
BX_WRITE_XMM_REG_CLEAR_HIGH(i->dst(), op);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VMOVSS: VEX.F2.0F 10 (VEX.W ignore) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVSD_VsdHpdWsdR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedXmmRegister op;
|
|
|
|
|
|
|
|
op.xmm64u(0) = BX_READ_XMM_REG_LO_QWORD(i->src2());
|
|
|
|
op.xmm64u(1) = BX_READ_XMM_REG_HI_QWORD(i->src1());
|
|
|
|
|
|
|
|
BX_WRITE_XMM_REG_CLEAR_HIGH(i->dst(), op);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VMOVAPS: VEX 0F 28 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
/* VMOVAPD: VEX.66.0F 28 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
/* VMOVDQA: VEX.66.0F 6F (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVAPS_VpsWpsR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), BX_READ_AVX_REG(i->src()), i->getVL());
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVAPS_VpsWpsM(bxInstruction_c *i)
|
|
|
|
{
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
#if BX_SUPPORT_EVEX
|
|
|
|
if (len == BX_VL512)
|
|
|
|
read_virtual_zmmword_aligned(i->seg(), eaddr, &BX_READ_AVX_REG(i->dst()));
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
if (len == BX_VL256) {
|
|
|
|
read_virtual_ymmword_aligned(i->seg(), eaddr, &BX_READ_YMM_REG(i->dst()));
|
|
|
|
BX_CLEAR_AVX_HIGH256(i->dst());
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
read_virtual_xmmword_aligned(i->seg(), eaddr, &BX_READ_XMM_REG(i->dst()));
|
|
|
|
BX_CLEAR_AVX_HIGH128(i->dst());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VMOVUPS: VEX 0F 10 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
/* VMOVUPD: VEX.66.0F 10 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
/* VMOVDQU: VEX.F3.0F 6F (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVUPS_VpsWpsM(bxInstruction_c *i)
|
|
|
|
{
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
#if BX_SUPPORT_EVEX
|
|
|
|
if (len == BX_VL512)
|
|
|
|
read_virtual_zmmword(i->seg(), eaddr, &BX_READ_AVX_REG(i->dst()));
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
if (len == BX_VL256) {
|
|
|
|
read_virtual_ymmword(i->seg(), eaddr, &BX_READ_YMM_REG(i->dst()));
|
|
|
|
BX_CLEAR_AVX_HIGH256(i->dst());
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
read_virtual_xmmword(i->seg(), eaddr, &BX_READ_XMM_REG(i->dst()));
|
|
|
|
BX_CLEAR_AVX_HIGH128(i->dst());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VMOVUPS: VEX 0F 11 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
/* VMOVUPD: VEX.66.0F 11 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
/* VMOVDQU: VEX.66.0F 7F (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVUPS_WpsVpsM(bxInstruction_c *i)
|
|
|
|
{
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
#if BX_SUPPORT_EVEX
|
|
|
|
if (len == BX_VL512)
|
|
|
|
write_virtual_zmmword(i->seg(), eaddr, &BX_READ_AVX_REG(i->src()));
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
if (len == BX_VL256)
|
|
|
|
write_virtual_ymmword(i->seg(), eaddr, &BX_READ_YMM_REG(i->src()));
|
|
|
|
else
|
|
|
|
write_virtual_xmmword(i->seg(), eaddr, &BX_READ_XMM_REG(i->src()));
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VMOVAPS: VEX 0F 29 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
/* VMOVAPD: VEX.66.0F 29 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
/* VMOVDQA: VEX.66.0F 7F (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVAPS_WpsVpsM(bxInstruction_c *i)
|
|
|
|
{
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
#if BX_SUPPORT_EVEX
|
|
|
|
if (len == BX_VL512)
|
|
|
|
write_virtual_zmmword_aligned(i->seg(), eaddr, &BX_READ_AVX_REG(i->src()));
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
if (len == BX_VL256)
|
|
|
|
write_virtual_ymmword_aligned(i->seg(), eaddr, &BX_READ_YMM_REG(i->src()));
|
|
|
|
else
|
|
|
|
write_virtual_xmmword_aligned(i->seg(), eaddr, &BX_READ_XMM_REG(i->src()));
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VEX.F2.0F 12 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVDDUP_VpdWpdR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedAvxRegister op = BX_READ_AVX_REG(i->src());
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < QWORD_ELEMENTS(len); n+=2) {
|
|
|
|
op.vmm64u(n+1) = op.vmm64u(n);
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), op, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VEX.F3.0F 12 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVSLDUP_VpsWpsR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedAvxRegister op = BX_READ_AVX_REG(i->src());
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < DWORD_ELEMENTS(len); n+=2) {
|
|
|
|
op.vmm32u(n+1) = op.vmm32u(n);
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), op, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VEX.F3.0F 12 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVSHDUP_VpsWpsR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedAvxRegister op = BX_READ_AVX_REG(i->src());
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < DWORD_ELEMENTS(len); n+=2) {
|
|
|
|
op.vmm32u(n) = op.vmm32u(n+1);
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), op, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VEX.0F 12 (VEX.W ignore) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVHLPS_VpsHpsWps(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedXmmRegister op;
|
|
|
|
|
|
|
|
op.xmm64u(0) = BX_READ_XMM_REG_HI_QWORD(i->src2());
|
|
|
|
op.xmm64u(1) = BX_READ_XMM_REG_HI_QWORD(i->src1());
|
|
|
|
|
|
|
|
BX_WRITE_XMM_REG_CLEAR_HIGH(i->dst(), op);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VEX.66.0F 12 (VEX.W ignore) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVLPD_VpdHpdMq(bxInstruction_c *i)
|
|
|
|
{
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
|
|
|
|
BxPackedXmmRegister op;
|
|
|
|
|
|
|
|
op.xmm64u(0) = read_virtual_qword(i->seg(), eaddr);
|
|
|
|
op.xmm64u(1) = BX_READ_XMM_REG_HI_QWORD(i->src1());
|
|
|
|
|
|
|
|
BX_WRITE_XMM_REG_CLEAR_HIGH(i->dst(), op);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VEX.0F 16 (VEX.W ignore) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVLHPS_VpsHpsWps(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedXmmRegister op;
|
|
|
|
|
|
|
|
op.xmm64u(0) = BX_READ_XMM_REG_LO_QWORD(i->src1());
|
|
|
|
op.xmm64u(1) = BX_READ_XMM_REG_LO_QWORD(i->src2());
|
|
|
|
|
|
|
|
BX_WRITE_XMM_REG_CLEAR_HIGH(i->dst(), op);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VEX.66.0F 16 (VEX.W ignore) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVHPD_VpdHpdMq(bxInstruction_c *i)
|
|
|
|
{
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
|
|
|
|
BxPackedXmmRegister op;
|
|
|
|
|
|
|
|
op.xmm64u(0) = BX_READ_XMM_REG_LO_QWORD(i->src1());
|
|
|
|
op.xmm64u(1) = read_virtual_qword(i->seg(), eaddr);
|
|
|
|
|
|
|
|
BX_WRITE_XMM_REG_CLEAR_HIGH(i->dst(), op);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VEX.0F 50 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVMSKPS_GdUps(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister op = BX_READ_YMM_REG(i->src());
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
Bit32u mask = 0;
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++)
|
|
|
|
mask |= xmm_pmovmskd(&op.ymm128(n)) << (4*n);
|
|
|
|
|
|
|
|
BX_WRITE_32BIT_REGZ(i->dst(), mask);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VEX.66.0F 50 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMOVMSKPD_GdUpd(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister op = BX_READ_YMM_REG(i->src());
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
Bit32u mask = 0;
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++)
|
|
|
|
mask |= xmm_pmovmskq(&op.ymm128(n)) << (2*n);
|
|
|
|
|
|
|
|
BX_WRITE_32BIT_REGZ(i->dst(), mask);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VEX.66.0F 50 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VPMOVMSKB_GdUdq(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister op = BX_READ_YMM_REG(i->src());
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
Bit32u mask = 0;
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++)
|
|
|
|
mask |= xmm_pmovmskb(&op.ymm128(n)) << (16*n);
|
|
|
|
|
|
|
|
BX_WRITE_32BIT_REGZ(i->dst(), mask);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.0F.C6 (VEX.W ignore) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VSHUFPS_VpsHpsWpsIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedAvxRegister op1 = BX_READ_AVX_REG(i->src1());
|
|
|
|
BxPackedAvxRegister op2 = BX_READ_AVX_REG(i->src2()), result;
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++)
|
|
|
|
xmm_shufps(&result.vmm128(n), &op1.vmm128(n), &op2.vmm128(n), i->Ib());
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), result, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.C6 (VEX.W ignore) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VSHUFPD_VpdHpdWpdIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedAvxRegister op1 = BX_READ_AVX_REG(i->src1());
|
|
|
|
BxPackedAvxRegister op2 = BX_READ_AVX_REG(i->src2()), result;
|
|
|
|
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
Bit8u order = i->Ib();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++) {
|
|
|
|
xmm_shufpd(&result.vmm128(n), &op1.vmm128(n), &op2.vmm128(n), order);
|
|
|
|
order >>= 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), result, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.38.17 (VEX.W ignore, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VPTEST_VdqWdqR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister op1 = BX_READ_YMM_REG(i->dst()), op2 = BX_READ_YMM_REG(i->src());
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
unsigned result = EFlagsZFMask | EFlagsCFMask;
|
|
|
|
|
|
|
|
for (unsigned n=0; n < QWORD_ELEMENTS(len); n++) {
|
|
|
|
if ((op2.ymm64u(n) & op1.ymm64u(n)) != 0) result &= ~EFlagsZFMask;
|
|
|
|
if ((op2.ymm64u(n) & ~op1.ymm64u(n)) != 0) result &= ~EFlagsCFMask;
|
|
|
|
}
|
|
|
|
|
|
|
|
setEFlagsOSZAPC(result);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.256.66.0F.38.1A (VEX.W=0, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VBROADCASTF128_VdqMdq(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedAvxRegister dst;
|
|
|
|
BxPackedXmmRegister src;
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
read_virtual_xmmword(i->seg(), eaddr, &src);
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++) {
|
|
|
|
dst.vmm128(n) = src;
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), dst, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.3A 0C (VEX.W ignore) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VBLENDPS_VpsHpsWpsIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister op1 = BX_READ_YMM_REG(i->src1()), op2 = BX_READ_YMM_REG(i->src2());
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
Bit8u mask = i->Ib();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++) {
|
|
|
|
xmm_blendps(&op1.ymm128(n), &op2.ymm128(n), mask);
|
|
|
|
mask >>= 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_WRITE_YMM_REGZ_VLEN(i->dst(), op1, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.3A 0D (VEX.W ignore) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VBLENDPD_VpdHpdWpdIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister op1 = BX_READ_YMM_REG(i->src1()), op2 = BX_READ_YMM_REG(i->src2());
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
Bit8u mask = i->Ib();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++) {
|
|
|
|
xmm_blendpd(&op1.ymm128(n), &op2.ymm128(n), mask);
|
|
|
|
mask >>= 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_WRITE_YMM_REGZ_VLEN(i->dst(), op1, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.3A 4A (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VBLENDVPS_VpsHpsWpsIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister op1 = BX_READ_YMM_REG(i->src1()), op2 = BX_READ_YMM_REG(i->src2()),
|
|
|
|
mask = BX_READ_YMM_REG(i->src3());
|
|
|
|
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++)
|
|
|
|
xmm_blendvps(&op1.ymm128(n), &op2.ymm128(n), &mask.ymm128(n));
|
|
|
|
|
|
|
|
BX_WRITE_YMM_REGZ_VLEN(i->dst(), op1, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.3A 4B (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VBLENDVPD_VpdHpdWpdIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister op1 = BX_READ_YMM_REG(i->src1()), op2 = BX_READ_YMM_REG(i->src2()),
|
|
|
|
mask = BX_READ_YMM_REG(i->src3());
|
|
|
|
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++)
|
|
|
|
xmm_blendvpd(&op1.ymm128(n), &op2.ymm128(n), &mask.ymm128(n));
|
|
|
|
|
|
|
|
BX_WRITE_YMM_REGZ_VLEN(i->dst(), op1, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.3A 4C (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VPBLENDVB_VdqHdqWdqIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister op1 = BX_READ_YMM_REG(i->src1()), op2 = BX_READ_YMM_REG(i->src2()),
|
|
|
|
mask = BX_READ_YMM_REG(i->src3());
|
|
|
|
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++)
|
|
|
|
xmm_pblendvb(&op1.ymm128(n), &op2.ymm128(n), &mask.ymm128(n));
|
|
|
|
|
|
|
|
BX_WRITE_YMM_REGZ_VLEN(i->dst(), op1, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.3A 18 (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VINSERTF128_VdqHdqWdqIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedAvxRegister op = BX_READ_AVX_REG(i->src1());
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
unsigned offset = i->Ib() & (len-1);
|
|
|
|
|
|
|
|
op.vmm128(offset) = BX_READ_XMM_REG(i->src2());
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), op, len);
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.3A 19 (VEX.W=0, VEX.VVV #UD) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VEXTRACTF128_WdqVdqIbM(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
unsigned len = i->getVL(), offset = i->Ib() & (len - 1);
|
|
|
|
BxPackedXmmRegister op = BX_READ_AVX_REG_LANE(i->src(), offset);
|
|
|
|
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
write_virtual_xmmword(i->seg(), eaddr, &op);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VEXTRACTF128_WdqVdqIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
unsigned len = i->getVL(), offset = i->Ib() & (len - 1);
|
|
|
|
BX_WRITE_XMM_REG_CLEAR_HIGH(i->dst(), BX_READ_AVX_REG_LANE(i->src(), offset));
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.38 0C (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VPERMILPS_VpsHpsWpsR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedAvxRegister op1 = BX_READ_AVX_REG(i->src1());
|
|
|
|
BxPackedAvxRegister op2 = BX_READ_AVX_REG(i->src2()), result;
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++)
|
|
|
|
xmm_permilps(&result.vmm128(n), &op1.vmm128(n), &op2.vmm128(n));
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), result, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.3A 05 (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VPERMILPD_VpdHpdWpdR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedAvxRegister op1 = BX_READ_AVX_REG(i->src1());
|
|
|
|
BxPackedAvxRegister op2 = BX_READ_AVX_REG(i->src2()), result;
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++)
|
|
|
|
xmm_permilpd(&result.vmm128(n), &op1.vmm128(n), &op2.vmm128(n));
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), result, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.3A 04 (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VPERMILPS_VpsWpsIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedAvxRegister op1 = BX_READ_AVX_REG(i->src()), result;
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++)
|
|
|
|
xmm_shufps(&result.vmm128(n), &op1.vmm128(n), &op1.vmm128(n), i->Ib());
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), result, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.3A 05 (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VPERMILPD_VpdWpdIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedAvxRegister op1 = BX_READ_AVX_REG(i->src()), result;
|
|
|
|
unsigned len = i->getVL();
|
|
|
|
Bit8u order = i->Ib();
|
|
|
|
|
|
|
|
for (unsigned n=0; n < len; n++) {
|
|
|
|
xmm_shufpd(&result.vmm128(n), &op1.vmm128(n), &op1.vmm128(n), order);
|
|
|
|
order >>= 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), result, len);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.3A 06 (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VPERM2F128_VdqHdqWdqIbR(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister op1 = BX_READ_YMM_REG(i->src1());
|
|
|
|
BxPackedYmmRegister op2 = BX_READ_YMM_REG(i->src2()), result;
|
|
|
|
Bit8u order = i->Ib();
|
|
|
|
|
|
|
|
for (unsigned n=0;n<2;n++) {
|
|
|
|
|
|
|
|
if (order & 0x8) {
|
|
|
|
result.ymm128(n).clear();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (order & 0x2)
|
|
|
|
result.ymm128(n) = op2.ymm128(order & 0x1);
|
|
|
|
else
|
|
|
|
result.ymm128(n) = op1.ymm128(order & 0x1);
|
|
|
|
}
|
|
|
|
|
|
|
|
order >>= 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_WRITE_YMM_REGZ(i->dst(), result);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.38 2C (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMASKMOVPS_VpsHpsMps(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister mask = BX_READ_YMM_REG(i->src1());
|
|
|
|
BxPackedAvxRegister result;
|
|
|
|
|
|
|
|
unsigned opmask = xmm_pmovmskd(&mask.ymm128(1));
|
|
|
|
opmask <<= 4;
|
|
|
|
opmask |= xmm_pmovmskd(&mask.ymm128(0));
|
|
|
|
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
|
|
|
|
avx_masked_load32(i, eaddr, &result, opmask);
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), result, i->getVL());
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.38 2D (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMASKMOVPD_VpdHpdMpd(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister mask = BX_READ_YMM_REG(i->src1());
|
|
|
|
BxPackedAvxRegister result;
|
|
|
|
|
|
|
|
unsigned opmask = xmm_pmovmskq(&mask.ymm128(1));
|
|
|
|
opmask <<= 2;
|
|
|
|
opmask |= xmm_pmovmskq(&mask.ymm128(0));
|
|
|
|
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
|
|
|
|
avx_masked_load64(i, eaddr, &result, opmask);
|
|
|
|
|
|
|
|
BX_WRITE_AVX_REGZ(i->dst(), result, i->getVL());
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.38 2C (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMASKMOVPS_MpsHpsVps(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister mask = BX_READ_YMM_REG(i->src1());
|
|
|
|
|
|
|
|
unsigned opmask = xmm_pmovmskd(&mask.ymm128(1));
|
|
|
|
opmask <<= 4;
|
|
|
|
opmask |= xmm_pmovmskd(&mask.ymm128(0));
|
|
|
|
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
|
|
|
|
avx_masked_store32(i, eaddr, &BX_READ_AVX_REG(i->src2()), opmask);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Opcode: VEX.66.0F.38 2D (VEX.W=0) */
|
|
|
|
BX_INSF_TYPE BX_CPP_AttrRegparmN(1) BX_CPU_C::VMASKMOVPD_MpdHpdVpd(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
BxPackedYmmRegister mask = BX_READ_YMM_REG(i->src1());
|
|
|
|
|
|
|
|
unsigned opmask = xmm_pmovmskq(&mask.ymm128(1));
|
|
|
|
opmask <<= 2;
|
|
|
|
opmask |= xmm_pmovmskq(&mask.ymm128(0));
|
|
|
|
|
2015-05-17 00:06:59 +03:00
|
|
|
bx_address eaddr = BX_CPU_RESOLVE_ADDR(i);
|
2014-07-25 12:35:06 +04:00
|
|
|
|
|
|
|
avx_masked_store64(i, eaddr, &BX_READ_AVX_REG(i->src2()), opmask);
|
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // BX_SUPPORT_AVX
|